04b1fc3024
Test Suite / test (24.15.0) (push) Has been cancelled
- Added a new API path for rematching recipe ingredients in `api_paths.dart`. - Implemented a manual product creation dialog in `inventory_screen.dart` to allow users to create new products directly. - Integrated the rematch functionality in `recipe_repository.dart` to handle rematching of recipe ingredients. - Updated the recipe detail screen to include a button for triggering the rematch process. - Introduced a new `RecipeMatchingService` in the backend to handle ingredient matching logic. - Added database migration to include `aiEngineEnabled` column in the User table. Co-authored-by: Copilot <copilot@github.com>
459 lines
13 KiB
TypeScript
459 lines
13 KiB
TypeScript
import { PrismaService } from '../prisma/prisma.service';
|
|
import { CreateProductDto } from './dto/create-product.dto';
|
|
import { UpdateProductDto } from './dto/update-product.dto';
|
|
import { UpsertNutritionDto } from './dto/upsert-nutrition.dto';
|
|
import { AiService } from '../ai/ai.service';
|
|
import { CategoriesService } from '../categories/categories.service';
|
|
export declare class ProductsService {
|
|
private readonly prisma;
|
|
private readonly aiService;
|
|
private readonly categoriesService;
|
|
constructor(prisma: PrismaService, aiService: AiService, categoriesService: CategoriesService);
|
|
findAll(filters?: {
|
|
tag?: string;
|
|
}): Promise<({
|
|
nutrition: {
|
|
id: number;
|
|
productId: number;
|
|
calories: number | null;
|
|
protein: number | null;
|
|
fat: number | null;
|
|
carbohydrates: number | null;
|
|
salt: number | null;
|
|
sugar: number | null;
|
|
fiber: number | null;
|
|
} | null;
|
|
tags: ({
|
|
tag: {
|
|
name: string;
|
|
id: number;
|
|
};
|
|
} & {
|
|
productId: number;
|
|
tagId: number;
|
|
})[];
|
|
categoryRef: ({
|
|
parent: ({
|
|
parent: {
|
|
name: string;
|
|
id: number;
|
|
parentId: number | null;
|
|
} | null;
|
|
} & {
|
|
name: string;
|
|
id: number;
|
|
parentId: number | null;
|
|
}) | null;
|
|
} & {
|
|
name: string;
|
|
id: number;
|
|
parentId: number | null;
|
|
}) | null;
|
|
} & {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
})[]>;
|
|
findByOwner(userId: number): Promise<{
|
|
name: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
canonicalName: string | null;
|
|
}[]>;
|
|
createPrivate(data: CreateProductDto, userId: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
findDuplicateCandidates(): Promise<{
|
|
normalizedName: string;
|
|
count: number;
|
|
products: {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}[];
|
|
}[]>;
|
|
findOne(id: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
create(data: CreateProductDto, ownerId?: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
update(id: number, data: UpdateProductDto): Promise<{
|
|
nutrition: {
|
|
id: number;
|
|
productId: number;
|
|
calories: number | null;
|
|
protein: number | null;
|
|
fat: number | null;
|
|
carbohydrates: number | null;
|
|
salt: number | null;
|
|
sugar: number | null;
|
|
fiber: number | null;
|
|
} | null;
|
|
tags: ({
|
|
tag: {
|
|
name: string;
|
|
id: number;
|
|
};
|
|
} & {
|
|
productId: number;
|
|
tagId: number;
|
|
})[];
|
|
} & {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
updateCanonicalName(id: number, canonicalName: string): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
findDeleted(): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}[]>;
|
|
remove(id: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
permanentDelete(id: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
restore(id: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
private findProductByIdOrThrow;
|
|
merge(sourceProductId: number, targetProductId: number): Promise<{
|
|
message: string;
|
|
sourceProductId: number;
|
|
targetProductId: number;
|
|
movedInventoryCount: number;
|
|
softDeletedSource: {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
};
|
|
}>;
|
|
previewMerge(sourceProductId: number, targetProductId: number): Promise<{
|
|
source: {
|
|
inventoryCount: number;
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
};
|
|
target: {
|
|
inventoryCount: number;
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
};
|
|
outcome: {
|
|
inventoryItemsToMove: number;
|
|
sourceWillBeSoftDeleted: boolean;
|
|
targetWillRemainActive: boolean;
|
|
};
|
|
}>;
|
|
backfillCanonicalNames(): Promise<{
|
|
message: string;
|
|
updatedCount: number;
|
|
products: {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}[];
|
|
}>;
|
|
setTags(productId: number, tagNames: string[]): Promise<({
|
|
nutrition: {
|
|
id: number;
|
|
productId: number;
|
|
calories: number | null;
|
|
protein: number | null;
|
|
fat: number | null;
|
|
carbohydrates: number | null;
|
|
salt: number | null;
|
|
sugar: number | null;
|
|
fiber: number | null;
|
|
} | null;
|
|
tags: ({
|
|
tag: {
|
|
name: string;
|
|
id: number;
|
|
};
|
|
} & {
|
|
productId: number;
|
|
tagId: number;
|
|
})[];
|
|
} & {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}) | null>;
|
|
upsertNutrition(productId: number, data: UpsertNutritionDto): Promise<{
|
|
id: number;
|
|
productId: number;
|
|
calories: number | null;
|
|
protein: number | null;
|
|
fat: number | null;
|
|
carbohydrates: number | null;
|
|
salt: number | null;
|
|
sugar: number | null;
|
|
fiber: number | null;
|
|
}>;
|
|
findAllTags(): Promise<{
|
|
name: string;
|
|
id: number;
|
|
}[]>;
|
|
resetAll(): Promise<{
|
|
ok: boolean;
|
|
}>;
|
|
bulkUpdate(ids: number[], data: {
|
|
categoryId?: number | null;
|
|
}): Promise<{
|
|
updated: number;
|
|
}>;
|
|
findUncategorized(): Promise<{
|
|
id: number;
|
|
name: string;
|
|
canonicalName: string | null;
|
|
}[]>;
|
|
aiCategorizeBulk(productIds?: number[]): Promise<{
|
|
productId: number;
|
|
productName: string;
|
|
suggestion: object;
|
|
}[]>;
|
|
findPending(): Promise<({
|
|
owner: {
|
|
id: number;
|
|
username: string;
|
|
};
|
|
categoryRef: ({
|
|
parent: {
|
|
name: string;
|
|
id: number;
|
|
parentId: number | null;
|
|
} | null;
|
|
} & {
|
|
name: string;
|
|
id: number;
|
|
parentId: number | null;
|
|
}) | null;
|
|
} & {
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
})[]>;
|
|
createPending(data: CreateProductDto, userId: number): Promise<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}>;
|
|
setStatus(id: number, status: string): import(".prisma/client").Prisma.Prisma__ProductClient<{
|
|
name: string;
|
|
category: string | null;
|
|
status: string;
|
|
id: number;
|
|
categoryId: number | null;
|
|
normalizedName: string;
|
|
canonicalName: string | null;
|
|
isActive: boolean;
|
|
deletedAt: Date | null;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
ownerId: number;
|
|
isPrivate: boolean;
|
|
}, never, import("@prisma/client/runtime/library").DefaultArgs, import(".prisma/client").Prisma.PrismaClientOptions>;
|
|
}
|