Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: refactor #79

Merged
merged 7 commits into from
Feb 21, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
77 changes: 0 additions & 77 deletions src/catalog/handler.js

This file was deleted.

34 changes: 20 additions & 14 deletions src/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,28 +12,33 @@

import { errorResponse } from './utils/http.js';
import { resolveConfig } from './utils/config.js';
import content from './content/handler.js';
import catalog from './catalog/handler.js';
import configHandler from './config/handler.js';
import handlers from './routes/index.js';

/**
* @type {Record<string, (ctx: Context, request: Request) => Promise<Response>>}
* @param {Request} req
*/
const handlers = {
content,
catalog,
config: configHandler,
// eslint-disable-next-line no-unused-vars
graphql: async (ctx) => errorResponse(501, 'not implemented'),
};
async function parseData(req) {
if (['GET', 'HEAD', 'OPTIONS'].includes(req.method)) {
return Object.fromEntries(new URL(req.url).searchParams.entries());
}
if (['POST', 'PUT', 'PATCH'].includes(req.method)) {
const text = await req.text();
try {
return JSON.parse(text);
} catch {
return text;
}
}
return null;
}

/**
* @param {import("@cloudflare/workers-types/experimental").ExecutionContext} pctx
* @param {Request} req
* @param {Env} env
* @returns {Context}
* @returns {Promise<Context>}
*/
export function makeContext(pctx, req, env) {
export async function makeContext(pctx, req, env) {
/** @type {Context} */
// @ts-ignore
const ctx = pctx;
Expand All @@ -49,6 +54,7 @@ export function makeContext(pctx, req, env) {
.map(([k, v]) => [k.toLowerCase(), v]),
),
};
ctx.data = await parseData(req);
return ctx;
}

