Skip to content

Commit

Permalink
Update CLI init command to create or link storefront (#1681)
Browse files Browse the repository at this point in the history
  • Loading branch information
aswamy authored Jan 25, 2024
1 parent 6f90c92 commit 1bc053c
Show file tree
Hide file tree
Showing 5 changed files with 100 additions and 51 deletions.
7 changes: 7 additions & 0 deletions .changeset/witty-clouds-sing.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
---
'@shopify/cli-hydrogen': patch
---

`init` command can link an existing storefront on Admin when creating a storefront from the CLI.

After logging into your Shop, the CLI will let you choose if you wish to create a new storefront on Admin or link an existing one. `npm create @shopify/hydrogen` command will also benefit from this change.
24 changes: 12 additions & 12 deletions packages/cli/src/commands/hydrogen/link.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,9 @@ describe('link', () => {
storefront: undefined,
};

const expectedStorefrontName = 'New Storefront';
const expectedJobId = 'gid://shopify/Job/1';

beforeEach(async () => {
vi.mocked(login).mockResolvedValue({
session: ADMIN_SESSION,
Expand All @@ -69,6 +72,15 @@ describe('link', () => {
]);

vi.mocked(renderSelectPrompt).mockResolvedValue(FULL_SHOPIFY_CONFIG.shop);

vi.mocked(createStorefront).mockResolvedValue({
storefront: {
id: 'gid://shopify/HydrogenStorefront/1',
title: expectedStorefrontName,
productionUrl: 'https://example.com',
},
jobId: expectedJobId,
});
});

afterEach(() => {
Expand Down Expand Up @@ -128,20 +140,8 @@ describe('link', () => {
});

describe('when you want to link a new Hydrogen storefront', () => {
const expectedStorefrontName = 'New Storefront';
const expectedJobId = 'gid://shopify/Job/1';

beforeEach(async () => {
vi.mocked(renderSelectPrompt).mockResolvedValue(null);

vi.mocked(createStorefront).mockResolvedValue({
storefront: {
id: 'gid://shopify/HydrogenStorefront/1',
title: expectedStorefrontName,
productionUrl: 'https://example.com',
},
jobId: expectedJobId,
});
});

it('chooses to create a new storefront given the directory path', async () => {
Expand Down
35 changes: 7 additions & 28 deletions packages/cli/src/commands/hydrogen/link.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import {basename} from '@shopify/cli-kit/node/path';

import {
renderConfirmationPrompt,
renderSelectPrompt,
renderSuccess,
renderTasks,
renderTextPrompt,
Expand All @@ -21,6 +20,10 @@ import {titleize} from '../../lib/string.js';
import {getCliCommand} from '../../lib/shell.js';
import {login} from '../../lib/auth.js';
import type {AdminSession} from '../../lib/auth.js';
import {
type HydrogenStorefront,
handleStorefrontSelection,
} from '../../lib/onboarding/common.js';

export default class Link extends Command {
static description =
Expand All @@ -47,12 +50,6 @@ export interface LinkStorefrontArguments {
storefront?: string;
}

interface HydrogenStorefront {
id: string;
title: string;
productionUrl: string;
}

export async function runLink({
force,
path: root = process.cwd(),
Expand Down Expand Up @@ -134,32 +131,14 @@ export async function linkStorefront(
return;
}
} else {
const choices = [
{
label: 'Create a new storefront',
value: null,
},
...storefronts.map(({id, title, productionUrl}) => ({
label: `${title} (${productionUrl})`,
value: id,
})),
];

const storefrontId = await renderSelectPrompt({
message: 'Select a Hydrogen storefront to link',
choices,
});
selectedStorefront = await handleStorefrontSelection(storefronts);

if (storefrontId) {
selectedStorefront = storefronts.find(({id}) => id === storefrontId);
} else {
if (!selectedStorefront) {
selectedStorefront = await createNewStorefront(root, session);
}
}

if (selectedStorefront) {
await setStorefront(root, selectedStorefront);
}
await setStorefront(root, selectedStorefront);

return selectedStorefront;
}
Expand Down
57 changes: 52 additions & 5 deletions packages/cli/src/lib/onboarding/common.ts
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ import {
renderRoutePrompt,
} from '../setups/routes/generate.js';
import {execAsync} from '../process.js';
import {getStorefronts} from '../graphql/admin/link-storefront.js';

export type InitOptions = {
path?: string;
Expand Down Expand Up @@ -221,6 +222,7 @@ export async function handleCliShortcut(
}

type StorefrontInfo = {
id?: string;
title: string;
shop: string;
shopName: string;
Expand All @@ -238,13 +240,58 @@ export async function handleStorefrontLink(
const {session, config} = await login();
renderLoginSuccess(config);

const title = await renderTextPrompt({
message: 'New storefront name',
defaultValue: titleize(config.shopName),
abortSignal: controller.signal,
const storefronts = await getStorefronts(session);

let selectedStorefront = await handleStorefrontSelection(storefronts);

let title;

if (selectedStorefront) {
title = selectedStorefront.title;
} else {
title = await renderTextPrompt({
message: 'New storefront name',
defaultValue: titleize(config.shopName),
abortSignal: controller.signal,
});
}

return {
...config,
id: selectedStorefront?.id,
title,
session,
};
}

export type HydrogenStorefront = {
id: string;
title: string;
productionUrl: string;
};

export async function handleStorefrontSelection(
storefronts: HydrogenStorefront[],
): Promise<HydrogenStorefront | undefined> {
const choices = [
{
label: 'Create a new storefront',
value: null,
},
...storefronts.map(({id, title, productionUrl}) => ({
label: `${title} (${productionUrl})`,
value: id,
})),
];

const storefrontId = await renderSelectPrompt({
message: 'Select a Hydrogen storefront to link',
choices,
});

return {...config, title, session};
return storefrontId
? storefronts.find(({id}) => id === storefrontId)
: undefined;
}

type Project = {
Expand Down
28 changes: 22 additions & 6 deletions packages/cli/src/lib/onboarding/local.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ export async function setupLocalStarterTemplate(

const createStorefrontPromise =
storefrontInfo &&
!storefrontInfo.id &&
createStorefront(storefrontInfo.session, storefrontInfo.title)
.then(async ({storefront, jobId}) => {
if (jobId) await waitForJob(storefrontInfo.session, jobId);
Expand Down Expand Up @@ -141,17 +142,25 @@ export async function setupLocalStarterTemplate(
'# Run `h2 link` to also inject environment variables from your storefront,\n' +
'# or `h2 env pull` to populate this file.';

if (storefrontInfo && createStorefrontPromise) {
let storefrontToLink: {id: string; title: string} | undefined;

if (storefrontInfo) {
promises.push(
// Save linked storefront in project
setUserAccount(project.directory, storefrontInfo),
createStorefrontPromise.then((storefront) =>
// Save linked storefront in project
setStorefront(project.directory, storefront),
),
// Write empty dotenv file to fallback to remote Oxygen variables
writeFile(joinPath(project.directory, '.env'), envLeadingComment),
);

if (storefrontInfo.id) {
storefrontToLink = {id: storefrontInfo.id, title: storefrontInfo.title};
} else if (createStorefrontPromise) {
promises.push(
createStorefrontPromise.then((createdStorefront) => {
storefrontToLink = createdStorefront;
}),
);
}
} else if (templateAction === 'mock') {
promises.push(
// Set required env vars
Expand All @@ -170,7 +179,14 @@ export async function setupLocalStarterTemplate(
);
}

return Promise.all(promises).catch(abort);
return Promise.all(promises)
.then(() => {
if (storefrontToLink) {
// Save linked storefront in project
return setStorefront(project.directory, storefrontToLink);
}
})
.catch(abort);
});

const {language, transpileProject} = await handleLanguage(
Expand Down

0 comments on commit 1bc053c

Please sign in to comment.