Start work on remote data sources; add infrastructure for host groups

This commit is contained in:
Garrett Mills 2023-10-09 23:40:14 -05:00
parent a7d273da1c
commit 17fda7b6ef
15 changed files with 549 additions and 68 deletions

View File

@ -1,4 +1,4 @@
import {OrderDirection, QueryRow} from '@extollo/lib' import {Awaitable, Constructable, Maybe, OrderDirection, QueryRow} from '@extollo/lib'
export enum FieldType { export enum FieldType {
text = 'text', text = 'text',
@ -34,17 +34,27 @@ const allResourceActions = [
ResourceAction.update, ResourceAction.delete, ResourceAction.update, ResourceAction.delete,
] ]
export type CobaltAction = {
slug: string,
title: string,
color: string,
icon: string,
type: 'route',
route: string,
overall?: boolean,
}
export type FieldBase = { export type FieldBase = {
key: string, key: string,
display: string, display: string,
type: FieldType, type: FieldType,
required: boolean, required: boolean,
queryable?: boolean,
sort?: 'asc' | 'desc', sort?: 'asc' | 'desc',
renderer?: Renderer, renderer?: Renderer,
readonly?: boolean, readonly?: boolean,
helpText?: string, helpText?: string,
placeholder?: string, placeholder?: string,
queryable?: boolean,
hideOn?: { hideOn?: {
form?: boolean, form?: boolean,
listing?: boolean, listing?: boolean,
@ -53,25 +63,45 @@ export type FieldBase = {
export type SelectOptions = {display: string, value: any}[] export type SelectOptions = {display: string, value: any}[]
export type RemoteSelectDefinition = {
displayFrom: string,
valueFrom: string,
source: DataSource,
}
export type FieldDefinition = FieldBase export type FieldDefinition = FieldBase
| FieldBase & { type: FieldType.select, options: SelectOptions } | FieldBase & { type: FieldType.select, options: SelectOptions }
| FieldBase & { type: FieldType.select } & RemoteSelectDefinition
export interface DataSourceController {
read(): Awaitable<QueryRow[]>
readOne(id: any): Awaitable<Maybe<QueryRow>>
insert(row: QueryRow): Awaitable<QueryRow>
update(id: any, row: QueryRow): Awaitable<QueryRow>
delete(id: any): Awaitable<void>
}
export type DataSource =
{ collection: string }
| { controller: Constructable<DataSourceController> }
| { method: Constructable<() => Awaitable<QueryRow[]>> }
export interface ResourceConfiguration { export interface ResourceConfiguration {
key: string, key: string,
collection: string, source: DataSource,
primaryKey: string, primaryKey: string,
orderField?: string, orderField?: string,
orderDirection?: OrderDirection, orderDirection?: OrderDirection,
generateKeyOnInsert?: () => string|number, generateKeyOnInsert?: () => string|number,
process?: { processBeforeInsert?: (row: QueryRow) => Awaitable<QueryRow>,
afterRead?: (row: QueryRow) => QueryRow, processAfterRead?: (row: QueryRow) => Awaitable<QueryRow>,
},
display: { display: {
field?: string, field?: string,
singular: string, singular: string,
plural: string, plural: string,
}, },
supportedActions: ResourceAction[], supportedActions: ResourceAction[],
otherActions: CobaltAction[],
fields: FieldDefinition[], fields: FieldDefinition[],
} }

View File

@ -1,4 +1,6 @@
import {FieldType, Renderer, ResourceAction, ResourceConfiguration} from '../cobalt' import {allResourceActions, FieldType, Renderer, ResourceAction, ResourceConfiguration} from '../cobalt'
import {constructable, QueryRow} from '@extollo/lib'
import {HostGroups} from '../http/controllers/resource/HostGroups.controller'
const tapRef = <T>(op: ((t: T) => unknown)): ((opd: T) => T) => { const tapRef = <T>(op: ((t: T) => unknown)): ((opd: T) => T) => {
return (opd: T): T => { return (opd: T): T => {
@ -9,10 +11,55 @@ const tapRef = <T>(op: ((t: T) => unknown)): ((opd: T) => T) => {
export default { export default {
resources: [ resources: [
{
key: 'hostgroup',
primaryKey: 'id',
source: { controller: constructable(HostGroups) },
display: {
field: 'name',
singular: 'Host Group',
plural: 'Host Groups',
},
orderField: 'name',
orderDirection: 'asc',
supportedActions: allResourceActions,
fields: [
{
key: 'id',
display: 'Host Group ID',
type: FieldType.number,
readonly: true,
required: false,
},
{
key: 'name',
display: 'Name',
type: FieldType.text,
required: true,
},
{
key: 'hosts',
display: 'Physical Hosts',
type: FieldType.select,
required: true,
displayFrom: 'pveDisplay',
valueFrom: 'pveHost',
hideOn: {
listing: true,
},
source: {
method: constructable<HostGroups>(HostGroups)
.tap(c => c.getBoundMethod('getPVEHosts')),
},
},
],
},
{ {
key: 'node', key: 'node',
primaryKey: 'id', primaryKey: 'id',
source: {
collection: 'p5x_nodes', collection: 'p5x_nodes',
},
display: { display: {
field: 'hostname', field: 'hostname',
singular: 'Instance', singular: 'Instance',
@ -21,9 +68,18 @@ export default {
orderField: 'pve_id', orderField: 'pve_id',
orderDirection: 'asc', orderDirection: 'asc',
supportedActions: [ResourceAction.read, ResourceAction.readOne], supportedActions: [ResourceAction.read, ResourceAction.readOne],
process: { otherActions: [
afterRead: tapRef(qr => qr.ip_display = `<pre><code>${qr.assigned_ip}/${qr.assigned_subnet}</code></pre>`), {
slug: 'provision',
title: 'Provision',
color: 'success',
icon: 'fa-plus',
type: 'route',
route: '/dash/provision',
overall: true,
}, },
],
processAfterRead: tapRef((qr: QueryRow) => qr.ip_display = `<pre><code>${qr.assigned_ip}/${qr.assigned_subnet}</code></pre>`),
fields: [ fields: [
{ {
key: 'id', key: 'id',

View File

@ -13,4 +13,10 @@ export class Dash extends Controller {
public main(): ResponseObject { public main(): ResponseObject {
return view('dash:template') return view('dash:template')
} }
public provision(): ResponseObject {
return view('dash:provision', {
})
}
} }

View File

@ -1,7 +1,7 @@
import { import {
api, api, ArrayIterable, AsyncCollection, AsyncPipe, Awaitable,
Builder, Builder,
collect, collect, Collection,
Config, Config,
Controller, Controller,
DatabaseService, DatabaseService,
@ -10,14 +10,64 @@ import {
HTTPError, HTTPError,
HTTPStatus, HTTPStatus,
Inject, Inject,
Injectable, Injectable, Iterable,
Maybe, Maybe,
QueryRow, QueryRow,
} from '@extollo/lib' } from '@extollo/lib'
import {FieldDefinition, FieldType, ResourceAction, ResourceConfiguration, SelectOptions} from '../../../cobalt' import {
DataSource,
FieldDefinition,
FieldType,
ResourceAction,
ResourceConfiguration,
SelectOptions,
} from '../../../cobalt'
const parser = require('any-date-parser') const parser = require('any-date-parser')
class AwaitableIterable<T> extends Iterable<T> {
public static lift<TInner>(source: Awaitable<Iterable<TInner>>): AwaitableIterable<TInner> {
return new AwaitableIterable<TInner>(source)
}
private resolved?: Iterable<T>
constructor(
protected source: Awaitable<Iterable<T>>
) {
super()
}
at(i: number): Promise<Maybe<T>> {
return this.resolve()
.then(x => x.at(i))
}
clone(): Iterable<T> {
return AwaitableIterable.lift(
this.resolve()
.then(x => x.clone())
)
}
range(start: number, end: number): Promise<Collection<T>> {
return this.resolve()
.then(x => x.range(start, end))
}
count(): Promise<number> {
return this.resolve()
.then(x => x.count())
}
private async resolve(): Promise<Iterable<T>> {
if ( !this.resolved ) {
this.resolved = await this.source
}
return this.resolved
}
}
@Injectable() @Injectable()
export class ResourceAPI extends Controller { export class ResourceAPI extends Controller {
@Inject() @Inject()
@ -26,8 +76,8 @@ export class ResourceAPI extends Controller {
@Inject() @Inject()
protected readonly db!: DatabaseService protected readonly db!: DatabaseService
public configure(key: string) { public async configure(key: string) {
const config = this.getResourceConfig(key) const config = await this.getResourceConfig(key)
if ( config ) { if ( config ) {
return api.one(config) return api.one(config)
} }
@ -36,54 +86,73 @@ export class ResourceAPI extends Controller {
} }
public async read(key: string) { public async read(key: string) {
const config = this.getResourceConfigOrFail(key) const config = await this.getResourceConfigOrFail(key)
this.checkAction(config, ResourceAction.read) this.checkAction(config, ResourceAction.read)
let result = await this.make<Builder>(Builder) let result: QueryRow[]
if ( hasOwnProperty(config.source, 'controller') ) {
const controller = config.source.controller.apply(this.request)
result = await controller.read()
} else if ( hasOwnProperty(config.source, 'method') ) {
const method = config.source.method.apply(this.request)
result = await method()
} else {
result = await this.make<Builder>(Builder)
.select(config.primaryKey, ...config.fields.filter(x => x.queryable !== false).map(x => x.key)) .select(config.primaryKey, ...config.fields.filter(x => x.queryable !== false).map(x => x.key))
.from(config.collection) .from(config.source.collection)
.limit(500) .limit(500)
.orderBy(config.orderField || config.primaryKey, config.orderDirection || 'asc') .orderBy(config.orderField || config.primaryKey, config.orderDirection || 'asc')
.connection(this.db.get()) .connection(this.db.get())
.get() .get()
.all() .all()
}
if ( config.process?.afterRead ) { if ( config.processAfterRead ) {
result = result.map(config.process.afterRead) result = await Promise.all(result.map(config.processAfterRead))
} }
return api.many(result) return api.many(result)
} }
public async readOne(key: string, id: number|string) { public async readOne(key: string, id: number|string) {
const config = this.getResourceConfigOrFail(key) const config = await this.getResourceConfigOrFail(key)
this.checkAction(config, ResourceAction.readOne) this.checkAction(config, ResourceAction.readOne)
let row = await this.make<Builder>(Builder) let row: Maybe<QueryRow>
if ( hasOwnProperty(config.source, 'controller') ) {
const controller = config.source.controller.apply(this.request)
row = await controller.readOne(id)
} else if ( hasOwnProperty(config.source, 'method') ) {
const method = config.source.method.apply(this.request)
row = collect(await method())
.firstWhere(config.primaryKey, '=', key)
} else {
row = await this.make<Builder>(Builder)
.select(config.primaryKey, ...config.fields.filter(x => x.queryable !== false).map(x => x.key)) .select(config.primaryKey, ...config.fields.filter(x => x.queryable !== false).map(x => x.key))
.from(config.collection) .from(config.source.collection)
.where(config.primaryKey, '=', id) .where(config.primaryKey, '=', id)
.limit(1) .limit(1)
.connection(this.db.get()) .connection(this.db.get())
.first() .first()
}
if ( !row ) { if ( !row ) {
throw new HTTPError(HTTPStatus.NOT_FOUND) throw new HTTPError(HTTPStatus.NOT_FOUND)
} }
if ( config.process?.afterRead ) { if ( config.processAfterRead ) {
row = config.process.afterRead(row) row = await config.processAfterRead(row)
} }
return api.one(row) return api.one(row)
} }
public async create(key: string, dataContainer: DataContainer) { public async create(key: string, dataContainer: DataContainer) {
const config = this.getResourceConfigOrFail(key) const config = await this.getResourceConfigOrFail(key)
this.checkAction(config, ResourceAction.create) this.checkAction(config, ResourceAction.create)
// Load input values // Load input values
const queryRow: QueryRow = {} let queryRow: QueryRow = {}
for ( const field of config.fields ) { for ( const field of config.fields ) {
const value = dataContainer.input(field.key) const value = dataContainer.input(field.key)
if ( field.required && typeof value === 'undefined' ) { if ( field.required && typeof value === 'undefined' ) {
@ -97,20 +166,32 @@ export class ResourceAPI extends Controller {
queryRow[config.primaryKey] = config.generateKeyOnInsert() queryRow[config.primaryKey] = config.generateKeyOnInsert()
} }
if ( config.processBeforeInsert ) {
queryRow = await config.processBeforeInsert(queryRow)
}
// Create insert query // Create insert query
const result = await this.make<Builder>(Builder) let result: Maybe<QueryRow>
.table(config.collection) if ( hasOwnProperty(config.source, 'controller') ) {
const controller = config.source.controller.apply(this.request)
result = await controller.insert(queryRow)
} else if ( hasOwnProperty(config.source, 'method') ) {
throw new Error('The "method" source type does not support creating records.')
} else {
result = await this.make<Builder>(Builder)
.table(config.source.collection)
.returning(config.primaryKey, ...config.fields.map(x => x.key)) .returning(config.primaryKey, ...config.fields.map(x => x.key))
.connection(this.db.get()) .connection(this.db.get())
.insert(queryRow) .insert(queryRow)
.then(x => x.rows.first()) .then(x => x.rows.first())
}
// Return result // Return result
return api.one(result) return api.one(result)
} }
public async update(key: string, id: number|string, dataContainer: DataContainer) { public async update(key: string, id: number|string, dataContainer: DataContainer) {
const config = this.getResourceConfigOrFail(key) const config = await this.getResourceConfigOrFail(key)
this.checkAction(config, ResourceAction.update) this.checkAction(config, ResourceAction.update)
// Load input values // Load input values
@ -128,31 +209,46 @@ export class ResourceAPI extends Controller {
await this.readOne(key, id) await this.readOne(key, id)
// Create update query // Create update query
const result = await this.make<Builder>(Builder) let result: Maybe<QueryRow>
.table(config.collection) if ( hasOwnProperty(config.source, 'controller') ) {
const controller = config.source.controller.apply(this.request)
result = await controller.update(id, queryRow)
} else if ( hasOwnProperty(config.source, 'method') ) {
throw new Error('The "method" source type does not support updating records.')
} else {
result = await this.make<Builder>(Builder)
.table(config.source.collection)
.returning(config.primaryKey, ...config.fields.map(x => x.key)) .returning(config.primaryKey, ...config.fields.map(x => x.key))
.connection(this.db.get()) .connection(this.db.get())
.where(config.primaryKey, '=', id) .where(config.primaryKey, '=', id)
.update(queryRow) .update(queryRow)
.then(x => x.rows.first()) .then(x => x.rows.first())
}
// Return the result // Return the result
return api.one(result) return api.one(result)
} }
public async delete(key: string, id: number|string) { public async delete(key: string, id: number|string) {
const config = this.getResourceConfigOrFail(key) const config = await this.getResourceConfigOrFail(key)
this.checkAction(config, ResourceAction.delete) this.checkAction(config, ResourceAction.delete)
// Make sure the row exists // Make sure the row exists
await this.readOne(key, id) await this.readOne(key, id)
// Execute the query // Execute the query
if ( hasOwnProperty(config.source, 'controller') ) {
const controller = config.source.controller.apply(this.request)
await controller.delete(id)
} else if ( hasOwnProperty(config.source, 'method') ) {
throw new Error('The "method" source type does not support deleting records.')
} else {
await this.make<Builder>(Builder) await this.make<Builder>(Builder)
.table(config.collection) .table(config.source.collection)
.connection(this.db.get()) .connection(this.db.get())
.where(config.primaryKey, '=', id) .where(config.primaryKey, '=', id)
.delete() .delete()
}
return { success: true } return { success: true }
} }
@ -224,20 +320,76 @@ export class ResourceAPI extends Controller {
} }
} }
protected getResourceConfigOrFail(key: string): ResourceConfiguration { protected async getResourceConfigOrFail(key: string): Promise<ResourceConfiguration> {
const config = this.getResourceConfig(key) const config = await this.getResourceConfig(key)
if ( !config ) { if ( !config ) {
throw new HTTPError(HTTPStatus.NOT_FOUND) throw new HTTPError(HTTPStatus.NOT_FOUND)
} }
return config return config
} }
protected getResourceConfig(key: string): Maybe<ResourceConfiguration> { protected async getResourceConfig(key: string): Promise<Maybe<ResourceConfiguration>> {
const configs = this.config.get('cobalt.resources') as ResourceConfiguration[] const configs = this.config.get('cobalt.resources') as ResourceConfiguration[]
for ( const config of configs ) { let config: Maybe<ResourceConfiguration> = undefined
if ( config.key === key ) { for ( const match of configs ) {
if ( match.key === key ) {
config = match
break
}
}
if ( !config ) {
return config return config
} }
// Resolve `select` fields with remote sources
config.fields = await collect(config.fields)
.map(async field => {
if ( field.type === FieldType.select && hasOwnProperty(field, 'source') ) {
return {
...field,
options: await this.readSource(field.source)
.map(qr => ({
display: qr[field.displayFrom],
value: qr[field.valueFrom],
}))
.then(c => c.all())
} as FieldDefinition
} }
return field
})
.awaitAll()
.then(x => x.all())
return config
}
protected readSource(source: DataSource): AsyncCollection<QueryRow> {
let iter: Iterable<QueryRow>
if ( hasOwnProperty(source, 'controller') ) {
iter = AwaitableIterable.lift(
AsyncPipe.wrap(source.controller.apply(this.request))
.tap(controller => controller.read())
.tap(rows => new ArrayIterable(rows))
.resolve()
)
} else if ( hasOwnProperty(source, 'method') ) {
iter = AwaitableIterable.lift(
AsyncPipe.wrap(source.method.apply(this.request))
.tap(method => method())
.tap(rows => new ArrayIterable(rows))
.resolve()
)
} else {
iter = this.make<Builder>(Builder)
.from(source.collection)
.limit(500)
.connection(this.db.get())
.getResultIterable()
}
return new AsyncCollection(iter)
} }
} }

View File

@ -0,0 +1,80 @@
import {AsyncPipe, Awaitable, Controller, Inject, Injectable, Maybe, QueryRow, Safe} from '@extollo/lib'
import {DataSourceController} from '../../../cobalt'
import {HostGroup} from '../../../models/HostGroup.model'
import {Provisioner} from '../../../services/Provisioner.service'
@Injectable()
export class HostGroups extends Controller implements DataSourceController {
@Inject()
protected readonly provisioner!: Provisioner
read(): Awaitable<QueryRow[]> {
return HostGroup.query<HostGroup>()
.orderBy('name')
.get()
.collect()
.then(x => x.mapCall('toCobalt').awaitAll())
.then(x => x.all())
}
readOne(id: any): Awaitable<Maybe<QueryRow>> {
return HostGroup.query<HostGroup>()
.whereKey(id)
.first()
.then(x => x?.toCobalt())
}
insert(row: QueryRow): Awaitable<QueryRow> {
const hosts = row.hosts
if ( !Array.isArray(hosts) || !hosts.every(i => typeof i === 'string') ) {
throw new Error('Invalid hosts: must be number[]')
}
return AsyncPipe.wrap(this.request.makeNew<HostGroup>(HostGroup))
.peek(g => g.name = (new Safe(row.name)).string())
.peek(g => g.save())
.peek(g => g.setHosts(hosts))
.tap(g => g.toCobalt())
.resolve()
}
async update(id: any, row: QueryRow): Promise<QueryRow> {
const hosts = row.hosts
if ( !Array.isArray(hosts) || !hosts.every(i => typeof i === 'string') ) {
throw new Error('Invalid hosts: must be number[]')
}
const hostGroup = await HostGroup.query<HostGroup>()
.whereKey(id)
.first()
if ( !hostGroup ) {
throw new Error('Invalid host group ID.')
}
hostGroup.name = (new Safe(row.name)).string()
await hostGroup.save()
await hostGroup.setHosts(hosts)
return hostGroup.toCobalt()
}
async delete(id: any): Promise<void> {
HostGroup.query<HostGroup>()
.whereKey(id)
.delete()
}
async getPVEHosts(): Promise<QueryRow[]> {
return AsyncPipe.wrap(this.provisioner)
.tap(p => p.getApi())
.tap(api => api.nodes.$get())
.tap(nodes =>
nodes.map(node => ({
pveHost: node.id || node.node,
pveDisplay: `${node.node} (CPUs: ${node.maxcpu}, RAM: ${Math.floor((node.maxmem || 0) / 1000000000)})`
}))
)
.resolve()
}
}

View File

@ -30,6 +30,9 @@ Route
.alias('@dash') .alias('@dash')
.calls<Dash>(Dash, dash => dash.main) .calls<Dash>(Dash, dash => dash.main)
Route.get('/provision')
.calls<Dash>(Dash, dash => dash.provision)
Route.group('/cobalt/resource', () => { Route.group('/cobalt/resource', () => {
Route.get('/:key/configure') Route.get('/:key/configure')
.parameterMiddleware(parseKey) .parameterMiddleware(parseKey)

View File

@ -0,0 +1,46 @@
import {Injectable, Migration, Inject, DatabaseService, FieldType} from '@extollo/lib'
/**
* CreatePhysicalHostsTableMigration
* ----------------------------------
* Put some description here.
*/
@Injectable()
export default class CreatePhysicalHostsTableMigration extends Migration {
@Inject()
protected readonly db!: DatabaseService
/**
* Apply the migration.
*/
async up(): Promise<void> {
const schema = this.db.get().schema()
const hostGroups = await schema.table('p5x_host_groups')
hostGroups.primaryKey('id')
hostGroups.column('name').type(FieldType.varchar)
await schema.commit(hostGroups)
const hostGroupHosts = await schema.table('p5x_host_group_hosts')
hostGroupHosts.primaryKey('id')
hostGroupHosts.column('pve_host').type(FieldType.varchar)
hostGroupHosts.column('host_group_id').type(FieldType.int)
hostGroupHosts.index('hostgrouphostid').field('host_group_id')
await schema.commit(hostGroupHosts)
}
/**
* Undo the migration.
*/
async down(): Promise<void> {
const schema = this.db.get().schema()
const hostGroups = await schema.table('p5x_host_groups')
hostGroups.dropIfExists()
await schema.commit(hostGroups)
const hostGroupHosts = await schema.table('p5x_host_group_hosts')
hostGroupHosts.dropIfExists()
await schema.commit(hostGroupHosts)
}
}

View File

@ -0,0 +1,48 @@
import {Injectable, Model, Field, FieldType, collect, Related, QueryRow} from '@extollo/lib'
import {HostGroupHost} from './HostGroupHost.model'
/**
* HostGroup Model
* -----------------------------------
* Put some description here.
*/
@Injectable()
export class HostGroup extends Model<HostGroup> {
protected static table = 'p5x_host_groups'
protected static key = 'id'
@Field(FieldType.serial)
public id?: number
@Field(FieldType.varchar)
public name!: string
@Related()
public hosts() {
return this.hasMany(HostGroupHost, 'id', 'hostGroupId')
}
public async setHosts(pveHosts: string[]): Promise<void> {
if ( !this.id ) {
throw new Error('Cannot setHosts on HostGroup that has not yet been persisted')
}
// Remove any existing hosts
await this.hosts().query().delete()
// Insert the host records
const rows = pveHosts.map(pveHost => ({
pveHost,
hostGroupId: this.key(),
}))
await HostGroupHost.query<HostGroupHost>()
.insert(rows)
}
public async toCobalt(): Promise<QueryRow> {
const obj = this.toQueryRow()
obj.hosts = await this.hosts().get().then(x => x.pluck('pveHost'))
return obj
}
}

View File

@ -0,0 +1,21 @@
import {Field, FieldType, Injectable, Model} from '@extollo/lib'
/**
* HostGroupHost Model
* -----------------------------------
* Put some description here.
*/
@Injectable()
export class HostGroupHost extends Model<HostGroupHost> {
protected static table = 'p5x_host_group_hosts'
protected static key = 'id'
@Field(FieldType.serial)
public id?: number
@Field(FieldType.varchar, 'pve_host')
public pveHost!: string
@Field(FieldType.int, 'host_group_id')
public hostGroupId!: number
}

View File

@ -32,6 +32,7 @@ export class Resource {
this.key = key this.key = key
this.configuration = { this.configuration = {
supportedActions: [], supportedActions: [],
otherActions: [],
} }
} }
@ -51,6 +52,10 @@ export class Resource {
return this.configuration.supportedActions return this.configuration.supportedActions
} }
getOtherActions() {
return this.configuration.otherActions || []
}
supports(action) { supports(action) {
return this.getSupportedActions().includes(action) return this.getSupportedActions().includes(action)
} }

View File

@ -21,7 +21,6 @@ const template = `
:id="id + field.key" :id="id + field.key"
:aria-describedby="id + field.key + '_help'" :aria-describedby="id + field.key + '_help'"
:readonly="mode === 'view' || field.readonly" :readonly="mode === 'view' || field.readonly"
:onclick="(mode === 'view' || field.readonly) ? 'return false;' : ''"
:required="field.required" :required="field.required"
> >
<label for="id + field.key">{{ field.display }}</label> <label for="id + field.key">{{ field.display }}</label>

View File

@ -119,7 +119,7 @@ export class ListingComponent extends Component {
action: 'update', action: 'update',
defer: true, defer: true,
}) })
} else if ( !reload && this.resource.supports(ResourceActions.read) ) { } else if ( !reload && this.resource.supports(ResourceActions.readOne) ) {
this.actions.push({ this.actions.push({
title: 'View', title: 'View',
color: 'primary', color: 'primary',
@ -143,6 +143,13 @@ export class ListingComponent extends Component {
}) })
} }
for ( const action of this.resource.getOtherActions() ) {
this.actions.push({
resource: this.resource.key,
...action,
})
}
this.rows = await this.resource.read() this.rows = await this.resource.read()
this.rows.forEach((row, idx) => row.idx = idx) this.rows.forEach((row, idx) => row.idx = idx)

View File

@ -14,6 +14,7 @@ export class ActionService {
async perform(action, data, onComplete = () => {}) { async perform(action, data, onComplete = () => {}) {
if ( action.type === 'back' ) this.goBack() if ( action.type === 'back' ) this.goBack()
if ( action.type === 'resource' ) await this.handleResourceAction(action, data, onComplete) if ( action.type === 'resource' ) await this.handleResourceAction(action, data, onComplete)
if ( action.type === 'route' ) await this.handleRouteAction(action, data, onComplete)
} }
async handleResourceAction(action, data, onComplete) { async handleResourceAction(action, data, onComplete) {
@ -60,6 +61,25 @@ export class ActionService {
location.assign(Session.get().url(`dash/cobalt/form/${key}${id ? '/' + id : ''}${view ? '/view' : ''}`)) location.assign(Session.get().url(`dash/cobalt/form/${key}${id ? '/' + id : ''}${view ? '/view' : ''}`))
} }
async handleRouteAction(action, data, onComplete = () => {}) {
let route = Session.get().url(action.route)
route = this.replacePlaceholders(route, { resourceKey: action.resource })
route = this.replacePlaceholders(route, data)
location.assign(route)
}
replacePlaceholders(str, data) {
for ( const key in data ) {
if ( !data.hasOwnProperty(key) ) {
continue
}
str = String(str).replaceAll(':' + key, data[key])
}
return str
}
goBack() { goBack() {
history.back() history.back()
} }

View File

@ -0,0 +1,4 @@
extends template
block content
h1 Provision!

View File

@ -21,6 +21,10 @@ head
| #{config('app.name')} | #{config('app.name')}
hr.sidebar-divider hr.sidebar-divider
.sidebar-heading Cluster .sidebar-heading Cluster
li.nav-item
a.nav-link(href=route('dash/cobalt/listing/hostgroup'))
i.fas.fa-fw.fa-server
span Host Groups
li.nav-item li.nav-item
a.nav-link(href=route('dash/cobalt/listing/node')) a.nav-link(href=route('dash/cobalt/listing/node'))
i.fas.fa-fw.fa-server i.fas.fa-fw.fa-server