2024-05-09 10:35:54 +00:00
|
|
|
import yargs from 'yargs';
|
|
|
|
import fs from 'fs';
|
|
|
|
import path from 'path';
|
2024-06-25 06:36:46 +00:00
|
|
|
import yaml from 'js-yaml';
|
2024-05-09 10:35:54 +00:00
|
|
|
import assert from 'assert';
|
|
|
|
import { hideBin } from 'yargs/helpers';
|
|
|
|
|
2024-06-24 07:27:39 +00:00
|
|
|
import { StdFee } from '@cosmjs/stargate';
|
2024-05-09 10:35:54 +00:00
|
|
|
import { Registry } from '@cerc-io/registry-sdk';
|
|
|
|
|
|
|
|
import { getConfig, getGasAndFees, getConnectionInfo, txOutput } from '../../src/util';
|
|
|
|
|
2024-06-24 07:27:39 +00:00
|
|
|
enum RecordType {
|
|
|
|
RepositoryRecord = 'RepositoryRecord',
|
|
|
|
ServiceRecord = 'ServiceRecord',
|
|
|
|
StackRecord = 'StackRecord',
|
|
|
|
SubgraphRecord = 'SubgraphRecord',
|
|
|
|
WatcherRecord = 'WatcherRecord',
|
2024-08-26 08:53:29 +00:00
|
|
|
DockerImageRecord = 'DockerImageRecord'
|
2024-06-24 07:27:39 +00:00
|
|
|
}
|
|
|
|
|
2024-05-09 10:35:54 +00:00
|
|
|
const recordTypeToRecordField = new Map<string, string>([
|
2024-06-24 07:27:39 +00:00
|
|
|
[RecordType.WatcherRecord, 'watcher'],
|
|
|
|
[RecordType.SubgraphRecord, 'subgraph'],
|
|
|
|
[RecordType.ServiceRecord, 'service']
|
2024-05-09 10:35:54 +00:00
|
|
|
]);
|
|
|
|
|
|
|
|
let registry: Registry;
|
|
|
|
let fee: any;
|
|
|
|
let userKey: string;
|
|
|
|
let bondId: string;
|
|
|
|
|
|
|
|
async function main () {
|
|
|
|
const argv = getArgs();
|
|
|
|
const { records: recordsDir, config } = argv;
|
|
|
|
|
|
|
|
const { services: { registry: registryConfig } } = getConfig(config as string);
|
|
|
|
|
|
|
|
if (registryConfig.userKey == null) {
|
|
|
|
throw new Error('userKey not set in config');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (registryConfig.bondId == null) {
|
|
|
|
throw new Error('bondId not set in config');
|
|
|
|
}
|
|
|
|
|
|
|
|
let rpcEndpoint, gqlEndpoint, chainId: string;
|
|
|
|
({ rpcEndpoint, gqlEndpoint, userKey, bondId, chainId } = getConnectionInfo(argv, registryConfig));
|
|
|
|
|
|
|
|
registry = new Registry(gqlEndpoint, rpcEndpoint, chainId);
|
|
|
|
fee = getGasAndFees(argv, registryConfig);
|
|
|
|
|
|
|
|
await processDir(path.resolve(recordsDir));
|
|
|
|
}
|
|
|
|
|
|
|
|
async function processDir (directoryPath: string): Promise<void> {
|
|
|
|
const files = fs.readdirSync(directoryPath);
|
|
|
|
|
2024-06-25 06:36:46 +00:00
|
|
|
const dirHasRecords = await publishRecordsFromDir(directoryPath);
|
|
|
|
if (dirHasRecords) {
|
2024-05-09 10:35:54 +00:00
|
|
|
// Skip further recursion in the current dir
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recursively iterate through subdirectories
|
|
|
|
for (let i = 0; i < files.length; i++) {
|
|
|
|
const file = files[i];
|
|
|
|
const filePath = path.join(directoryPath, file);
|
|
|
|
|
|
|
|
if (fs.statSync(filePath).isDirectory()) {
|
|
|
|
await processDir(filePath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-25 06:36:46 +00:00
|
|
|
async function publishRecordsFromDir (recordsDir: string): Promise<boolean> {
|
2024-05-09 10:35:54 +00:00
|
|
|
// List record files
|
|
|
|
const files = fs.readdirSync(recordsDir);
|
2024-06-25 06:36:46 +00:00
|
|
|
const recordFiles = files.filter(file => ['.json', '.yaml', '.yml'].includes(path.extname(file).toLowerCase()));
|
|
|
|
|
|
|
|
if (recordFiles.length === 0) {
|
|
|
|
return false;
|
|
|
|
}
|
2024-05-09 10:35:54 +00:00
|
|
|
|
|
|
|
// Read record from each JSON file
|
|
|
|
console.log('**************************************');
|
|
|
|
console.log(`Publishing records from ${recordsDir}`);
|
|
|
|
|
|
|
|
let recordType;
|
2024-06-25 06:36:46 +00:00
|
|
|
for (let i = 0; i < recordFiles.length; i++) {
|
|
|
|
const file = recordFiles[i];
|
2024-05-09 10:35:54 +00:00
|
|
|
|
|
|
|
const filePath = path.resolve(recordsDir, file);
|
2024-06-25 06:36:46 +00:00
|
|
|
const record = await readRecord(filePath);
|
2024-05-09 10:35:54 +00:00
|
|
|
|
|
|
|
// Publish record
|
2024-06-24 07:27:39 +00:00
|
|
|
const result = await publishRecord(userKey, bondId, fee, record);
|
2024-05-09 10:35:54 +00:00
|
|
|
|
|
|
|
console.log(`Published record ${file}`);
|
|
|
|
txOutput(result, JSON.stringify(result, undefined, 2), '', false);
|
|
|
|
|
|
|
|
recordType = record.type;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if deployment record files exist
|
|
|
|
const deploymentRecordsDir = path.resolve(recordsDir, 'deployments');
|
2024-06-24 07:27:39 +00:00
|
|
|
if (!fs.existsSync(deploymentRecordsDir) || !fs.statSync(deploymentRecordsDir).isDirectory()) {
|
2024-06-25 06:36:46 +00:00
|
|
|
return true;
|
2024-05-09 10:35:54 +00:00
|
|
|
}
|
|
|
|
console.log('--------------------------------------');
|
|
|
|
console.log(`Publishing deployment records from ${deploymentRecordsDir}`);
|
|
|
|
|
|
|
|
// List record files
|
|
|
|
const deploymentFiles = fs.readdirSync(deploymentRecordsDir);
|
|
|
|
const deploymentJsonFiles = deploymentFiles.filter(file => path.extname(file).toLowerCase() === '.json');
|
|
|
|
|
|
|
|
for (let i = 0; i < deploymentJsonFiles.length; i++) {
|
|
|
|
const file = deploymentJsonFiles[i];
|
|
|
|
|
|
|
|
const filePath = path.resolve(deploymentRecordsDir, file);
|
2024-06-25 06:36:46 +00:00
|
|
|
const deploymentRecord = await readRecord(filePath);
|
2024-05-09 10:35:54 +00:00
|
|
|
|
|
|
|
// Find record using name and given type
|
|
|
|
const recordName = deploymentRecord.name;
|
|
|
|
assert(recordType, 'recordType could not be resolved');
|
|
|
|
const queryResult = await registry.queryRecords({ type: recordType, name: recordName }, true);
|
|
|
|
if (queryResult.length === 0) {
|
|
|
|
throw new Error(`Record not found, type: ${recordType}, name: ${recordName}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assume the first query result
|
|
|
|
const recordId = queryResult[0].id;
|
|
|
|
|
|
|
|
// Set record field to record id
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
|
|
deploymentRecord[recordTypeToRecordField.get(recordType)!] = recordId;
|
|
|
|
|
|
|
|
// Publish record
|
|
|
|
const deploymentResult = await registry.setRecord({ privateKey: userKey, record: deploymentRecord, bondId }, userKey, fee);
|
|
|
|
|
|
|
|
console.log(`Published record ${file}`);
|
|
|
|
txOutput(deploymentResult, JSON.stringify(deploymentResult, undefined, 2), '', false);
|
|
|
|
}
|
2024-06-25 06:36:46 +00:00
|
|
|
|
|
|
|
return true;
|
2024-05-09 10:35:54 +00:00
|
|
|
}
|
|
|
|
|
2024-06-25 06:36:46 +00:00
|
|
|
async function readRecord (filePath: string): Promise<any> {
|
2024-05-09 10:35:54 +00:00
|
|
|
let record;
|
|
|
|
try {
|
2024-06-25 06:36:46 +00:00
|
|
|
const fileExt = path.extname(filePath).toLowerCase();
|
2024-05-09 10:35:54 +00:00
|
|
|
const data = fs.readFileSync(filePath, 'utf8');
|
2024-06-25 06:36:46 +00:00
|
|
|
|
|
|
|
if (fileExt === '.json') {
|
|
|
|
// JSON file
|
|
|
|
record = JSON.parse(data);
|
|
|
|
} else {
|
|
|
|
// YAML file
|
|
|
|
({ record } = await yaml.load(data) as any);
|
|
|
|
|
|
|
|
// Convert sub-objects (other than arrays) to a JSON automatically.
|
|
|
|
for (const [k, v] of Object.entries(record)) {
|
|
|
|
if (v && typeof v === 'object' && !Array.isArray(v)) {
|
|
|
|
record[k] = JSON.stringify(v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-05-09 10:35:54 +00:00
|
|
|
} catch (err) {
|
|
|
|
console.error(`Error reading file ${filePath}:`, err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return record;
|
|
|
|
}
|
|
|
|
|
2024-06-24 07:27:39 +00:00
|
|
|
async function publishRecord (userKey: string, bondId: string, fee: StdFee, record: any): Promise<any> {
|
2024-06-25 06:36:46 +00:00
|
|
|
// Replace repository URL with record id (if type is one of RecordType)
|
|
|
|
if (record.repository && Object.values(RecordType).includes(record.type)) {
|
2024-06-24 07:27:39 +00:00
|
|
|
const repoUrl = record.repository;
|
|
|
|
|
|
|
|
const queryResult = await registry.queryRecords({ type: RecordType.RepositoryRecord, url: repoUrl }, true);
|
|
|
|
if (queryResult.length === 0) {
|
|
|
|
throw new Error(`Record not found, type: ${RecordType.RepositoryRecord}, url: ${repoUrl}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assume the first query result
|
|
|
|
const repoRecordId = queryResult[0].id;
|
|
|
|
|
|
|
|
// Replace repository URL with the repo record id
|
|
|
|
record.repository = repoRecordId;
|
|
|
|
}
|
|
|
|
|
2024-08-26 08:53:29 +00:00
|
|
|
// For stack records, check for attributes
|
|
|
|
if (record.type === RecordType.StackRecord) {
|
|
|
|
const watcherName = record.meta?.watcher;
|
|
|
|
const dockerImages = record.docker_images;
|
|
|
|
|
|
|
|
// If .docker_images present, check for image records
|
|
|
|
if (Array.isArray(dockerImages) && dockerImages.length > 0) {
|
|
|
|
const dockerImageRecordsIdPromises = dockerImages.map(async (dockerImage) => {
|
|
|
|
// Find the required docker image record
|
|
|
|
const queryResult = await registry.queryRecords({ type: RecordType.DockerImageRecord, name: dockerImage }, true);
|
|
|
|
if (queryResult.length === 0) {
|
|
|
|
throw new Error(`Record not found, type: ${RecordType.DockerImageRecord}, name: ${dockerImage}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assume the first query result
|
|
|
|
const dockerImageRecordId = queryResult[0].id;
|
|
|
|
|
|
|
|
// Replace watcher name with the watcher record id
|
|
|
|
return dockerImageRecordId;
|
|
|
|
});
|
|
|
|
|
|
|
|
record.docker_images = await Promise.all(dockerImageRecordsIdPromises);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If .meta.watcher present, check for watcher record
|
|
|
|
if (watcherName) {
|
|
|
|
// Find the required watcher record
|
|
|
|
const queryResult = await registry.queryRecords({ type: RecordType.WatcherRecord, name: watcherName }, true);
|
|
|
|
if (queryResult.length === 0) {
|
|
|
|
throw new Error(`Record not found, type: ${RecordType.WatcherRecord}, name: ${watcherName}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assume the first query result
|
|
|
|
const watcherRecordId = queryResult[0].id;
|
|
|
|
|
|
|
|
// Replace watcher name with the watcher record id
|
|
|
|
record.meta.watcher = watcherRecordId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-24 07:27:39 +00:00
|
|
|
return registry.setRecord({ privateKey: userKey, record, bondId }, userKey, fee);
|
|
|
|
}
|
|
|
|
|
2024-05-09 10:35:54 +00:00
|
|
|
function getArgs (): any {
|
|
|
|
return yargs(hideBin(process.argv)).parserConfiguration({
|
|
|
|
'parse-numbers': false
|
|
|
|
}).usage('Usage: $0 [options]')
|
|
|
|
.option('config', {
|
|
|
|
alias: 'c',
|
|
|
|
describe: 'Config',
|
|
|
|
type: 'string',
|
|
|
|
demandOption: true
|
|
|
|
})
|
|
|
|
.option('records', {
|
|
|
|
alias: 'r',
|
|
|
|
describe: 'Records dir path',
|
|
|
|
type: 'string',
|
|
|
|
demandOption: true
|
|
|
|
})
|
|
|
|
.help().argv;
|
|
|
|
}
|
|
|
|
|
|
|
|
main()
|
|
|
|
.catch(err => {
|
|
|
|
console.error(err);
|
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
console.log('Done');
|
|
|
|
});
|