Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[PoC] - Datx types generator #43

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
40 changes: 40 additions & 0 deletions __generated__/types.ts

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions datx-generator.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
module.exports = {
models: './src/models/*.ts',
output: './__generated__',
};
58 changes: 30 additions & 28 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 4 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,8 @@
"postinstall": "npm run gen:theme-typings",
"i18n:generate": "node ./scripts/i18next-cli.js",
"gen:component": "plop component",
"gen:theme": "plop theme"
"gen:theme": "plop theme",
"gen:datx": "node ./scripts/datx-generator.js"
},
"dependencies": {
"@chakra-ui/react": "~2.3.4",
Expand Down Expand Up @@ -50,6 +51,8 @@
"swr": "~1.3.0"
},
"devDependencies": {
"@babel/generator": "~7.20.5",
"@babel/parser": "~7.20.5",
"@chakra-ui/cli": "~2.1.8",
"@chakra-ui/storybook-addon": "~4.0.12",
"@infinum/plop-next-ts-generators": "~0.0.1-5",
Expand Down
115 changes: 115 additions & 0 deletions scripts/datx-generator.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,115 @@
const path = require('path');
const fs = require('fs');
const glob = require('glob');
const parser = require('@babel/parser');
const babelGenerate = require('@babel/generator').default;

const configFileName = 'datx-generator.config.js';

const parserConfig = {
sourceType: 'module',
plugins: ['typescript', 'decorators-legacy'],
};

const { models: modelsGlob, output } = require(`${process.cwd()}/${configFileName}`);

// getType is a function that returns the type of the model based on the source code using babel parser
const getType = (source) => {
const ast = parser.parse(source, parserConfig);

const type = ast.program.body.find((node) => node.type === 'ExportNamedDeclaration');

return type.declaration.body.body.find((node) => node.key.name === 'type').value.value;
};

// getAttributes is a function that returns the attributes of the model based on the source code using babel parser
const getAttributes = (source) => {
const ast = parser.parse(source, parserConfig);

// find all props with @Attribute decorator without any options
const arguments = ast.program.body
.filter((node) => node.type === 'ExportNamedDeclaration')
.map((node) => node.declaration.body.body)
.flat()
.filter((node) => node.decorators)
.filter((node) => node.decorators[0].expression.callee.name === 'Attribute')
.filter((node) => node.decorators[0].expression.arguments.length === 0);

if (arguments.length === 0) {
return '';
}

return `
attributes: {
${arguments
.map((node) => `${node.key.name}: ${babelGenerate(node.typeAnnotation.typeAnnotation).code};`)
.join('\n\t\t')}
};`;
};

// getRelationships is a function that returns the relationships of the model based on the source code using babel parser finding all props with @Attribute decorator with toOne or toMany options
const getRelationships = (source) => {
const ast = parser.parse(source, parserConfig);

// find all props with @Attribute decorator with toOne or toMany options
const arguments = ast.program.body
.filter((node) => node.type === 'ExportNamedDeclaration')
.map((node) => node.declaration.body.body)
.flat()
.filter((node) => node.decorators)
.filter((node) => node.decorators[0].expression.callee.name === 'Attribute')
.filter((node) => node.decorators[0].expression.arguments.length > 0)
.filter(
(node) =>
node.decorators[0].expression.arguments[0]?.properties[0].key.name === 'toOne' ||
node.decorators[0].expression.arguments[0]?.properties[0].key.name === 'toMany'
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TODO: loop through all the attributes, not just the first one [0]

);

if (arguments.length === 0) {
return '';
}

return `
relationships: {
${arguments
.map(
(node) => `${node.key.name}: {
data: ${babelGenerate(node.typeAnnotation.typeAnnotation).code};
};`
)
.join('\n\t\t')}
};`;
};

const generate = (models) => `// NOTE: this file is generated with datx generator

${models
.map(
({ basename, source }) => `export type ${basename} = {
type: '${getType(source)}';
id?: string;
lid?: string;${getAttributes(source)}${getRelationships(source)}
}
`
)
.join('\n')}
`;

const models = glob.sync(modelsGlob).map((model) => ({
basename: path.basename(model, '.ts'),
source: fs.readFileSync(model, 'utf8'),
}));

const template = generate(models);

if (!fs.existsSync(path.join(process.cwd(), output))) {
fs.mkdirSync(path.join(process.cwd(), output));
}

fs.writeFile(path.join(process.cwd(), output, 'types.ts'), template, (err) => {
if (err) {
return console.log(`Unable to save: ${err}`);
}

console.log('The file was saved!');
});