From 74cd16e75e5a3f7c31ab6dc7e1ffe55fa5245d28 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Sun, 15 Dec 2024 11:00:43 +0100 Subject: [PATCH 01/11] Update Trip endpoints --- .../tripRoutesValidator.ts | 37 +++++++++----- server/src/controllers/trip/addTrip.ts | 14 ++--- server/src/controllers/trip/editTrip.ts | 11 ++-- server/src/controllers/trip/index.ts | 1 + server/src/db/schema.ts | 4 +- server/src/drizzle/methods/Geojson.ts | 26 ++++++++-- server/src/drizzle/methods/trip.ts | 51 ++++++++++--------- server/src/routes/trpcRouter.ts | 3 ++ server/src/services/trip/addTripService.ts | 36 +++++++++---- server/src/services/trip/trip.service.ts | 1 + 10 files changed, 116 insertions(+), 68 deletions(-) diff --git a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts index 27afb355a..960554e0e 100644 --- a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts +++ b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts @@ -5,14 +5,13 @@ const tripActivityValues = Object.values(TripActivity) as [string, ...string[]]; export const addTripForm = z.object({ name: z.string(), - description: z.string(), - activity: z.enum(tripActivityValues), + description: z.string().optional().nullable(), + activity: z.enum(tripActivityValues).optional(), is_public: z.union([z.literal('0'), z.literal('1')]), }); -// @ts-ignore const coordinateSchema = z.lazy(() => - z.union([z.number(), z.array(coordinateSchema)]), + z.union([z.number(), z.array(z.number())]), ); const baseGeometrySchema = z.object({ @@ -50,31 +49,41 @@ export const getTripById = z.object({ export const addTripDetails = z.object({ start_date: z.string(), end_date: z.string(), - destination: z.string(), - activity: z.enum(tripActivityValues), + activity: z.enum(tripActivityValues).optional(), parks: z.string().optional(), trails: z.string().optional(), + packId: z.string(), geoJSON: z.string(), - owner_id: z.string(), - pack_id: z.string(), - bounds: z.tuple([z.array(z.number()), z.array(z.number())]), }); -export const addTrip = addTripDetails.merge(addTripForm); +export const addTrip = addTripDetails.merge(addTripForm).omit({ + is_public: true, +}); +export type AddTripType = z.infer; export const editTrip = z.object({ - id: z.string(), + id: z.string().min(1), name: z.string().optional(), description: z.string().optional(), - activity: z.enum(tripActivityValues).optional(), start_date: z.string().optional(), end_date: z.string().optional(), - destination: z.string().optional(), + packId: z.string().optional(), + parks: z.string().optional(), + trails: z.string().optional(), + activity: z.enum(tripActivityValues).optional(), + geoJSON: z.string().optional(), +}); + +export type EditTripType = z.infer; + +export const setTripVisibility = z.object({ + tripId: z.string().min(1), is_public: z.union([z.literal('0'), z.literal('1')]), }); +export type SetTripVisibilityType = z.infer; export const deleteTrip = z.object({ - tripId: z.string().nonempty(), + tripId: z.string().min(1), }); export const queryTrip = z.object({ diff --git a/server/src/controllers/trip/addTrip.ts b/server/src/controllers/trip/addTrip.ts index feba2d7db..d0d3d2135 100644 --- a/server/src/controllers/trip/addTrip.ts +++ b/server/src/controllers/trip/addTrip.ts @@ -1,11 +1,13 @@ -import { publicProcedure, protectedProcedure } from '../../trpc'; -import { addTripService } from '../../services/trip/addTripService'; +import { protectedProcedure } from '../../trpc'; +import { Context } from 'hono'; +import { addTripService } from '../../services/trip/trip.service'; import * as validator from '@packrat/validations'; -export const addTrip = async (c) => { +export const addTrip = async (c: Context) => { try { - const tripData = await c.req.json(); - const trip = await addTripService(tripData); + const requestData = (await c.req.json()) satisfies validator.AddTripType; + const tripData = { ...requestData, ownerId: c.user.id }; + const trip = await addTripService(tripData, c.executionCtx); return c.json(trip, 200); } catch (error) { return c.json({ error: `${error.message}` }, 500); @@ -14,7 +16,7 @@ export const addTrip = async (c) => { export function addTripRoute() { return protectedProcedure.input(validator.addTrip).mutation(async (opts) => { - const tripData = opts.input; + const tripData = { ownerId: opts.ctx.user.id, ...opts.input }; return await addTripService(tripData, opts.ctx.executionCtx); }); } diff --git a/server/src/controllers/trip/editTrip.ts b/server/src/controllers/trip/editTrip.ts index a509c8391..e84403a2c 100644 --- a/server/src/controllers/trip/editTrip.ts +++ b/server/src/controllers/trip/editTrip.ts @@ -1,12 +1,12 @@ import { protectedProcedure } from '../../trpc'; import * as validator from '@packrat/validations'; -import { Trip } from '../../drizzle/methods/trip'; +import { editTripService } from '../../services/trip/trip.service'; +import { Context } from 'hono'; -export const editTrip = async (c) => { +export const editTrip = async (c: Context) => { try { const tripData = await c.req.json(); - const tripClass = new Trip(); - const trip = await tripClass.update(tripData); + const trip = await editTripService(tripData, c.executionCtx); return c.json({ trip }, 200); } catch (error) { return c.json({ error: `${error.message}` }, 500); @@ -16,8 +16,7 @@ export const editTrip = async (c) => { export function editTripRoute() { return protectedProcedure.input(validator.editTrip).mutation(async (opts) => { const tripData = { ...opts.input }; - const tripClass = new Trip(); - const trip = await tripClass.update(tripData); + const trip = await editTripService(tripData, opts.ctx.executionCtx); return trip; }); } diff --git a/server/src/controllers/trip/index.ts b/server/src/controllers/trip/index.ts index 85037928d..ff3db090a 100644 --- a/server/src/controllers/trip/index.ts +++ b/server/src/controllers/trip/index.ts @@ -4,3 +4,4 @@ export * from './editTrip'; export * from './getPublicTrips'; export * from './getTrip'; export * from './getTripById'; +export * from './setTripVisibility'; diff --git a/server/src/db/schema.ts b/server/src/db/schema.ts index d1d1a39b2..c03b5e4f1 100644 --- a/server/src/db/schema.ts +++ b/server/src/db/schema.ts @@ -403,7 +403,8 @@ export const trip = sqliteTable('trip', { .primaryKey() .$defaultFn(() => createId()), name: text('name').notNull(), - description: text('description').notNull(), + description: text('description'), + // destination: text('destination').notNull(), parks: text('parks', { mode: 'json' }).$type< Array<{ id: string; name: string }> >(), @@ -412,7 +413,6 @@ export const trip = sqliteTable('trip', { >(), start_date: text('start_date').notNull(), end_date: text('end_date').notNull(), - destination: text('destination').notNull(), owner_id: text('owner_id').references(() => user.id, { onDelete: 'cascade', }), diff --git a/server/src/drizzle/methods/Geojson.ts b/server/src/drizzle/methods/Geojson.ts index def55775c..ea7e2b4d6 100644 --- a/server/src/drizzle/methods/Geojson.ts +++ b/server/src/drizzle/methods/Geojson.ts @@ -1,3 +1,4 @@ +import { eq } from 'drizzle-orm'; import { DbClient } from '../../db/client'; import { type InsertGeoJson, geojson } from '../../db/schema'; @@ -5,14 +6,29 @@ export class GeoJson { async create(geoJSON: InsertGeoJson) { try { const db = DbClient.instance; - const record = await db - .insert(geojson) - .values({ geoJSON }) - .returning() - .get(); + const record = await db.insert(geojson).values(geoJSON).returning().get(); return record; } catch (error) { throw new Error(`Failed to create geojson record: ${error.message}`); } } + + async update( + id: string, + data: Partial, + filter = eq(geojson.id, id), + ) { + try { + const geojsonValue = await DbClient.instance + .update(geojson) + .set(data) + .where(filter) + .returning() + .get(); + + return geojsonValue; + } catch (error) { + throw new Error(`Failed to geojson record: ${error.message}`); + } + } } diff --git a/server/src/drizzle/methods/trip.ts b/server/src/drizzle/methods/trip.ts index e032bf1c6..fb047acbb 100644 --- a/server/src/drizzle/methods/trip.ts +++ b/server/src/drizzle/methods/trip.ts @@ -1,6 +1,10 @@ import { eq } from 'drizzle-orm'; import { DbClient } from '../../db/client'; -import { type InsertTrip, trip as TripTable } from '../../db/schema'; +import { + type InsertTrip, + type Trip as SelectTrip, + trip as TripTable, +} from '../../db/schema'; export class Trip { async update(trip: Partial) { @@ -48,7 +52,7 @@ export class Trip { async findById(id: string) { try { - const trip = await DbClient.instance.query.trip.findFirst({ + return DbClient.instance.query.trip.findFirst({ where: eq(TripTable.id, id), with: { owner: { @@ -59,27 +63,27 @@ export class Trip { }, packs: { columns: { id: true, name: true }, - with: { - itemPacks: { - columns: { packId: true }, - with: { - item: { - columns: { - id: true, - name: true, - weight: true, - quantity: true, - unit: true, - }, - with: { - category: { - columns: { id: true, name: true }, - }, - }, - }, - }, - }, - }, + // with: { + // itemPacks: { + // columns: { packId: true }, + // with: { + // item: { + // columns: { + // id: true, + // name: true, + // weight: true, + // quantity: true, + // unit: true, + // }, + // with: { + // category: { + // columns: { id: true, name: true }, + // }, + // }, + // }, + // }, + // }, + // }, }, tripGeojsons: { columns: {}, @@ -89,7 +93,6 @@ export class Trip { }, }, }); - return trip; } catch (error) { throw new Error(`Failed to find trip by id: ${error.message}`); } diff --git a/server/src/routes/trpcRouter.ts b/server/src/routes/trpcRouter.ts index 165874968..4f3b87c60 100644 --- a/server/src/routes/trpcRouter.ts +++ b/server/src/routes/trpcRouter.ts @@ -29,6 +29,7 @@ import { getPublicTripsRoute, getTripByIdRoute, getTripsRoute, + setTripVisibilityRoute, } from '../controllers/trip'; import { addTemplateRoute, @@ -111,6 +112,7 @@ import { import { router as trpcRouter } from '../trpc'; import { getOfflineMapsRoute, saveOfflineMapRoute } from '../modules/map'; +import { setTripVisibility } from '@packrat/validations'; export const appRouter = trpcRouter({ getUserById: getUserByIdRoute(), @@ -144,6 +146,7 @@ export const appRouter = trpcRouter({ getTripById: getTripByIdRoute(), addTrip: addTripRoute(), editTrip: editTripRoute(), + setTripVisibility: setTripVisibilityRoute(), deleteTrip: deleteTripRoute(), // templates routes getPackTemplates: getPackTemplatesRoute(), diff --git a/server/src/services/trip/addTripService.ts b/server/src/services/trip/addTripService.ts index cece53d6d..c572a634d 100644 --- a/server/src/services/trip/addTripService.ts +++ b/server/src/services/trip/addTripService.ts @@ -1,3 +1,4 @@ +import * as validator from '@packrat/validations'; import { calculateTripScore } from 'src/utils/scoreTrip'; import { GeoJson } from '../../drizzle/methods/Geojson'; import { TripGeoJson } from '../../drizzle/methods/TripGeoJson'; @@ -5,35 +6,48 @@ import { Trip } from '../../drizzle/methods/trip'; import { validateGeojsonId, validateGeojsonType } from '../../utils/geojson'; import { GeojsonStorageService } from '../geojsonStorage'; import { scoreTripService } from './scoreTripService'; +import { trip } from 'src/db/schema'; export const addTripService = async ( - tripData: any, + tripData: validator.AddTripType & { ownerId: string }, executionCtx: ExecutionContext, ) => { try { - const { geoJSON, ...otherTripData } = tripData; + const geoJSON = tripData.geoJSON; const tripClass = new Trip(); + // Create Trip const newTrip = await tripClass.create({ - ...otherTripData, - trails: otherTripData.trails ? JSON.parse(otherTripData.trails) : null, - parks: otherTripData.parks ? JSON.parse(otherTripData.parks) : null, + name: tripData.name, + description: tripData.description, + start_date: tripData.start_date, + end_date: tripData.end_date, + activity: tripData.activity, + owner_id: tripData.ownerId, + is_public: false, + trails: tripData.trails ? JSON.parse(tripData.trails) : null, + parks: tripData.parks ? JSON.parse(tripData.parks) : null, }); + await scoreTripService(newTrip.id); + const geojsonClass = new GeoJson(); const tripGeoJsonClass = new TripGeoJson(); if (!geoJSON) { throw new Error("Geojson data doesn't exist"); } - // const insertedGeoJson = await geojsonClass.create(geoJSON); - // await tripGeoJsonClass.create({ - // tripId: newTrip.id, - // geojsonId: insertedGeoJson.id, - // }); + const insertedGeoJson = await geojsonClass.create({ + geoJSON, + }); + + await tripGeoJsonClass.create({ + tripId: newTrip.id, + geojsonId: insertedGeoJson.id, + }); executionCtx.waitUntil( - GeojsonStorageService.save('trip', JSON.stringify(geoJSON), newTrip.id), + GeojsonStorageService.save('trip', geoJSON, newTrip.id), ); return newTrip; diff --git a/server/src/services/trip/trip.service.ts b/server/src/services/trip/trip.service.ts index 270d8768f..d40c87337 100644 --- a/server/src/services/trip/trip.service.ts +++ b/server/src/services/trip/trip.service.ts @@ -1,4 +1,5 @@ export * from './addTripService'; +export * from './editTripService'; export * from './getPublicTripService'; export * from './getTripByIdService'; export * from './getTripsService'; From 3c179129986c17dfc938f4fc2a05376207d8552a Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Sun, 15 Dec 2024 11:09:40 +0100 Subject: [PATCH 02/11] --amend --- .../src/controllers/trip/setTripVisibility.ts | 12 +++ server/src/db/schema.ts | 2 - server/src/services/trip/editTripService.ts | 79 +++++++++++++++++++ 3 files changed, 91 insertions(+), 2 deletions(-) create mode 100644 server/src/controllers/trip/setTripVisibility.ts create mode 100644 server/src/services/trip/editTripService.ts diff --git a/server/src/controllers/trip/setTripVisibility.ts b/server/src/controllers/trip/setTripVisibility.ts new file mode 100644 index 000000000..c3f8ed013 --- /dev/null +++ b/server/src/controllers/trip/setTripVisibility.ts @@ -0,0 +1,12 @@ +import { protectedProcedure } from '../../trpc'; +import * as validator from '@packrat/validations'; +import { setTripVisibilityService } from '../../services/trip/trip.service'; + +export function setTripVisibilityRoute() { + return protectedProcedure + .input(validator.setTripVisibility) + .mutation(async (opts) => { + const trip = await setTripVisibilityService(opts.input); + return trip; + }); +} diff --git a/server/src/db/schema.ts b/server/src/db/schema.ts index c03b5e4f1..9ff007b19 100644 --- a/server/src/db/schema.ts +++ b/server/src/db/schema.ts @@ -404,7 +404,6 @@ export const trip = sqliteTable('trip', { .$defaultFn(() => createId()), name: text('name').notNull(), description: text('description'), - // destination: text('destination').notNull(), parks: text('parks', { mode: 'json' }).$type< Array<{ id: string; name: string }> >(), @@ -462,7 +461,6 @@ export const tripRelations = relations(trip, ({ one, many }) => ({ fields: [trip.pack_id], references: [pack.id], }), - // geojsons: many(tripGeojsons), tripGeojsons: many(tripGeojsons), })); diff --git a/server/src/services/trip/editTripService.ts b/server/src/services/trip/editTripService.ts new file mode 100644 index 000000000..9152190ef --- /dev/null +++ b/server/src/services/trip/editTripService.ts @@ -0,0 +1,79 @@ +import * as validator from '@packrat/validations'; +import { GeoJson } from '../../drizzle/methods/Geojson'; +import { TripGeoJson } from '../../drizzle/methods/TripGeoJson'; +import { Trip } from '../../drizzle/methods/trip'; +import { GeojsonStorageService } from '../geojsonStorage'; +import { scoreTripService } from './scoreTripService'; + +export const editTripService = async ( + tripData: validator.EditTripType, + executionCtx: ExecutionContext, +) => { + try { + const tripClass = new Trip(); + const selectedTrip = await tripClass.findById(tripData.id); + const updatedTrip = await tripClass.update({ + name: tripData.name || selectedTrip.name, + description: tripData.description || selectedTrip.description, + start_date: tripData.start_date || selectedTrip.start_date, + end_date: tripData.end_date || selectedTrip.end_date, + activity: tripData.activity || selectedTrip.activity, + trails: tripData.trails + ? JSON.parse(tripData.trails) + : selectedTrip.trails, + parks: tripData.parks ? JSON.parse(tripData.parks) : selectedTrip.parks, + }); + + await scoreTripService(selectedTrip.id); + + const serializedGeoJSON = tripData.geoJSON; + if (!serializedGeoJSON) { + return updatedTrip; + } + + const geojsonClass = new GeoJson(); + + const tripGeoJSONs = selectedTrip.tripGeojsons; + if (tripGeoJSONs.length > 0) { + await geojsonClass.update(tripGeoJSONs[0].geojson.id, { + geoJSON: serializedGeoJSON, + }); + } else { + const insertedGeoJson = await geojsonClass.create({ + geoJSON: serializedGeoJSON, + }); + + const tripGeoJsonClass = new TripGeoJson(); + await tripGeoJsonClass.create({ + tripId: selectedTrip.id, + geojsonId: insertedGeoJson.id, + }); + } + + executionCtx.waitUntil( + GeojsonStorageService.save('trip', serializedGeoJSON, selectedTrip.id), + ); + + return updatedTrip; + } catch (error) { + console.error(error); + throw new Error('Unable to add trip and GeoJSON data'); + } +}; + +export const setTripVisibilityService = async ( + tripData: validator.SetTripVisibilityType, +) => { + try { + const tripClass = new Trip(); + const selectedTrip = await tripClass.findById(tripData.id); + const updatedTrip = await tripClass.update({ + id: selectedTrip.id, + is_public: tripData.is_public === '1', + }); + return updatedTrip; + } catch (error) { + console.error(error); + throw new Error('Unable to add trip and GeoJSON data'); + } +}; From 5e9251c095b13bd1935f1a616d087fc409a68a39 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Sun, 15 Dec 2024 11:43:39 +0100 Subject: [PATCH 03/11] Add bounds as optional in the request --- .../tripRoutesValidator.ts | 26 +++++++++---------- server/src/services/trip/addTripService.ts | 5 +++- server/src/services/trip/editTripService.ts | 3 +++ 3 files changed, 20 insertions(+), 14 deletions(-) diff --git a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts index 960554e0e..c3134bba6 100644 --- a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts +++ b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts @@ -7,7 +7,7 @@ export const addTripForm = z.object({ name: z.string(), description: z.string().optional().nullable(), activity: z.enum(tripActivityValues).optional(), - is_public: z.union([z.literal('0'), z.literal('1')]), + is_public: z.union([z.literal('0'), z.literal('1')]).optional(), }); const coordinateSchema = z.lazy(() => @@ -47,31 +47,31 @@ export const getTripById = z.object({ }); export const addTripDetails = z.object({ - start_date: z.string(), - end_date: z.string(), activity: z.enum(tripActivityValues).optional(), + bounds: z.tuple([z.array(z.number()), z.array(z.number())]).optional(), + end_date: z.string(), + geoJSON: z.string(), + packId: z.string(), parks: z.string().optional(), + start_date: z.string(), trails: z.string().optional(), - packId: z.string(), - geoJSON: z.string(), }); -export const addTrip = addTripDetails.merge(addTripForm).omit({ - is_public: true, -}); +export const addTrip = addTripDetails.merge(addTripForm); export type AddTripType = z.infer; export const editTrip = z.object({ - id: z.string().min(1), - name: z.string().optional(), + activity: z.enum(tripActivityValues).optional(), + bounds: z.array(z.array(z.number())).length(2).optional(), description: z.string().optional(), - start_date: z.string().optional(), end_date: z.string().optional(), + geoJSON: z.string().optional(), + name: z.string().optional(), packId: z.string().optional(), parks: z.string().optional(), + start_date: z.string().optional(), trails: z.string().optional(), - activity: z.enum(tripActivityValues).optional(), - geoJSON: z.string().optional(), + id: z.string().min(1), }); export type EditTripType = z.infer; diff --git a/server/src/services/trip/addTripService.ts b/server/src/services/trip/addTripService.ts index c572a634d..de3d09a12 100644 --- a/server/src/services/trip/addTripService.ts +++ b/server/src/services/trip/addTripService.ts @@ -24,9 +24,12 @@ export const addTripService = async ( end_date: tripData.end_date, activity: tripData.activity, owner_id: tripData.ownerId, - is_public: false, + is_public: tripData.is_public === '0', trails: tripData.trails ? JSON.parse(tripData.trails) : null, parks: tripData.parks ? JSON.parse(tripData.parks) : null, + ...(tripData.bounds && { + bounds: [tripData.bounds[0], tripData.bounds[1]], + }), }); await scoreTripService(newTrip.id); diff --git a/server/src/services/trip/editTripService.ts b/server/src/services/trip/editTripService.ts index 9152190ef..78bdedbd9 100644 --- a/server/src/services/trip/editTripService.ts +++ b/server/src/services/trip/editTripService.ts @@ -22,6 +22,9 @@ export const editTripService = async ( ? JSON.parse(tripData.trails) : selectedTrip.trails, parks: tripData.parks ? JSON.parse(tripData.parks) : selectedTrip.parks, + ...(tripData.bounds && { + bounds: [tripData.bounds[0], tripData.bounds[1]], + }), }); await scoreTripService(selectedTrip.id); From eca561ebe8e82a95f7a690636a05c8b59865eb97 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Thu, 12 Dec 2024 23:23:08 +0100 Subject: [PATCH 04/11] Fix broken items import from csv --- packages/app/hooks/offline/useOfflineQueue.ts | 12 ++---------- packages/app/hooks/offline/useRunQueueItem.ts | 5 +++++ 2 files changed, 7 insertions(+), 10 deletions(-) diff --git a/packages/app/hooks/offline/useOfflineQueue.ts b/packages/app/hooks/offline/useOfflineQueue.ts index f508aca19..a5ea87c4b 100644 --- a/packages/app/hooks/offline/useOfflineQueue.ts +++ b/packages/app/hooks/offline/useOfflineQueue.ts @@ -8,15 +8,7 @@ type Request = { }; export const useOfflineQueue = () => { - const { - isConnected, - requests, - setRequests, - }: { - isConnected: boolean; - requests: Request[]; - setRequests: React.Dispatch>; - } = useOfflineStore(); + const { connectionStatus, requests, setRequests } = useOfflineStore(); const runQueryItem = useRunQueryItem(); const processQueue = () => { @@ -54,7 +46,7 @@ export const useOfflineQueue = () => { }; return { - isConnected, + isConnected: connectionStatus === 'connected', addOfflineRequest: handleAddOfflineRequest, processQueue, }; diff --git a/packages/app/hooks/offline/useRunQueueItem.ts b/packages/app/hooks/offline/useRunQueueItem.ts index cad66c230..f7a24dbb5 100644 --- a/packages/app/hooks/offline/useRunQueueItem.ts +++ b/packages/app/hooks/offline/useRunQueueItem.ts @@ -1,12 +1,17 @@ import { queryTrpc } from 'app/trpc'; export const useRunQueryItem = () => { + const { mutateAsync: importItemsGlobal } = + queryTrpc.importItemsGlobal.useMutation(); const { mutateAsync: addItem } = queryTrpc.addItemGlobal.useMutation(); const { mutateAsync: editItem } = queryTrpc.editItem.useMutation(); const { mutateAsync: deleteItem } = queryTrpc.deleteGlobalItem.useMutation(); const runMutation = async (key, data) => { switch (key) { + case 'importItemsGlobal': { + return importItemsGlobal(data); + } case 'addItemGlobal': { return addItem(data); } From 0e187064e0eace260d18043eb879542ec1efcc5a Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Thu, 12 Dec 2024 23:29:56 +0100 Subject: [PATCH 05/11] Fix invalid method call --- packages/app/modules/item/hooks/useImportItem.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/app/modules/item/hooks/useImportItem.ts b/packages/app/modules/item/hooks/useImportItem.ts index f673aa7d8..ea2d1b448 100644 --- a/packages/app/modules/item/hooks/useImportItem.ts +++ b/packages/app/modules/item/hooks/useImportItem.ts @@ -36,7 +36,7 @@ export const useImportItem = () => { }); } - addOfflineRequest('addItemGlobal', newItem); + addOfflineRequest('importItemsGlobal', newItem); // Optionally, handle offline case here if needed }, From 00b0f828d4df2d235946de8c791ef08d0f8f63d1 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Tue, 10 Dec 2024 01:42:10 +0100 Subject: [PATCH 06/11] Create public endPoint to import pack template --- .../DownloadMapBtn/DownloadMapBtn.web.tsx | 7 -- .../packTemplateRoutesValidator.ts | 27 +++++- server/package.json | 2 + .../packTemplates/addPackTemplate.ts | 43 ++++++++++ .../packTemplates/getPackTemplate.ts | 3 +- server/src/controllers/packTemplates/index.ts | 1 + server/src/db/schema.ts | 20 +++-- .../src/drizzle/methods/ItemPackTemplate.ts | 39 +++++++++ server/src/drizzle/methods/PackTemplate.ts | 63 ++++++++++++-- server/src/routes/trpcRouter.ts | 4 + .../itemPackTemplate/addItemPackTemplate.ts | 28 ++++++ .../itemPackTemplate.service.ts | 1 + .../packTemplate/addPackTemplateService.ts | 53 ++++++++++++ .../createPackFromTemplateService.ts | 5 +- .../packTemplate/getPackTemplateService.ts | 6 +- .../packTemplate/packTemplate.service.ts | 1 + server/src/tests/routes/packTemplate.spec.ts | 15 +++- yarn.lock | 86 ++++++++++++++++++- 18 files changed, 373 insertions(+), 31 deletions(-) delete mode 100644 packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx create mode 100644 server/src/controllers/packTemplates/addPackTemplate.ts create mode 100644 server/src/drizzle/methods/ItemPackTemplate.ts create mode 100644 server/src/services/itemPackTemplate/addItemPackTemplate.ts create mode 100644 server/src/services/itemPackTemplate/itemPackTemplate.service.ts create mode 100644 server/src/services/packTemplate/addPackTemplateService.ts diff --git a/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx b/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx deleted file mode 100644 index 6e6a23f51..000000000 --- a/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx +++ /dev/null @@ -1,7 +0,0 @@ -import React, { type FC } from 'react'; -import { View } from 'react-native'; - -export const DownloadMapBtn: FC = () => { - // Returns empty component on the on the web implementation. - return ; -}; diff --git a/packages/validations/src/validations/packTemplateRoutesValidator.ts b/packages/validations/src/validations/packTemplateRoutesValidator.ts index a98162019..3a977d1b6 100644 --- a/packages/validations/src/validations/packTemplateRoutesValidator.ts +++ b/packages/validations/src/validations/packTemplateRoutesValidator.ts @@ -13,11 +13,32 @@ export const getPackTemplates = z.object({ }), }); -export const getPackTemplate = z.object({ - id: z.string().min(1), -}); +export const getPackTemplate = z.union([ + z.object({ + id: z.string().min(1), + name: z.string().optional(), + }), + z.object({ + name: z.string(), + id: z.string().optional(), + }), +] as const); export const createPackFromTemplate = z.object({ packTemplateId: z.string().min(1), newPackName: z.string().min(1), }); + +export const addPackTemplate = z.object({ + name: z.string(), + description: z.string(), + type: z.string(), + itemPackTemplates: z.array( + z.object({ + itemId: z.string(), + quantity: z.number(), + }), + ), +}); + +export const addPackTemplates = z.array(addPackTemplate); diff --git a/server/package.json b/server/package.json index 5c36092a6..f89f15b21 100644 --- a/server/package.json +++ b/server/package.json @@ -3,6 +3,7 @@ "version": "1.1.1", "description": "", "main": "index.js", + "type": "module", "private": true, "scripts": { "build": "NODE_OPTIONS=--max_old_space_size=4096 tsc", @@ -143,6 +144,7 @@ "expect.js": "^0.3.1", "jest": "^29.7.0", "migrate-mongo": "^10.0.0", + "miniflare": "^3.20241205.0", "mocha": "^10.2.0", "openapi-generator": "^0.1.39", "prettier": "^3.2.5", diff --git a/server/src/controllers/packTemplates/addPackTemplate.ts b/server/src/controllers/packTemplates/addPackTemplate.ts new file mode 100644 index 000000000..5c28299b7 --- /dev/null +++ b/server/src/controllers/packTemplates/addPackTemplate.ts @@ -0,0 +1,43 @@ +import { addPackTemplateService } from '../../services/packTemplate/packTemplate.service'; +import * as validator from '@packrat/validations'; +import { publicProcedure } from '../../trpc'; +import { th } from '@faker-js/faker'; + +export function importPackTemplatesRoute() { + return publicProcedure + .input(validator.addPackTemplates) + .mutation(async (opts) => { + const array = opts.input; + const packTemplates = []; + for (let idx = 0; idx < array.length; idx++) { + const { name, description, type, itemPackTemplates } = array[idx]; + try { + const template = await addPackTemplateService({ + name, + description, + type, + itemPackTemplates, + }); + packTemplates.push(template); + } catch (error) { + console.log(error); + throw error; + } + } + }); +} + +export function addPackTemplateRoute() { + return publicProcedure + .input(validator.addPackTemplate) + .mutation(async (opts) => { + const { name, description, type, itemPackTemplates } = opts.input; + const packTemplate = await addPackTemplateService({ + name, + description, + type, + itemPackTemplates + }); + return packTemplate; + }); +} diff --git a/server/src/controllers/packTemplates/getPackTemplate.ts b/server/src/controllers/packTemplates/getPackTemplate.ts index 41713c24a..03942e002 100644 --- a/server/src/controllers/packTemplates/getPackTemplate.ts +++ b/server/src/controllers/packTemplates/getPackTemplate.ts @@ -6,6 +6,7 @@ export function getPackTemplateRoute() { return protectedProcedure .input(validator.getPackTemplate) .query(async ({ input }) => { - return await getPackTemplateService(input.id); + const param = input.id ? { id: input.id } : { name: input.name }; + return await getPackTemplateService(param); }); } diff --git a/server/src/controllers/packTemplates/index.ts b/server/src/controllers/packTemplates/index.ts index eca2b3dd7..31d9ba1a1 100644 --- a/server/src/controllers/packTemplates/index.ts +++ b/server/src/controllers/packTemplates/index.ts @@ -1,3 +1,4 @@ +export * from './addPackTemplate'; export * from './getPackTemplates'; export * from './getPackTemplate'; export * from './createPackFromTemplate'; diff --git a/server/src/db/schema.ts b/server/src/db/schema.ts index d1d1a39b2..47770046c 100644 --- a/server/src/db/schema.ts +++ b/server/src/db/schema.ts @@ -182,10 +182,10 @@ export const packTemplate = sqliteTable('pack_template', { }); export const packTemplateRelations = relations(packTemplate, ({ many }) => ({ - itemPackTemplates: many(itemPackTemplates), + itemPackTemplates: many(itemPackTemplate), })); -export const itemPackTemplates = sqliteTable( +export const itemPackTemplate = sqliteTable( 'item_pack_templates', { itemId: text('item_id').references(() => item.id, { onDelete: 'cascade' }), @@ -205,14 +205,14 @@ export const itemPackTemplates = sqliteTable( ); export const itemPackTemplatesRelations = relations( - itemPackTemplates, + itemPackTemplate, ({ one }) => ({ packTemplate: one(packTemplate, { - fields: [itemPackTemplates.packTemplateId], + fields: [itemPackTemplate.packTemplateId], references: [packTemplate.id], }), item: one(item, { - fields: [itemPackTemplates.itemId], + fields: [itemPackTemplate.itemId], references: [item.id], }), }), @@ -369,6 +369,7 @@ export const itemRelations = relations(item, ({ one, many }) => ({ images: many(itemImage), itemOwners: many(itemOwners), itemPacks: many(itemPacks), + itemPackTemplates: many(itemPackTemplate), })); export const template = sqliteTable('template', { @@ -611,8 +612,17 @@ export const insertTemplateSchema = createInsertSchema(template); export const selectTemplateSchema = createSelectSchema(template); export type PackTemplate = InferSelectModel; +export type InsertPackTemplate = InferInsertModel; +export const insertPackTemplateSchema = createInsertSchema(packTemplate); export const selectPackTemplateSchema = createSelectSchema(packTemplate); +export type ItemPackTemplate = InferSelectModel; +export type InsertItemPackTemplate = InferInsertModel; +export const insertItemPackTemplateSchema = + createInsertSchema(itemPackTemplate); +export const selectItemPackTemplateSchema = + createSelectSchema(itemPackTemplate); + export type Pack = InferSelectModel; export type InsertPack = InferInsertModel; export const insertPackSchema = createInsertSchema(pack); diff --git a/server/src/drizzle/methods/ItemPackTemplate.ts b/server/src/drizzle/methods/ItemPackTemplate.ts new file mode 100644 index 000000000..5dfef9e9f --- /dev/null +++ b/server/src/drizzle/methods/ItemPackTemplate.ts @@ -0,0 +1,39 @@ +import { DbClient } from '../../db/client'; +import { eq, and } from 'drizzle-orm'; +import { + type InsertItemPackTemplate, + itemPackTemplate as ItemPackTemplateTable, +} from '../../db/schema'; + +export class ItemPackTemplate { + async create(data: InsertItemPackTemplate) { + try { + const item = await DbClient.instance + .insert(ItemPackTemplateTable) + .values(data) + .returning() + .get(); + + return item; + } catch (error) { + throw new Error(`Failed to create item: ${error.message}`); + } + } + + async findByItemIdAndPackTemplateId({ + itemId, + packTemplateId, + }: { + itemId: string; + packTemplateId: string; + }) { + const itemFilter = eq(ItemPackTemplateTable.itemId, itemId); + const packFilter = eq(ItemPackTemplateTable.packTemplateId, packTemplateId); + + const filter = and(itemFilter, packFilter); + + return await DbClient.instance.query.itemPackTemplate.findFirst({ + where: filter, + }); + } +} diff --git a/server/src/drizzle/methods/PackTemplate.ts b/server/src/drizzle/methods/PackTemplate.ts index 7cf285c7c..d9ac6b475 100644 --- a/server/src/drizzle/methods/PackTemplate.ts +++ b/server/src/drizzle/methods/PackTemplate.ts @@ -1,7 +1,12 @@ import { asc, count, desc, eq, like, sql } from 'drizzle-orm'; import { DbClient } from '../../db/client'; import { convertWeight, type WeightUnit } from 'src/utils/convertWeight'; -import { packTemplate, item, itemPackTemplates } from 'src/db/schema'; +import { + packTemplate, + item, + itemPackTemplate, + InsertPackTemplate, +} from 'src/db/schema'; import { PaginationParams } from 'src/helpers/pagination'; export interface Filter { @@ -27,17 +32,17 @@ export class PackTemplate { type: packTemplate.type, description: packTemplate.description, total_weight: - sql`SUM(${item.weight} * ${itemPackTemplates.quantity})`.as( + sql`SUM(${item.weight} * ${itemPackTemplate.quantity})`.as( 'total_weight', ), - quantity: sql`SUM(${itemPackTemplates.quantity})`, + quantity: sql`SUM(${itemPackTemplate.quantity})`, }) .from(packTemplate) .leftJoin( - itemPackTemplates, - eq(packTemplate.id, itemPackTemplates.packTemplateId), + itemPackTemplate, + eq(packTemplate.id, itemPackTemplate.packTemplateId), ) - .leftJoin(item, eq(itemPackTemplates.itemId, item.id)) + .leftJoin(item, eq(itemPackTemplate.itemId, item.id)) .groupBy(packTemplate.id); if (filter?.searchQuery) { @@ -68,15 +73,54 @@ export class PackTemplate { }; } - async findPackTemplate(id: string) { + /** + * Creates a pack template. + * @param data The date used to create a pack template. + */ + async create(data: InsertPackTemplate) { + try { + const createdPackTemplate = await DbClient.instance + .insert(packTemplate) + .values(data) + .returning() + .get(); + return createdPackTemplate; + } catch (error) { + throw new Error(`Failed to create a pack: ${error.message}`); + } + } + + /** + * Finds a pack template by its ID or name. + * @param params The parameters to search for a pack template. + * @param params.id The ID of the pack template to search for. + * @param params.name The name of the pack template to search for. + * @returns + */ + async findPackTemplate( + params: { id: string; name?: undefined } | { name: string; id?: undefined }, + ) { + const { id, name } = params; + let filter; + if (id) { + filter = eq(packTemplate.id, id); + } else if (name) { + filter = eq(packTemplate.name, name); + } else { + throw new Error('Either id or name must be provided'); + } const packTemplateResult = await DbClient.instance.query.packTemplate.findFirst({ - where: eq(packTemplate.id, id), + where: filter, with: { itemPackTemplates: { with: { item: { with: { category: {} } } } }, }, }); + if (!packTemplateResult) { + return packTemplateResult; + } + const items = packTemplateResult.itemPackTemplates.map( (itemPackTemplate) => ({ ...itemPackTemplate.item, @@ -91,13 +135,14 @@ export class PackTemplate { ); return sum + weightInGrams * item.quantity; }, 0); + const quantity = items.reduce((sum, item) => sum + item.quantity, 0); delete packTemplateResult.itemPackTemplates; return { ...packTemplateResult, - items, + itemsPackTemplate: items, total_weight, quantity, }; diff --git a/server/src/routes/trpcRouter.ts b/server/src/routes/trpcRouter.ts index 165874968..b7236cc71 100644 --- a/server/src/routes/trpcRouter.ts +++ b/server/src/routes/trpcRouter.ts @@ -106,6 +106,8 @@ import { getPackTemplatesRoute, getPackTemplateRoute, createPackFromTemplateRoute, + addPackTemplateRoute, + importPackTemplatesRoute, } from '../controllers/packTemplates'; import { router as trpcRouter } from '../trpc'; @@ -149,6 +151,8 @@ export const appRouter = trpcRouter({ getPackTemplates: getPackTemplatesRoute(), getPackTemplate: getPackTemplateRoute(), createPackFromTemplate: createPackFromTemplateRoute(), + addPackTemplate: addPackTemplateRoute(), + addPackTemplates: importPackTemplatesRoute(), getTemplates: getTemplatesRoute(), getTemplateById: getTemplateByIdRoute(), addTemplate: addTemplateRoute(), diff --git a/server/src/services/itemPackTemplate/addItemPackTemplate.ts b/server/src/services/itemPackTemplate/addItemPackTemplate.ts new file mode 100644 index 000000000..fefd556c4 --- /dev/null +++ b/server/src/services/itemPackTemplate/addItemPackTemplate.ts @@ -0,0 +1,28 @@ +import { ItemPackTemplate } from '../../drizzle/methods/ItemPackTemplate'; + +/** + * Adds a item to the pack template service. + * @param {string} packTemplateId - The ID of the pack template. + * @param {string} itemId - The ID reference of the item. + * @param {number} quantity - The ID of the owner. + * @return {Promise} - A promise that resolves to the added item. + */ +export const addItemPackTemplate = async (params: { + packTemplateId: string; + itemId: string; + quantity: number; +}) => { + const { packTemplateId, itemId, quantity } = params; + const itemPackTemplate = new ItemPackTemplate(); + const item = await itemPackTemplate.findByItemIdAndPackTemplateId({ + packTemplateId, + itemId, + }); + + if (item) { + throw new Error( + 'A template item with the same item and pack references already exists', + ); + } + await itemPackTemplate.create({ itemId, packTemplateId, quantity }); +}; diff --git a/server/src/services/itemPackTemplate/itemPackTemplate.service.ts b/server/src/services/itemPackTemplate/itemPackTemplate.service.ts new file mode 100644 index 000000000..0c6fa2785 --- /dev/null +++ b/server/src/services/itemPackTemplate/itemPackTemplate.service.ts @@ -0,0 +1 @@ +export * from './addItemPackTemplate'; diff --git a/server/src/services/packTemplate/addPackTemplateService.ts b/server/src/services/packTemplate/addPackTemplateService.ts new file mode 100644 index 000000000..4ced9cb47 --- /dev/null +++ b/server/src/services/packTemplate/addPackTemplateService.ts @@ -0,0 +1,53 @@ +import { PackTemplate as PackTemplateClass } from '../../drizzle/methods/PackTemplate'; +import * as ItemPackTemplateService from '../itemPackTemplate/itemPackTemplate.service'; +import { PackTemplate } from 'src/db/schema'; + +/** + * Adds a new pack template service. + * @param {Object} packTemplateData - The data for the new pack template. + * @param {string} packTemplateData.name - The name of the pack. + * @param {string} packTemplateData.description - The description of the pack. + * @param {string} packTemplateData.type - Whether the pack is public or not. + * @return {Object} An object containing the created pack. + */ +export const addPackTemplateService = async (packTemplateData: { + name: string; + description: string; + type: string; + itemPackTemplates: Array<{ + itemId: string; + quantity: number; + }>; +}): Promise => { + const { name, description, type } = packTemplateData; + console.log({ packTemplateData }); + const packTemplateClass = new PackTemplateClass(); + let existingPack: PackTemplate | null = + await packTemplateClass.findPackTemplate({ name }); + + if (!existingPack) { + existingPack = await packTemplateClass.create({ + name, + description, + type, + }); + } else { + console.log( + 'Pack template already exists. Skipping creation and proceeding with update of items', + ); + } + + for (const itemPackTemplate of packTemplateData.itemPackTemplates) { + try { + await ItemPackTemplateService.addItemPackTemplate({ + itemId: itemPackTemplate.itemId, + quantity: itemPackTemplate.quantity, + packTemplateId: existingPack.id, + }); + } catch (error) { + console.log(error); + } + } + + return existingPack; +}; diff --git a/server/src/services/packTemplate/createPackFromTemplateService.ts b/server/src/services/packTemplate/createPackFromTemplateService.ts index a7828c5dc..df93a0312 100644 --- a/server/src/services/packTemplate/createPackFromTemplateService.ts +++ b/server/src/services/packTemplate/createPackFromTemplateService.ts @@ -11,8 +11,9 @@ export const createPackFromTemplateService = async ( ) => { const packTemplateRepository = new PackTemplateRepository(); - const packTemplate = - await packTemplateRepository.findPackTemplate(packTemplateId); + const packTemplate = await packTemplateRepository.findPackTemplate({ + id: packTemplateId, + }); // TODO - creating pack and adding items to it should ideally be transactional const createdPack = await addPackService( diff --git a/server/src/services/packTemplate/getPackTemplateService.ts b/server/src/services/packTemplate/getPackTemplateService.ts index e03ea3a9e..cb86e2846 100644 --- a/server/src/services/packTemplate/getPackTemplateService.ts +++ b/server/src/services/packTemplate/getPackTemplateService.ts @@ -1,5 +1,7 @@ import { PackTemplate } from 'src/drizzle/methods/PackTemplate'; -export async function getPackTemplateService(packTemplateId: string) { - return await new PackTemplate().findPackTemplate(packTemplateId); +export async function getPackTemplateService( + params: { id: string; name?: undefined } | { name: string; id?: undefined }, +) { + return await new PackTemplate().findPackTemplate(params); } diff --git a/server/src/services/packTemplate/packTemplate.service.ts b/server/src/services/packTemplate/packTemplate.service.ts index 40fd201b4..191893919 100644 --- a/server/src/services/packTemplate/packTemplate.service.ts +++ b/server/src/services/packTemplate/packTemplate.service.ts @@ -1,3 +1,4 @@ +export * from './addPackTemplateService'; export * from './getPackTemplatesService'; export * from './getPackTemplateService'; export * from './createPackFromTemplateService'; diff --git a/server/src/tests/routes/packTemplate.spec.ts b/server/src/tests/routes/packTemplate.spec.ts index 59d02867a..5af0be490 100644 --- a/server/src/tests/routes/packTemplate.spec.ts +++ b/server/src/tests/routes/packTemplate.spec.ts @@ -7,7 +7,7 @@ import type { trpcCaller } from '../testHelpers'; import { type PackTemplate, type Item, - itemPackTemplates as itemPackTemplatesTable, + itemPackTemplate as itemPackTemplatesTable, packTemplate as packTemplateTable, } from 'src/db/schema'; import { DbClient } from '../../db/client'; @@ -117,6 +117,19 @@ describe('Pack template routes', () => { }); }); + describe('addPackTemplate', () => { + it('should add pack template', async () => { + const packTemplate = await caller.addPackTemplate({ + name: 'test', + description: 'pack template description', + type: 'pack', + }); + expect(packTemplate).toMatchObject([ + { ...packTemplate, items: packTemplateItems }, + ]); + }); + }); + describe('getPackTemplate', () => { it('should get a pack template', async () => { const packTemplateResult = await caller.getPackTemplate({ diff --git a/yarn.lock b/yarn.lock index 9c0a2e1ca..fa86d6778 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2345,6 +2345,13 @@ __metadata: languageName: node linkType: hard +"@cloudflare/workerd-darwin-64@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "@cloudflare/workerd-darwin-64@npm:1.20241205.0" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + "@cloudflare/workerd-darwin-arm64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-darwin-arm64@npm:1.20240405.0" @@ -2359,6 +2366,13 @@ __metadata: languageName: node linkType: hard +"@cloudflare/workerd-darwin-arm64@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "@cloudflare/workerd-darwin-arm64@npm:1.20241205.0" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + "@cloudflare/workerd-linux-64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-linux-64@npm:1.20240405.0" @@ -2373,6 +2387,13 @@ __metadata: languageName: node linkType: hard +"@cloudflare/workerd-linux-64@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "@cloudflare/workerd-linux-64@npm:1.20241205.0" + conditions: os=linux & cpu=x64 + languageName: node + linkType: hard + "@cloudflare/workerd-linux-arm64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-linux-arm64@npm:1.20240405.0" @@ -2387,6 +2408,13 @@ __metadata: languageName: node linkType: hard +"@cloudflare/workerd-linux-arm64@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "@cloudflare/workerd-linux-arm64@npm:1.20241205.0" + conditions: os=linux & cpu=arm64 + languageName: node + linkType: hard + "@cloudflare/workerd-windows-64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-windows-64@npm:1.20240405.0" @@ -2401,6 +2429,13 @@ __metadata: languageName: node linkType: hard +"@cloudflare/workerd-windows-64@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "@cloudflare/workerd-windows-64@npm:1.20241205.0" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + "@cloudflare/workers-shared@npm:0.5.3": version: 0.5.3 resolution: "@cloudflare/workers-shared@npm:0.5.3" @@ -28779,6 +28814,28 @@ __metadata: languageName: node linkType: hard +"miniflare@npm:^3.20241205.0": + version: 3.20241205.0 + resolution: "miniflare@npm:3.20241205.0" + dependencies: + "@cspotcode/source-map-support": "npm:0.8.1" + acorn: "npm:^8.8.0" + acorn-walk: "npm:^8.2.0" + capnp-ts: "npm:^0.7.0" + exit-hook: "npm:^2.2.1" + glob-to-regexp: "npm:^0.4.1" + stoppable: "npm:^1.1.0" + undici: "npm:^5.28.4" + workerd: "npm:1.20241205.0" + ws: "npm:^8.18.0" + youch: "npm:^3.2.2" + zod: "npm:^3.22.3" + bin: + miniflare: bootstrap.js + checksum: 10/0efbb456890f0293af129b4850c6e420530384d674ed02e95b9195fdabba270b6016bd44cc52b92a4d9be037b0e522949d6ab463c17dfadc86014fefcba461b7 + languageName: node + linkType: hard + "minimalistic-assert@npm:^1.0.0, minimalistic-assert@npm:^1.0.1": version: 1.0.1 resolution: "minimalistic-assert@npm:1.0.1" @@ -34924,6 +34981,7 @@ __metadata: jsonwebtoken: "npm:^9.0.1" langchain: "npm:^0.0.203" migrate-mongo: "npm:^10.0.0" + miniflare: "npm:^3.20241205.0" mocha: "npm:^10.2.0" mock-aws-s3: "npm:^4.0.2" mongoose: "npm:^7.4.0" @@ -39271,6 +39329,32 @@ __metadata: languageName: node linkType: hard +"workerd@npm:1.20241205.0": + version: 1.20241205.0 + resolution: "workerd@npm:1.20241205.0" + dependencies: + "@cloudflare/workerd-darwin-64": "npm:1.20241205.0" + "@cloudflare/workerd-darwin-arm64": "npm:1.20241205.0" + "@cloudflare/workerd-linux-64": "npm:1.20241205.0" + "@cloudflare/workerd-linux-arm64": "npm:1.20241205.0" + "@cloudflare/workerd-windows-64": "npm:1.20241205.0" + dependenciesMeta: + "@cloudflare/workerd-darwin-64": + optional: true + "@cloudflare/workerd-darwin-arm64": + optional: true + "@cloudflare/workerd-linux-64": + optional: true + "@cloudflare/workerd-linux-arm64": + optional: true + "@cloudflare/workerd-windows-64": + optional: true + bin: + workerd: bin/workerd + checksum: 10/c66c79f290a247f86f28182218f4d1c54736b6a954827a447299fd7b38a4d9483b3ee10c936335a336594001c174e62d578f7a95d2044f56b56d1162eb9aa719 + languageName: node + linkType: hard + "workerpool@npm:^6.5.1": version: 6.5.1 resolution: "workerpool@npm:6.5.1" @@ -39446,7 +39530,7 @@ __metadata: languageName: node linkType: hard -"ws@npm:^8.11.0, ws@npm:^8.12.1, ws@npm:^8.13.0, ws@npm:^8.17.1": +"ws@npm:^8.11.0, ws@npm:^8.12.1, ws@npm:^8.13.0, ws@npm:^8.17.1, ws@npm:^8.18.0": version: 8.18.0 resolution: "ws@npm:8.18.0" peerDependencies: From 188531953b8526d56bd31bf21b81f2ec017700b6 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Tue, 10 Dec 2024 01:44:46 +0100 Subject: [PATCH 07/11] revert the minifar installation --- .../DownloadMapBtn/DownloadMapBtn.web.tsx | 7 ++ server/package.json | 2 - yarn.lock | 86 +------------------ 3 files changed, 8 insertions(+), 87 deletions(-) create mode 100644 packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx diff --git a/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx b/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx new file mode 100644 index 000000000..6e6a23f51 --- /dev/null +++ b/packages/app/modules/map/components/DownloadMapBtn/DownloadMapBtn.web.tsx @@ -0,0 +1,7 @@ +import React, { type FC } from 'react'; +import { View } from 'react-native'; + +export const DownloadMapBtn: FC = () => { + // Returns empty component on the on the web implementation. + return ; +}; diff --git a/server/package.json b/server/package.json index f89f15b21..5c36092a6 100644 --- a/server/package.json +++ b/server/package.json @@ -3,7 +3,6 @@ "version": "1.1.1", "description": "", "main": "index.js", - "type": "module", "private": true, "scripts": { "build": "NODE_OPTIONS=--max_old_space_size=4096 tsc", @@ -144,7 +143,6 @@ "expect.js": "^0.3.1", "jest": "^29.7.0", "migrate-mongo": "^10.0.0", - "miniflare": "^3.20241205.0", "mocha": "^10.2.0", "openapi-generator": "^0.1.39", "prettier": "^3.2.5", diff --git a/yarn.lock b/yarn.lock index fa86d6778..9c0a2e1ca 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2345,13 +2345,6 @@ __metadata: languageName: node linkType: hard -"@cloudflare/workerd-darwin-64@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "@cloudflare/workerd-darwin-64@npm:1.20241205.0" - conditions: os=darwin & cpu=x64 - languageName: node - linkType: hard - "@cloudflare/workerd-darwin-arm64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-darwin-arm64@npm:1.20240405.0" @@ -2366,13 +2359,6 @@ __metadata: languageName: node linkType: hard -"@cloudflare/workerd-darwin-arm64@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "@cloudflare/workerd-darwin-arm64@npm:1.20241205.0" - conditions: os=darwin & cpu=arm64 - languageName: node - linkType: hard - "@cloudflare/workerd-linux-64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-linux-64@npm:1.20240405.0" @@ -2387,13 +2373,6 @@ __metadata: languageName: node linkType: hard -"@cloudflare/workerd-linux-64@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "@cloudflare/workerd-linux-64@npm:1.20241205.0" - conditions: os=linux & cpu=x64 - languageName: node - linkType: hard - "@cloudflare/workerd-linux-arm64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-linux-arm64@npm:1.20240405.0" @@ -2408,13 +2387,6 @@ __metadata: languageName: node linkType: hard -"@cloudflare/workerd-linux-arm64@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "@cloudflare/workerd-linux-arm64@npm:1.20241205.0" - conditions: os=linux & cpu=arm64 - languageName: node - linkType: hard - "@cloudflare/workerd-windows-64@npm:1.20240405.0": version: 1.20240405.0 resolution: "@cloudflare/workerd-windows-64@npm:1.20240405.0" @@ -2429,13 +2401,6 @@ __metadata: languageName: node linkType: hard -"@cloudflare/workerd-windows-64@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "@cloudflare/workerd-windows-64@npm:1.20241205.0" - conditions: os=win32 & cpu=x64 - languageName: node - linkType: hard - "@cloudflare/workers-shared@npm:0.5.3": version: 0.5.3 resolution: "@cloudflare/workers-shared@npm:0.5.3" @@ -28814,28 +28779,6 @@ __metadata: languageName: node linkType: hard -"miniflare@npm:^3.20241205.0": - version: 3.20241205.0 - resolution: "miniflare@npm:3.20241205.0" - dependencies: - "@cspotcode/source-map-support": "npm:0.8.1" - acorn: "npm:^8.8.0" - acorn-walk: "npm:^8.2.0" - capnp-ts: "npm:^0.7.0" - exit-hook: "npm:^2.2.1" - glob-to-regexp: "npm:^0.4.1" - stoppable: "npm:^1.1.0" - undici: "npm:^5.28.4" - workerd: "npm:1.20241205.0" - ws: "npm:^8.18.0" - youch: "npm:^3.2.2" - zod: "npm:^3.22.3" - bin: - miniflare: bootstrap.js - checksum: 10/0efbb456890f0293af129b4850c6e420530384d674ed02e95b9195fdabba270b6016bd44cc52b92a4d9be037b0e522949d6ab463c17dfadc86014fefcba461b7 - languageName: node - linkType: hard - "minimalistic-assert@npm:^1.0.0, minimalistic-assert@npm:^1.0.1": version: 1.0.1 resolution: "minimalistic-assert@npm:1.0.1" @@ -34981,7 +34924,6 @@ __metadata: jsonwebtoken: "npm:^9.0.1" langchain: "npm:^0.0.203" migrate-mongo: "npm:^10.0.0" - miniflare: "npm:^3.20241205.0" mocha: "npm:^10.2.0" mock-aws-s3: "npm:^4.0.2" mongoose: "npm:^7.4.0" @@ -39329,32 +39271,6 @@ __metadata: languageName: node linkType: hard -"workerd@npm:1.20241205.0": - version: 1.20241205.0 - resolution: "workerd@npm:1.20241205.0" - dependencies: - "@cloudflare/workerd-darwin-64": "npm:1.20241205.0" - "@cloudflare/workerd-darwin-arm64": "npm:1.20241205.0" - "@cloudflare/workerd-linux-64": "npm:1.20241205.0" - "@cloudflare/workerd-linux-arm64": "npm:1.20241205.0" - "@cloudflare/workerd-windows-64": "npm:1.20241205.0" - dependenciesMeta: - "@cloudflare/workerd-darwin-64": - optional: true - "@cloudflare/workerd-darwin-arm64": - optional: true - "@cloudflare/workerd-linux-64": - optional: true - "@cloudflare/workerd-linux-arm64": - optional: true - "@cloudflare/workerd-windows-64": - optional: true - bin: - workerd: bin/workerd - checksum: 10/c66c79f290a247f86f28182218f4d1c54736b6a954827a447299fd7b38a4d9483b3ee10c936335a336594001c174e62d578f7a95d2044f56b56d1162eb9aa719 - languageName: node - linkType: hard - "workerpool@npm:^6.5.1": version: 6.5.1 resolution: "workerpool@npm:6.5.1" @@ -39530,7 +39446,7 @@ __metadata: languageName: node linkType: hard -"ws@npm:^8.11.0, ws@npm:^8.12.1, ws@npm:^8.13.0, ws@npm:^8.17.1, ws@npm:^8.18.0": +"ws@npm:^8.11.0, ws@npm:^8.12.1, ws@npm:^8.13.0, ws@npm:^8.17.1": version: 8.18.0 resolution: "ws@npm:8.18.0" peerDependencies: From 7d87bdd3b0a03bb6f464bb9a1dcef04348ef8e79 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Wed, 11 Dec 2024 12:20:52 +0100 Subject: [PATCH 08/11] Update the input data --- .../src/validations/itemRoutesValidator.ts | 16 +++ .../packTemplateRoutesValidator.ts | 4 +- .../src/controllers/item/importItemsGlobal.ts | 4 +- .../packTemplates/addPackTemplate.ts | 27 ++-- .../packTemplates/getPackTemplate.ts | 4 +- server/src/controllers/user/getUsers.ts | 4 +- server/src/drizzle/methods/Item.ts | 17 +++ server/src/drizzle/methods/PackTemplate.ts | 1 + .../src/services/item/addItemGlobalService.ts | 98 ++++---------- server/src/services/item/addItemService.ts | 121 +++++++++++++++++- .../packTemplate/addPackTemplateService.ts | 50 +++++--- 11 files changed, 226 insertions(+), 120 deletions(-) diff --git a/packages/validations/src/validations/itemRoutesValidator.ts b/packages/validations/src/validations/itemRoutesValidator.ts index 890257491..0e0ba3c2a 100644 --- a/packages/validations/src/validations/itemRoutesValidator.ts +++ b/packages/validations/src/validations/itemRoutesValidator.ts @@ -70,6 +70,22 @@ export const addItemGlobal = z.object({ ownerId: z.string(), }); +export const addNewItemGlobal = z.object({ + name: z.string(), + weight: z.number(), + unit: z.string(), + category: z.enum(['Food', 'Water', 'Essentials']), + sku: z.string(), + productUrl: z.string(), + description: z.string(), + productDetails: z.record( + z.string(), + z.union([z.string(), z.number(), z.boolean()]), + ), + seller: z.string(), + imageUrls: z.string(), +}); + export const importItemsGlobal = z.object({ content: z.string(), ownerId: z.string(), diff --git a/packages/validations/src/validations/packTemplateRoutesValidator.ts b/packages/validations/src/validations/packTemplateRoutesValidator.ts index 3a977d1b6..2458c9db5 100644 --- a/packages/validations/src/validations/packTemplateRoutesValidator.ts +++ b/packages/validations/src/validations/packTemplateRoutesValidator.ts @@ -1,4 +1,5 @@ import { z } from 'zod'; +import { addNewItemGlobal } from './itemRoutesValidator'; export const getPackTemplates = z.object({ filter: z @@ -33,9 +34,10 @@ export const addPackTemplate = z.object({ name: z.string(), description: z.string(), type: z.string(), + itemsOwnerId: z.string(), itemPackTemplates: z.array( z.object({ - itemId: z.string(), + item: addNewItemGlobal, quantity: z.number(), }), ), diff --git a/server/src/controllers/item/importItemsGlobal.ts b/server/src/controllers/item/importItemsGlobal.ts index e29dc511d..fc15e1de9 100644 --- a/server/src/controllers/item/importItemsGlobal.ts +++ b/server/src/controllers/item/importItemsGlobal.ts @@ -117,6 +117,8 @@ export function importItemsGlobalRoute() { let idx = 0; await addItemGlobalServiceBatch( results.data, + true, + opts.ctx.executionCtx, (item) => { const productDetailsStr = `${item.techs}` .replace(/'([^']*)'\s*:/g, '"$1":') // Replace single quotes keys with double quotes. @@ -153,8 +155,6 @@ export function importItemsGlobalRoute() { productDetails: JSON.parse(productDetailsStr), }; }, - false, - opts.ctx.executionCtx, ); return resolve('items'); } catch (error) { diff --git a/server/src/controllers/packTemplates/addPackTemplate.ts b/server/src/controllers/packTemplates/addPackTemplate.ts index 5c28299b7..310551123 100644 --- a/server/src/controllers/packTemplates/addPackTemplate.ts +++ b/server/src/controllers/packTemplates/addPackTemplate.ts @@ -1,7 +1,6 @@ import { addPackTemplateService } from '../../services/packTemplate/packTemplate.service'; import * as validator from '@packrat/validations'; import { publicProcedure } from '../../trpc'; -import { th } from '@faker-js/faker'; export function importPackTemplatesRoute() { return publicProcedure @@ -10,20 +9,19 @@ export function importPackTemplatesRoute() { const array = opts.input; const packTemplates = []; for (let idx = 0; idx < array.length; idx++) { - const { name, description, type, itemPackTemplates } = array[idx]; + const input = array[idx]; try { - const template = await addPackTemplateService({ - name, - description, - type, - itemPackTemplates, - }); - packTemplates.push(template); + const packTemplate = await addPackTemplateService( + input, + opts.ctx.executionCtx, + ); + packTemplates.push(packTemplate); } catch (error) { console.log(error); throw error; } } + return packTemplates; }); } @@ -31,13 +29,10 @@ export function addPackTemplateRoute() { return publicProcedure .input(validator.addPackTemplate) .mutation(async (opts) => { - const { name, description, type, itemPackTemplates } = opts.input; - const packTemplate = await addPackTemplateService({ - name, - description, - type, - itemPackTemplates - }); + const packTemplate = await addPackTemplateService( + opts.input, + opts.ctx.executionCtx, + ); return packTemplate; }); } diff --git a/server/src/controllers/packTemplates/getPackTemplate.ts b/server/src/controllers/packTemplates/getPackTemplate.ts index 03942e002..69eeda2e2 100644 --- a/server/src/controllers/packTemplates/getPackTemplate.ts +++ b/server/src/controllers/packTemplates/getPackTemplate.ts @@ -1,9 +1,9 @@ import * as validator from '@packrat/validations'; -import { protectedProcedure } from 'src/trpc'; +import { publicProcedure } from 'src/trpc'; import { getPackTemplateService } from 'src/services/packTemplate/packTemplate.service'; export function getPackTemplateRoute() { - return protectedProcedure + return publicProcedure .input(validator.getPackTemplate) .query(async ({ input }) => { const param = input.id ? { id: input.id } : { name: input.name }; diff --git a/server/src/controllers/user/getUsers.ts b/server/src/controllers/user/getUsers.ts index 6721b0655..ece62d01e 100644 --- a/server/src/controllers/user/getUsers.ts +++ b/server/src/controllers/user/getUsers.ts @@ -1,4 +1,4 @@ -import { protectedProcedure } from '../../trpc'; +import { protectedProcedure, publicProcedure } from '../../trpc'; import { responseHandler } from '../../helpers/responseHandler'; import { User } from '../../drizzle/methods/User'; @@ -13,7 +13,7 @@ export const getUsers = async (c) => { }; export function getUsersRoute() { - return protectedProcedure.query(async (opts) => { + return publicProcedure.query(async (opts) => { const userClass = new User(); const users = await userClass.findMany(); return users; diff --git a/server/src/drizzle/methods/Item.ts b/server/src/drizzle/methods/Item.ts index f2e5bbc18..7dee75417 100644 --- a/server/src/drizzle/methods/Item.ts +++ b/server/src/drizzle/methods/Item.ts @@ -5,6 +5,7 @@ import { ITEM_TABLE_NAME, itemPacks, item as ItemTable, + itemImage as itemImageTable, } from '../../db/schema'; import { scorePackService } from '../../services/pack/scorePackService'; import { ItemPacks } from './ItemPacks'; @@ -25,6 +26,22 @@ export class Item { } } + async insertImage(itemId: string, url: string) { + try { + const itemImage = await DbClient.instance + .insert(itemImageTable) + .values({ + itemId, + url, + }) + .run(); + + return itemImage; + } catch (error) { + throw new Error(`Failed to update item: ${error.message}`); + } + } + async createPackItem(data: InsertItem, packId: string, quantity: number) { // TODO wrap in transaction const item = await this.create(data); diff --git a/server/src/drizzle/methods/PackTemplate.ts b/server/src/drizzle/methods/PackTemplate.ts index d9ac6b475..8a7679469 100644 --- a/server/src/drizzle/methods/PackTemplate.ts +++ b/server/src/drizzle/methods/PackTemplate.ts @@ -117,6 +117,7 @@ export class PackTemplate { }, }); + if (!packTemplateResult) { return packTemplateResult; } diff --git a/server/src/services/item/addItemGlobalService.ts b/server/src/services/item/addItemGlobalService.ts index 01b8af03c..567c8390b 100644 --- a/server/src/services/item/addItemGlobalService.ts +++ b/server/src/services/item/addItemGlobalService.ts @@ -12,6 +12,7 @@ import { convertWeight, SMALLEST_WEIGHT_UNIT } from 'src/utils/convertWeight'; import { DbClient } from 'src/db/client'; import { itemImage as itemImageTable } from '../../db/schema'; import { summarizeItem } from 'src/utils/item'; +import { addNewItemService, bulkAddNewItemsService } from './addItemService'; // import { prisma } from '../../prisma'; interface AddItemGlobalServiceParams { @@ -117,87 +118,36 @@ export const addItemGlobalService = async ( */ export const addItemGlobalServiceBatch = async ( rawItems: T[], - transform: (rawItem: T) => AddItemGlobalServiceParams, continueOnError = false, executionCtx: ExecutionContext, + transform: (rawItem: T) => AddItemGlobalServiceParams, ) => { - const errors = []; + const errors: Error[] = []; - const createdItemsInOrder: Array<[itemIndex: number, item: Item]> = []; - for (let idx = 0; idx < rawItems.length; idx++) { - const item = transform(rawItems[idx]); - let category: InsertItemCategory | null; - if (!categories.includes(item.type)) { - const error = new Error( - `[${item.sku}#${item.name}]: Category must be one of: ${categories.join(', ')}`, - ); - if (continueOnError) { - errors.push(error); - continue; - } else { - throw error; - } - } + const createdItemsInOrder: Array< + Awaited> + > = []; - const itemClass = new ItemClass(); - const itemCategoryClass = new ItemCategory(); - category = - (await itemCategoryClass.findItemCategory({ name: item.type })) || null; - if (!category) { - category = await itemCategoryClass.create({ name: item.type }); + function* itemIterator() { + for (let idx = 0; idx < rawItems.length; idx++) { + const item = transform(rawItems[idx]); + yield { ...item, category: item.type, imageUrls: item.image_urls }; } + } - const newItem = await itemClass.create({ - name: item.name, - weight: convertWeight( - Number(item.weight), - item.unit as any, - SMALLEST_WEIGHT_UNIT, - ), - unit: item.unit, - categoryId: category.id, - global: true, - ownerId: item.ownerId, - sku: item.sku, - productUrl: item.productUrl, - description: item.description, - productDetails: item.productDetails, - seller: item.seller, - }); - - createdItemsInOrder.push([idx, newItem]); - - if (item.image_urls) { - const urls = item.image_urls.split(','); - - const newItemImages = []; - for (const url of urls) { - newItemImages.push({ - itemId: newItem.id, - url, - }); + await bulkAddNewItemsService({ + items: itemIterator(), + executionCtx, + onItemCreationError: (error) => { + if (!continueOnError) { + throw error; } + errors.push(error); + }, + }); - await DbClient.instance - .insert(itemImageTable) - .values(newItemImages) - .run(); - } - } - - // Format Item for vector indexes - const vectorData = []; - for (const [idx, item] of createdItemsInOrder) { - item.category = { name: transform(rawItems[idx]).type }; - vectorData.push({ - id: item.id, - content: summarizeItem(item), - namespace: ITEM_TABLE_NAME, - metadata: { - isPublic: item.global, - ownerId: item.ownerId, - }, - }); - } - executionCtx.waitUntil(VectorClient.instance.syncRecords(vectorData)); + return { + createdItemsInOrder, + errors, + }; }; diff --git a/server/src/services/item/addItemService.ts b/server/src/services/item/addItemService.ts index 6488095e9..eb9344dfd 100644 --- a/server/src/services/item/addItemService.ts +++ b/server/src/services/item/addItemService.ts @@ -1,12 +1,17 @@ -// import { prisma } from '../../prisma'; -import { Item } from '../../drizzle/methods/Item'; +import { type ExecutionContext } from 'hono'; +import * as validator from '@packrat/validations'; +import { Item as ItemClass } from '../../drizzle/methods/Item'; import { ItemCategory } from '../../drizzle/methods/itemcategory'; import { ItemOwners } from '../../drizzle/methods/ItemOwners'; import { ItemCategory as categories } from '../../utils/itemCategory'; -import { type InsertItemCategory } from '../../db/schema'; +import { + Item, + ITEM_TABLE_NAME, + type InsertItemCategory, +} from '../../db/schema'; import { VectorClient } from '../../vector/client'; import { convertWeight, SMALLEST_WEIGHT_UNIT } from 'src/utils/convertWeight'; -import { type ExecutionContext } from 'hono'; +import { summarizeItem } from 'src/utils/item'; /** * Generates a new item and adds it to a pack based on the given parameters. @@ -34,7 +39,7 @@ export const addItemService = async ( throw new Error(`Category must be one of: ${categories.join(', ')}`); } const itemCategoryClass = new ItemCategory(); - const itemClass = new Item(); + const itemClass = new ItemClass(); const itemOwnersClass = new ItemOwners(); category = (await itemCategoryClass.findItemCategory({ name: type })) || null; if (!category) { @@ -87,3 +92,109 @@ export const addItemService = async ( return item; }; + +/** + * Creates a new item. + * @param {Object} item - The data for the new item. + * @return {object} An object containing the newly created item + */ +export const addNewItemService = async ( + item: typeof validator.addNewItemGlobal._type, + ownerId: string, +): Promise => { + let category: InsertItemCategory | null; + if (!categories.includes(item.category)) { + const error = new Error( + `[${item.sku}#${item.name}]: Category must be one of: ${categories.join(', ')}`, + ); + throw error; + } + + const itemClass = new ItemClass(); + const itemCategoryClass = new ItemCategory(); + category = + (await itemCategoryClass.findItemCategory({ name: item.category })) || null; + if (!category) { + category = await itemCategoryClass.create({ name: item.category }); + } + + const newItem = await itemClass.create({ + name: item.name, + weight: convertWeight( + Number(item.weight), + item.unit as any, + SMALLEST_WEIGHT_UNIT, + ), + unit: item.unit, + categoryId: category.id, + global: true, + ownerId, + sku: item.sku, + productUrl: item.productUrl, + description: item.description, + productDetails: item.productDetails, + seller: item.seller, + }); + + if (item.imageUrls) { + const urls = item.imageUrls.split(','); + for (const url of urls) { + await itemClass.insertImage(newItem.id, url); + } + } + + return { ...newItem, category }; +}; + +/** + * Adds a list of items to the global inventory and indexes them in the vector database. + * @param {Object} bulkPram: the config parameter + * @param {Object} bulkPram.items - The list of items to add. + * @param {string} bulkPram.ownerId - The owner of all items + * @param {ExecutionContext} bulkParam.onItemCreated - A callback function to be called when an item is created. + * @param {Function} bulkParam.onItemCreated - A callback function to be called when an item is created. + */ +export const bulkAddNewItemsService = async (bulkParam: { + items: Iterable< + typeof validator.addNewItemGlobal._type & { ownerId: string } + >; + executionCtx: ExecutionContext; + onItemCreated?: ( + item: Awaited>, + bulkIndex: number, + ) => void; + onItemCreationError?: (error: Error, bulkIndex: number) => void; +}): Promise>>> => { + const { items, executionCtx, onItemCreated, onItemCreationError } = bulkParam; + const createdItems: Array>> = []; + const vectorData = []; + + let idx = -1; + for (const item of items) { + idx += 1; + try { + const createdItem = await addNewItemService(item, item.ownerId); + if (onItemCreated) { + onItemCreated(createdItem, idx); + } + createdItems.push(createdItem); + + vectorData.push({ + id: createdItem.id, + content: summarizeItem(createdItem), + namespace: ITEM_TABLE_NAME, + metadata: { + isPublic: createdItem.global, + ownerId: createdItem.ownerId, + }, + }); + } catch (error) { + if (onItemCreationError) { + onItemCreationError(error as Error, idx); + } + } + } + + executionCtx.waitUntil(VectorClient.instance.syncRecords(vectorData)); + return createdItems; +}; diff --git a/server/src/services/packTemplate/addPackTemplateService.ts b/server/src/services/packTemplate/addPackTemplateService.ts index 4ced9cb47..49549c619 100644 --- a/server/src/services/packTemplate/addPackTemplateService.ts +++ b/server/src/services/packTemplate/addPackTemplateService.ts @@ -1,6 +1,13 @@ +import type { ExecutionContext } from 'hono'; +import * as validator from '@packrat/validations'; + import { PackTemplate as PackTemplateClass } from '../../drizzle/methods/PackTemplate'; import * as ItemPackTemplateService from '../itemPackTemplate/itemPackTemplate.service'; -import { PackTemplate } from 'src/db/schema'; +import * as ItemService from '../item/item.service'; +import { ITEM_TABLE_NAME, PackTemplate } from 'src/db/schema'; +import { summarizeItem } from 'src/utils/item'; +import { VectorClient } from 'src/vector/client'; +import { i } from 'vitest/dist/reporters-QGe8gs4b.js'; /** * Adds a new pack template service. @@ -10,15 +17,10 @@ import { PackTemplate } from 'src/db/schema'; * @param {string} packTemplateData.type - Whether the pack is public or not. * @return {Object} An object containing the created pack. */ -export const addPackTemplateService = async (packTemplateData: { - name: string; - description: string; - type: string; - itemPackTemplates: Array<{ - itemId: string; - quantity: number; - }>; -}): Promise => { +export const addPackTemplateService = async ( + packTemplateData: typeof validator.addPackTemplate._type, + executionCtx: ExecutionContext, +): Promise => { const { name, description, type } = packTemplateData; console.log({ packTemplateData }); const packTemplateClass = new PackTemplateClass(); @@ -37,17 +39,29 @@ export const addPackTemplateService = async (packTemplateData: { ); } - for (const itemPackTemplate of packTemplateData.itemPackTemplates) { - try { + function* itemIterator() { + for (const itemPackTemplate of packTemplateData.itemPackTemplates) { + yield { + ...itemPackTemplate.item, + ownerId: packTemplateData.itemsOwnerId, + }; + } + } + + await ItemService.bulkAddNewItemsService({ + items: itemIterator(), + executionCtx, + onItemCreationError: (error, idx) => { + console.error(`Error creating item at ${idx}:`, error); + }, + onItemCreated: async (createdItem, idx) => { await ItemPackTemplateService.addItemPackTemplate({ - itemId: itemPackTemplate.itemId, - quantity: itemPackTemplate.quantity, + itemId: createdItem.id, + quantity: packTemplateData.itemPackTemplates[idx].quantity, packTemplateId: existingPack.id, }); - } catch (error) { - console.log(error); - } - } + }, + }); return existingPack; }; From eef47453e927d3b31cea88162dd803a92e25afbe Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Thu, 12 Dec 2024 23:50:06 +0100 Subject: [PATCH 09/11] Update and use existing item definition --- .../src/validations/itemRoutesValidator.ts | 26 ++- .../packTemplateRoutesValidator.ts | 8 +- .../src/controllers/item/importItemsGlobal.ts | 111 +++++++----- .../packTemplates/getPackTemplate.ts | 4 +- server/src/controllers/user/getUsers.ts | 4 +- server/src/db/schema.ts | 5 - .../src/services/item/addItemGlobalService.ts | 166 ++++++------------ server/src/services/item/addItemService.ts | 114 +----------- .../services/item/bulkAddGlobalItemService.ts | 145 +++++---------- .../packTemplate/addPackTemplateService.ts | 6 +- server/src/tests/routes/packTemplate.spec.ts | 4 + 11 files changed, 193 insertions(+), 400 deletions(-) diff --git a/packages/validations/src/validations/itemRoutesValidator.ts b/packages/validations/src/validations/itemRoutesValidator.ts index 0e0ba3c2a..95eb2bb6f 100644 --- a/packages/validations/src/validations/itemRoutesValidator.ts +++ b/packages/validations/src/validations/itemRoutesValidator.ts @@ -66,25 +66,19 @@ export const addItemGlobal = z.object({ name: z.string(), weight: z.number(), unit: z.string(), - type: z.string(), + type: z.enum(['Food', 'Water', 'Essentials']), ownerId: z.string(), + sku: z.string().optional(), + productUrl: z.string().optional(), + description: z.string().optional(), + productDetails: z + .record(z.string(), z.union([z.string(), z.number(), z.boolean()])) + .optional(), + seller: z.string().optional(), + image_urls: z.string().optional(), }); -export const addNewItemGlobal = z.object({ - name: z.string(), - weight: z.number(), - unit: z.string(), - category: z.enum(['Food', 'Water', 'Essentials']), - sku: z.string(), - productUrl: z.string(), - description: z.string(), - productDetails: z.record( - z.string(), - z.union([z.string(), z.number(), z.boolean()]), - ), - seller: z.string(), - imageUrls: z.string(), -}); +export type AddItemGlobalType = z.infer; export const importItemsGlobal = z.object({ content: z.string(), diff --git a/packages/validations/src/validations/packTemplateRoutesValidator.ts b/packages/validations/src/validations/packTemplateRoutesValidator.ts index 2458c9db5..de1c47541 100644 --- a/packages/validations/src/validations/packTemplateRoutesValidator.ts +++ b/packages/validations/src/validations/packTemplateRoutesValidator.ts @@ -1,5 +1,5 @@ import { z } from 'zod'; -import { addNewItemGlobal } from './itemRoutesValidator'; +import { addItemGlobal } from './itemRoutesValidator'; export const getPackTemplates = z.object({ filter: z @@ -23,7 +23,7 @@ export const getPackTemplate = z.union([ name: z.string(), id: z.string().optional(), }), -] as const); +]); export const createPackFromTemplate = z.object({ packTemplateId: z.string().min(1), @@ -37,10 +37,12 @@ export const addPackTemplate = z.object({ itemsOwnerId: z.string(), itemPackTemplates: z.array( z.object({ - item: addNewItemGlobal, + item: addItemGlobal.omit({ ownerId: true }), quantity: z.number(), }), ), }); +export type AddPackTemplateType = z.infer; + export const addPackTemplates = z.array(addPackTemplate); diff --git a/server/src/controllers/item/importItemsGlobal.ts b/server/src/controllers/item/importItemsGlobal.ts index fc15e1de9..50014a0e0 100644 --- a/server/src/controllers/item/importItemsGlobal.ts +++ b/server/src/controllers/item/importItemsGlobal.ts @@ -1,7 +1,7 @@ import { type Context } from 'hono'; import { addItemGlobalService, - addItemGlobalServiceBatch, + bulkAddItemsGlobalService, } from '../../services/item/item.service'; import { protectedProcedure } from '../../trpc'; import * as validator from '@packrat/validations'; @@ -72,6 +72,61 @@ export const importItemsGlobal = async (c: Context) => { } }; +/** + * Converts a list of raw CSV items into an iterable of validated items. + * @param {Array>} csvRawItems - The raw CSV items. + * @param {string} ownerId - The ID of the owner. + * @returns {Iterable} An iterable that yields the validated items. + */ +function* sanitizeItemsIterator( + csvRawItems: Array>, + ownerId: string, +): Generator { + for (let idx = 0; idx < csvRawItems.length; idx++) { + const item = csvRawItems[idx]; + + const productDetailsStr = `${item.techs}` + .replace(/'([^']*)'\s*:/g, '"$1":') // Replace single quotes keys with double quotes. + .replace(/:\s*'([^']*)'/g, ': "$1"') // Replace single quotes values with double quotes. + .replace(/\\x([0-9A-Fa-f]{2})/g, (match, hex) => { + // Replace hex escape sequences with UTF-8 characters + const codePoint = parseInt(hex, 16); + return String.fromCharCode(codePoint); + }); + + console.log(`${idx} / ${csvRawItems.length}`); + let parsedProductDetails: + | validator.AddItemGlobalType['productDetails'] + | null = null; + try { + parsedProductDetails = JSON.parse(productDetailsStr); + } catch (e) { + console.log( + `${productDetailsStr}\nFailed to parse product details for item ${item.Name}: ${e.message}`, + ); + throw e; + } + + const validatedItem: validator.AddItemGlobalType = { + name: String(item.Name), + weight: Number(item.Weight), + unit: String(item.Unit), + type: String(item.Category) as ItemCategoryEnum, + ownerId, + image_urls: item.image_urls && String(item.image_urls), + sku: item.sku && String(item.sku), + productUrl: item.product_url && String(item.product_url), + description: item.description && String(item.description), + seller: item.seller && String(item.seller), + }; + + if (parsedProductDetails) { + validatedItem.productDetails = parsedProductDetails; + } + + yield validatedItem; + } +} export function importItemsGlobalRoute() { const expectedHeaders = [ 'Name', @@ -114,49 +169,23 @@ export function importItemsGlobalRoute() { results.data.pop(); } - let idx = 0; - await addItemGlobalServiceBatch( - results.data, - true, + const errors: Error[] = []; + const createdItems = await bulkAddItemsGlobalService( + sanitizeItemsIterator(results.data, ownerId), opts.ctx.executionCtx, - (item) => { - const productDetailsStr = `${item.techs}` - .replace(/'([^']*)'\s*:/g, '"$1":') // Replace single quotes keys with double quotes. - .replace(/:\s*'([^']*)'/g, ': "$1"') // Replace single quotes values with double quotes. - .replace(/\\x([0-9A-Fa-f]{2})/g, (match, hex) => { - // Replace hex escape sequences with UTF-8 characters - const codePoint = parseInt(hex, 16); - return String.fromCharCode(codePoint); - }); - - idx++; - console.log(`${idx} / ${results.data.length}`); - try { - const parsedProductDetails = JSON.parse(productDetailsStr); - } catch (e) { - console.log( - `${productDetailsStr}\nFailed to parse product details for item ${item.Name}: ${e.message}`, - ); - throw e; - } - - return { - name: String(item.Name), - weight: Number(item.Weight), - unit: String(item.Unit), - type: String(item.Category) as ItemCategoryEnum, - ownerId, - executionCtx: opts.ctx.executionCtx, - image_urls: item.image_urls && String(item.image_urls), - sku: item.sku && String(item.sku), - productUrl: item.product_url && String(item.product_url), - description: item.description && String(item.description), - seller: item.seller && String(item.seller), - productDetails: JSON.parse(productDetailsStr), - }; + { + onItemCreationError: (error) => { + errors.push(error); + }, }, ); - return resolve('items'); + + return resolve({ + status: 'success', + items: createdItems, + errorsCount: errors.length, + errors, + }); } catch (error) { console.error(error); return reject(new Error(`Failed to add items: ${error.message}`)); diff --git a/server/src/controllers/packTemplates/getPackTemplate.ts b/server/src/controllers/packTemplates/getPackTemplate.ts index 69eeda2e2..03942e002 100644 --- a/server/src/controllers/packTemplates/getPackTemplate.ts +++ b/server/src/controllers/packTemplates/getPackTemplate.ts @@ -1,9 +1,9 @@ import * as validator from '@packrat/validations'; -import { publicProcedure } from 'src/trpc'; +import { protectedProcedure } from 'src/trpc'; import { getPackTemplateService } from 'src/services/packTemplate/packTemplate.service'; export function getPackTemplateRoute() { - return publicProcedure + return protectedProcedure .input(validator.getPackTemplate) .query(async ({ input }) => { const param = input.id ? { id: input.id } : { name: input.name }; diff --git a/server/src/controllers/user/getUsers.ts b/server/src/controllers/user/getUsers.ts index ece62d01e..6721b0655 100644 --- a/server/src/controllers/user/getUsers.ts +++ b/server/src/controllers/user/getUsers.ts @@ -1,4 +1,4 @@ -import { protectedProcedure, publicProcedure } from '../../trpc'; +import { protectedProcedure } from '../../trpc'; import { responseHandler } from '../../helpers/responseHandler'; import { User } from '../../drizzle/methods/User'; @@ -13,7 +13,7 @@ export const getUsers = async (c) => { }; export function getUsersRoute() { - return publicProcedure.query(async (opts) => { + return protectedProcedure.query(async (opts) => { const userClass = new User(); const users = await userClass.findMany(); return users; diff --git a/server/src/db/schema.ts b/server/src/db/schema.ts index 47770046c..db8e3a331 100644 --- a/server/src/db/schema.ts +++ b/server/src/db/schema.ts @@ -613,15 +613,10 @@ export const selectTemplateSchema = createSelectSchema(template); export type PackTemplate = InferSelectModel; export type InsertPackTemplate = InferInsertModel; -export const insertPackTemplateSchema = createInsertSchema(packTemplate); export const selectPackTemplateSchema = createSelectSchema(packTemplate); export type ItemPackTemplate = InferSelectModel; export type InsertItemPackTemplate = InferInsertModel; -export const insertItemPackTemplateSchema = - createInsertSchema(itemPackTemplate); -export const selectItemPackTemplateSchema = - createSelectSchema(itemPackTemplate); export type Pack = InferSelectModel; export type InsertPack = InferInsertModel; diff --git a/server/src/services/item/addItemGlobalService.ts b/server/src/services/item/addItemGlobalService.ts index 567c8390b..b9371af67 100644 --- a/server/src/services/item/addItemGlobalService.ts +++ b/server/src/services/item/addItemGlobalService.ts @@ -1,4 +1,5 @@ import { type ExecutionContext } from 'hono'; +import * as validator from '@packrat/validations'; import { Item, ITEM_TABLE_NAME, @@ -9,145 +10,76 @@ import { ItemCategory } from '../../drizzle/methods/itemcategory'; import { ItemCategory as categories } from '../../utils/itemCategory'; import { VectorClient } from '../../vector/client'; import { convertWeight, SMALLEST_WEIGHT_UNIT } from 'src/utils/convertWeight'; -import { DbClient } from 'src/db/client'; -import { itemImage as itemImageTable } from '../../db/schema'; import { summarizeItem } from 'src/utils/item'; -import { addNewItemService, bulkAddNewItemsService } from './addItemService'; -// import { prisma } from '../../prisma'; -interface AddItemGlobalServiceParams { - /** The name of the item. */ - name: string; - /** The description of the item. */ - description?: string; - /** The weight of the item. */ - weight: number; - /** The unit of measurement for the item. */ - unit: string; - /** The category of the item. */ - type: (typeof categories)[number]; - /** The ID of the owner of the item. */ - ownerId: string; - /** The URLs of the images of the item. */ - image_urls?: string; - /** The SKU of the item. */ - sku?: string; - /** The URL of the product of the item. */ - productUrl?: string; - /** The product details of the item. */ - productDetails?: Record; - /** The seller of the item. */ - seller?: string; -} +type ItemWithCategory = Item & { category?: InsertItemCategory }; /** * Adds an item to the global service. + * @param {Object} item - The data for the new item. + * @param {ExecutionContext} executionCtx - Then vector database execution context * @return {Promise} The newly created item. */ export const addItemGlobalService = async ( - { - name, - weight, - unit, - type, - ownerId, - image_urls, - sku, - productUrl, - description, - productDetails, - seller, - }: AddItemGlobalServiceParams, - executionCtx: ExecutionContext, -) => { + item: validator.AddItemGlobalType, + executionCtx?: ExecutionContext, +): Promise => { let category: InsertItemCategory | null; - if (!categories.includes(type)) { - throw new Error(`Category must be one of: ${categories.join(', ')}`); + if (!categories.includes(item.type)) { + const error = new Error( + `[${item.sku}#${item.name}]: Category must be one of: ${categories.join(', ')}`, + ); + throw error; } + const itemClass = new ItemClass(); const itemCategoryClass = new ItemCategory(); - category = (await itemCategoryClass.findItemCategory({ name: type })) || null; + category = + (await itemCategoryClass.findItemCategory({ name: item.type })) || null; if (!category) { - category = await itemCategoryClass.create({ name: type }); + category = await itemCategoryClass.create({ name: item.type }); } - const newItem = await itemClass.create({ - name, - weight: convertWeight(Number(weight), unit as any, SMALLEST_WEIGHT_UNIT), - unit, + + const newItem = (await itemClass.create({ + name: item.name, + weight: convertWeight( + Number(item.weight), + item.unit as any, + SMALLEST_WEIGHT_UNIT, + ), + unit: item.unit, categoryId: category.id, global: true, - ownerId, - sku, - productUrl, - description, - productDetails, - seller, - }); + ownerId: item.ownerId, + sku: item.sku, + productUrl: item.productUrl, + description: item.description, + productDetails: item.productDetails, + seller: item.seller, + })) as ItemWithCategory; - if (image_urls) { - const urls = image_urls.split(','); + if (item.image_urls) { + const urls = item.image_urls.split(','); for (const url of urls) { - const newItemImage = { - itemId: newItem.id, - url, - }; - await DbClient.instance.insert(itemImageTable).values(newItemImage).run(); + await itemClass.insertImage(newItem.id, url); } } - executionCtx.waitUntil( - VectorClient.instance.syncRecord({ - id: newItem.id, - content: `product_name: ${name}, category: ${type}, description: ${description}, productDetails: ${JSON.stringify( - productDetails, - )}`, - namespace: ITEM_TABLE_NAME, - metadata: { - isPublic: newItem.global, - ownerId, - }, - }), - ); + newItem.category = category; - return newItem; -}; - -/** - * Adds list of items to the global service. - * @return {Promise} The newly created item. - */ -export const addItemGlobalServiceBatch = async ( - rawItems: T[], - continueOnError = false, - executionCtx: ExecutionContext, - transform: (rawItem: T) => AddItemGlobalServiceParams, -) => { - const errors: Error[] = []; - - const createdItemsInOrder: Array< - Awaited> - > = []; - - function* itemIterator() { - for (let idx = 0; idx < rawItems.length; idx++) { - const item = transform(rawItems[idx]); - yield { ...item, category: item.type, imageUrls: item.image_urls }; - } + if (executionCtx) { + executionCtx.waitUntil( + VectorClient.instance.syncRecord({ + id: newItem.id, + content: summarizeItem(newItem), + namespace: ITEM_TABLE_NAME, + metadata: { + isPublic: newItem.global, + ownerId: newItem.ownerId, + }, + }), + ); } - await bulkAddNewItemsService({ - items: itemIterator(), - executionCtx, - onItemCreationError: (error) => { - if (!continueOnError) { - throw error; - } - errors.push(error); - }, - }); - - return { - createdItemsInOrder, - errors, - }; + return newItem; }; diff --git a/server/src/services/item/addItemService.ts b/server/src/services/item/addItemService.ts index eb9344dfd..59f0652d2 100644 --- a/server/src/services/item/addItemService.ts +++ b/server/src/services/item/addItemService.ts @@ -1,17 +1,11 @@ import { type ExecutionContext } from 'hono'; -import * as validator from '@packrat/validations'; import { Item as ItemClass } from '../../drizzle/methods/Item'; import { ItemCategory } from '../../drizzle/methods/itemcategory'; import { ItemOwners } from '../../drizzle/methods/ItemOwners'; import { ItemCategory as categories } from '../../utils/itemCategory'; -import { - Item, - ITEM_TABLE_NAME, - type InsertItemCategory, -} from '../../db/schema'; +import { type InsertItemCategory } from '../../db/schema'; import { VectorClient } from '../../vector/client'; import { convertWeight, SMALLEST_WEIGHT_UNIT } from 'src/utils/convertWeight'; -import { summarizeItem } from 'src/utils/item'; /** * Generates a new item and adds it to a pack based on the given parameters. @@ -92,109 +86,3 @@ export const addItemService = async ( return item; }; - -/** - * Creates a new item. - * @param {Object} item - The data for the new item. - * @return {object} An object containing the newly created item - */ -export const addNewItemService = async ( - item: typeof validator.addNewItemGlobal._type, - ownerId: string, -): Promise => { - let category: InsertItemCategory | null; - if (!categories.includes(item.category)) { - const error = new Error( - `[${item.sku}#${item.name}]: Category must be one of: ${categories.join(', ')}`, - ); - throw error; - } - - const itemClass = new ItemClass(); - const itemCategoryClass = new ItemCategory(); - category = - (await itemCategoryClass.findItemCategory({ name: item.category })) || null; - if (!category) { - category = await itemCategoryClass.create({ name: item.category }); - } - - const newItem = await itemClass.create({ - name: item.name, - weight: convertWeight( - Number(item.weight), - item.unit as any, - SMALLEST_WEIGHT_UNIT, - ), - unit: item.unit, - categoryId: category.id, - global: true, - ownerId, - sku: item.sku, - productUrl: item.productUrl, - description: item.description, - productDetails: item.productDetails, - seller: item.seller, - }); - - if (item.imageUrls) { - const urls = item.imageUrls.split(','); - for (const url of urls) { - await itemClass.insertImage(newItem.id, url); - } - } - - return { ...newItem, category }; -}; - -/** - * Adds a list of items to the global inventory and indexes them in the vector database. - * @param {Object} bulkPram: the config parameter - * @param {Object} bulkPram.items - The list of items to add. - * @param {string} bulkPram.ownerId - The owner of all items - * @param {ExecutionContext} bulkParam.onItemCreated - A callback function to be called when an item is created. - * @param {Function} bulkParam.onItemCreated - A callback function to be called when an item is created. - */ -export const bulkAddNewItemsService = async (bulkParam: { - items: Iterable< - typeof validator.addNewItemGlobal._type & { ownerId: string } - >; - executionCtx: ExecutionContext; - onItemCreated?: ( - item: Awaited>, - bulkIndex: number, - ) => void; - onItemCreationError?: (error: Error, bulkIndex: number) => void; -}): Promise>>> => { - const { items, executionCtx, onItemCreated, onItemCreationError } = bulkParam; - const createdItems: Array>> = []; - const vectorData = []; - - let idx = -1; - for (const item of items) { - idx += 1; - try { - const createdItem = await addNewItemService(item, item.ownerId); - if (onItemCreated) { - onItemCreated(createdItem, idx); - } - createdItems.push(createdItem); - - vectorData.push({ - id: createdItem.id, - content: summarizeItem(createdItem), - namespace: ITEM_TABLE_NAME, - metadata: { - isPublic: createdItem.global, - ownerId: createdItem.ownerId, - }, - }); - } catch (error) { - if (onItemCreationError) { - onItemCreationError(error as Error, idx); - } - } - } - - executionCtx.waitUntil(VectorClient.instance.syncRecords(vectorData)); - return createdItems; -}; diff --git a/server/src/services/item/bulkAddGlobalItemService.ts b/server/src/services/item/bulkAddGlobalItemService.ts index 1592d5f1a..c0aa1bb49 100644 --- a/server/src/services/item/bulkAddGlobalItemService.ts +++ b/server/src/services/item/bulkAddGlobalItemService.ts @@ -1,109 +1,60 @@ import { type ExecutionContext } from 'hono'; -import { type InsertItemCategory } from '../../db/schema'; -import { ItemCategory } from '../../drizzle/methods/itemcategory'; -import { DbClient } from 'src/db/client'; -import { - item as ItemTable, - itemImage as itemImageTable, -} from '../../db/schema'; -import { convertWeight, SMALLEST_WEIGHT_UNIT } from 'src/utils/convertWeight'; -import { eq } from 'drizzle-orm'; +import * as validator from '@packrat/validations'; +import { ITEM_TABLE_NAME } from '../../db/schema'; import { VectorClient } from 'src/vector/client'; - +import { addItemGlobalService } from './addItemGlobalService'; +import { summarizeItem } from 'src/utils/item'; + +/** + * Adds a list of items to the global inventory and indexes them in the vector database. + * @param {Object} items - The list of items to add. + * @param {string} executionCtx - The execution context. + * @param {ExecutionContext} callbacks.onItemCreated - A callback function to be called when an item is created. + * @param {Function} callbacks.onItemCreationError - A callback function to be called when an item creation fails. + * @return {Promise} A promise that resolves to an array of the created items. + */ export const bulkAddItemsGlobalService = async ( - items: Array<{ - name: string; - weight: number; - unit: string; - type: 'Food' | 'Water' | 'Essentials'; - ownerId: string; - image_urls?: string; - sku?: string; - productUrl?: string; - description?: string; - productDetails?: Record; - seller?: string; - }>, + items: Iterable, executionCtx: ExecutionContext, -) => { - const categories = ['Food', 'Water', 'Essentials']; - - const itemCategoryClass = new ItemCategory(); - const insertedItems = []; - - for (const itemData of items) { - const { name, weight, unit, type, ownerId, image_urls } = itemData; - if (!categories.includes(type)) { - throw new Error(`Category must be one of: ${categories.join(', ')}`); - } - - let category: InsertItemCategory | null; - category = - (await itemCategoryClass.findItemCategory({ name: type })) || null; - if (!category) { - category = await itemCategoryClass.create({ name: type }); - } - - // Check if item with the same name already exists - const existingItem = await DbClient.instance - .select() - .from(ItemTable) - .where(eq(ItemTable.name, name)) - .get(); - - if (existingItem) { - continue; - } - - const newItem = { - name, - weight: convertWeight(Number(weight), unit as any, SMALLEST_WEIGHT_UNIT), - unit, - categoryId: category.id, - global: true, - ownerId, - sku, - productUrl, - description, - productDetails, - seller, - }; - - const item = await DbClient.instance - .insert(ItemTable) - .values(newItem) - .returning() - .get(); + callbacks?: { + onItemCreated?: ( + item: Awaited>, + bulkIndex: number, + ) => void; + onItemCreationError?: (error: Error, bulkIndex: number) => void; + }, +): Promise>>> => { + const { onItemCreated, onItemCreationError } = callbacks; + const createdItems: Array>> = + []; + const vectorData = []; + + let idx = -1; + for (const item of items) { + idx += 1; + try { + const createdItem = await addItemGlobalService(item); + if (onItemCreated) { + onItemCreated(createdItem, idx); + } + createdItems.push(createdItem); - executionCtx.waitUntil( - VectorClient.instance.syncRecord({ - id: item.id, - content: name, - namespace: 'items', + vectorData.push({ + id: createdItem.id, + content: summarizeItem(createdItem), + namespace: ITEM_TABLE_NAME, metadata: { - isPublic: item.global, - ownerId, + isPublic: createdItem.global, + ownerId: createdItem.ownerId, }, - }), - ); - - if (image_urls) { - const urls = image_urls.split(','); - for (const url of urls) { - const newItemImage = { - itemId: item.id, - url, - }; - await DbClient.instance - .insert(itemImageTable) - .values(newItemImage) - .run(); + }); + } catch (error) { + if (onItemCreationError) { + onItemCreationError(error as Error, idx); } - console.log('Added image urls for item:', item.id); } - - insertedItems.push(item); } - return insertedItems; + executionCtx.waitUntil(VectorClient.instance.syncRecords(vectorData)); + return createdItems; }; diff --git a/server/src/services/packTemplate/addPackTemplateService.ts b/server/src/services/packTemplate/addPackTemplateService.ts index 49549c619..637a8083c 100644 --- a/server/src/services/packTemplate/addPackTemplateService.ts +++ b/server/src/services/packTemplate/addPackTemplateService.ts @@ -18,7 +18,7 @@ import { i } from 'vitest/dist/reporters-QGe8gs4b.js'; * @return {Object} An object containing the created pack. */ export const addPackTemplateService = async ( - packTemplateData: typeof validator.addPackTemplate._type, + packTemplateData: validator.AddPackTemplateType, executionCtx: ExecutionContext, ): Promise => { const { name, description, type } = packTemplateData; @@ -48,9 +48,7 @@ export const addPackTemplateService = async ( } } - await ItemService.bulkAddNewItemsService({ - items: itemIterator(), - executionCtx, + await ItemService.bulkAddItemsGlobalService(itemIterator(), executionCtx, { onItemCreationError: (error, idx) => { console.error(`Error creating item at ${idx}:`, error); }, diff --git a/server/src/tests/routes/packTemplate.spec.ts b/server/src/tests/routes/packTemplate.spec.ts index 5af0be490..d914ee981 100644 --- a/server/src/tests/routes/packTemplate.spec.ts +++ b/server/src/tests/routes/packTemplate.spec.ts @@ -123,6 +123,10 @@ describe('Pack template routes', () => { name: 'test', description: 'pack template description', type: 'pack', + itemPackTemplates: packTemplateItems.map((item) => ({ + itemId: item.id, + quantity: 1, + })), }); expect(packTemplate).toMatchObject([ { ...packTemplate, items: packTemplateItems }, From 45c8a9b9317e190725a85e866ac15bb4bca5eaf0 Mon Sep 17 00:00:00 2001 From: Taron Date: Sun, 15 Dec 2024 21:43:02 +0400 Subject: [PATCH 10/11] enhance settings screen --- packages/app/components/Fab/FabWeb.tsx | 4 - .../app/components/LayoutCard/LayoutCard.tsx | 1 - .../components/navigation/Navbar/Navbar.tsx | 3 +- .../hooks/common/usePreviewResourceState.ts | 1 + .../app/hooks/singletrips/useUserTrips.ts | 2 - .../components/HeroSection/HeroSection.tsx | 5 +- .../SectionHeader/SectionHeader.tsx | 10 +- .../DashboardScreen/DashboardScreen.tsx | 13 +- .../feed/hooks/useFetchUserFavorites.ts | 8 +- .../item/screens/ItemDetailsScreen.tsx | 11 +- .../app/modules/pack/hooks/useUserPacks.ts | 45 +-- .../app/modules/trip/hooks/useUserTrips.ts | 41 +- .../user/components/UserDetailList.tsx | 32 +- packages/app/modules/user/hooks/useProfile.ts | 4 + .../modules/user/widgets/ProfileContainer.tsx | 357 +++++++----------- .../user/widgets/UserDataContainer.tsx | 134 +++---- packages/ui/src/dialog/BaseDialog.tsx | 4 +- .../controllers/favorite/getUserFavorites.ts | 9 +- .../favorite/getUserFavoritesService.ts | 5 +- 19 files changed, 277 insertions(+), 412 deletions(-) diff --git a/packages/app/components/Fab/FabWeb.tsx b/packages/app/components/Fab/FabWeb.tsx index 360fc3d03..66766fade 100644 --- a/packages/app/components/Fab/FabWeb.tsx +++ b/packages/app/components/Fab/FabWeb.tsx @@ -46,12 +46,8 @@ const loadStyles = (theme) => { alignSelf: 'stretch', zIndex: 2, }, - fabIcon: { - color: currentTheme.colors.tertiaryBlue, - }, fabText: { fontSize: 17, - color: currentTheme.colors.tertiaryBlue, fontWeight: 'bold', }, }; diff --git a/packages/app/components/LayoutCard/LayoutCard.tsx b/packages/app/components/LayoutCard/LayoutCard.tsx index 48227094d..f8b660c9c 100644 --- a/packages/app/components/LayoutCard/LayoutCard.tsx +++ b/packages/app/components/LayoutCard/LayoutCard.tsx @@ -21,7 +21,6 @@ export const LayoutCard: FC = ({ style={[ { backgroundColor: currentTheme.colors.card, - boxShadow: '0px 4px 8px rgba(0,0,0,0.1)', borderWidth: 1, borderColor: isDark ? '#424242' : '#e5e7eb', padding: 24, diff --git a/packages/app/components/navigation/Navbar/Navbar.tsx b/packages/app/components/navigation/Navbar/Navbar.tsx index fdaeabd87..aadc8f869 100644 --- a/packages/app/components/navigation/Navbar/Navbar.tsx +++ b/packages/app/components/navigation/Navbar/Navbar.tsx @@ -122,7 +122,8 @@ const loadStyles = (currentTheme, isScrolled, screenWidth) => { }, container: { width: '100vw', - maxWidth: '100%', // Ensure container does not exceed the viewport width + maxWidth: 1440, + margin: 'auto', flex: 1, // Ensure container can grow to fit content backgroundColor, borderRadius: NavbarStyles.floatingRadius, diff --git a/packages/app/hooks/common/usePreviewResourceState.ts b/packages/app/hooks/common/usePreviewResourceState.ts index c820d6b46..dc457e634 100644 --- a/packages/app/hooks/common/usePreviewResourceState.ts +++ b/packages/app/hooks/common/usePreviewResourceState.ts @@ -10,6 +10,7 @@ interface PreviewResourceState { export interface PreviewResourceStateWithData extends PreviewResourceState, Partial { + resourceName: string; isPreviewLoading: boolean; previewData: any; isAllQueryLoading: boolean; diff --git a/packages/app/hooks/singletrips/useUserTrips.ts b/packages/app/hooks/singletrips/useUserTrips.ts index a074ce322..b0fc33f16 100644 --- a/packages/app/hooks/singletrips/useUserTrips.ts +++ b/packages/app/hooks/singletrips/useUserTrips.ts @@ -14,8 +14,6 @@ export const useUserTrips = (ownerId: string | undefined) => { }, ); - console.log('data', data); - // Extract trips or set an empty array if data is undefined. // const trips = data?.trips || []; diff --git a/packages/app/modules/dashboard/components/HeroSection/HeroSection.tsx b/packages/app/modules/dashboard/components/HeroSection/HeroSection.tsx index 64a6db02e..419d16ffd 100644 --- a/packages/app/modules/dashboard/components/HeroSection/HeroSection.tsx +++ b/packages/app/modules/dashboard/components/HeroSection/HeroSection.tsx @@ -116,19 +116,20 @@ const loadStyles = (theme: any) => { }, stack: { width: '100%', - alignItems: 'flex-start', // Align items to the start (left) + alignItems: 'flex-start', + paddingTop: Platform.OS === 'web' ? 24 : 16, }, title: { fontSize: 24, fontWeight: 'bold', marginBottom: 20, - color: currentTheme.colors.tertiaryBlue, textAlign: 'left', // Align text to the left width: '100%', // Ensure it takes full width for left alignment }, searchContainer: { width: '100%', alignItems: 'flex-start', + padding: 2, }, searchBar: { width: '100%', diff --git a/packages/app/modules/dashboard/components/SectionHeader/SectionHeader.tsx b/packages/app/modules/dashboard/components/SectionHeader/SectionHeader.tsx index 039f133dc..c2b8fb9ef 100644 --- a/packages/app/modules/dashboard/components/SectionHeader/SectionHeader.tsx +++ b/packages/app/modules/dashboard/components/SectionHeader/SectionHeader.tsx @@ -8,14 +8,9 @@ import useCustomStyles from 'app/hooks/useCustomStyles'; interface SectionHeaderProps { iconName: keyof typeof Ionicons.glyphMap; text: string; - textStyle: any; } -export const SectionHeader = ({ - iconName, - text, - textStyle, -}: SectionHeaderProps) => { +export const SectionHeader = ({ iconName, text }: SectionHeaderProps) => { const styles = useCustomStyles(loadStyles); const navigate = useNavigate(); @@ -40,8 +35,7 @@ const loadStyles = (theme: any) => { width: '100%', }, text: { - color: currentTheme.colors.tertiaryBlue, - fontSize: 25, + fontSize: 24, fontWeight: 'bold', }, icon: { diff --git a/packages/app/modules/dashboard/screens/DashboardScreen/DashboardScreen.tsx b/packages/app/modules/dashboard/screens/DashboardScreen/DashboardScreen.tsx index 739c37ca2..6772a411f 100644 --- a/packages/app/modules/dashboard/screens/DashboardScreen/DashboardScreen.tsx +++ b/packages/app/modules/dashboard/screens/DashboardScreen/DashboardScreen.tsx @@ -4,7 +4,6 @@ import { RStack, RScrollView } from '@packrat/ui'; import { HeroSection, Section, SectionHeader } from '../../components'; import useCustomStyles from 'app/hooks/useCustomStyles'; import Layout from 'app/components/layout/Layout'; -import { SCREEN_WIDTH } from 'app/constants/breakpoint'; import { useScreenWidth } from 'app/hooks/common'; import FAB from 'app/components/Fab/Fab'; import { FeedPreview } from 'app/modules/feed'; @@ -32,7 +31,7 @@ export const DashboardScreen = () => { flexDirection: 'row', alignItems: 'center', alignSelf: 'flex-end', - paddingRight: 16, + marginTop: 16, gap: 8, }} > @@ -49,11 +48,7 @@ export const DashboardScreen = () => {
- +
@@ -85,6 +80,7 @@ const loadStyles = (theme) => { }, gridContainer: { flexDirection: 'row', + marginTop: 32, flexWrap: 'wrap', justifyContent: 'space-between', }, @@ -92,8 +88,5 @@ const loadStyles = (theme) => { flexBasis: '100%', backgroundColor: currentTheme.colors.background, }, - sectionHeaderText: { - color: currentTheme.colors.text, - }, }; }; diff --git a/packages/app/modules/feed/hooks/useFetchUserFavorites.ts b/packages/app/modules/feed/hooks/useFetchUserFavorites.ts index f9c1e73d7..121b915fc 100644 --- a/packages/app/modules/feed/hooks/useFetchUserFavorites.ts +++ b/packages/app/modules/feed/hooks/useFetchUserFavorites.ts @@ -12,7 +12,7 @@ import { export const useFetchUserFavorites = ( userId: string, - { queryEnabled = true, isPreview = false, searchTerm = '' } = {}, + { queryEnabled = true, isPreview = false, searchTerm = '', isPublic } = {}, ) => { const enabled = !!userId && queryEnabled; const [pagination, setPagination] = useState( @@ -21,7 +21,7 @@ export const useFetchUserFavorites = ( const { data, error, isLoading, refetch } = queryTrpc.getUserFavorites.useQuery( - { userId, pagination, isPreview, searchTerm }, + { userId, pagination, isPreview, searchTerm, isPublic }, { enabled, refetchOnWindowFocus: false, @@ -64,11 +64,12 @@ interface FetchUserFavoritesReturn extends PreviewResourceStateWithData { export const useFetchUserFavoritesWithPreview = ( userId: string, searchTerm: string, + isPublic?: boolean, ): FetchUserFavoritesReturn => { const { isAllQueryEnabled, ...previewResourceState } = usePreviewResourceState(); const { data: previewData, isLoading: isPreviewLoading } = - useFetchUserFavorites(userId, { isPreview: true }); + useFetchUserFavorites(userId, { isPreview: true, isPublic }); const { data: allQueryData, @@ -87,6 +88,7 @@ export const useFetchUserFavoritesWithPreview = ( return { ...previewResourceState, + resourceName: 'Favorites', isAllQueryEnabled, previewData, isPreviewLoading, diff --git a/packages/app/modules/item/screens/ItemDetailsScreen.tsx b/packages/app/modules/item/screens/ItemDetailsScreen.tsx index 9c539f255..0280517d7 100644 --- a/packages/app/modules/item/screens/ItemDetailsScreen.tsx +++ b/packages/app/modules/item/screens/ItemDetailsScreen.tsx @@ -9,6 +9,7 @@ import useCustomStyles from 'app/hooks/useCustomStyles'; import { TouchableOpacity, Platform } from 'react-native'; import { useRouter } from 'app/hooks/router'; import RLink from '@packrat/ui/src/RLink'; +import Layout from 'app/components/layout/Layout'; export function ItemDetailsScreen() { const { currentTheme } = useTheme(); @@ -22,13 +23,7 @@ export function ItemDetailsScreen() { } return ( - + {Platform.OS === 'web' ? ( @@ -80,7 +75,7 @@ export function ItemDetailsScreen() { - + ); } diff --git a/packages/app/modules/pack/hooks/useUserPacks.ts b/packages/app/modules/pack/hooks/useUserPacks.ts index e6babf6ee..5ed78a401 100644 --- a/packages/app/modules/pack/hooks/useUserPacks.ts +++ b/packages/app/modules/pack/hooks/useUserPacks.ts @@ -8,7 +8,7 @@ import { type PreviewResourceStateWithData, usePreviewResourceState, } from 'app/hooks/common'; -import { useEffect, useState } from 'react'; +import { useEffect, useMemo, useState } from 'react'; interface QueryOptions { isPublic?: boolean; @@ -29,25 +29,26 @@ export const useUserPacks = ( ); const utils = queryTrpc.useContext(); const enabled = queryEnabled && !!ownerId; + const queryParams = useMemo( + () => ({ + ownerId, + isPublic, + queryBy: queryString, + pagination, + searchTerm, + isPreview, + itemId, + }), + [isPublic, isPreview, searchTerm, pagination, ownerId, itemId, queryString], + ); const { data, error, isLoading, refetch } = - queryTrpc.getUserPacksFeed.useQuery( - { - ownerId, - isPublic, - queryBy: queryString, - pagination, - searchTerm, - isPreview, - itemId, - }, - { - enabled, - refetchOnWindowFocus: false, - staleTime: 5 * 60, - cacheTime: 60 * 60 * 24, - networkMode: 'offlineFirst', - }, - ); + queryTrpc.getUserPacksFeed.useQuery(queryParams, { + enabled, + refetchOnWindowFocus: false, + staleTime: 5 * 60, + cacheTime: 60 * 60 * 24, + networkMode: 'offlineFirst', + }); utils.getPacks.setData({ ownerId: ownerId || '', queryBy: queryString, @@ -92,12 +93,13 @@ interface FetchUserPacksPreviewReturn extends PreviewResourceStateWithData { export const useUserPacksWithPreview = ( userId: string, searchTerm: string, + isPublic?: boolean, ): FetchUserPacksPreviewReturn => { const { isAllQueryEnabled, ...previewResourceState } = usePreviewResourceState(); const { data: previewData, isLoading: isPreviewLoading } = useUserPacks( userId, - { isPublic: true, isPreview: true }, + { isPreview: true, isPublic }, 'Most Recent', true, ); @@ -112,10 +114,11 @@ export const useUserPacksWithPreview = ( hasNextPage, currentPage, totalPages, - } = useUserPacks(userId, { isPublic: true, searchTerm }, 'Most Recent', true); + } = useUserPacks(userId, { isPublic, searchTerm }, 'Most Recent', true); return { ...previewResourceState, + resourceName: 'Packs', isAllQueryEnabled, previewData, isPreviewLoading, diff --git a/packages/app/modules/trip/hooks/useUserTrips.ts b/packages/app/modules/trip/hooks/useUserTrips.ts index 0b90197f4..c1ff57a39 100644 --- a/packages/app/modules/trip/hooks/useUserTrips.ts +++ b/packages/app/modules/trip/hooks/useUserTrips.ts @@ -8,7 +8,7 @@ import { type PreviewResourceStateWithData, usePreviewResourceState, } from 'app/hooks/common'; -import { useEffect, useState } from 'react'; +import { useEffect, useMemo, useState } from 'react'; export const useUserTrips = ( ownerId: string | undefined, @@ -20,23 +20,24 @@ export const useUserTrips = ( ); const enabled = queryEnabled && !!ownerId; const { searchTerm, isPublic, isPreview } = params || {}; + const queryParams = useMemo( + () => ({ + ownerId, + searchTerm, + queryBy: 'Most Recent', + pagination, + isPublic, + isPreview, + }), + [isPublic, isPreview, searchTerm, pagination, ownerId], + ); // Leverage the query hook provided by tRPC const { data, error, isLoading, refetch } = - queryTrpc.getUserTripsFeed.useQuery( - { - ownerId, - searchTerm, - queryBy: 'Most Recent', - pagination, - isPublic, - isPreview, - }, - { - enabled, // This query will run only if 'enabled' is true. - refetchOnWindowFocus: false, - }, - ); + queryTrpc.getUserTripsFeed.useQuery(queryParams, { + enabled, // This query will run only if 'enabled' is true. + refetchOnWindowFocus: false, + }); const { fetchPrevPage, fetchNextPage } = usePagination( refetch, @@ -77,12 +78,13 @@ interface FetchUserTripsPreviewReturn extends PreviewResourceStateWithData { export const useUserTripsWithPreview = ( userId: string, searchTerm: string, + isPublic?: boolean, ): FetchUserTripsPreviewReturn => { const { isAllQueryEnabled, ...previewResourceState } = usePreviewResourceState(); const { data: previewData, isLoading: isPreviewLoading } = useUserTrips( userId, - { isPublic: true, searchTerm }, + { isPublic, isPreview: true, searchTerm }, true, ); @@ -96,14 +98,11 @@ export const useUserTripsWithPreview = ( totalPages, hasPrevPage, hasNextPage, - } = useUserTrips( - userId, - { isPublic: true, searchTerm: '' }, - isAllQueryEnabled, - ); + } = useUserTrips(userId, { isPublic, searchTerm: '' }, isAllQueryEnabled); return { ...previewResourceState, + resourceName: 'Trips', isAllQueryEnabled, previewData, isPreviewLoading, diff --git a/packages/app/modules/user/components/UserDetailList.tsx b/packages/app/modules/user/components/UserDetailList.tsx index 240e56a8b..ce646f209 100644 --- a/packages/app/modules/user/components/UserDetailList.tsx +++ b/packages/app/modules/user/components/UserDetailList.tsx @@ -10,6 +10,7 @@ import { FeedCard, FeedSearchFilter } from 'app/modules/feed'; import { fuseSearch } from 'app/utils/fuseSearch'; import { BaseDialog, BaseModal, Pagination, RButton } from '@packrat/ui'; import { type PreviewResourceStateWithData } from 'app/hooks/common'; +import RSecondaryButton from 'app/components/RSecondaryButton'; interface DataListProps { resource: PreviewResourceStateWithData; @@ -27,20 +28,9 @@ export const UserDataList = ({ return ( <> {Platform.OS == 'web' ? ( - resource.setIsSeeAllModalOpen(true)} - onClose={() => resource.setIsSeeAllModalOpen(false)} - footerButtons={[ - { - label: 'Cancel', - color: '#B22222', - onClick: (_, closeModal) => closeModal(), - }, - ]} - footerComponent={undefined} + } > ) : null} - + ) : ( closeModal(), - }, - ]} - footerComponent={undefined} + title={`${resource.resourceName} List`} + triggerComponent={} > { const authUser = useAuthUser(); + const isAuthUserProfile = !id || id === authUser?.id; + console.log('isAuthUserProfile', isAuthUserProfile); const userId = id ?? authUser?.id; const [searchTerms, setSearchTerms] = useState< Record @@ -22,10 +24,12 @@ export const useProfile = (id = null) => { const userPacksQuery = useUserPacksWithPreview( userId as string, searchTerms.packs, + !isAuthUserProfile ? true : undefined, ); const userTripsQuery = useUserTripsWithPreview( userId as string, searchTerms.trips, + !isAuthUserProfile ? true : undefined, ); const onSearchChange = useCallback( diff --git a/packages/app/modules/user/widgets/ProfileContainer.tsx b/packages/app/modules/user/widgets/ProfileContainer.tsx index 674ddcf93..1e45eb636 100644 --- a/packages/app/modules/user/widgets/ProfileContainer.tsx +++ b/packages/app/modules/user/widgets/ProfileContainer.tsx @@ -5,6 +5,7 @@ import { RStack, RText as OriginalRText, RSkeleton, + XStack, } from '@packrat/ui'; import { ScrollView } from 'react-native-gesture-handler'; import { UserDataContainer } from './UserDataContainer'; @@ -16,6 +17,8 @@ import useCustomStyles from 'app/hooks/useCustomStyles'; import Avatar from '../../../components/Avatar/Avatar'; import { useProfile } from '../hooks'; import Layout from 'app/components/layout/Layout'; +import { AsyncView } from 'app/components/AsyncView'; +import { LayoutCard } from 'app/components/LayoutCard'; const RText: any = OriginalRText; @@ -55,7 +58,6 @@ const Header = ({ const { enableDarkMode, enableLightMode, isDark, isLight, currentTheme } = useTheme(); const styles = useCustomStyles(loadStyles); - const profileImage = user?.profileImage ?? null; const userRealName = user?.name ?? null; const userEmail = user?.email ?? null; const userEmailSplitFirstHalf = userEmail?.split('@')[0] ?? null; @@ -64,124 +66,89 @@ const Header = ({ : `@${userEmailSplitFirstHalf}`; return ( - - - - {isCurrentUser && !isLoading && ( - - - - )} - - - {isLoading ? ( - <> - - - - ) : ( - <> - - - {userRealName} - - {username} - - )} - - - {isCurrentUser && !isLoading && } - - - {isLoading ? ( - <> - - - - - - ) : ( - <> - - Trips - {tripsCount} - - - Packs - - {packsCount} - - - - - Favorites - - - {favoritesCount} - - - - - Certified - - + + {isCurrentUser && !isLoading && ( + + + + )} + + + {isLoading ? ( + <> + - - - )} + + + ) : ( + <> + + + {userRealName} + + {username} + + )} + - {error ? {error} : null} - - - ); -}; - -// Skeleton version of the UserDataCard component -const SkeletonUserDataCard = () => { - return ( - - - - + {isCurrentUser && !isLoading && } + + {error}} + > + + + + Trips + + {tripsCount} + + + + Packs + + {packsCount} + + + + Favorites + + {favoritesCount} + + + {/* TODO add guide users */} + {/* + Certified + + */} + + + ); }; @@ -204,100 +171,62 @@ export function ProfileContainer({ id = null }) { error, } = useProfile(id); + useEffect(() => { + console.log('mount'); + return () => { + console.log('unmount'); + }; + }, [user]); + return ( - - - -
+ +
+ + {favoritesQuery?.previewData?.length > 0 ? ( + - - - {isLoading && } - - - {favoritesQuery?.previewData?.length > 0 ? ( - - ) : ( - - No favorites yet - - )} - {userPacksQuery?.previewData?.length > 0 && ( - - )} - {userTripsQuery?.previewData?.length > 0 && ( - - )} - - - - + ) : null} + {userPacksQuery?.previewData?.length > 0 && ( + + )} + {userTripsQuery?.previewData?.length > 0 && ( + + )} + + ); } const loadStyles = (theme) => { const { currentTheme } = theme; return { - mainContainer: { - backgroundColor: currentTheme.colors.background, - // flex: 1, - alignItems: 'center', - justifyItems: 'center', - padding: 20, - }, - infoSection: { - flexDirection: 'column', - alignSelf: 'center', - backgroundColor: currentTheme.colors.background, - alignItems: 'center', - borderRadius: 12, - shadowColor: '#000', - shadowOffset: { - width: 0, - height: 1, - }, - shadowOpacity: 0.18, - shadowRadius: 1.0, - elevation: 1, - justifyContent: 'center', - }, userInfo: { flexDirection: 'column', alignItems: 'center', @@ -314,15 +243,6 @@ const loadStyles = (theme) => { color: currentTheme.colors.text, textAlign: 'center', }, - card: { - flexDirection: 'row', - justifyContent: 'space-around', - width: '100%', - padding: 15, - borderRadius: 12, - backgroundColor: currentTheme.colors.card, - marginVertical: 15, - }, cardInfo: { alignItems: 'center', }, @@ -330,20 +250,5 @@ const loadStyles = (theme) => { width: '100%', flex: 1, }, - - userDataCard: { - borderRadius: 15, - backgroundColor: currentTheme.colors.card, - padding: 10, - margin: 5, - shadowColor: '#000', - shadowOffset: { - width: 0, - height: 1, - }, - shadowOpacity: 0.18, - shadowRadius: 1.0, - elevation: 1, - }, }; }; diff --git a/packages/app/modules/user/widgets/UserDataContainer.tsx b/packages/app/modules/user/widgets/UserDataContainer.tsx index b97a1e6cd..8d957f2ec 100644 --- a/packages/app/modules/user/widgets/UserDataContainer.tsx +++ b/packages/app/modules/user/widgets/UserDataContainer.tsx @@ -2,16 +2,14 @@ import { RLink } from '@packrat/ui'; import { RStack, RText, RButton, RSkeleton } from '@packrat/ui'; import { Platform, VirtualizedList } from 'react-native'; import { UserDataCard, UserDataList } from '../components'; -import React, { memo, useEffect, useState } from 'react'; -import LargeCard from 'app/components/card/LargeCard'; +import React, { memo } from 'react'; import useTheme from 'app/hooks/useTheme'; -import { hexToRGBA } from 'app/utils/colorFunctions'; import { View } from 'react-native'; import { useAuthUser } from 'app/modules/auth'; -import Layout from 'app/components/layout/Layout'; import { SearchProvider } from 'app/modules/feed'; import { type PreviewResourceStateWithData } from 'app/hooks/common'; import type { PreviewListType } from '../model'; +import Carousel from 'app/components/carousel'; // Skeleton version of the UserDataCard component const SkeletonUserDataCard = () => { @@ -55,7 +53,7 @@ export const UserDataContainer = memo(function UserDataContainer({ const typeUppercaseSingular = typeUppercase.slice(0, -1); const differentUser = userId && currentUser && userId !== currentUser.id; - const Card = ({ item, index }) => { + const Card = ({ item }) => { return ; }; @@ -81,87 +79,73 @@ export const UserDataContainer = memo(function UserDataContainer({ } return ( - - + - - - {differentUser ? `${typeUppercase}` : `Your ${typeUppercase}`} - - - onSearchChange(search, type)} - /> - - - - {isLoading ? ( - skeletonCards - ) : resource?.previewData && resource?.previewData.length > 0 ? ( - <> - resource.previewData.length} - getItem={(data, index) => data[index]} - data={resource.previewData} - keyExtractor={(item) => item.id} - renderItem={Card} - scrollEnabled={true} - maxToRenderPerBatch={2} - horizontal={true} - nestedScrollEnabled={true} - contentContainerStyle={{ - paddingHorizontal: 10, - }} - ItemSeparatorComponent={() => } - /> - - ) : currentUser?.id === userId ? ( - - - {`Create your first ${typeUppercaseSingular}`} - - - ) : ( - <> - )} - + {differentUser ? `${typeUppercase}` : `Your ${typeUppercase}`} + + + onSearchChange(search, type)} + /> + - - + + {isLoading ? ( + skeletonCards + ) : resource?.previewData && resource?.previewData.length > 0 ? ( + <> + + {resource?.previewData?.map((item) => ( + + + + ))} + + + ) : currentUser?.id === userId ? ( + + + {`Create your first ${typeUppercaseSingular}`} + + + ) : ( + <> + )} + + + ); }); diff --git a/packages/ui/src/dialog/BaseDialog.tsx b/packages/ui/src/dialog/BaseDialog.tsx index 3975dac00..d4e2d6e2e 100644 --- a/packages/ui/src/dialog/BaseDialog.tsx +++ b/packages/ui/src/dialog/BaseDialog.tsx @@ -16,6 +16,7 @@ interface BaseDialogProps { title: string; description: string; trigger: string; + triggerComponent?: React.ReactNode; children: string | JSX.Element | JSX.Element[] | (() => JSX.Element); } @@ -23,6 +24,7 @@ export const BaseDialog = ({ title, description, trigger, + triggerComponent, children, }: BaseDialogProps) => { const [open, setOpen] = useState(false); @@ -35,7 +37,7 @@ export const BaseDialog = ({ }} > - {trigger} + {triggerComponent || {trigger}} diff --git a/server/src/controllers/favorite/getUserFavorites.ts b/server/src/controllers/favorite/getUserFavorites.ts index 17e280aca..de43b11f2 100644 --- a/server/src/controllers/favorite/getUserFavorites.ts +++ b/server/src/controllers/favorite/getUserFavorites.ts @@ -22,12 +22,17 @@ export function getUserFavoritesRoute() { pagination: z.object({ limit: z.number(), offset: z.number() }), searchTerm: z.string().optional(), isPreview: z.boolean().optional(), + isPublic: z.boolean().optional(), }), ) .query(async (opts) => { - const { userId, pagination, searchTerm } = opts.input; + const { userId, pagination, searchTerm, isPublic } = opts.input; const { data, totalCount, currentPagination } = - await getUserFavoritesService(userId, { searchTerm }, pagination); + await getUserFavoritesService( + userId, + { searchTerm, isPublic }, + pagination, + ); return { data, diff --git a/server/src/services/favorite/getUserFavoritesService.ts b/server/src/services/favorite/getUserFavoritesService.ts index ba27732ea..81fa13e94 100644 --- a/server/src/services/favorite/getUserFavoritesService.ts +++ b/server/src/services/favorite/getUserFavoritesService.ts @@ -10,10 +10,10 @@ import { PaginationParams } from 'src/helpers/pagination'; */ export const getUserFavoritesService = async ( userId: string, - options?: { searchTerm?: string }, + options?: { searchTerm?: string; isPublic?: boolean }, pagination?: PaginationParams, ) => { - const { searchTerm } = options || {}; + const { searchTerm, isPublic } = options || {}; const userClass = new User(); const feedClass = new Feed(); const user = (await userClass.findUser({ @@ -30,6 +30,7 @@ export const getUserFavoritesService = async ( { includeUserFavoritesOnly: true, searchTerm, + isPublic, ownerId: userId, }, 'trips', From c349373b4d9004ba968d61d842e3e82e555d2783 Mon Sep 17 00:00:00 2001 From: Tadjaur Date: Sun, 15 Dec 2024 19:31:36 +0100 Subject: [PATCH 11/11] Edit schema --- .../tripRoutesValidator.ts | 20 ++++++------------- server/src/drizzle/methods/trip.ts | 2 +- server/src/services/trip/addTripService.ts | 6 ++---- server/src/services/trip/editTripService.ts | 1 + 4 files changed, 10 insertions(+), 19 deletions(-) diff --git a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts index c3134bba6..0a3d5c1de 100644 --- a/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts +++ b/packages/validations/src/validations/tripRoutesValidator/tripRoutesValidator.ts @@ -51,7 +51,7 @@ export const addTripDetails = z.object({ bounds: z.tuple([z.array(z.number()), z.array(z.number())]).optional(), end_date: z.string(), geoJSON: z.string(), - packId: z.string(), + pack_id: z.string(), parks: z.string().optional(), start_date: z.string(), trails: z.string().optional(), @@ -60,19 +60,11 @@ export const addTripDetails = z.object({ export const addTrip = addTripDetails.merge(addTripForm); export type AddTripType = z.infer; -export const editTrip = z.object({ - activity: z.enum(tripActivityValues).optional(), - bounds: z.array(z.array(z.number())).length(2).optional(), - description: z.string().optional(), - end_date: z.string().optional(), - geoJSON: z.string().optional(), - name: z.string().optional(), - packId: z.string().optional(), - parks: z.string().optional(), - start_date: z.string().optional(), - trails: z.string().optional(), - id: z.string().min(1), -}); +export const editTrip = addTrip.merge( + z.object({ + id: z.string().min(1), + }), +); export type EditTripType = z.infer; diff --git a/server/src/drizzle/methods/trip.ts b/server/src/drizzle/methods/trip.ts index fb047acbb..8e242e100 100644 --- a/server/src/drizzle/methods/trip.ts +++ b/server/src/drizzle/methods/trip.ts @@ -7,7 +7,7 @@ import { } from '../../db/schema'; export class Trip { - async update(trip: Partial) { + async update(trip: InsertTrip) { try { if (!trip.id) { throw new Error('Trip id is required for update operation'); diff --git a/server/src/services/trip/addTripService.ts b/server/src/services/trip/addTripService.ts index de3d09a12..286368252 100644 --- a/server/src/services/trip/addTripService.ts +++ b/server/src/services/trip/addTripService.ts @@ -1,12 +1,9 @@ import * as validator from '@packrat/validations'; -import { calculateTripScore } from 'src/utils/scoreTrip'; import { GeoJson } from '../../drizzle/methods/Geojson'; import { TripGeoJson } from '../../drizzle/methods/TripGeoJson'; import { Trip } from '../../drizzle/methods/trip'; -import { validateGeojsonId, validateGeojsonType } from '../../utils/geojson'; import { GeojsonStorageService } from '../geojsonStorage'; import { scoreTripService } from './scoreTripService'; -import { trip } from 'src/db/schema'; export const addTripService = async ( tripData: validator.AddTripType & { ownerId: string }, @@ -22,8 +19,9 @@ export const addTripService = async ( description: tripData.description, start_date: tripData.start_date, end_date: tripData.end_date, - activity: tripData.activity, + activity: tripData.activity || 'trip', owner_id: tripData.ownerId, + pack_id: tripData.pack_id, is_public: tripData.is_public === '0', trails: tripData.trails ? JSON.parse(tripData.trails) : null, parks: tripData.parks ? JSON.parse(tripData.parks) : null, diff --git a/server/src/services/trip/editTripService.ts b/server/src/services/trip/editTripService.ts index 78bdedbd9..f7083bb43 100644 --- a/server/src/services/trip/editTripService.ts +++ b/server/src/services/trip/editTripService.ts @@ -18,6 +18,7 @@ export const editTripService = async ( start_date: tripData.start_date || selectedTrip.start_date, end_date: tripData.end_date || selectedTrip.end_date, activity: tripData.activity || selectedTrip.activity, + pack_id: tripData.pack_id || selectedTrip.pack_id, trails: tripData.trails ? JSON.parse(tripData.trails) : selectedTrip.trails,