feat: update inventory item structure and improve error handling

This commit is contained in:
THIS ONE IS A LITTLE BIT TRICKY KRUB 2025-04-03 20:01:44 +07:00
parent 01bc37a136
commit f037bf766f
7 changed files with 297 additions and 208 deletions

View File

@ -100,7 +100,6 @@ type HarvestUnit struct {
type CreateInventoryItemInput struct { type CreateInventoryItemInput struct {
Header string `header:"Authorization" required:"true" example:"Bearer token"` Header string `header:"Authorization" required:"true" example:"Bearer token"`
UserID string `header:"userId" required:"true" example:"user-uuid"`
Body struct { Body struct {
Name string `json:"name" required:"true"` Name string `json:"name" required:"true"`
CategoryID int `json:"categoryId" required:"true"` CategoryID int `json:"categoryId" required:"true"`
@ -119,7 +118,6 @@ type CreateInventoryItemOutput struct {
type UpdateInventoryItemInput struct { type UpdateInventoryItemInput struct {
Header string `header:"Authorization" required:"true" example:"Bearer token"` Header string `header:"Authorization" required:"true" example:"Bearer token"`
UserID string `header:"userId" required:"true" example:"user-uuid"`
ID string `path:"id"` ID string `path:"id"`
Body struct { Body struct {
Name string `json:"name"` Name string `json:"name"`
@ -137,7 +135,6 @@ type UpdateInventoryItemOutput struct {
type GetInventoryItemsInput struct { type GetInventoryItemsInput struct {
Header string `header:"Authorization" required:"true" example:"Bearer token"` Header string `header:"Authorization" required:"true" example:"Bearer token"`
UserID string `header:"userId" required:"true" example:"user-uuid"`
CategoryID int `query:"categoryId"` CategoryID int `query:"categoryId"`
StatusID int `query:"statusId"` StatusID int `query:"statusId"`
StartDate time.Time `query:"startDate" format:"date-time"` StartDate time.Time `query:"startDate" format:"date-time"`
@ -186,8 +183,9 @@ type GetHarvestUnitsOutput struct {
} }
func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInventoryItemInput) (*CreateInventoryItemOutput, error) { func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInventoryItemInput) (*CreateInventoryItemOutput, error) {
userID, err := a.getUserIDFromHeader(input.Header)
item := &domain.InventoryItem{ item := &domain.InventoryItem{
UserID: input.UserID, UserID: userID,
Name: input.Body.Name, Name: input.Body.Name,
CategoryID: input.Body.CategoryID, CategoryID: input.Body.CategoryID,
Quantity: input.Body.Quantity, Quantity: input.Body.Quantity,
@ -200,7 +198,7 @@ func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInven
return nil, huma.Error422UnprocessableEntity(err.Error()) return nil, huma.Error422UnprocessableEntity(err.Error())
} }
err := a.inventoryRepo.CreateOrUpdate(ctx, item) err = a.inventoryRepo.CreateOrUpdate(ctx, item)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -211,8 +209,9 @@ func (a *api) createInventoryItemHandler(ctx context.Context, input *CreateInven
} }
func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInventoryItemsInput) (*GetInventoryItemsOutput, error) { func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInventoryItemsInput) (*GetInventoryItemsOutput, error) {
userID, err := a.getUserIDFromHeader(input.Header)
filter := domain.InventoryFilter{ filter := domain.InventoryFilter{
UserID: input.UserID, UserID: userID,
CategoryID: input.CategoryID, CategoryID: input.CategoryID,
StatusID: input.StatusID, StatusID: input.StatusID,
StartDate: input.StartDate, StartDate: input.StartDate,
@ -225,7 +224,7 @@ func (a *api) getInventoryItemsByUserHandler(ctx context.Context, input *GetInve
Direction: input.SortOrder, Direction: input.SortOrder,
} }
items, err := a.inventoryRepo.GetByUserID(ctx, input.UserID, filter, sort) items, err := a.inventoryRepo.GetByUserID(ctx, userID, filter, sort)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -286,7 +285,8 @@ func (a *api) getInventoryItemHandler(ctx context.Context, input *GetInventoryIt
} }
func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInventoryItemInput) (*UpdateInventoryItemOutput, error) { func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInventoryItemInput) (*UpdateInventoryItemOutput, error) {
item, err := a.inventoryRepo.GetByID(ctx, input.ID, input.UserID) userID, err := a.getUserIDFromHeader(input.Header)
item, err := a.inventoryRepo.GetByID(ctx, input.ID, userID)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -319,7 +319,7 @@ func (a *api) updateInventoryItemHandler(ctx context.Context, input *UpdateInven
return nil, err return nil, err
} }
updatedItem, err := a.inventoryRepo.GetByID(ctx, input.ID, input.UserID) updatedItem, err := a.inventoryRepo.GetByID(ctx, input.ID, userID)
if err != nil { if err != nil {
return nil, err return nil, err
} }

View File

@ -6,6 +6,7 @@ import type {
CreateInventoryItemInput, CreateInventoryItemInput,
UpdateInventoryItemInput, UpdateInventoryItemInput,
} from "@/types"; } from "@/types";
import { AxiosError } from "axios";
/** /**
* Simulates an API call to fetch inventory items. * Simulates an API call to fetch inventory items.
@ -46,7 +47,6 @@ export async function fetchInventoryItems(): Promise<InventoryItem[]> {
} catch (error) { } catch (error) {
console.error("Error while fetching inventory items! " + error); console.error("Error while fetching inventory items! " + error);
throw error; throw error;
} }
} }
@ -59,9 +59,32 @@ export async function createInventoryItem(
item item
); );
return response.data; return response.data;
} catch (error) { } catch (error: unknown) {
console.error("Error while creating Inventory Item! " + error); // Cast error to AxiosError to safely access response properties
throw new Error("Failed to create inventory item: " + error); if (error instanceof AxiosError && error.response) {
// Log the detailed error message
console.error("Error while creating Inventory Item!");
console.error("Response Status:", error.response.status); // e.g., 422
console.error("Error Detail:", error.response.data?.detail); // Custom error message from backend
console.error("Full Error Response:", error.response.data); // Entire error object (including details)
// Throw a new error with a more specific message
throw new Error(
`Failed to create inventory item: ${
error.response.data?.detail || error.message
}`
);
} else {
// Handle other errors (e.g., network errors or unknown errors)
console.error(
"Error while creating Inventory Item, unknown error:",
error
);
throw new Error(
"Failed to create inventory item: " +
(error instanceof Error ? error.message : "Unknown error")
);
}
} }
} }

View File

@ -3,8 +3,7 @@
import { useState } from "react"; import { useState } from "react";
import { CalendarIcon } from "lucide-react"; import { CalendarIcon } from "lucide-react";
import { format } from "date-fns"; import { format } from "date-fns";
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import { useMutation, useQueryClient } from "@tanstack/react-query";
import { Button } from "@/components/ui/button"; import { Button } from "@/components/ui/button";
import { Calendar } from "@/components/ui/calendar"; import { Calendar } from "@/components/ui/calendar";
import { import {
@ -59,29 +58,51 @@ export function AddInventoryItem({
const [itemQuantity, setItemQuantity] = useState(0); const [itemQuantity, setItemQuantity] = useState(0);
const [itemUnit, setItemUnit] = useState(""); const [itemUnit, setItemUnit] = useState("");
const [itemStatus, setItemStatus] = useState(""); const [itemStatus, setItemStatus] = useState("");
const [isSubmitted, setIsSubmitted] = useState(false);
const [successMessage, setSuccessMessage] = useState("");
const [errorMessage, setErrorMessage] = useState("");
const queryClient = useQueryClient();
const mutation = useMutation({ const mutation = useMutation({
mutationFn: (item: CreateInventoryItemInput) => createInventoryItem(item), mutationFn: (item: CreateInventoryItemInput) => createInventoryItem(item),
onSuccess: () => { onSuccess: () => {
// invalidate queries to refresh inventory data. // invalidate queries to refresh inventory data
const queryClient = useQueryClient();
queryClient.invalidateQueries({ queryKey: ["inventoryItems"] }); queryClient.invalidateQueries({ queryKey: ["inventoryItems"] });
// reset form fields and close dialog.
setItemName(""); setItemName("");
setItemCategory(""); setItemCategory("");
setItemQuantity(0); setItemQuantity(0);
setItemUnit(""); setItemUnit("");
setDate(undefined); setDate(undefined);
setIsSubmitted(true);
setSuccessMessage("Item created successfully!");
// reset success message after 3 seconds
setTimeout(() => {
setIsSubmitted(false);
setSuccessMessage("");
setOpen(false); setOpen(false);
}, 3000);
},
onError: (error: any) => {
console.error("Error creating item: ", error);
setErrorMessage(
"There was an error creating the item. Please try again."
);
// reset success message after 3 seconds
setTimeout(() => {
setErrorMessage("");
}, 3000);
}, },
}); });
const inputStates = [itemName, itemCategory, itemUnit, itemStatus, date]; const inputStates = [itemName, itemCategory, itemUnit, itemStatus, date];
const isInputValid = inputStates.every((input) => input); const isInputValid = inputStates.every((input) => input);
const handleSave = () => { const handleSave = () => {
if (!isInputValid) { if (!isInputValid) {
console.error("All fields are required"); setErrorMessage(
"There was an error creating the item. Please try again."
);
return; return;
} }
@ -93,13 +114,14 @@ export function AddInventoryItem({
unitId: harvestUnits.find((item) => item.name === itemUnit)?.id || 0, unitId: harvestUnits.find((item) => item.name === itemUnit)?.id || 0,
statusId: statusId:
inventoryStatus.find((item) => item.name === itemStatus)?.id || 0, inventoryStatus.find((item) => item.name === itemStatus)?.id || 0,
lastUpdated: date ? date.toISOString() : new Date().toISOString(), dateAdded: date ? date.toISOString() : new Date().toISOString(),
}; };
console.table(newItem); // console.table(newItem);
mutation.mutate(newItem); mutation.mutate(newItem);
}; };
return ( return (
<>
<Dialog open={open} onOpenChange={setOpen}> <Dialog open={open} onOpenChange={setOpen}>
<DialogTrigger asChild> <DialogTrigger asChild>
<Button>Add New Item</Button> <Button>Add New Item</Button>
@ -134,8 +156,11 @@ export function AddInventoryItem({
<SelectContent> <SelectContent>
<SelectGroup> <SelectGroup>
<SelectLabel>Category</SelectLabel> <SelectLabel>Category</SelectLabel>
{inventoryCategory.map((categoryItem, _) => ( {inventoryCategory.map((categoryItem) => (
<SelectItem key={categoryItem.id} value={categoryItem.name}> <SelectItem
key={categoryItem.id}
value={categoryItem.name}
>
{categoryItem.name} {categoryItem.name}
</SelectItem> </SelectItem>
))} ))}
@ -154,7 +179,7 @@ export function AddInventoryItem({
<SelectContent> <SelectContent>
<SelectGroup> <SelectGroup>
<SelectLabel>Status</SelectLabel> <SelectLabel>Status</SelectLabel>
{inventoryStatus.map((statusItem, _) => ( {inventoryStatus.map((statusItem) => (
<SelectItem key={statusItem.id} value={statusItem.name}> <SelectItem key={statusItem.id} value={statusItem.name}>
{statusItem.name} {statusItem.name}
</SelectItem> </SelectItem>
@ -186,7 +211,7 @@ export function AddInventoryItem({
<SelectContent> <SelectContent>
<SelectGroup> <SelectGroup>
<SelectLabel>Unit</SelectLabel> <SelectLabel>Unit</SelectLabel>
{harvestUnits.map((unit, _) => ( {harvestUnits.map((unit) => (
<SelectItem key={unit.id} value={unit.name}> <SelectItem key={unit.id} value={unit.name}>
{unit.name} {unit.name}
</SelectItem> </SelectItem>
@ -224,11 +249,26 @@ export function AddInventoryItem({
</div> </div>
</div> </div>
<DialogFooter> <DialogFooter>
<Button type="submit" onClick={handleSave}> <div className="flex flex-col items-center w-full space-y-2">
<Button type="button" onClick={handleSave} className="w-full">
Save Item Save Item
</Button> </Button>
<div className="flex flex-col items-center space-y-2">
{isSubmitted && (
<p className="text-green-500 text-sm">
{successMessage} You may close this window.
</p>
)}
{errorMessage && (
<p className="text-red-500 text-sm">{errorMessage}</p>
)}
</div>
</div>
</DialogFooter> </DialogFooter>
</DialogContent> </DialogContent>
</Dialog> </Dialog>
</>
); );
} }

View File

@ -34,9 +34,9 @@ import type { UpdateInventoryItemInput } from "@/types";
export interface EditInventoryItemProps { export interface EditInventoryItemProps {
id: string; id: string;
name: string; name: string;
categoryId: string; categoryId: number;
statusId: string; statusId: number;
unitId: string; unitId: number;
quantity: number; quantity: number;
fetchedInventoryStatus: InventoryStatus[]; fetchedInventoryStatus: InventoryStatus[];
fetchedInventoryCategory: InventoryItemCategory[]; fetchedInventoryCategory: InventoryItemCategory[];
@ -58,19 +58,16 @@ export function EditInventoryItem({
const [itemName, setItemName] = useState(name); const [itemName, setItemName] = useState(name);
const [itemCategory, setItemCategory] = useState( const [itemCategory, setItemCategory] = useState(
fetchedInventoryCategory.find( fetchedInventoryCategory.find(
(categoryItem) => categoryItem.id.toString() === categoryId (categoryItem) => categoryItem.id === categoryId
)?.name )?.name
); );
const [itemQuantity, setItemQuantity] = useState(quantity); const [itemQuantity, setItemQuantity] = useState(quantity);
const [itemUnit, setItemUnit] = useState( const [itemUnit, setItemUnit] = useState(
fetchedHarvestUnits.find( fetchedHarvestUnits.find((harvestItem) => harvestItem.id === unitId)?.name
(harvestItem) => harvestItem.id.toString() === unitId
)?.name
); );
const [itemStatus, setItemStatus] = useState( const [itemStatus, setItemStatus] = useState(
fetchedInventoryStatus.find( fetchedInventoryStatus.find((statusItem) => statusItem.id === statusId)
(statusItem) => statusItem.id.toString() === statusId ?.name
)?.name
); );
const [error, setError] = useState<string | null>(null); const [error, setError] = useState<string | null>(null);
@ -137,7 +134,7 @@ export function EditInventoryItem({
statusId: statusId:
fetchedInventoryStatus.find((status) => status.name === itemStatus) fetchedInventoryStatus.find((status) => status.name === itemStatus)
?.id ?? 0, ?.id ?? 0,
lastUpdated: new Date().toISOString(), dateAdded: new Date().toISOString(),
}); });
}; };

View File

@ -63,6 +63,8 @@ export default function InventoryPage() {
queryFn: fetchInventoryItems, queryFn: fetchInventoryItems,
staleTime: 60 * 1000, staleTime: 60 * 1000,
}); });
// console.table(inventoryItems);
// console.log(inventoryItems);
const { const {
data: inventoryStatus = [], data: inventoryStatus = [],
@ -73,6 +75,8 @@ export default function InventoryPage() {
queryFn: fetchInventoryStatus, queryFn: fetchInventoryStatus,
staleTime: 60 * 1000, staleTime: 60 * 1000,
}); });
// console.log(inventoryStatus);
const { const {
data: inventoryCategory = [], data: inventoryCategory = [],
isLoading: isLoadingCategory, isLoading: isLoadingCategory,
@ -101,24 +105,23 @@ export default function InventoryPage() {
return inventoryItems return inventoryItems
.map((item) => ({ .map((item) => ({
...item, ...item,
status: status: item.status.name,
inventoryStatus.find( category: item.category.name,
(statusItem) => statusItem.id.toString() === item.statusId.toString() categoryId: item.categoryId,
)?.name || "", unit: item.unit.name,
category: unitId: item.unitId,
inventoryCategory.find( statusId: item.statusId,
(categoryItem) =>
categoryItem.id.toString() === item.categoryId.toString()
)?.name || "",
categoryId: item.categoryId.toString(),
unit:
harvestUnits.find((unit) => unit.id === item.unitId)
?.name || "",
unitId: item.unitId.toString(),
statusId: item.statusId.toString(),
fetchedInventoryStatus: inventoryStatus, fetchedInventoryStatus: inventoryStatus,
fetchedInventoryCategory: inventoryCategory, fetchedInventoryCategory: inventoryCategory,
fetchedHarvestUnits: harvestUnits, fetchedHarvestUnits: harvestUnits,
lastUpdated: new Date(item.updatedAt).toLocaleString("en-US", {
year: "numeric",
month: "short",
day: "numeric",
hour: "2-digit",
minute: "2-digit",
hour12: true,
}),
})) }))
.filter((item) => .filter((item) =>
item.name.toLowerCase().includes(searchTerm.toLowerCase()) item.name.toLowerCase().includes(searchTerm.toLowerCase())
@ -139,12 +142,18 @@ export default function InventoryPage() {
cell: (info: { getValue: () => string }) => { cell: (info: { getValue: () => string }) => {
const status = info.getValue(); const status = info.getValue();
let statusClass = ""; // default status class let statusClass = "";
if (status === "Low Stock") { if (status === "In Stock") {
statusClass = "bg-yellow-300"; // yellow for low stock statusClass = "bg-green-500 hover:bg-green-600 text-white";
} else if (status === "Out Of Stock") { } else if (status === "Low Stock") {
statusClass = "bg-red-500 text-white"; // red for out of stock statusClass = "bg-yellow-300 hover:bg-yellow-400";
} else if (status === "Out of Stock") {
statusClass = "bg-red-500 hover:bg-red-600 text-white";
} else if (status === "Expired") {
statusClass = "bg-gray-500 hover:bg-gray-600 text-white";
} else if (status === "Reserved") {
statusClass = "bg-blue-500 hover:bg-blue-600 text-white";
} }
return ( return (
@ -201,6 +210,19 @@ export default function InventoryPage() {
const isLoading = loadingStates.some((loading) => loading); const isLoading = loadingStates.some((loading) => loading);
const isError = errorStates.some((error) => error); const isError = errorStates.some((error) => error);
if (isLoading)
return (
<div className="flex min-h-screen items-center justify-center">
Loading...
</div>
);
if (isError)
return (
<div className="flex min-h-screen items-center justify-center">
Error loading inventory data.
</div>
);
if (inventoryItems.length === 0) { if (inventoryItems.length === 0) {
return ( return (
@ -210,26 +232,23 @@ export default function InventoryPage() {
<TriangleAlertIcon className="h-6 w-6 text-red-500 mb-2" /> <TriangleAlertIcon className="h-6 w-6 text-red-500 mb-2" />
<AlertTitle>No Inventory Data</AlertTitle> <AlertTitle>No Inventory Data</AlertTitle>
<AlertDescription> <AlertDescription>
<div>
You currently have no inventory items. Add a new item to get You currently have no inventory items. Add a new item to get
started! started!
</div>
<div className="mt-5">
<AddInventoryItem
inventoryCategory={inventoryCategory}
inventoryStatus={inventoryStatus}
harvestUnits={harvestUnits}
/>
</div>
</AlertDescription> </AlertDescription>
</div> </div>
</Alert> </Alert>
</div> </div>
); );
} }
if (isLoading)
return (
<div className="flex min-h-screen items-center justify-center">
Loading...
</div>
);
if (isError)
return (
<div className="flex min-h-screen items-center justify-center">
Error loading inventory data.
</div>
);
return ( return (
<div className="flex min-h-screen bg-background"> <div className="flex min-h-screen bg-background">

View File

@ -1,5 +1,9 @@
lockfileVersion: '6.0' lockfileVersion: '6.0'
settings:
autoInstallPeers: true
excludeLinksFromLockfile: false
dependencies: dependencies:
'@hookform/resolvers': '@hookform/resolvers':
specifier: ^4.0.0 specifier: ^4.0.0
@ -4889,7 +4893,3 @@ packages:
/zod@3.24.2: /zod@3.24.2:
resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==} resolution: {integrity: sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==}
dev: false dev: false
settings:
autoInstallPeers: true
excludeLinksFromLockfile: false

View File

@ -156,7 +156,7 @@ export type CreateInventoryItemInput = {
categoryId: number; categoryId: number;
quantity: number; quantity: number;
unitId: number; unitId: number;
lastUpdated: string; dateAdded: string;
statusId: number; statusId: number;
}; };
@ -239,22 +239,32 @@ export interface SetOverlayAction {
export type Action = ActionWithTypeOnly | SetOverlayAction; export type Action = ActionWithTypeOnly | SetOverlayAction;
export function isCircle(overlay: OverlayGeometry): overlay is google.maps.Circle { export function isCircle(
overlay: OverlayGeometry
): overlay is google.maps.Circle {
return (overlay as google.maps.Circle).getCenter !== undefined; return (overlay as google.maps.Circle).getCenter !== undefined;
} }
export function isMarker(overlay: OverlayGeometry): overlay is google.maps.Marker { export function isMarker(
overlay: OverlayGeometry
): overlay is google.maps.Marker {
return (overlay as google.maps.Marker).getPosition !== undefined; return (overlay as google.maps.Marker).getPosition !== undefined;
} }
export function isPolygon(overlay: OverlayGeometry): overlay is google.maps.Polygon { export function isPolygon(
overlay: OverlayGeometry
): overlay is google.maps.Polygon {
return (overlay as google.maps.Polygon).getPath !== undefined; return (overlay as google.maps.Polygon).getPath !== undefined;
} }
export function isPolyline(overlay: OverlayGeometry): overlay is google.maps.Polyline { export function isPolyline(
overlay: OverlayGeometry
): overlay is google.maps.Polyline {
return (overlay as google.maps.Polyline).getPath !== undefined; return (overlay as google.maps.Polyline).getPath !== undefined;
} }
export function isRectangle(overlay: OverlayGeometry): overlay is google.maps.Rectangle { export function isRectangle(
overlay: OverlayGeometry
): overlay is google.maps.Rectangle {
return (overlay as google.maps.Rectangle).getBounds !== undefined; return (overlay as google.maps.Rectangle).getBounds !== undefined;
} }