The Internet, 2023/08/06

Introduction

In this tutorial, we will create a Shopify Checkout UI Extension, which allows merchants to include a personalized message on their checkout page. We will achieve this by building an API using ChatGPT.

However, our primary goal is to evoke a nostalgic feeling from the 2000s, when you were using Word and Clippy™ would appear, offering assistance while you tried to write a document.

Ol' Clippy™

The Goal

Goal Plot

Our approach involves creating a Checkout UI extension that establishes a connection with the app’s API. This API will interact with ChatGPT to generate a personalized message. Subsequently, this message will be presented to the customer on the checkout page.

Getting Started

A great starting point is the Shopify documentation. You can access the Checkout UI Extension documentation here.

Building the App

To begin, let’s create a new app using the Shopify CLI. We’ll name the app “Clippify” and utilize Remix for the application. The first command will take a few minutes to complete.

$ yarn create "@shopify/app"
$ cd clippify
$ yarn shopify app generate extension

We select Checkout UI as Type of extension, we put clippify-checkout-ui as the name of the extension, and we will use Typescript React.

Fixing typing issue and eslint

Maybe it’s just me, but out of the box I had an error with the React components. I had to add the following to extension/clippify-checkout-ui/package.json:

{
  "devDependencies": {
    "@types/react": "^17.0.0",
    "typescript": "^5.0.0",
    "@shopify/eslint-plugin": "^42.1.0"
  }
}

I also had an issue with eslint, so I had to add the following to .eslintrc.js:

  plugins: ["@shopify/eslint-plugin"]

After the Generation

With the generation process complete, let’s take a look at the project structure that lies before us:


clippify
├── Dockerfile
├── README.md
├── app
│ ├── db.server.js
│ ├── entry.server.jsx
│ ├── root.jsx
│ ├── routes
│ │ ├── app._index.jsx
│ │ ├── app.additional.jsx
│ │ ├── app.jsx
│ │ ├── auth.$.jsx
│ │ ├── auth.login
│ │ │ ├── error.server.jsx
│ │ │ └── route.jsx
│ │ └── webhooks.jsx
│ └── shopify.server.js
├── extensions
│ └── clippify-checkout-ui
│     ├── README.md
│     ├── locales
│     ├── package.json
│     ├── shopify.extension.toml
│     ├── src
│     │ └── Checkout.tsx
│     └── tsconfig.json
├── package.json
├── prisma
├── public
│ └── favicon.ico
├── remix.config.js
├── shopify.app.toml
├── shopify.web.toml
├── tsconfig.json
└── yarn.lock

The Checkout UI

Creating the Block with Picture and Message

Our goal is to design a block featuring an image and a message, ideally situated in the bottom right corner of the checkout page. As we explore the available extension points, we find that purchase.checkout.block.render aligns perfectly with our requirements. Lucky for us, this extension is already included in the template.

To accomplish our goal, we need to identify the most suitable components for the task. The list of available components can be found here.

In our strategy, we will leverage the Grid component to carefully position the image and message elements.

Our Clippy™ Assistant!

Let’s find a mascot! We’ll head over to Lottie Files to discover a cute animation.

Who could resist a puppy tiger? Take a look: Tiger

Modifying Checkout.tsx

As a first step, we will simply add a straightforward message and our mascot. While we’re there, we’ll also retrieve the cart lines with the useCartLines hook (we will need them later):

function Extension() {
    const cartLines = useCartLines()
    // later we will use a metaobject to retrieve the picture asset. (Couldn't find a way to make the local path work)
    return (
        <Grid columns={['30%', 'fill']} spacing={'none'}>
            <View>
                <Image source="https://filippi.dev/tutorials/clippify/tiger.gif"/>
            </View>
            <View>
                <TextBlock>
                    It looks like you want to buy {cartLines[0].merchandise.title}!
                </TextBlock>
            </View>
        </Grid>
    );
}

Testing the App

We have now some code we can test, let’s start the app:

$ yarn dev

This process will generate a link enabling us to install the app on our development store. extension.

First Result

Not bad for a first result, the position needs to be fixed, the message display is not great but we’ll get there. Our focus can now shift to creating the right message.

The API

During the generation of the app, a remix project with a server is created. We’ll leverage this server to establish an API that simply provides a message. Our parameters for this API will include the customer (which could be null) and the cart lines.

ChatGPT Integration

To obtain the ChatGPT code, the most effective approach is to visit the Playground. Through experimentation, we can fine-tune our desired sentence and then extract the corresponding code. Here’s what we came up with:

const {Configuration, OpenAIApi} = require("openai");
const configuration = new Configuration({
    apiKey: process.env.OPENAI_API_KEY,
});
const openai = new OpenAIApi(configuration);

const response = await openai.createChatCompletion({
    model: "gpt-4",
    messages: [
        {
            "role": "user",
            "content": "You are a shop assistant that is helping a customer. Their name is\nFabio. You want them to be happy about their purchase.\\\\n  They have purchased White shirt and Yellow Tie. And you should explain how good they go together. Say\nthat in a brief way.\n"
        }
    ],
    temperature: 1,
    max_tokens: 256,
    top_p: 1,
    frequency_penalty: 0,
    presence_penalty: 0,
});

We will replace the parameters in our code and that’s it.

Remix

We create a new file app/routes/ai.jsx which define a straightforward GET endpoint, returning a message. For the purpose of this tutorial (and to save time), we will not implement a request handler for the entire cart and customer. Instead, we will focus on the specific parameters we need for the current task.

