Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/iliashad/visual-product-search-app

Visual Product Search API using Shopify API, Algolia, Google Vision API, Remix.js and React Native
https://github.com/iliashad/visual-product-search-app

google-cloud react-native shopify

Last synced: 27 days ago
JSON representation

Visual Product Search API using Shopify API, Algolia, Google Vision API, Remix.js and React Native

Awesome Lists containing this project

README

        

# React Native: Visual Product Search API using Shopify API

## Overview

![visual-search](https://github.com/user-attachments/assets/f8477560-2a6f-4807-b139-19ea81c28d07)

| License | Mobile Technology | Image Labeling | eCommerce CMS | Search API |
| :---: | :---: | :---: | :---: | :---: |
| [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT) | ![React Native](https://img.shields.io/badge/react_native-%2320232a.svg?style=for-the-badge&logo=react&logoColor=%2361DAFB) | [![Google Cloud](https://img.shields.io/badge/Google%20Cloud-%234285F4.svg?logo=google-cloud&logoColor=white)](#) | [![Shopify](https://img.shields.io/badge/Shopify-7AB55C?logo=shopify&logoColor=fff)](#) | Algolia Search API

| Browsing | Image Searching | Displaying Results |
| :---: | :---: | :---: |
|![IMG_7232](https://github.com/user-attachments/assets/73118f63-ec74-44af-8ff0-dc8edbe216dd) | ![IMG_7234](https://github.com/user-attachments/assets/9ac14cdb-9b64-40a2-a006-d952a29e3935) | ![IMG_7233](https://github.com/user-attachments/assets/179fa312-5293-4c74-9a49-c791efe3d34a)|

### Developer Note: Keeping it Simple

While building this demo, I intentionally simplified many aspects that would be crucial for a production system. Here's why:

The main goal was to demonstrate how to:

- Connect Google Cloud Vision API with Shopify product search
- Process product images to enable visual search
- Create a basic mobile interface for image uploads

I deliberately omitted production concerns like error handling, security, and scalability because:

1. This is a learning demo focused on the core visual search workflow
2. Adding production-grade features would obscure the main technical concepts
3. Each production requirement (auth, caching, monitoring etc.) deserves its own deep dive

Think of this implementation as a "proof of concept" that shows:

- The basic architecture works
- Visual search can enhance product discovery
- The core APIs integrate successfully

For anyone looking to build on this demo, you'd want to address production concerns based on your specific needs - like authentication, error handling, and performance optimization. But those additions shouldn't overshadow understanding the core visual search mechanics demonstrated here.

The goal was to keep the code approachable and focused on demonstrating value, rather than building a production-ready system. This lets developers understand the core concepts first, then layer in production requirements as needed.

### Demo Video

https://github.com/user-attachments/assets/28e109bb-14d1-40e8-a1ec-1b8415c54a19

## Features

- 📸 Visual Product Search
- 🚀 Instant Product Matching
- 📱 Cross-Platform Mobile Experience (iOS & Android)

## Tech Stack

- **Frontend**: React Native
- **Image Recognition**: Google Cloud Platform Vision API
- **Search**: Algolia Search API
- **Backend**: Remix.js

## Prerequisites

- Node.js (v16+ recommended)
- npm or Yarn
- React Native CLI
- Google Cloud Platform Account
- Algolia Account
- Shopify Partner Account or a Shopify development store

## Get started

1. Install dependencies

```bash
npm install
```

2. Start the app

```bash
npx expo start
```

In the output, you'll find options to open the app in a

- [development build](https://docs.expo.dev/develop/development-builds/introduction/)
- [Android emulator](https://docs.expo.dev/workflow/android-studio-emulator/)
- [iOS simulator](https://docs.expo.dev/workflow/ios-simulator/)
- [Expo Go](https://expo.dev/go), a limited sandbox for trying out app development with Expo

You can start developing by editing the files inside the **app** directory. This project uses [file-based routing](https://docs.expo.dev/router/introduction).

3. Set up environment variables:
Create a `.env` file in the project root with the following:

```
EXPO_PUBLIC_APP_HOST=""
EXPO_PUBLIC_SHOPIFY_API_SECRET=""
EXPO_PUBLIC_SHOPIFY_DOMAIN="example.myshopify.com"
```

## Configuration

### Google Cloud Platform Vision API
1. Create a GCP project
2. Enable Vision API
3. Generate and download service account credentials

### Algolia
1. Create an Algolia account
2. Set up your product index
3. Configure search settings

### Shopify Storefront API
1. Create a Shopify development store
2. Set up Shopify Storefront API
3. Get your API key

## Running the App

### iOS
```bash
npm run ios
```

### Android
```bash
npm run android
```
## Workflow

1. The user uploads a product photo
2. GCP Vision API analyzes the image
3. Extracted tags/labels sent to Algolia
4. Algolia returns matching products
5. Results displayed to the user

## Server Setup

I used Remix.js [Cloudflare Template](https://remix.run/resources/remix-worker-template)

I made an API route for searching using an image in `base64` format using [Algolia Search API](https://www.algolia.com/doc/rest-api/search/) which I used before to index our Shopify product image with image labels we got from Google Cloud Vision API. I used the Shopify product variant id and product id as an index.

```tsx
// app/routes/search.tsx
import type { LoaderFunctionArgs } from '@remix-run/cloudflare';
import { json } from '@remix-run/react';
import {
getImageLabels,
reduceLabelsToFilters,
} from '../services/vision.server';
import { algoliaClient } from '../services/algolia.server';

export async function action({ request }: LoaderFunctionArgs) {
const formData = await request.formData();
const image = formData.get('image') as string; // Image Base64
if (!image) {
return json({ error: 'No image provided' }, { status: 400 });
}

const classifiedImage = await getImageLabels(image);
const labels = reduceLabelsToFilters(classifiedImage.labels);

const indexName = 'products';
const { results } = await algoliaClient.search({
requests: [
{
indexName,
optionalFilters: labels,
query: classifiedImage.labels.map(label => label.description).join(' '),
},
],
});

// ObjectID is a unique identifier for each image in Algolia, Example: "1234-54355", first part is the product ID and the second part is the variant ID
// We're indexing each variant image separately, and we're using the product ID to get parent product information
const products = results[0].hits.map(
product => product.objectID && product.objectID.split('-')[0],
);

return json({ results: products });
}
```

Here's our [Algolia Search API](https://www.algolia.com/doc/rest-api/search/) client:

```tsx
// app/services/algolia.server.ts
import { searchClient } from '@algolia/client-search';
import { configDotenv } from 'dotenv';
configDotenv({
path: '.dev.vars',
});
if (!process.env.ALGOLIA_APP_ID) {
throw new Error(
'Missing Algolia App ID. Please provide it in the .env file.',
);
}
if (!process.env.ALGOLIA_SEARCH_KEY) {
throw new Error(
'Missing Algolia Search Key. Please provide it in the .env file.',
);
}
export const algoliaClient = searchClient(
process.env.ALGOLIA_APP_ID,
process.env.ALGOLIA_SEARCH_KEY,
);

```

Here's our Shopify Storefront API client:

```tsx
// app/services/shopify.server.ts

import { createStorefrontApiClient } from '@shopify/storefront-api-client';
import { configDotenv } from 'dotenv';
configDotenv({
path: '.dev.vars',
});
if (!process.env.STORE_DOMAIN) {
throw new Error('Missing Store Domain. Please provide it in the .env file.');
}
if (!process.env.STOREFRONT_ACCESS_TOKEN) {
throw new Error(
'Missing Storefront Access Token. Please provide it in the .env file.',
);
}

export const ShopifyClient = createStorefrontApiClient({
storeDomain: process.env.STORE_DOMAIN,
publicAccessToken: process.env.STOREFRONT_ACCESS_TOKEN,
apiVersion: '2024-10',
});

```
And our Google Cloud Vision API to get image labels by passing an image in `base64` format

```tsx
// app/services/vision.server.ts

import vision from '@google-cloud/vision';
import { configDotenv } from 'dotenv';
configDotenv({
path: '.dev.vars',
});
if (!process.env.GOOGLE_APPLICATION_CREDENTIALS_BASE64) {
throw new Error(
'Missing Google Application Credentials. Please provide it in the .env file.',
);
}

const credentials = JSON.parse(
Buffer.from(
process.env.GOOGLE_APPLICATION_CREDENTIALS_BASE64,
'base64',
).toString('utf-8'),
);

const client = new vision.ImageAnnotatorClient({
credentials,
});

type Label = {
description: string | null | undefined;
score: number | null | undefined;
};
export const getImageLabels = async (imageBase64: string) => {
const [result] = await client.annotateImage({
image: {
content: imageBase64,
},
features: [
{
type: 'LABEL_DETECTION',
maxResults: 10,
},
],
});
if (!result.labelAnnotations) {
return { labels: [] };
}

const labels = result.labelAnnotations
.filter(label => label.score && label.score > 0.5)
.map(label => ({
description: label.description,
score: label.score,
}));
return { labels };
};

export const reduceLabelsToFilters = (labels: Label[]) => {
const optionalFilters = labels.map(
label => `labels.description:'${label.description}'`,
);
return optionalFilters;
};

```
## Directory Structure
```
shopify-search-app/
│
├── app/ # App configuration and entry points
│ ├── _layout.tsx # Navigation layout
│ └── (tabs)/ # Grouped tab navigation
│ ├── _layout.tsx # Navigation layout
│ └── index.tsx # Main Entry view
│ └── +not-found.tsx # Not found view

├── assets/ # Static assets
│ ├── images/
│ ├── fonts/
│
├── components/ # Reusable UI components
│ │ ├── Button.tsx
│ │ ├── Container.tsx
│ │ ├── Text.tsx
│ │ ├── Card.tsx
│ │ ├── QueryClient.tsx
│ │ ├── ProductList.tsx
│ │ ├── ProductCard.tsx
│ │ ├── SearchBar.tsx
│ │ ├── SearchResults.tsx
│ │ ├── SelectedImagePreview.tsx
│ │ ├── NotFound.tsx
│ │ ├── Grid.tsx
│ │ ├── ExternalLink.tsx
│ │ ├── HapticTab.tsx
│ │ ├── LoadingResults.tsx
│ │ └── SearchImageResults.tsx
│
├── constants/ # App-wide constants
│ ├── Colors.ts
│ ├── Query.ts
│
├── hooks/ # Custom React hooks
│ ├── useHomeViewState.ts
│ ├── useProducts.ts
│ ├── useImageSearch.ts
│ └── useProductSearch.ts
│
├── navigation/ # Navigation configuration
│ ├── AppNavigator.tsx
│ └── types.ts
│
├── screens/ # Full screen components
│ ├── HomeScreen.tsx
│ ├── LoginScreen.tsx
│ └── ProfileScreen.tsx
│
├── services/ # API and external service calls
│ ├── search.ts
│ ├── shopify.ts
│ └── StorageService.ts
│
│
├── types/ # TypeScript type definitions
│ └── shopify.ts
│
│
├── .env # Environment variables
├── app.json # Expo app configuration
├── package.json # Project dependencies and scripts
├── tsconfig.json # TypeScript configuration
└── README.md # Project documentation
```

## Future Roadmap

- [ ] Advanced ML-powered visual search

## Contributing

1. Fork the repository
2. Create your feature branch (`git checkout -b feature/AmazingFeature`)
3. Commit your changes (`git commit -m 'Add some AmazingFeature'`)
4. Push to the branch (`git push origin feature/AmazingFeature`)
5. Open a Pull Request

## License

Distributed under the MIT License. See `LICENSE` for more information.