2021-09-23 11:25:46 +00:00
|
|
|
//
|
|
|
|
// Copyright 2021 Vulcanize, Inc.
|
|
|
|
//
|
|
|
|
|
|
|
|
import fs from 'fs';
|
|
|
|
import path from 'path';
|
|
|
|
import assert from 'assert';
|
2021-09-27 12:33:04 +00:00
|
|
|
import yaml from 'js-yaml';
|
2021-09-23 11:25:46 +00:00
|
|
|
import Handlebars from 'handlebars';
|
|
|
|
import { Writable } from 'stream';
|
|
|
|
|
2021-11-12 12:39:03 +00:00
|
|
|
import { getTsForSol, getPgForTs, getTsForGql } from './utils/type-mappings';
|
2021-09-23 11:25:46 +00:00
|
|
|
import { Param } from './utils/types';
|
|
|
|
|
|
|
|
const TEMPLATE_FILE = './templates/entity-template.handlebars';
|
2021-09-27 12:33:04 +00:00
|
|
|
const TABLES_DIR = './data/entities';
|
2021-09-23 11:25:46 +00:00
|
|
|
|
|
|
|
export class Entity {
|
|
|
|
_entities: Array<any>;
|
|
|
|
_templateString: string;
|
|
|
|
|
|
|
|
constructor () {
|
|
|
|
this._entities = [];
|
|
|
|
this._templateString = fs.readFileSync(path.resolve(__dirname, TEMPLATE_FILE)).toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates an entity object from the query and stores to be passed to the template.
|
|
|
|
* @param name Name of the query.
|
|
|
|
* @param params Parameters to the query.
|
|
|
|
* @param returnType Return type for the query.
|
|
|
|
*/
|
|
|
|
addQuery (name: string, params: Array<Param>, returnType: string): void {
|
|
|
|
// Check if the query is already added.
|
|
|
|
if (this._entities.some(entity => entity.className.toLowerCase() === name.toLowerCase())) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const entityObject: any = {
|
2021-10-04 05:34:06 +00:00
|
|
|
className: '',
|
2021-09-27 04:43:50 +00:00
|
|
|
indexOn: [],
|
|
|
|
columns: [],
|
|
|
|
imports: []
|
2021-09-23 11:25:46 +00:00
|
|
|
};
|
|
|
|
|
2021-10-04 05:34:06 +00:00
|
|
|
// eth_call mode: Capitalize first letter of entity name (balanceOf -> BalanceOf).
|
|
|
|
// storage mode: Capiltalize second letter of entity name (_balances -> _Balances).
|
|
|
|
entityObject.className = (name.charAt(0) === '_')
|
|
|
|
? `_${name.charAt(1).toUpperCase()}${name.slice(2)}`
|
|
|
|
: `${name.charAt(0).toUpperCase()}${name.slice(1)}`;
|
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
entityObject.imports.push(
|
|
|
|
{
|
2021-09-27 12:33:04 +00:00
|
|
|
toImport: new Set(['Entity', 'PrimaryGeneratedColumn', 'Column', 'Index']),
|
2021-09-27 04:43:50 +00:00
|
|
|
from: 'typeorm'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const indexObject = {
|
|
|
|
columns: ['blockHash', 'contractAddress'],
|
|
|
|
unique: true
|
|
|
|
};
|
|
|
|
indexObject.columns = indexObject.columns.concat(
|
|
|
|
params.map((param) => {
|
|
|
|
return param.name;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
entityObject.indexOn.push(indexObject);
|
|
|
|
|
2021-11-12 12:39:03 +00:00
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'id',
|
|
|
|
tsType: 'number',
|
|
|
|
columnType: 'PrimaryGeneratedColumn',
|
|
|
|
columnOptions: []
|
|
|
|
});
|
2021-09-27 04:43:50 +00:00
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'blockHash',
|
|
|
|
pgType: 'varchar',
|
|
|
|
tsType: 'string',
|
|
|
|
columnType: 'Column',
|
|
|
|
columnOptions: [
|
|
|
|
{
|
|
|
|
option: 'length',
|
|
|
|
value: 66
|
|
|
|
}
|
|
|
|
]
|
|
|
|
});
|
2021-10-21 07:32:23 +00:00
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'blockNumber',
|
|
|
|
pgType: 'integer',
|
|
|
|
tsType: 'number',
|
|
|
|
columnType: 'Column'
|
|
|
|
});
|
2021-09-27 04:43:50 +00:00
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'contractAddress',
|
|
|
|
pgType: 'varchar',
|
|
|
|
tsType: 'string',
|
|
|
|
columnType: 'Column',
|
|
|
|
columnOptions: [
|
|
|
|
{
|
|
|
|
option: 'length',
|
|
|
|
value: 42
|
|
|
|
}
|
|
|
|
]
|
2021-09-23 11:25:46 +00:00
|
|
|
});
|
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
entityObject.columns = entityObject.columns.concat(
|
|
|
|
params.map((param) => {
|
|
|
|
const name = param.name;
|
2021-09-23 11:25:46 +00:00
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
const tsType = getTsForSol(param.type);
|
|
|
|
assert(tsType);
|
2021-09-23 11:25:46 +00:00
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
const pgType = getPgForTs(tsType);
|
|
|
|
assert(pgType);
|
2021-09-23 11:25:46 +00:00
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
const columnOptions = [];
|
|
|
|
|
|
|
|
if (param.type === 'address') {
|
|
|
|
columnOptions.push(
|
|
|
|
{
|
|
|
|
option: 'length',
|
|
|
|
value: 42
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
name,
|
|
|
|
pgType,
|
|
|
|
tsType,
|
|
|
|
columnType: 'Column',
|
|
|
|
columnOptions
|
|
|
|
};
|
|
|
|
})
|
|
|
|
);
|
2021-09-23 11:25:46 +00:00
|
|
|
|
|
|
|
const tsReturnType = getTsForSol(returnType);
|
|
|
|
assert(tsReturnType);
|
|
|
|
|
|
|
|
const pgReturnType = getPgForTs(tsReturnType);
|
|
|
|
assert(pgReturnType);
|
|
|
|
|
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'value',
|
|
|
|
pgType: pgReturnType,
|
2021-09-27 04:43:50 +00:00
|
|
|
tsType: tsReturnType,
|
2021-09-27 12:33:04 +00:00
|
|
|
columnType: 'Column',
|
|
|
|
columnOptions: []
|
2021-09-27 04:43:50 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'proof',
|
|
|
|
pgType: 'text',
|
|
|
|
tsType: 'string',
|
|
|
|
columnType: 'Column',
|
|
|
|
columnOptions: [
|
|
|
|
{
|
|
|
|
option: 'nullable',
|
|
|
|
value: true
|
|
|
|
}
|
|
|
|
]
|
2021-09-23 11:25:46 +00:00
|
|
|
});
|
|
|
|
|
2021-11-12 12:39:03 +00:00
|
|
|
// Add bigintTransformer column option if required.
|
|
|
|
this._addBigIntTransformerOption(entityObject);
|
2021-09-27 12:33:04 +00:00
|
|
|
|
2021-09-23 11:25:46 +00:00
|
|
|
this._entities.push(entityObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes the generated entity files in the given directory.
|
|
|
|
* @param entityDir Directory to write the entities to.
|
|
|
|
*/
|
|
|
|
exportEntities (entityDir: string): void {
|
2021-09-27 04:43:50 +00:00
|
|
|
this._addEventEntity();
|
|
|
|
this._addSyncStatusEntity();
|
|
|
|
this._addContractEntity();
|
|
|
|
this._addBlockProgressEntity();
|
2021-10-12 10:32:56 +00:00
|
|
|
this._addIPLDBlockEntity();
|
2021-10-18 09:20:41 +00:00
|
|
|
this._addHookStatusEntity();
|
2021-09-27 04:43:50 +00:00
|
|
|
|
2021-09-23 11:25:46 +00:00
|
|
|
const template = Handlebars.compile(this._templateString);
|
|
|
|
this._entities.forEach(entityObj => {
|
|
|
|
const entity = template(entityObj);
|
|
|
|
const outStream: Writable = entityDir
|
|
|
|
? fs.createWriteStream(path.join(entityDir, `${entityObj.className}.ts`))
|
|
|
|
: process.stdout;
|
|
|
|
outStream.write(entity);
|
|
|
|
});
|
|
|
|
}
|
2021-09-27 04:43:50 +00:00
|
|
|
|
2021-11-12 12:39:03 +00:00
|
|
|
addSubgraphEntities (subgraphSchemaDocument: any): void {
|
|
|
|
const subgraphTypeDefs = subgraphSchemaDocument.definitions;
|
|
|
|
|
|
|
|
subgraphTypeDefs.forEach((def: any) => {
|
|
|
|
// TODO Handle enum types.
|
|
|
|
if (def.kind !== 'ObjectTypeDefinition') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let entityObject: any = {
|
|
|
|
className: def.name.value,
|
|
|
|
indexOn: [],
|
|
|
|
columns: [],
|
|
|
|
imports: []
|
|
|
|
};
|
|
|
|
|
|
|
|
entityObject.imports.push(
|
|
|
|
{
|
|
|
|
toImport: new Set(['Entity', 'PrimaryColumn', 'Column']),
|
|
|
|
from: 'typeorm'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
// Add common columns.
|
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'id',
|
|
|
|
pgType: 'varchar',
|
|
|
|
tsType: 'string',
|
|
|
|
columnType: 'PrimaryColumn',
|
|
|
|
columnOptions: []
|
|
|
|
});
|
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'blockHash',
|
|
|
|
pgType: 'varchar',
|
|
|
|
tsType: 'string',
|
|
|
|
columnType: 'PrimaryColumn',
|
|
|
|
columnOptions: [
|
|
|
|
{
|
|
|
|
option: 'length',
|
|
|
|
value: 66
|
|
|
|
}
|
|
|
|
]
|
|
|
|
});
|
|
|
|
entityObject.columns.push({
|
|
|
|
name: 'blockNumber',
|
|
|
|
pgType: 'integer',
|
|
|
|
tsType: 'number',
|
|
|
|
columnType: 'Column'
|
|
|
|
});
|
|
|
|
|
|
|
|
// Add subgraph entity specific columns.
|
2021-12-22 11:24:51 +00:00
|
|
|
entityObject = this._addSubgraphColumns(subgraphTypeDefs, entityObject, def);
|
2021-11-12 12:39:03 +00:00
|
|
|
|
|
|
|
// Add bigintTransformer column option if required.
|
|
|
|
this._addBigIntTransformerOption(entityObject);
|
|
|
|
|
|
|
|
this._entities.push(entityObject);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-09-27 04:43:50 +00:00
|
|
|
_addEventEntity (): void {
|
2021-09-27 12:33:04 +00:00
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'Event.yaml'), 'utf8'));
|
2021-09-27 04:43:50 +00:00
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
|
|
|
|
|
|
|
_addSyncStatusEntity (): void {
|
2021-09-27 12:33:04 +00:00
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'SyncStatus.yaml'), 'utf8'));
|
2021-09-27 04:43:50 +00:00
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
|
|
|
|
|
|
|
_addContractEntity (): void {
|
2021-09-27 12:33:04 +00:00
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'Contract.yaml'), 'utf8'));
|
2021-09-27 04:43:50 +00:00
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
|
|
|
|
|
|
|
_addBlockProgressEntity (): void {
|
2021-09-27 12:33:04 +00:00
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'BlockProgress.yaml'), 'utf8'));
|
2021-09-27 04:43:50 +00:00
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
2021-10-12 10:32:56 +00:00
|
|
|
|
|
|
|
_addIPLDBlockEntity (): void {
|
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'IPLDBlock.yaml'), 'utf8'));
|
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
2021-10-14 10:38:45 +00:00
|
|
|
|
2021-10-18 09:20:41 +00:00
|
|
|
_addHookStatusEntity (): void {
|
|
|
|
const entity = yaml.load(fs.readFileSync(path.resolve(__dirname, TABLES_DIR, 'HookStatus.yaml'), 'utf8'));
|
2021-10-14 10:38:45 +00:00
|
|
|
this._entities.push(entity);
|
|
|
|
}
|
2021-11-12 12:39:03 +00:00
|
|
|
|
|
|
|
_addBigIntTransformerOption (entityObject: any): void {
|
|
|
|
entityObject.columns.forEach((column: any) => {
|
|
|
|
if (column.tsType === 'bigint') {
|
|
|
|
column.columnOptions.push(
|
|
|
|
{
|
|
|
|
option: 'transformer',
|
|
|
|
value: 'bigintTransformer'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const importObject = entityObject.imports.find((element: any) => {
|
|
|
|
return element.from === '@vulcanize/util';
|
|
|
|
});
|
|
|
|
|
|
|
|
if (importObject) {
|
|
|
|
importObject.toImport.add('bigintTransformer');
|
|
|
|
} else {
|
|
|
|
entityObject.imports.push(
|
|
|
|
{
|
|
|
|
toImport: new Set(['bigintTransformer']),
|
|
|
|
from: '@vulcanize/util'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-12-22 11:24:51 +00:00
|
|
|
_addSubgraphColumns (subgraphTypeDefs: any, entityObject: any, def: any): any {
|
2021-11-12 12:39:03 +00:00
|
|
|
def.fields.forEach((field: any) => {
|
2021-12-22 11:24:51 +00:00
|
|
|
let name = field.name.value;
|
2021-11-12 12:39:03 +00:00
|
|
|
|
2021-12-22 11:24:51 +00:00
|
|
|
// Column id is already added.
|
|
|
|
if (name === 'id') {
|
2021-11-12 12:39:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-12-22 11:24:51 +00:00
|
|
|
// Handle column with existing name.
|
|
|
|
if (['blockHash', 'blockNumber'].includes(name)) {
|
|
|
|
name = `_${name}`;
|
|
|
|
}
|
|
|
|
|
2021-11-12 12:39:03 +00:00
|
|
|
const columnObject: any = {
|
|
|
|
name,
|
2021-12-22 11:24:51 +00:00
|
|
|
columnOptions: [],
|
|
|
|
columnType: 'Column'
|
2021-11-12 12:39:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const { typeName, array, nullable } = this._getFieldType(field.type);
|
|
|
|
let tsType = getTsForGql(typeName);
|
|
|
|
|
|
|
|
if (tsType) {
|
|
|
|
// Handle basic array types.
|
|
|
|
if (array) {
|
|
|
|
columnObject.columnOptions.push({
|
|
|
|
option: 'array',
|
|
|
|
value: 'true'
|
|
|
|
});
|
|
|
|
|
|
|
|
columnObject.tsType = `${tsType}[]`;
|
|
|
|
} else {
|
|
|
|
columnObject.tsType = tsType;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// TODO Handle array of custom types.
|
|
|
|
tsType = typeName;
|
|
|
|
columnObject.tsType = tsType;
|
|
|
|
}
|
|
|
|
|
|
|
|
const pgType = getPgForTs(tsType);
|
|
|
|
|
2021-12-22 11:24:51 +00:00
|
|
|
// If basic type: create a column.
|
2021-11-12 12:39:03 +00:00
|
|
|
if (pgType) {
|
|
|
|
columnObject.pgType = pgType;
|
|
|
|
} else {
|
2021-12-22 11:24:51 +00:00
|
|
|
if (subgraphTypeDefs.some((typeDef: any) => typeDef.kind === 'EnumTypeDefinition' && typeDef.name.value === typeName)) {
|
|
|
|
// Create enum type column.
|
|
|
|
|
|
|
|
const entityImport = entityObject.imports.find(({ from }: any) => from === '../types');
|
|
|
|
|
|
|
|
if (!entityImport) {
|
|
|
|
entityObject.imports.push(
|
|
|
|
{
|
|
|
|
toImport: new Set([typeName]),
|
|
|
|
from: '../types'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
entityImport.toImport.add(typeName);
|
|
|
|
}
|
2021-11-12 12:39:03 +00:00
|
|
|
|
2021-12-22 11:24:51 +00:00
|
|
|
columnObject.columnOptions.push(
|
|
|
|
{
|
|
|
|
option: 'type',
|
|
|
|
value: "'enum'"
|
|
|
|
},
|
2021-11-12 12:39:03 +00:00
|
|
|
{
|
2021-12-22 11:24:51 +00:00
|
|
|
option: 'enum',
|
|
|
|
value: typeName
|
2021-11-12 12:39:03 +00:00
|
|
|
}
|
|
|
|
);
|
2021-12-22 11:24:51 +00:00
|
|
|
} else {
|
|
|
|
// Create a relation.
|
|
|
|
|
|
|
|
columnObject.columnType = 'ManyToOne';
|
|
|
|
columnObject.lhs = '()';
|
|
|
|
columnObject.rhs = tsType;
|
|
|
|
|
|
|
|
entityObject.imports[0].toImport.add('ManyToOne');
|
|
|
|
|
|
|
|
// Check if type import already added.
|
|
|
|
const importObject = entityObject.imports.find((element: any) => {
|
|
|
|
return element.from === `./${tsType}`;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!importObject) {
|
|
|
|
entityObject.imports.push(
|
|
|
|
{
|
|
|
|
toImport: new Set([tsType]),
|
|
|
|
from: `./${tsType}`
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2021-11-12 12:39:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nullable) {
|
|
|
|
columnObject.columnOptions.push({
|
|
|
|
option: 'nullable',
|
|
|
|
value: 'true'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
entityObject.columns.push(columnObject);
|
|
|
|
});
|
|
|
|
|
|
|
|
return entityObject;
|
|
|
|
}
|
|
|
|
|
|
|
|
_getFieldType (typeNode: any): { typeName: string, array: boolean, nullable: boolean } {
|
|
|
|
if (typeNode.kind === 'ListType') {
|
|
|
|
return { typeName: this._getFieldType(typeNode.type).typeName, array: true, nullable: true };
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeNode.kind === 'NonNullType') {
|
|
|
|
const fieldType = this._getFieldType(typeNode.type);
|
|
|
|
|
|
|
|
return { typeName: fieldType.typeName, array: fieldType.array, nullable: false };
|
|
|
|
}
|
|
|
|
|
|
|
|
// If 'NamedType'.
|
|
|
|
return { typeName: typeNode.name.value, array: false, nullable: true };
|
|
|
|
}
|
2021-09-23 11:25:46 +00:00
|
|
|
}
|