export const loader = async ({request}) => {
    const url = new URL(request.url);
    const firstName = url.searchParams.get("firstName");
    const title1 = url.searchParams.get("title1");
    const title2 = url.searchParams.get("title2");
    const customerName = firstName ? "Their name is " + firstName + "." : ""
    let cartMessage;
    if (title2) {
        cartMessage = `They have purchased ${title1} and ${title2}. And you should explain how good they go together.`
    } else {
        cartMessage = `They have purchased ${title1} and you should find a good reason to use it.`
    }
    const prompt = `You are a shop assistant that is helping a customer. ${customerName} You want them to be happy about their purchase.
  ${cartMessage} Say that in a short sentence suggesting that they should definitely but it. Be assertive.`

    let res = await getChatStream({ // this is a wrapper of the previous code
        message: prompt
    }).then(res => res.choices[0]?.message?.content || "Sorry, your taste is too good, I'm speechless.");

    return new Response(res, {headers: withCors()}); // we need to add CORS headers, I'll explain down here
};

Putting It All Together

Now, we can retrieve the generated message and display it. However, fetching the right URL of our API is a bit more complex than it might initially seem.

Fetch Call

As explained here, the checkout UI has a few limitations concerning network access. In summary:

  1. The UI runs in a Web Worker, resulting in a null origin. (This is why CORS headers are necessary for the API.)
  2. The UI lacks a secure method to authorize requests.
  3. App Proxy cannot be utilized for password-protected stores.

In a real production app, I would utilize the App Proxy. Unfortunately, it’s unavailable in a development store, forcing me to hardcode the URL (environment variables aren’t available neither). The lack of authentication is not a real problem, since we aren’t handling sensitive information.

const cartLines = useCartLines()
const {buyerIdentity} = useApi();
const [message, setMessage] = useState(null);
const firstName = buyerIdentity?.customer?.current?.firstName || "";
const title1 = cartLines[0]?.merchandise?.title;
const title2 = cartLines[1]?.merchandise?.title || "";
useMemo(() => {
    fetch(`https://mydeployedapp/ai?firstName=${encodeURIComponent(firstName)}&title1=${encodeURIComponent(title1)}&title2=${encodeURIComponent(title2)}`)
        .then(res => res.text()).then(text => {
        setMessage(text);
    })
}, [firstName, title1, title2]);

Completing the UI

At this stage, you might assume we’re almost there, just a bit of CSS, and we’re finished! However, that’s not entirely accurate. The React API (based on remote-ui) of the checkout UI only permits us to build functional components on top of the existing ones. Unfortunately, CSS isn’t supported. We’re limited to using the predefined parameters of the components, which hampers our ability to create the exact interface we desire.

Maybe this will change in the future, maybe not.

Image Generation Process

To solve that problem we will generate an image, on the server, with the message! This image will then be returned as a base64-encoded string.

If the concept of displaying text within an image doesn’t evoke a sense of nostalgia from the 2000s, I’m not sure what will.

To achieve this, we will utilize CanvasJS for image generation. The code responsible for creating the image is quite extensive, and you can review it in the repository. However, it’s not a critical aspect of the tutorial. I find the result quite satisfactory:

Message generated

Final UI

The final UI is quite straightforward; our main task is to display the GIF along with the message (picture).

By default, the image will contain a default message, which will be replaced once the fetch operation is successful.

const [message, setMessage] = useState("");

return <Grid columns={["30%", "fill"]}>
    <View>
        <Image source="https://filippi.dev/tutorials/clippify/tiger.gif"/>
    </View>
    <View>
        <Image source={message}/>
    </View>
</Grid>

Demo

Deploy

We are deploying our app on Heroku. Following (this)[https://shopify.dev/docs/apps/deployment/web]

$ heroku login
$ heroku container:login
$ heroku create -a clippify -s container
$ yarn shopify app env show

We create a heroku.yml file with the following content:

build:
  docker:
    web: Dockerfile
  config:
    SHOPIFY_API_KEY: 6.......................c

Because we are using canvasjs that uses glibc we can’t use alpine (unless we install the glibc package). So we are modifying Dockerfile replacing node:18-alpine with node:18-bullseye-slim and adding RUN apt-get update && apt-get install -y -q libfontconfig1 to install the fonts.

We need to update the heroku environment variables with the ones from the shopify app and our open api key.

$ heroku config:set -a clippify SCOPES=write_products SHOPIFY_APP_URL=<SHOPIFY_APP_URL> SHOPIFY_API_KEY=<SHOPIFY_API_KEY> SHOPIFY_API_SECRET=<SHOPIFY_API_SECRET>
$ heroku config:set -a clippify OPEN_API_KEY=<my open ai api key>
$ git push heroku

The app is now on heroku. We can add the block in our checkout and test it.

Positioning the block

I struggled a lot to make the app appear in the block list under the checkout customization. The simplified deployment introduced a “Development Store Preview” that needs to be OFF when installing the app to make it visible in the checkout block list. To understand that took about 20% of the time spent on this project. But now our puppy is in the right spot…

block

The Result

Pretty, pretty, pretty good, not bad!

Conclusion

It was really nice to work on Checkout UI, it’s fun. The UI limitations are a real struggle, and when the checkout.liquid customization will come to an end it will be fun to see how shops will React.

The Code

This is the repository if you want the full code: https://github.com/faaabio1618/clippify

Clippy is a registered trademark of Microsoft Corporation in the United States and/or other countries. The use of the Clippy image is for parody purposes only and is not endorsed by Microsoft. I’m not affiliated with Microsoft (buy I can send a CV if asked).