Skip to content
Merged
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
1 change: 1 addition & 0 deletions esbuild.package.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ esbuild
entryPoints: [
path.resolve(__dirname, 'forward_engineering', 'api.js'),
path.resolve(__dirname, 'forward_engineering', 'ddlProvider.js'),
path.resolve(__dirname, 'forward_engineering', 'dbtProvider.js'),
path.resolve(__dirname, 'reverse_engineering', 'api.js'),
],
bundle: true,
Expand Down
76 changes: 76 additions & 0 deletions forward_engineering/dbtProvider.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
/**
* @typedef {import('./types').AppInstance} AppInstance
* @typedef {import('./types').ColumnDefinition} ColumnDefinition
* @typedef {import('./types').JsonSchema} JsonSchema
* @typedef {import('./types').ConstraintDto} ConstraintDto
*/
const { toLower, toUpper, identity } = require('lodash');

const types = require('./configs/types');
const defaultTypes = require('./configs/defaultTypes');
const getKeyHelper = require('./helpers/keyHelper');
const getColumnDefinitionHelper = require('./helpers/columnDefinitionHelper');

class DbtProvider {
/**
* @returns {DbtProvider}
*/
static createDbtProvider() {
return new DbtProvider();
}

/**
* @param {string} type
* @returns {string | undefined}
*/
getDefaultType(type) {
return defaultTypes[type];
}

/**
* @returns {Record<string, object>}
*/
getTypesDescriptors() {
return types;
}

/**
* @param {string} type
* @returns {boolean}
*/
hasType(type) {
return Object.keys(types).map(toLower).includes(toLower(type));
}

/**
* @param {{ type: string; columnDefinition: ColumnDefinition }}
* @returns {string}
*/
decorateType({ type, columnDefinition }) {
const columnDefinitionHelper = getColumnDefinitionHelper(identity);

return columnDefinitionHelper.decorateType(toUpper(type), columnDefinition);
}

/**
* @param {{ jsonSchema: JsonSchema }}
* @returns {ConstraintDto[]}
*/
getCompositeKeyConstraints({ jsonSchema }) {
const keyHelper = getKeyHelper(identity);

return keyHelper.getCompositeKeyConstraints({ jsonSchema });
}

/**
* @param {{ columnDefinition: ColumnDefinition }}
* @returns {ConstraintDto[]}
*/
getColumnConstraints({ columnDefinition }) {
const keyHelper = getKeyHelper(identity);

return keyHelper.getColumnConstraints({ columnDefinition });
}
}

module.exports = DbtProvider;
6 changes: 3 additions & 3 deletions forward_engineering/ddlProvider.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
const _ = require('lodash');
const defaultTypes = require('./configs/defaultTypes');
const types = require('./configs/types');
const templates = require('./configs/templates');
Expand All @@ -6,7 +7,6 @@ const dropStatementProxy = require('./helpers/dropStatementProxy');
const { joinActivatedAndDeactivatedStatements } = require('./utils/joinActivatedAndDeactivatedStatements');

