Skip to content

Commit

Permalink
Add prototype code
Browse files Browse the repository at this point in the history
Signed-off-by: Ryan Nowak <[email protected]>
  • Loading branch information
rynowak committed Dec 24, 2024
1 parent 465ec28 commit aee275e
Show file tree
Hide file tree
Showing 19 changed files with 323 additions and 38 deletions.
8 changes: 8 additions & 0 deletions .github/depdendabot.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
version: 2
updates:
- package-ecosystem: 'gitsubmodule'
directory: '/'
- package-ecosystem: 'npm'
directory: '/'
schedule:
interval: 'monthly'
6 changes: 2 additions & 4 deletions .github/workflows/build.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,6 @@ on:
- main
- features/*
- release/*
env:
CI_PUBLISH:
jobs:
build:
runs-on: ubuntu-latest
Expand All @@ -25,7 +23,7 @@ jobs:
- name: Set up Node.js
uses: actions/setup-node@v2
with:
node-version: "18"
node-version: '18'
- name: Enable Corepack
run: |
corepack enable
Expand Down Expand Up @@ -57,7 +55,7 @@ jobs:
- name: Set up Node.js
uses: actions/setup-node@v2
with:
node-version: "18"
node-version: '18'
registry-url: 'https://npm.pkg.github.com'
scope: '@radius-project'
- name: Enable Corepack
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/test.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ jobs:
- name: Set up Node.js
uses: actions/setup-node@v2
with:
node-version: "18"
node-version: '18'
- name: Enable Corepack
run: |
corepack enable
Expand Down
3 changes: 3 additions & 0 deletions .gitmodules
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
[submodule "bicep-types"]
path = bicep-types
url = https://github.com/Azure/bicep-types
1 change: 1 addition & 0 deletions .prettierignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
bicep-types/
4 changes: 2 additions & 2 deletions .prettierrc
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"trailingComma": "es5",
"tabWidth": 4,
"tabWidth": 2,
"semi": false,
"singleQuote": true
}
}
1 change: 1 addition & 0 deletions bicep-types
Submodule bicep-types added at 3676a8
15 changes: 10 additions & 5 deletions eslint.config.mjs
Original file line number Diff line number Diff line change
@@ -1,10 +1,15 @@
// @ts-check
import eslint from "@eslint/js";
import tseslint from "typescript-eslint";
import eslint from '@eslint/js'
import tseslint from 'typescript-eslint'

export default tseslint.config(
{
ignores: ["**/jest.config.js", "**/eslint.config.mjs", "**/dist/**"],
ignores: [
'**/jest.config.js',
'**/eslint.config.mjs',
'**/dist/**',
'**/bicep-types/**',
],
},
eslint.configs.recommended,
tseslint.configs.strict,
Expand All @@ -16,5 +21,5 @@ export default tseslint.config(
tsconfigRootDir: import.meta.dirname,
},
},
},
);
}
)
8 changes: 4 additions & 4 deletions jest.config.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
/** @type {import('ts-jest').JestConfigWithTsJest} **/
module.exports = {
projects: ["<rootDir>/packages/*"],
testEnvironment: "node",
projects: ['<rootDir>/packages/*'],
testEnvironment: 'node',
transform: {
"^.+.tsx?$": ["ts-jest", {}],
'^.+.tsx?$': ['ts-jest', {}],
},
};
}
14 changes: 7 additions & 7 deletions packages/manifest-to-bicep-extension/jest.config.js
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
module.exports = {
displayName: "manifest-to-bicep-extension",
roots: ["<rootDir>"],
displayName: 'manifest-to-bicep-extension',
roots: ['<rootDir>'],
transform: {
"^.+\\.tsx?$": "ts-jest",
'^.+\\.tsx?$': 'ts-jest',
},
testRegex: "(/__tests__/.*|(\\.|/)(test|spec))\\.tsx?$",
moduleFileExtensions: ["ts", "tsx", "js", "jsx", "json", "node"],
testRegex: '(/__tests__/.*|(\\.|/)(test|spec))\\.tsx?$',
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
moduleNameMapper: {
"^src/(.*)": "<rootDir>/src/$1",
'^src/(.*)': '<rootDir>/src/$1',
},
};
}
13 changes: 13 additions & 0 deletions packages/manifest-to-bicep-extension/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,24 @@
"dist/"
],
"main": "dist/src/index.js",
"dependencies": {
"bicep-types": "file:../../bicep-types/src/bicep-types",
"yaml": "^2.6.0",
"yargs": "^17.7.2"
},
"devDependencies": {
"@types/node": "^22.7.5",
"@types/yargs": "^17.0.33"
},
"bundledDependencies": [
"bicep-types"
],
"scripts": {
"build": "tsc",
"watch": "tsc -w",
"publish": "npm publish",
"prepublishOnly": "yarn build",
"preinstall": "cd ../../bicep-types/src/bicep-types && npm ci && npm run build",
"version": "npm pkg set version=${0}"
}
}
148 changes: 148 additions & 0 deletions packages/manifest-to-bicep-extension/src/converter.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
import {
buildIndex,
ObjectTypeProperty,
ObjectTypePropertyFlags,
ResourceFlags,
ScopeType,
TypeFactory,
TypeFile,
TypeReference,
writeIndexJson,
writeIndexMarkdown,
writeTypesJson,
} from 'bicep-types'
import { ResourceProvider, Schema } from './manifest'

