@graphprotocol/graph-cli
Version:
CLI for building for and deploying to The Graph
135 lines (134 loc) • 4.97 kB
JavaScript
import { print, prompt } from 'gluegun';
import open from 'open';
import { Args, Command, Flags, ux } from '@oclif/core';
// eslint-disable-next-line no-restricted-imports
import { URL, URLSearchParams } from '@whatwg-node/fetch';
import { createCompiler } from '../command-helpers/compiler.js';
import * as DataSourcesExtractor from '../command-helpers/data-sources.js';
import { DEFAULT_IPFS_URL } from '../command-helpers/ipfs.js';
import Protocol from '../protocols/index.js';
export default class PublishCommand extends Command {
static description = 'Publish to the Graph Network';
static args = {
'subgraph-manifest': Args.string({
default: 'subgraph.yaml',
}),
};
static flags = {
help: Flags.help({
char: 'h',
}),
'subgraph-id': Flags.string({
summary: 'Subgraph ID to publish to.',
required: false,
}),
'protocol-network': Flags.string({
summary: 'The network to use for the subgraph deployment.',
options: ['arbitrum-one', 'arbitrum-sepolia'],
default: 'arbitrum-one',
required: false,
}),
ipfs: Flags.string({
summary: 'Upload build results to an IPFS node.',
char: 'i',
default: DEFAULT_IPFS_URL,
}),
'ipfs-hash': Flags.string({
summary: 'IPFS hash of the subgraph manifest to deploy.',
required: false,
}),
'webapp-url': Flags.string({
summary: 'URL of the web UI you want to use to deploy.',
required: false,
default: 'https://cli.thegraph.com/publish',
}),
'api-key': Flags.string({
summary: 'The API key to use for the Subgraph queries.',
required: false,
}),
};
/**
* Prompt the user to open up the browser to continue publishing the subgraph
*/
async publishWithBrowser({ ipfsHash, webapp, subgraphId, protocolNetwork, apiKey, }) {
const { openBrowser } = await prompt
.ask([
{
type: 'confirm',
name: 'openBrowser',
message: () => `Open up the browser to continue publishing ?`,
initial: true,
required: true,
},
])
.catch(() => this.exit(1));
if (!openBrowser) {
this.exit(0);
}
const url = new URL(webapp);
const searchParams = new URLSearchParams(url.search);
searchParams.set('id', ipfsHash);
if (subgraphId) {
searchParams.set('subgraphId', subgraphId);
}
if (protocolNetwork) {
searchParams.set('network', protocolNetwork);
}
if (apiKey) {
searchParams.set('apiKey', apiKey);
}
url.search = searchParams.toString();
const openUrl = url.toString();
print.success(`Finalize the publish of the subgraph from the Graph CLI publish page. Opening up the browser to continue publishing at ${openUrl}`);
await open(openUrl);
return;
}
async run() {
const { args: { 'subgraph-manifest': manifest }, flags: { 'ipfs-hash': ipfsHash, 'webapp-url': webUiUrl, ipfs, 'subgraph-id': subgraphId, 'protocol-network': protocolNetwork, 'api-key': apiKey, }, } = await this.parse(PublishCommand);
if (subgraphId && !apiKey) {
ux.error('API key is required to publish to an existing subgraph (`--api-key`).\nSee https://thegraph.com/docs/en/deploying/subgraph-studio-faqs/#2-how-do-i-create-an-api-key', { exit: 1 });
}
if (ipfsHash) {
await this.publishWithBrowser({
ipfsHash,
webapp: webUiUrl,
subgraphId,
protocolNetwork,
apiKey,
});
return;
}
let protocol;
try {
const dataSourcesAndTemplates = await DataSourcesExtractor.fromFilePath(manifest);
protocol = Protocol.fromDataSources(dataSourcesAndTemplates);
}
catch (e) {
this.error(e, { exit: 1 });
}
const compiler = createCompiler(manifest, {
ipfs,
outputDir: 'build/',
outputFormat: 'wasm',
skipMigrations: false,
protocol,
});
// Exit with an error code if the compiler couldn't be created
if (!compiler) {
this.exit(1);
}
const result = await compiler.compile({ validate: true });
if (result === undefined || result === false) {
// Compilation failed, not deploying.
this.exit(1);
}
await this.publishWithBrowser({
ipfsHash: result,
webapp: webUiUrl,
subgraphId,
protocolNetwork,
apiKey,
});
return;
}
}