module.exports = (baseProvider, options, app) => {
const _ = app.require('lodash');
const {
tab,
commentIfDeactivated,
Expand All @@ -19,7 +19,7 @@ module.exports = (baseProvider, options, app) => {
} = app.require('@hackolade/ddl-fe-utils').general;
const { assignTemplates, compareGroupItems } = app.require('@hackolade/ddl-fe-utils');
const { decorateDefault, decorateType, canBeNational, getSign, createGeneratedColumn, canHaveAutoIncrement } =
require('./helpers/columnDefinitionHelper')(_, wrap);
require('./helpers/columnDefinitionHelper')(wrap);
const { getTableName, getTableOptions, getPartitions, getViewData, getCharacteristics, escapeQuotes } =
require('./helpers/general')(_, wrap);
const { generateConstraintsString, foreignKeysToString, foreignActiveKeysToString, createKeyConstraint } =
Expand All @@ -31,7 +31,7 @@ module.exports = (baseProvider, options, app) => {
assignTemplates,
escapeQuotes,
});
const keyHelper = require('./helpers/keyHelper')(_, clean);
const keyHelper = require('./helpers/keyHelper')(clean);
const { processIndexKeyName } = require('./helpers/indexHelper')(wrap);
const additionalOptions = getAdditionalOptions(options.additionalOptions);

Expand Down
4 changes: 3 additions & 1 deletion forward_engineering/helpers/columnDefinitionHelper.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
module.exports = (_, wrap) => {
const _ = require('lodash');

module.exports = wrap => {
const addLength = (type, length) => {
return `${type}(${length})`;
};
Expand Down
66 changes: 65 additions & 1 deletion forward_engineering/helpers/keyHelper.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,12 @@
module.exports = (_, clean) => {
/**
* @typedef {import('../types').ColumnDefinition} ColumnDefinition
* @typedef {import('../types').JsonSchema} JsonSchema
* @typedef {import('../types').ConstraintDto} ConstraintDto
*/

const _ = require('lodash');

module.exports = clean => {
const mapProperties = (jsonSchema, iteratee) => {
return Object.entries(jsonSchema.properties).map(iteratee);
};
Expand Down Expand Up @@ -166,11 +174,67 @@ module.exports = (_, clean) => {
];
};

/**
* @param {{ jsonSchema: JsonSchema }}
* @returns {ConstraintDto[]}
*/
const getCompositeKeyConstraints = ({ jsonSchema }) => {
const compositePrimaryKeys = getCompositePrimaryKeys(jsonSchema);
const compositeUniqueKeys = getCompositeUniqueKeys(jsonSchema);

return [...compositePrimaryKeys, ...compositeUniqueKeys];
};

/**
* @param {{ columnDefinition: ColumnDefinition }}
* @returns {ConstraintDto | undefined}
*/
const getPrimaryKeyConstraint = ({ columnDefinition }) => {
if (!isPrimaryKey(columnDefinition)) {
return;
}

return hydratePrimaryKeyOptions(columnDefinition.primaryKeyOptions ?? {}, '', columnDefinition.isActivated);
};

/**
* @param {{ columnDefinition: ColumnDefinition }}
* @returns {ConstraintDto[]}
*/
const getUniqueKeyConstraints = ({ columnDefinition }) => {
if (!isUniqueKey(columnDefinition)) {
return [];
}

if (isInlineUnique(columnDefinition)) {
const constraint = hydrateUniqueOptions({}, '', columnDefinition.isActivated);

return [constraint];
}

return columnDefinition.uniqueKeyOptions.map(uniqueKeyOption => {
return hydrateUniqueOptions(uniqueKeyOption, '', columnDefinition.isActivated);
});
};

/**
* @param {{ columnDefinition: ColumnDefinition }}
* @returns {ConstraintDto[]}
*/
const getColumnConstraints = ({ columnDefinition }) => {
const primaryKeyConstraint = getPrimaryKeyConstraint({ columnDefinition });
const uniqueKeyConstraints = getUniqueKeyConstraints({ columnDefinition });

return [primaryKeyConstraint, ...uniqueKeyConstraints].filter(Boolean);
};

return {
getTableKeyConstraints,
isInlineUnique,
isInlinePrimaryKey,
hydratePrimaryKeyOptions,
hydrateUniqueOptions,
getColumnConstraints,
getCompositeKeyConstraints,
};
};
33 changes: 33 additions & 0 deletions forward_engineering/types.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
export type ColumnDefinition = {
name: string;
type: string;
isActivated: boolean;
length?: number;
precision?: number;
primaryKey?: boolean;
scale?: number;
timePrecision?: number;
unique?: boolean;
primaryKeyOptions?: { GUID: string; constraintName: string };
uniqueKeyOptions?: Array<{ GUID: string; constraintName: string }>;
};

export type AppInstance = {
require: (packageName: string) => unknown;
general: object;
}

export type ConstraintDtoColumn = {
name: string;
isActivated: boolean;
};

export type KeyType = 'PRIMARY KEY' | 'UNIQUE';

export type ConstraintDto = {
keyType: KeyType;
name: string;
columns?: ConstraintDtoColumn[];
};

export type JsonSchema = Record<string, unknown>;
8 changes: 7 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,13 @@
"nestedCollections": false,
"disablePatternField": true,
"disableMultipleTypes": true,
"enableForwardEngineering": true,
"enableForwardEngineering": {
"jsonDocument": true,
"jsonSchema": true,
"excel": true,
"plugin": true,
"dbt": true
},
"disableReverseEngineering": false,
"disableChoices": true,
"enableJsonType": true,
Expand Down