export function convert(manifest: ResourceProvider): {
typesContent: string
indexContent: string
documentationContent: string
} {
const factory = new TypeFactory()

for (const resourceType of manifest.types) {
for (const apiVersion of resourceType.apiVersions) {
const qualifiedName = `${manifest.name}/${resourceType.name}@${apiVersion.name}`

const propertyType = factory.addObjectType(
`${resourceType.name}Properties`,
schemaProperties(apiVersion.schema, factory)
)

const bodyType = factory.addObjectType(qualifiedName, {
name: {
type: factory.addStringType(),
flags:
ObjectTypePropertyFlags.Required |
ObjectTypePropertyFlags.Identifier,
description: 'The resource name.',
},
location: {
type: factory.addStringType(),
flags: ObjectTypePropertyFlags.None,
description: 'The resource location.',
},
properties: {
type: propertyType,
flags: ObjectTypePropertyFlags.Required,
description: 'The resource properties.',
},
apiVersion: {
type: factory.addStringLiteralType(apiVersion.name),
flags:
ObjectTypePropertyFlags.ReadOnly |
ObjectTypePropertyFlags.DeployTimeConstant,
description: 'The API version.',
},
type: {
type: factory.addStringLiteralType(
`${manifest.name}/${resourceType.name}`
),
flags:
ObjectTypePropertyFlags.ReadOnly |
ObjectTypePropertyFlags.DeployTimeConstant,
description: 'The resource type.',
},
id: {
type: factory.addStringType(),
flags: ObjectTypePropertyFlags.ReadOnly,
description: 'The resource id.',
},
})
factory.addResourceType(
qualifiedName,
ScopeType.Unknown,
undefined,
bodyType,
ResourceFlags.None,
{}
)
}
}

const typeFiles: TypeFile[] = []
typeFiles.push({
relativePath: 'types.json',
types: factory.types,
})

const indexContent = buildIndex(typeFiles, (log) => console.log(log), {
name: manifest.name.toLowerCase().replace('.', ''),
version: '0.0.1',
isSingleton: false,
})

return {
typesContent: writeTypesJson(factory.types),
indexContent: writeIndexJson(indexContent),
documentationContent: writeIndexMarkdown(indexContent),
}
}

function schemaProperties(
parent: Schema,
factory: TypeFactory
): Record<string, ObjectTypeProperty> {
const results: Record<string, ObjectTypeProperty> = {}
for (const [key, value] of Object.entries(parent.properties ?? {})) {
results[key] = addSchemaProperty(parent, key, value, factory)
}
return results
}

function addSchemaProperty(
parent: Schema,
key: string,
property: Schema,
factory: TypeFactory
): ObjectTypeProperty {
const propertyType = addSchema(property, key, factory)

let flags = ObjectTypePropertyFlags.None
if (parent.required?.includes(key)) {
flags |= ObjectTypePropertyFlags.Required
}
if (property.readOnly === true) {
flags |= ObjectTypePropertyFlags.ReadOnly
}

return {
description: property.description,
type: propertyType,
flags: flags,
}
}

function addSchema(
schema: Schema,
name: string,
factory: TypeFactory
): TypeReference {
if (schema.type === 'string') {
return factory.addStringType()
} else if (schema.type === 'object') {
return factory.addObjectType(name, schemaProperties(schema, factory))
} else {
throw new Error(`Unsupported schema type: ${schema.type}`)
}
}
5 changes: 3 additions & 2 deletions packages/manifest-to-bicep-extension/src/main.ts
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
import { add } from './math';
add(9, 3);
// eslint-disable-next-line @typescript-eslint/no-require-imports
const program = require('./program')
program.run()
24 changes: 24 additions & 0 deletions packages/manifest-to-bicep-extension/src/manifest.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
export interface ResourceProvider {
name: string
types: ResourceType[]
}

export interface ResourceType {
name: string
defaultApiVersion?: string
apiVersions: APIVersion[]
}

export interface APIVersion {
name: string
schema: Schema
capabilities?: string[]
}

export interface Schema {
type: 'string' | 'object'
description?: string
properties?: Schema[]
required?: string[]
readOnly?: boolean
}
2 changes: 1 addition & 1 deletion packages/manifest-to-bicep-extension/src/math.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
export function add(a: number, b: number): number {
return a + b;
return a + b
}
60 changes: 60 additions & 0 deletions packages/manifest-to-bicep-extension/src/program.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import { parse } from 'yaml'
import yargs from 'yargs'
import { hideBin } from 'yargs/helpers'
import fs from 'node:fs'
import { convert } from './converter'

async function generate(manifest: string, output: string) {
const data = fs.readFileSync(manifest, 'utf8')
const parsed = parse(data)
const converted = convert(parsed)

fs.rmSync(`${output}/types.json`, { force: true })
fs.rmSync(`${output}/index.json`, { force: true })
fs.rmSync(`${output}/index.md`, { force: true })

console.log(`Writing types to ${output}/types.json`)
fs.writeFileSync(`${output}/types.json`, converted.typesContent, {
encoding: 'utf8',
})

console.log(`Writing index to ${output}/index.json`)
fs.writeFileSync(`${output}/index.json`, converted.indexContent, {
encoding: 'utf8',
})

console.log(`Writing documentation to ${output}/index.md`)
fs.writeFileSync(`${output}/index.md`, converted.documentationContent, {
encoding: 'utf8',
})
}

export async function run() {
return await yargs(hideBin(process.argv))
.usage('Usage: $0 <command> [options]')
.showHelpOnFail(true)
.demandCommand(1, 'Command name is required')
.command(
'generate <manifest> <output> [options]',
'Generate Bicep extension from Radius Resource Provider manifest.',
(yargs) => {
return yargs
.positional('manifest', {
demandOption: true,
type: 'string',
})
.positional('output', {
demandOption: true,
type: 'string',
})
},
async (yargs) => {
try {
await generate(yargs.manifest, yargs.output)
} catch (err) {
console.error(err)
}
}
)
.parseAsync()
}
Loading

0 comments on commit aee275e

Please sign in to comment.