@salesforce/plugin-org
Version:
Commands to interact with Salesforce orgs
111 lines • 5.49 kB
JavaScript
/*
* Copyright (c) 2020, salesforce.com, inc.
* All rights reserved.
* Licensed under the BSD 3-Clause license.
* For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause
*/
import path from 'node:path';
import { Flags, loglevel, orgApiVersionFlagWithDeprecations, requiredOrgFlagWithDeprecations, } from '@salesforce/sf-plugins-core';
import { Messages } from '@salesforce/core';
import { MetadataResolver } from '@salesforce/source-deploy-retrieve';
import { env } from '@salesforce/kit';
import { buildFrontdoorUrl } from '../../shared/orgOpenUtils.js';
import { OrgOpenCommandBase } from '../../shared/orgOpenCommandBase.js';
Messages.importMessagesDirectoryFromMetaUrl(import.meta.url);
const messages = Messages.loadMessages('@salesforce/plugin-org', 'open');
const sharedMessages = Messages.loadMessages('@salesforce/plugin-org', 'messages');
export class OrgOpenCommand extends OrgOpenCommandBase {
static summary = messages.getMessage('summary');
static description = messages.getMessage('description');
static examples = messages.getMessages('examples');
static aliases = ['force:org:open', 'force:source:open'];
static deprecateAliases = true;
static flags = {
...OrgOpenCommandBase.flags,
'target-org': requiredOrgFlagWithDeprecations,
'api-version': orgApiVersionFlagWithDeprecations,
private: Flags.boolean({
summary: messages.getMessage('flags.private.summary'),
exclusive: ['url-only', 'browser'],
}),
browser: Flags.option({
char: 'b',
summary: messages.getMessage('flags.browser.summary'),
options: ['chrome', 'edge', 'firefox'], // These are ones supported by "open" package
exclusive: ['url-only', 'private'],
})(),
path: Flags.string({
char: 'p',
summary: messages.getMessage('flags.path.summary'),
env: 'FORCE_OPEN_URL',
exclusive: ['source-file'],
parse: (input) => Promise.resolve(encodeURIComponent(decodeURIComponent(input))),
}),
'url-only': Flags.boolean({
char: 'r',
summary: messages.getMessage('flags.url-only.summary'),
aliases: ['urlonly'],
deprecateAliases: true,
}),
loglevel,
'source-file': Flags.file({
char: 'f',
aliases: ['sourcefile'],
exclusive: ['path'],
deprecateAliases: true,
summary: messages.getMessage('flags.source-file.summary'),
}),
};
async run() {
this.warn(sharedMessages.getMessage('BehaviorChangeWarning'));
const { flags } = await this.parse(OrgOpenCommand);
this.org = flags['target-org'];
this.connection = this.org.getConnection(flags['api-version']);
const singleUseEnvVar = env.getBoolean('SF_SINGLE_USE_ORG_OPEN_URL');
const singleUseMode = singleUseEnvVar ? singleUseEnvVar : !(flags['url-only'] || this.jsonEnabled());
const [frontDoorUrl, retUrl] = await Promise.all([
buildFrontdoorUrl(this.org, this.connection, singleUseMode),
flags['source-file'] ? generateFileUrl(flags['source-file'], this.connection) : flags.path,
]);
return this.openOrgUI(flags, frontDoorUrl, retUrl);
}
}
const generateFileUrl = async (file, conn) => {
try {
const metadataResolver = new MetadataResolver();
const components = metadataResolver.getComponentsFromPath(file);
const typeName = components[0]?.type?.name;
switch (typeName) {
case 'Bot':
return `AiCopilot/copilotStudio.app#/copilot/builder?copilotId=${await botFileNameToId(conn, file)}`;
case 'ApexPage':
return `/apex/${path.basename(file).replace('.page-meta.xml', '').replace('.page', '')}`;
case 'Flow':
return `/builder_platform_interaction/flowBuilder.app?flowId=${await flowFileNameToId(conn, file)}`;
case 'FlexiPage':
return `/visualEditor/appBuilder.app?pageId=${await flexiPageFilenameToId(conn, file)}`;
default:
return 'lightning/setup/FlexiPageList/home';
}
}
catch (error) {
if (error instanceof Error && error.name === 'FlowIdNotFoundError') {
throw error;
}
return 'lightning/setup/FlexiPageList/home';
}
};
const botFileNameToId = async (conn, filePath) => (await conn.singleRecordQuery(`SELECT id FROM BotDefinition WHERE DeveloperName='${path.basename(filePath, '.bot-meta.xml')}'`)).Id;
/** query flexipage via toolingPAI to get its ID (starts with 0M0) */
const flexiPageFilenameToId = async (conn, filePath) => (await conn.singleRecordQuery(`SELECT id FROM flexipage WHERE DeveloperName='${path.basename(filePath, '.flexipage-meta.xml')}'`, { tooling: true })).Id;
/** query the rest API to turn a flow's filepath into a FlowId (starts with 301) */
const flowFileNameToId = async (conn, filePath) => {
try {
const flow = await conn.singleRecordQuery(`SELECT DurableId FROM FlowVersionView WHERE FlowDefinitionView.ApiName = '${path.basename(filePath, '.flow-meta.xml')}' ORDER BY VersionNumber DESC LIMIT 1`);
return flow.DurableId;
}
catch (error) {
throw messages.createError('FlowIdNotFound', [filePath]);
}
};
//# sourceMappingURL=open.js.map