Expand All @@ -60,7 +66,7 @@ export default {
* @returns {Promise<Response>}
*/
async fetch(request, env, pctx) {
const ctx = makeContext(pctx, request, env);
const ctx = await makeContext(pctx, request, env);

try {
const overrides = Object.fromEntries(ctx.url.searchParams.entries());
Expand Down
133 changes: 101 additions & 32 deletions src/catalog/storage/r2.js → src/routes/catalog/StorageClient.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright 2024 Adobe. All rights reserved.
* Copyright 2025 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
Expand All @@ -14,17 +14,31 @@ import { callPreviewPublish } from '../../utils/admin.js';
import { BatchProcessor } from '../../utils/batch.js';
import { errorWithResponse } from '../../utils/http.js';

/* eslint-disable no-await-in-loop, max-len */

export default class StorageClient {
/**
* Constructs a new StorageClient instance.
* @param {Context} ctx - The context object
* @param {Config} config - The configuration object.
* @param {Context} ctx
* @returns {StorageClient}
*/
static fromContext(ctx) {
if (!ctx.attributes.storageClient) {
ctx.attributes.storageClient = new StorageClient(ctx);
}
return ctx.attributes.storageClient;
}

/** @type {Context} */
ctx = undefined;

/**
* @param {Context} ctx
*/
constructor(ctx, config) {
constructor(ctx) {
this.ctx = ctx;
this.config = config;
}

/** @type {Config} */
get config() {
return this.ctx.config;
}

/**
Expand All @@ -33,11 +47,21 @@ export default class StorageClient {
* @returns {Promise<Product>} - A promise that resolves to the product.
*/
async fetchProduct(sku) {
const { log } = this.ctx;
const key = `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/products/${sku}.json`;
const {
log,
env,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

const key = `${org}/${site}/${storeCode}/${storeViewCode}/products/${sku}.json`;
log.debug('Fetching product from R2:', key);

const object = await this.ctx.env.CATALOG_BUCKET.get(key);
const object = await env.CATALOG_BUCKET.get(key);
if (!object) {
// Product not found in R2
throw errorWithResponse(404, 'Product not found');
Expand Down Expand Up @@ -74,29 +98,38 @@ export default class StorageClient {
* @returns {Promise<Partial<BatchResult>[]>} - Resolves with an array of save results.
*/
async storeProductsBatch(batch) {
const {
env,
log,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

const storePromises = batch.map(async (product) => {
const { sku, name } = product;
const key = `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/products/${sku}.json`;
const { sku, name, urlKey } = product;
const key = `${org}/${site}/${storeCode}/${storeViewCode}/products/${sku}.json`;
const body = JSON.stringify(product);

try {
const customMetadata = { sku, name };

const { urlKey } = product;
if (urlKey) {
customMetadata.urlKey = urlKey;
}

// Attempt to save the product
const putResponse = await this.ctx.env.CATALOG_BUCKET.put(key, body, {
const putResponse = await env.CATALOG_BUCKET.put(key, body, {
httpMetadata: { contentType: 'application/json' },
customMetadata,
});

// If urlKey exists, save the urlKey metadata
if (urlKey) {
const metadataKey = `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/urlkeys/${urlKey}`;
await this.ctx.env.CATALOG_BUCKET.put(metadataKey, '', {
const metadataKey = `${org}/${site}/${storeCode}/${storeViewCode}/urlkeys/${urlKey}`;
await env.CATALOG_BUCKET.put(metadataKey, '', {
httpMetadata: { contentType: 'application/octet-stream' },
customMetadata,
});
Expand All @@ -116,7 +149,7 @@ export default class StorageClient {

return result;
} catch (error) {
this.ctx.log.error(`Error storing product SKU: ${sku}:`, error);
log.error(`Error storing product SKU: ${sku}:`, error);
return {
sku,
status: error.code || 500,
Expand Down Expand Up @@ -152,10 +185,16 @@ export default class StorageClient {
* @returns {Promise<Partial<BatchResult>[]>} - Resolves with an array of deletion results.
*/
async deleteProductsBatch(batch) {
const { log, env } = this.ctx;
const {
org, site, storeCode, storeViewCode,
} = this.config;
log,
env,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

const deletionPromises = batch.map(async (sku) => {
try {
Expand All @@ -178,7 +217,7 @@ export default class StorageClient {
await env.CATALOG_BUCKET.delete(urlKeyPath);
}

const adminResponse = await callPreviewPublish(this.config, 'DELETE', sku, urlKey);
const adminResponse = await callPreviewPublish(this.ctx.config, 'DELETE', sku, urlKey);
/**
* @type {Partial<BatchResult>}
*/
Expand Down Expand Up @@ -209,9 +248,19 @@ export default class StorageClient {
* @returns {Promise<string>} - A promise that resolves to the SKU.
*/
async lookupSku(urlKey) {
const {
env,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

// Make a HEAD request to retrieve the SKU from metadata based on the URL key
const urlKeyPath = `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/urlkeys/${urlKey}`;
const headResponse = await this.ctx.env.CATALOG_BUCKET.head(urlKeyPath);
const urlKeyPath = `${org}/${site}/${storeCode}/${storeViewCode}/urlkeys/${urlKey}`;
const headResponse = await env.CATALOG_BUCKET.head(urlKeyPath);

if (!headResponse || !headResponse.customMetadata?.sku) {
// SKU not found for the provided URL key
Expand All @@ -227,10 +276,20 @@ export default class StorageClient {
* @returns {Promise<string | undefined>} - A promise that resolves to the URL key or undefined.
*/
async lookupUrlKey(sku) {
const {
env,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

// Construct the path to the product JSON file
const productKey = `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/products/${sku}.json`;
const productKey = `${org}/${site}/${storeCode}/${storeViewCode}/products/${sku}.json`;

const headResponse = await this.ctx.env.CATALOG_BUCKET.head(productKey);
const headResponse = await env.CATALOG_BUCKET.head(productKey);
if (!headResponse || !headResponse.customMetadata) {
return undefined;
}
Expand All @@ -249,9 +308,18 @@ export default class StorageClient {
* @returns {Promise<Product[]>} - A promise that resolves to the products.
*/
async listAllProducts() {
const bucket = this.ctx.env.CATALOG_BUCKET;
const listResponse = await bucket.list({
prefix: `${this.config.org}/${this.config.site}/${this.config.storeCode}/${this.config.storeViewCode}/products/`,
const {
env,
config: {
org,
site,
storeCode,
storeViewCode,
},
} = this.ctx;

const listResponse = await env.CATALOG_BUCKET.list({
prefix: `${org}/${site}/${storeCode}/${storeViewCode}/products/`,
});
const files = listResponse.objects;

Expand All @@ -271,18 +339,19 @@ export default class StorageClient {

// Process each chunk sequentially
for (const chunk of fileChunks) {
// eslint-disable-next-line no-await-in-loop
const chunkResults = await Promise.all(
chunk.map(async (file) => {
const objectKey = file.key;

const headResponse = await bucket.head(objectKey);
const headResponse = await env.CATALOG_BUCKET.head(objectKey);
if (headResponse) {
const { customMetadata } = headResponse;
const { sku } = customMetadata;
return {
...customMetadata,
links: {
product: `${this.ctx.url.origin}/${this.config.org}/${this.config.site}/catalog/${this.config.storeCode}/${this.config.storeViewCode}/product/${sku}`,
product: `${this.ctx.url.origin}/${org}/${site}/catalog/${storeCode}/${storeViewCode}/product/${sku}`,
},
};
} else {
Expand Down
Loading