From 1edaef0f8842b87446be1ea5b477a02f6c79ea1f Mon Sep 17 00:00:00 2001 From: AML - A Laycock Date: Thu, 10 Oct 2024 19:01:55 +0100 Subject: [PATCH 1/4] feat: start unique asset fields --- app/routes/app.$assetslug.$entry.edit.tsx | 28 ++++++++++++-- .../app.asset-manager.$asset.$assetfield.tsx | 37 +++++++++++++++++-- .../migration.sql | 20 ++++++++++ prisma/schema.prisma | 1 + 4 files changed, 79 insertions(+), 7 deletions(-) create mode 100644 prisma/migrations/20241009135947_add_unique_to_asset_field/migration.sql diff --git a/app/routes/app.$assetslug.$entry.edit.tsx b/app/routes/app.$assetslug.$entry.edit.tsx index 0518807..bd6d26c 100644 --- a/app/routes/app.$assetslug.$entry.edit.tsx +++ b/app/routes/app.$assetslug.$entry.edit.tsx @@ -6,7 +6,7 @@ import { redirect, unstable_parseMultipartFormData } from '@remix-run/node' -import {asyncForEach, indexedBy, invariant} from '@arcath/utils' +import {asyncMap, indexedBy, invariant} from '@arcath/utils' import {ensureUser} from '~/lib/utils/ensure-user' import {getPrisma} from '~/lib/prisma.server' @@ -80,9 +80,14 @@ export const action = async ({request, params}: ActionFunctionArgs) => { data: {aclId: acl} }) - await asyncForEach( + const results = await asyncMap( asset.assetFields, - async ({fieldId, field, id}): Promise => { + async ({ + fieldId, + field, + id, + unique + }): Promise<{error: string} | boolean> => { const entryValue = await prisma.value.findFirst({ where: {entryId: params.entry!, fieldId} }) @@ -93,6 +98,19 @@ export const action = async ({request, params}: ActionFunctionArgs) => { entryValue ? entryValue.value : '' ) + switch (unique) { + case 2: + const withinFieldCount = await prisma.value.count({ + where: {fieldId, value} + }) + if (withinFieldCount > 0) { + return {error: 'Value is not unique'} + } + case 0: + default: + break + } + if (entryValue) { if (value !== entryValue.value) { await prisma.valueHistory.create({ @@ -110,12 +128,14 @@ export const action = async ({request, params}: ActionFunctionArgs) => { data: {value} }) - return + return true } await prisma.value.create({ data: {entryId: params.entry!, fieldId, value, lastEditedById: user.id} }) + + return true } ) diff --git a/app/routes/app.asset-manager.$asset.$assetfield.tsx b/app/routes/app.asset-manager.$asset.$assetfield.tsx index 62bc2f7..a6353fe 100644 --- a/app/routes/app.asset-manager.$asset.$assetfield.tsx +++ b/app/routes/app.asset-manager.$asset.$assetfield.tsx @@ -11,7 +11,13 @@ import {invariant} from '@arcath/utils' import {ensureUser} from '~/lib/utils/ensure-user' import {getPrisma} from '~/lib/prisma.server' import {Button} from '~/lib/components/button' -import {Label, Input, HelperText, Checkbox} from '~/lib/components/input' +import { + Label, + Input, + HelperText, + Checkbox, + Select +} from '~/lib/components/input' import {pageTitle} from '~/lib/utils/page-title' export const loader = async ({request, params}: LoaderFunctionArgs) => { @@ -39,20 +45,30 @@ export const action = async ({request, params}: ActionFunctionArgs) => { const order = formData.get('order') as string | undefined const displayOnTable = formData.get('displayontable') as null | 'on' const hidden = formData.get('hidden') as null | 'on' + const unique = formData.get('unique') as string | undefined invariant(helper) invariant(order) + invariant(unique) - await prisma.assetField.update({ + const updatedAssetField = await prisma.assetField.update({ where: {id: params.assetfield}, data: { helperText: helper, order: parseInt(order), displayOnTable: displayOnTable === 'on', - hidden: hidden === 'on' + hidden: hidden === 'on', + unique: parseInt(unique) } }) + if (parseInt(unique) === 2) { + await prisma.assetField.updateMany({ + where: {fieldId: updatedAssetField.fieldId}, + data: {unique: 2} + }) + } + return redirect(`/app/asset-manager/${params.asset}`) } @@ -109,6 +125,21 @@ const AssetManagerAddFieldToAsset = () => { Hide this field from the display (not revisions) and forms. + diff --git a/prisma/migrations/20241009135947_add_unique_to_asset_field/migration.sql b/prisma/migrations/20241009135947_add_unique_to_asset_field/migration.sql new file mode 100644 index 0000000..1ba9a24 --- /dev/null +++ b/prisma/migrations/20241009135947_add_unique_to_asset_field/migration.sql @@ -0,0 +1,20 @@ +-- RedefineTables +PRAGMA defer_foreign_keys=ON; +PRAGMA foreign_keys=OFF; +CREATE TABLE "new_AssetField" ( + "id" TEXT NOT NULL PRIMARY KEY, + "order" INTEGER NOT NULL, + "helperText" TEXT NOT NULL DEFAULT '', + "displayOnTable" BOOLEAN NOT NULL DEFAULT false, + "hidden" BOOLEAN NOT NULL DEFAULT false, + "unique" INTEGER NOT NULL DEFAULT 0, + "assetId" TEXT NOT NULL, + "fieldId" TEXT NOT NULL, + CONSTRAINT "AssetField_assetId_fkey" FOREIGN KEY ("assetId") REFERENCES "Asset" ("id") ON DELETE CASCADE ON UPDATE CASCADE, + CONSTRAINT "AssetField_fieldId_fkey" FOREIGN KEY ("fieldId") REFERENCES "Field" ("id") ON DELETE CASCADE ON UPDATE CASCADE +); +INSERT INTO "new_AssetField" ("assetId", "displayOnTable", "fieldId", "helperText", "hidden", "id", "order") SELECT "assetId", "displayOnTable", "fieldId", "helperText", "hidden", "id", "order" FROM "AssetField"; +DROP TABLE "AssetField"; +ALTER TABLE "new_AssetField" RENAME TO "AssetField"; +PRAGMA foreign_keys=ON; +PRAGMA defer_foreign_keys=OFF; diff --git a/prisma/schema.prisma b/prisma/schema.prisma index bbfab9b..267af33 100644 --- a/prisma/schema.prisma +++ b/prisma/schema.prisma @@ -49,6 +49,7 @@ model AssetField { helperText String @default("") displayOnTable Boolean @default(false) hidden Boolean @default(false) + unique Int @default(0) // Unique Modes: 0 - Not, 1 - Within Asset, 2 - Unique within Field asset Asset @relation(fields: [assetId], references: [id], onDelete: Cascade) assetId String From 9d93c6c0d52a972b41c228697577adda6923c8e4 Mon Sep 17 00:00:00 2001 From: Adam Laycock Date: Fri, 11 Oct 2024 13:26:13 +0100 Subject: [PATCH 2/4] feat: unique fields --- app/routes/app.$assetslug.$entry.edit.tsx | 46 +++++++++++++- app/routes/app.$assetslug.add.tsx | 74 ++++++++++++++++++++-- prisma/sql/getUniqueCountForAssetField.sql | 13 ++++ 3 files changed, 126 insertions(+), 7 deletions(-) create mode 100644 prisma/sql/getUniqueCountForAssetField.sql diff --git a/app/routes/app.$assetslug.$entry.edit.tsx b/app/routes/app.$assetslug.$entry.edit.tsx index bd6d26c..6957d01 100644 --- a/app/routes/app.$assetslug.$entry.edit.tsx +++ b/app/routes/app.$assetslug.$entry.edit.tsx @@ -6,11 +6,12 @@ import { redirect, unstable_parseMultipartFormData } from '@remix-run/node' +import {useLoaderData, useActionData} from '@remix-run/react' import {asyncMap, indexedBy, invariant} from '@arcath/utils' +import {getUniqueCountForAssetField} from '@prisma/client/sql' import {ensureUser} from '~/lib/utils/ensure-user' import {getPrisma} from '~/lib/prisma.server' -import {useLoaderData} from '@remix-run/react' import {FIELDS} from '~/lib/fields/field' import {Button} from '~/lib/components/button' import {pageTitle} from '~/lib/utils/page-title' @@ -87,7 +88,7 @@ export const action = async ({request, params}: ActionFunctionArgs) => { field, id, unique - }): Promise<{error: string} | boolean> => { + }): Promise<{error: string; field: string} | boolean> => { const entryValue = await prisma.value.findFirst({ where: {entryId: params.entry!, fieldId} }) @@ -99,12 +100,25 @@ export const action = async ({request, params}: ActionFunctionArgs) => { ) switch (unique) { + case 1: + const [withinAssetCount] = await prisma.$queryRawTyped( + getUniqueCountForAssetField(params.entry!, fieldId, value) + ) + if (withinAssetCount['COUNT(*)'] > 0) { + return { + error: `Value is not unique across all ${asset.name}`, + field: fieldId + } + } case 2: const withinFieldCount = await prisma.value.count({ where: {fieldId, value} }) if (withinFieldCount > 0) { - return {error: 'Value is not unique'} + return { + error: 'Value is not unique across all of the documentation', + field: fieldId + } } case 0: default: @@ -139,6 +153,12 @@ export const action = async ({request, params}: ActionFunctionArgs) => { } ) + const flags = results.filter(v => v !== true) + + if (flags.length > 0) { + return json({errors: flags}) + } + return redirect(`/app/${params.assetslug}/${params.entry}`) } @@ -148,14 +168,34 @@ export const meta: MetaFunction = ({data}) => { const Asset = () => { const {entry, acls} = useLoaderData() + const actionData = useActionData() const {asset} = entry const fieldValues = indexedBy('fieldId', entry.values) + const fields = indexedBy('fieldId', asset.assetFields) return (

Edit {asset.singular}

+ {actionData && actionData.errors ? ( +
+

Save Errors

+
    + {actionData.errors + .filter(v => v !== false) + .map(({field, error}) => { + return ( +
  • + {fields[field].field.name}: {error} +
  • + ) + })} +
+
+ ) : ( + '' + )}
{asset.assetFields.map(({id, helperText, field}) => { const FieldComponent = (params: { diff --git a/app/routes/app.$assetslug.add.tsx b/app/routes/app.$assetslug.add.tsx index dac100a..f726979 100644 --- a/app/routes/app.$assetslug.add.tsx +++ b/app/routes/app.$assetslug.add.tsx @@ -6,8 +6,9 @@ import { redirect, unstable_parseMultipartFormData } from '@remix-run/node' -import {useLoaderData} from '@remix-run/react' -import {asyncForEach} from '@arcath/utils' +import {useLoaderData, useActionData} from '@remix-run/react' +import {asyncMap, indexedBy} from '@arcath/utils' +import {getUniqueCountForAssetField} from '@prisma/client/sql' import {ensureUser} from '~/lib/utils/ensure-user' import {getPrisma} from '~/lib/prisma.server' @@ -52,21 +53,64 @@ export const action = async ({request, params}: ActionFunctionArgs) => { data: {assetId: asset.id, aclId: asset.aclId} }) - await asyncForEach( + const results = await asyncMap( asset.assetFields, - async ({fieldId, field, id}): Promise => { + async ({ + fieldId, + field, + id, + unique + }): Promise<{error: string; field: string} | boolean> => { const value = await FIELD_HANDLERS[field.type].valueSetter( formData, id, '' ) + switch (unique) { + case 1: + const [withinAssetCount] = await prisma.$queryRawTyped( + getUniqueCountForAssetField(params.entry!, fieldId, value) + ) + if (withinAssetCount['COUNT(*)'] > 0) { + return { + error: `Value is not unique across all ${asset.name}`, + field: fieldId + } + } + case 2: + const withinFieldCount = await prisma.value.count({ + where: {fieldId, value} + }) + if (withinFieldCount > 0) { + return { + error: 'Value is not unique across all of the documentation', + field: fieldId + } + } + case 0: + default: + break + } + await prisma.value.create({ data: {entryId: entry.id, fieldId, value, lastEditedById: user.id} }) + + return true } ) + // If validation fails, need to delete the new entry. + const flags = results.filter(v => v !== true) + + if (flags.length > 0) { + await prisma.value.deleteMany({where: {entryId: entry.id}}) + await prisma.entry.delete({where: {id: entry.id}}) + + return json({errors: flags}) + } + return redirect(`/app/${params.assetslug}/${entry.id}`) } @@ -81,9 +125,31 @@ export const meta: MetaFunction = ({data}) => { const Asset = () => { const {asset} = useLoaderData() + const actionData = useActionData() + + const fields = indexedBy('fieldId', asset.assetFields) + return (

Add {asset.singular}

+ {actionData && actionData.errors ? ( +
+

Save Errors

+
    + {actionData.errors + .filter(v => v !== false) + .map(({field, error}) => { + return ( +
  • + {fields[field].field.name}: {error} +
  • + ) + })} +
+
+ ) : ( + '' + )} {asset.assetFields.map(({id, helperText, field}) => { const FieldComponent = (params: { diff --git a/prisma/sql/getUniqueCountForAssetField.sql b/prisma/sql/getUniqueCountForAssetField.sql new file mode 100644 index 0000000..a30971c --- /dev/null +++ b/prisma/sql/getUniqueCountForAssetField.sql @@ -0,0 +1,13 @@ +-- @param {String} $1:entryId The ID of the entry being checked +-- @param {String} $2:fieldId The ID of the field being checked +-- @param {String} $3:value The value to be checked +SELECT + COUNT(*) +FROM + Value +WHERE + Value.entryId IN (SELECT Entry.id FROM Entry WHERE Entry.assetId = (SELECT Asset.id FROM Asset WHERE Asset.id = (SELECT Entry.assetId FROM Entry WHERE Entry.id = $1))) +AND + Value.fieldId = $2 +AND + Value.value = $3 \ No newline at end of file From 842060d573f7bfa3fe62fd6494bfa9828c9e70b4 Mon Sep 17 00:00:00 2001 From: Adam Laycock Date: Fri, 11 Oct 2024 13:47:02 +0100 Subject: [PATCH 3/4] docs: document field creation. --- docs/docs/concepts/assets.md | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/docs/docs/concepts/assets.md b/docs/docs/concepts/assets.md index 60b2866..e8521d2 100644 --- a/docs/docs/concepts/assets.md +++ b/docs/docs/concepts/assets.md @@ -19,3 +19,10 @@ another asset. Assets have an icon which is any emoji. Be aware that emoji does render differently on each OS, even within versions. + +## Fields + +Any field can be added to an asset. When adding a field you set the _Helper +Text_ which is the text that will appear under the field in the editors. The +order is a numerical order for the field in the forms and display. A field can +be set to unique within the asset, or globally within the whole of Net-Doc. From 10588803afbc8f3fe29c436b6a8b79b227f6cb88 Mon Sep 17 00:00:00 2001 From: Adam Laycock Date: Fri, 11 Oct 2024 13:55:49 +0100 Subject: [PATCH 4/4] fix(lint): address lint issues --- app/routes/app.$assetslug.$entry.edit.tsx | 8 ++++++-- app/routes/app.$assetslug.add.tsx | 8 ++++++-- app/routes/app.asset-manager.$asset.$assetfield.tsx | 6 +++--- 3 files changed, 15 insertions(+), 7 deletions(-) diff --git a/app/routes/app.$assetslug.$entry.edit.tsx b/app/routes/app.$assetslug.$entry.edit.tsx index 6957d01..0055a9a 100644 --- a/app/routes/app.$assetslug.$entry.edit.tsx +++ b/app/routes/app.$assetslug.$entry.edit.tsx @@ -100,7 +100,7 @@ export const action = async ({request, params}: ActionFunctionArgs) => { ) switch (unique) { - case 1: + case 1: { const [withinAssetCount] = await prisma.$queryRawTyped( getUniqueCountForAssetField(params.entry!, fieldId, value) ) @@ -110,7 +110,9 @@ export const action = async ({request, params}: ActionFunctionArgs) => { field: fieldId } } - case 2: + break + } + case 2: { const withinFieldCount = await prisma.value.count({ where: {fieldId, value} }) @@ -120,6 +122,8 @@ export const action = async ({request, params}: ActionFunctionArgs) => { field: fieldId } } + break + } case 0: default: break diff --git a/app/routes/app.$assetslug.add.tsx b/app/routes/app.$assetslug.add.tsx index f726979..28c4dac 100644 --- a/app/routes/app.$assetslug.add.tsx +++ b/app/routes/app.$assetslug.add.tsx @@ -68,7 +68,7 @@ export const action = async ({request, params}: ActionFunctionArgs) => { ) switch (unique) { - case 1: + case 1: { const [withinAssetCount] = await prisma.$queryRawTyped( getUniqueCountForAssetField(params.entry!, fieldId, value) ) @@ -78,7 +78,9 @@ export const action = async ({request, params}: ActionFunctionArgs) => { field: fieldId } } - case 2: + break + } + case 2: { const withinFieldCount = await prisma.value.count({ where: {fieldId, value} }) @@ -88,6 +90,8 @@ export const action = async ({request, params}: ActionFunctionArgs) => { field: fieldId } } + break + } case 0: default: break diff --git a/app/routes/app.asset-manager.$asset.$assetfield.tsx b/app/routes/app.asset-manager.$asset.$assetfield.tsx index a6353fe..5822373 100644 --- a/app/routes/app.asset-manager.$asset.$assetfield.tsx +++ b/app/routes/app.asset-manager.$asset.$assetfield.tsx @@ -135,9 +135,9 @@ const AssetManagerAddFieldToAsset = () => { - Control the uniqueness of this field. If you choose "Unique within - all assets using this field" all other assets will be updated to the - same setting. + Control the uniqueness of this field. If you choose "Unique + within all assets using this field" all other assets will be + updated to the same setting.