mirror of
https://github.com/Start9Labs/start-sdk.git
synced 2026-03-26 02:11:56 +00:00
wip: Creating an sdk builder that has all the generics we need in one place
This commit is contained in:
248
lib/StartSDK.ts
Normal file
248
lib/StartSDK.ts
Normal file
@@ -0,0 +1,248 @@
|
||||
import { AnyParser } from "ts-matches"
|
||||
import { SDKManifest } from "./manifest/ManifestTypes"
|
||||
import { RequiredDefault, Value } from "./config/builder/value"
|
||||
import { Config, ExtractConfigType, LazyBuild } from "./config/builder/config"
|
||||
import {
|
||||
DefaultString,
|
||||
Pattern,
|
||||
RandomString,
|
||||
ValueSpecDatetime,
|
||||
ValueSpecText,
|
||||
} from "./config/configTypes"
|
||||
import { Variants } from "./config/builder/variants"
|
||||
import { createAction } from "./actions/createAction"
|
||||
import { ActionMetaData, Effects, ActionResult, Metadata } from "./types"
|
||||
import { Utils } from "./util"
|
||||
|
||||
// prettier-ignore
|
||||
type AnyNeverCond<T extends any[], Then, Else> =
|
||||
T extends [] ? Else :
|
||||
T extends [never, ...Array<any>] ? Then :
|
||||
T extends [any, ...infer U] ? AnyNeverCond<U,Then, Else> :
|
||||
never
|
||||
|
||||
class StartSDK<Manifest extends SDKManifest, Store> {
|
||||
private constructor() {}
|
||||
private anyOf<A>(
|
||||
a: A,
|
||||
): AnyNeverCond<[Manifest, Store], "Build not ready", A> {
|
||||
return a as any
|
||||
}
|
||||
|
||||
static of() {
|
||||
return new StartSDK<never, never>()
|
||||
}
|
||||
withManifest<Manifest extends SDKManifest = never>() {
|
||||
return new StartSDK<Manifest, Store>()
|
||||
}
|
||||
withStore<Store extends Record<string, any>>() {
|
||||
return new StartSDK<Manifest, Store>()
|
||||
}
|
||||
|
||||
build() {
|
||||
return this.anyOf({
|
||||
// TODO AutoConfig
|
||||
// TODO Backup
|
||||
// TODO Config
|
||||
// TODO configConstants
|
||||
// TODO configDependencies
|
||||
createAction: <
|
||||
Store,
|
||||
ConfigType extends
|
||||
| Record<string, any>
|
||||
| Config<any, any>
|
||||
| Config<any, never>,
|
||||
Type extends Record<string, any> = ExtractConfigType<ConfigType>,
|
||||
>(
|
||||
metaData: Omit<ActionMetaData, "input"> & {
|
||||
input: Config<Type, Store> | Config<Type, never>
|
||||
},
|
||||
fn: (options: {
|
||||
effects: Effects
|
||||
utils: Utils<Store>
|
||||
input: Type
|
||||
}) => Promise<ActionResult>,
|
||||
) => createAction<Store, ConfigType, Type>(metaData, fn),
|
||||
// TODO Daemons
|
||||
// TODO HealthCheck
|
||||
// TODO healthCheckFns
|
||||
// TODO List
|
||||
// TODO mainNetwork
|
||||
// TODO Migration
|
||||
// TODO setupActions
|
||||
// TODO setupAutoConfig
|
||||
// TODO setupBackup
|
||||
// TODO setupInit
|
||||
// TODO setupInstall
|
||||
// TODO setupMain
|
||||
// TODO setupManifest
|
||||
// TODO setupMigrations
|
||||
// TODO setupUninstall
|
||||
// TODO trigger changeOnFirstSuccess, cooldown, default
|
||||
Value: {
|
||||
toggle: Value.toggle,
|
||||
text: Value.text,
|
||||
textarea: Value.textarea,
|
||||
number: Value.number,
|
||||
color: Value.color,
|
||||
datetime: Value.datetime,
|
||||
select: Value.select,
|
||||
multiselect: Value.multiselect,
|
||||
object: Value.object,
|
||||
union: Value.union,
|
||||
list: Value.list,
|
||||
dynamicToggle: (
|
||||
a: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
default: boolean
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicToggle<Store>(a),
|
||||
dynamicText: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: RequiredDefault<DefaultString>
|
||||
|
||||
/** Default = false */
|
||||
masked?: boolean
|
||||
placeholder?: string | null
|
||||
minLength?: number | null
|
||||
maxLength?: number | null
|
||||
patterns?: Pattern[]
|
||||
/** Default = 'text' */
|
||||
inputmode?: ValueSpecText["inputmode"]
|
||||
generate?: null | RandomString
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicText<Store>(getA),
|
||||
dynamicTextarea: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: boolean
|
||||
minLength?: number | null
|
||||
maxLength?: number | null
|
||||
placeholder?: string | null
|
||||
disabled?: false | string
|
||||
generate?: null | RandomString
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicTextarea<Store>(getA),
|
||||
dynamicNumber: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: RequiredDefault<number>
|
||||
min?: number | null
|
||||
max?: number | null
|
||||
/** Default = '1' */
|
||||
step?: string | null
|
||||
integer: boolean
|
||||
units?: string | null
|
||||
placeholder?: string | null
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicNumber<Store>(getA),
|
||||
dynamicColor: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: RequiredDefault<string>
|
||||
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicColor<Store>(getA),
|
||||
dynamicDatetime: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: RequiredDefault<string>
|
||||
/** Default = 'datetime-local' */
|
||||
inputmode?: ValueSpecDatetime["inputmode"]
|
||||
min?: string | null
|
||||
max?: string | null
|
||||
step?: string | null
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicDatetime<Store>(getA),
|
||||
dynamicSelect: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: RequiredDefault<string>
|
||||
values: Record<string, string>
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicSelect<Store>(getA),
|
||||
dynamicMultiselect: (
|
||||
getA: LazyBuild<
|
||||
Store,
|
||||
{
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
default: string[]
|
||||
values: Record<string, string>
|
||||
minLength?: number | null
|
||||
maxLength?: number | null
|
||||
disabled?: false | string
|
||||
}
|
||||
>,
|
||||
) => Value.dynamicMultiselect<Store>(getA),
|
||||
filteredUnion: <
|
||||
Required extends RequiredDefault<string>,
|
||||
Type extends Record<string, any>,
|
||||
>(
|
||||
getDisabledFn: LazyBuild<Store, string[]>,
|
||||
a: {
|
||||
name: string
|
||||
description?: string | null
|
||||
warning?: string | null
|
||||
required: Required
|
||||
},
|
||||
aVariants: Variants<Type, Store> | Variants<Type, never>,
|
||||
) =>
|
||||
Value.filteredUnion<Required, Type, Store>(
|
||||
getDisabledFn,
|
||||
a,
|
||||
aVariants,
|
||||
),
|
||||
},
|
||||
// TODO Variants
|
||||
})
|
||||
}
|
||||
}
|
||||
// TODO Test output.ts with sdk
|
||||
|
||||
// const test = StartSDK.of()
|
||||
// .withManifest<any>()
|
||||
// .withStore<{}>()
|
||||
// .Value.dynamicToggle({} as any, {} as any)
|
||||
@@ -1,56 +1,51 @@
|
||||
import { Config, ExtractConfigType } from "../config/builder/config"
|
||||
import { ActionMetaData, ActionResult, Effects, ExportedAction } from "../types"
|
||||
import { Utils, createUtils, utils } from "../util"
|
||||
import { WrapperDataContract } from "../wrapperData/wrapperDataContract"
|
||||
|
||||
export class CreatedAction<
|
||||
WD,
|
||||
ConfigType extends Record<string, any> | Config<any, WD> | Config<any, never>,
|
||||
Store,
|
||||
ConfigType extends
|
||||
| Record<string, any>
|
||||
| Config<any, Store>
|
||||
| Config<any, never>,
|
||||
Type extends Record<string, any> = ExtractConfigType<ConfigType>,
|
||||
> {
|
||||
private constructor(
|
||||
readonly wrapperDataContract: WrapperDataContract<WD>,
|
||||
public readonly myMetaData: ActionMetaData,
|
||||
readonly fn: (options: {
|
||||
effects: Effects
|
||||
utils: Utils<WD>
|
||||
utils: Utils<Store>
|
||||
input: Type
|
||||
}) => Promise<ActionResult>,
|
||||
readonly input: Config<Type, WD> | Config<Type, never>,
|
||||
readonly input: Config<Type, Store> | Config<Type, never>,
|
||||
) {}
|
||||
public validator = this.input.validator
|
||||
|
||||
static of<
|
||||
WD,
|
||||
Store,
|
||||
ConfigType extends
|
||||
| Record<string, any>
|
||||
| Config<any, any>
|
||||
| Config<any, never>,
|
||||
Type extends Record<string, any> = ExtractConfigType<ConfigType>,
|
||||
>(
|
||||
wrapperDataContract: WrapperDataContract<WD>,
|
||||
metaData: Omit<ActionMetaData, "input"> & {
|
||||
input: Config<Type, WD> | Config<Type, never>
|
||||
input: Config<Type, Store> | Config<Type, never>
|
||||
},
|
||||
fn: (options: {
|
||||
effects: Effects
|
||||
utils: Utils<WD>
|
||||
utils: Utils<Store>
|
||||
input: Type
|
||||
}) => Promise<ActionResult>,
|
||||
) {
|
||||
const { input, ...rest } = metaData
|
||||
return new CreatedAction<WD, ConfigType, Type>(
|
||||
wrapperDataContract,
|
||||
rest,
|
||||
fn,
|
||||
input,
|
||||
)
|
||||
return new CreatedAction<Store, ConfigType, Type>(rest, fn, input)
|
||||
}
|
||||
|
||||
exportedAction: ExportedAction = ({ effects, input }) => {
|
||||
return this.fn({
|
||||
effects,
|
||||
utils: createUtils(this.wrapperDataContract, effects),
|
||||
utils: createUtils(effects),
|
||||
input: this.validator.unsafeCast(input),
|
||||
})
|
||||
}
|
||||
@@ -58,7 +53,7 @@ export class CreatedAction<
|
||||
run = async ({ effects, input }: { effects: Effects; input?: Type }) => {
|
||||
return this.fn({
|
||||
effects,
|
||||
utils: createUtils(this.wrapperDataContract, effects),
|
||||
utils: createUtils(effects),
|
||||
input: this.validator.unsafeCast(input),
|
||||
})
|
||||
}
|
||||
@@ -66,7 +61,7 @@ export class CreatedAction<
|
||||
async getConfig({ effects }: { effects: Effects }) {
|
||||
return this.input.build({
|
||||
effects,
|
||||
utils: createUtils(this.wrapperDataContract, effects) as any,
|
||||
utils: createUtils(effects) as any,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
@@ -24,9 +24,8 @@ import {
|
||||
unknown,
|
||||
} from "ts-matches"
|
||||
import { once } from "../../util/once"
|
||||
import { WrapperDataContract } from "../../wrapperData/wrapperDataContract"
|
||||
|
||||
type RequiredDefault<A> =
|
||||
export type RequiredDefault<A> =
|
||||
| false
|
||||
| {
|
||||
default: A | null
|
||||
@@ -96,7 +95,7 @@ const username = Value.string({
|
||||
```
|
||||
*/
|
||||
export class Value<Type, WD> {
|
||||
private constructor(
|
||||
protected constructor(
|
||||
public build: LazyBuild<WD, ValueSpec>,
|
||||
public validator: Parser<unknown, Type>,
|
||||
) {}
|
||||
@@ -122,7 +121,6 @@ export class Value<Type, WD> {
|
||||
)
|
||||
}
|
||||
static dynamicToggle<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
a: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -186,7 +184,6 @@ export class Value<Type, WD> {
|
||||
)
|
||||
}
|
||||
static dynamicText<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getA: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -258,7 +255,6 @@ export class Value<Type, WD> {
|
||||
)
|
||||
}
|
||||
static dynamicTextarea<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getA: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -325,7 +321,6 @@ export class Value<Type, WD> {
|
||||
)
|
||||
}
|
||||
static dynamicNumber<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getA: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -387,7 +382,6 @@ export class Value<Type, WD> {
|
||||
}
|
||||
|
||||
static dynamicColor<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getA: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -445,7 +439,6 @@ export class Value<Type, WD> {
|
||||
)
|
||||
}
|
||||
static dynamicDatetime<WD = never>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getA: LazyBuild<
|
||||
WD,
|
||||
{
|
||||
@@ -642,7 +635,6 @@ export class Value<Type, WD> {
|
||||
Type extends Record<string, any>,
|
||||
WD = never,
|
||||
>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
getDisabledFn: LazyBuild<WD, string[]>,
|
||||
a: {
|
||||
name: string
|
||||
|
||||
@@ -1,12 +1,14 @@
|
||||
import { SDKManifest } from "../manifest/ManifestTypes"
|
||||
import { Dependency } from "../types"
|
||||
|
||||
export type Dependencies<T extends SDKManifest> = {
|
||||
export type ConfigDependencies<T extends SDKManifest> = {
|
||||
exists(id: keyof T["dependencies"]): Dependency
|
||||
running(id: keyof T["dependencies"]): Dependency
|
||||
}
|
||||
|
||||
export const dependenciesSet = <T extends SDKManifest>(): Dependencies<T> => ({
|
||||
export const configDependenciesSet = <
|
||||
T extends SDKManifest,
|
||||
>(): ConfigDependencies<T> => ({
|
||||
exists(id: keyof T["dependencies"]) {
|
||||
return {
|
||||
id,
|
||||
@@ -1,5 +1,5 @@
|
||||
import "./builder"
|
||||
|
||||
import "./setupConfig"
|
||||
import "./dependencies"
|
||||
import "./constants"
|
||||
import "./configDependencies"
|
||||
import "./configConstants"
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { Effects, ExpectedExports } from "../types"
|
||||
import { SDKManifest } from "../manifest/ManifestTypes"
|
||||
import * as D from "./dependencies"
|
||||
import * as D from "./configDependencies"
|
||||
import { Config, ExtractConfigType } from "./builder/config"
|
||||
import { Utils, utils } from "../util"
|
||||
import nullIfEmpty from "../util/nullIfEmpty"
|
||||
@@ -22,7 +22,7 @@ export type Save<
|
||||
effects: Effects
|
||||
input: ExtractConfigType<A> & Record<string, any>
|
||||
utils: Utils<WD>
|
||||
dependencies: D.Dependencies<Manifest>
|
||||
dependencies: D.ConfigDependencies<Manifest>
|
||||
}) => Promise<{
|
||||
dependenciesReceipt: DependenciesReceipt
|
||||
restart: boolean
|
||||
@@ -70,7 +70,7 @@ export function setupConfig<
|
||||
input: JSON.parse(JSON.stringify(input)),
|
||||
effects,
|
||||
utils: utils(wrapperDataContract, effects),
|
||||
dependencies: D.dependenciesSet<Manifest>(),
|
||||
dependencies: D.configDependenciesSet<Manifest>(),
|
||||
})
|
||||
if (restart) {
|
||||
await effects.restart()
|
||||
|
||||
@@ -19,3 +19,5 @@ import "./inits"
|
||||
export * as matches from "ts-matches"
|
||||
export * as YAML from "yaml"
|
||||
export * as TOML from "@iarna/toml"
|
||||
|
||||
export class<Manifest extends SDKManifest>
|
||||
@@ -84,7 +84,6 @@ export type Utils<WD, WrapperOverWrite = { const: never }> = {
|
||||
nullIfEmpty: typeof nullIfEmpty
|
||||
}
|
||||
export const utils = <WD = never, WrapperOverWrite = { const: never }>(
|
||||
_wrapperDataContract: WrapperDataContract<WD>,
|
||||
effects: T.Effects,
|
||||
): Utils<WD, WrapperOverWrite> => ({
|
||||
createOrUpdateVault: async ({
|
||||
|
||||
@@ -1,14 +0,0 @@
|
||||
export declare const wrapperDataContractType: unique symbol
|
||||
export type WrapperDataContract<A> = {
|
||||
[wrapperDataContractType]: A
|
||||
}
|
||||
export const neverWrapperDataContract: WrapperDataContract<never> = null as any
|
||||
/**
|
||||
* Used to indicate the type of the wrapper data. To be used in areas where
|
||||
* we need to know the wrapper data value
|
||||
*/
|
||||
export function createWrapperDataContract<A = never>(): A extends never
|
||||
? "Wrapper Data Contract must be created with a generic"
|
||||
: WrapperDataContract<A> {
|
||||
return null as any
|
||||
}
|
||||
Reference in New Issue
Block a user