@selvajs/local-provider 0.11.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +123 -0
- package/dist/auth/LocalAuthProvider.d.ts +28 -0
- package/dist/auth/LocalAuthProvider.d.ts.map +1 -0
- package/dist/auth/LocalAuthProvider.js +142 -0
- package/dist/auth/LocalAuthProvider.js.map +1 -0
- package/dist/auth/__tests__/conformance.test.d.ts +2 -0
- package/dist/auth/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/auth/__tests__/conformance.test.js +36 -0
- package/dist/auth/__tests__/conformance.test.js.map +1 -0
- package/dist/auth/hmac.d.ts +18 -0
- package/dist/auth/hmac.d.ts.map +1 -0
- package/dist/auth/hmac.js +41 -0
- package/dist/auth/hmac.js.map +1 -0
- package/dist/auth/index.d.ts +6 -0
- package/dist/auth/index.d.ts.map +1 -0
- package/dist/auth/index.js +4 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/users.d.ts +38 -0
- package/dist/auth/users.d.ts.map +1 -0
- package/dist/auth/users.js +100 -0
- package/dist/auth/users.js.map +1 -0
- package/dist/compute/FilesystemComputeProvider.d.ts +16 -0
- package/dist/compute/FilesystemComputeProvider.d.ts.map +1 -0
- package/dist/compute/FilesystemComputeProvider.js +51 -0
- package/dist/compute/FilesystemComputeProvider.js.map +1 -0
- package/dist/compute/SingleComputeServerProvider.d.ts +15 -0
- package/dist/compute/SingleComputeServerProvider.d.ts.map +1 -0
- package/dist/compute/SingleComputeServerProvider.js +26 -0
- package/dist/compute/SingleComputeServerProvider.js.map +1 -0
- package/dist/compute/types.d.ts +30 -0
- package/dist/compute/types.d.ts.map +1 -0
- package/dist/compute/types.js +2 -0
- package/dist/compute/types.js.map +1 -0
- package/dist/computeServer/FilesystemComputeServerStore.d.ts +14 -0
- package/dist/computeServer/FilesystemComputeServerStore.d.ts.map +1 -0
- package/dist/computeServer/FilesystemComputeServerStore.js +35 -0
- package/dist/computeServer/FilesystemComputeServerStore.js.map +1 -0
- package/dist/computeServer/LocalComputeServerProvider.d.ts +18 -0
- package/dist/computeServer/LocalComputeServerProvider.d.ts.map +1 -0
- package/dist/computeServer/LocalComputeServerProvider.js +29 -0
- package/dist/computeServer/LocalComputeServerProvider.js.map +1 -0
- package/dist/computeServer/__tests__/conformance.test.d.ts +2 -0
- package/dist/computeServer/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/computeServer/__tests__/conformance.test.js +20 -0
- package/dist/computeServer/__tests__/conformance.test.js.map +1 -0
- package/dist/computeServer/index.d.ts +2 -0
- package/dist/computeServer/index.d.ts.map +1 -0
- package/dist/computeServer/index.js +2 -0
- package/dist/computeServer/index.js.map +1 -0
- package/dist/data/LocalComputeServerStore.d.ts +72 -0
- package/dist/data/LocalComputeServerStore.d.ts.map +1 -0
- package/dist/data/LocalComputeServerStore.js +207 -0
- package/dist/data/LocalComputeServerStore.js.map +1 -0
- package/dist/data/LocalDataProvider.d.ts +47 -0
- package/dist/data/LocalDataProvider.d.ts.map +1 -0
- package/dist/data/LocalDataProvider.js +118 -0
- package/dist/data/LocalDataProvider.js.map +1 -0
- package/dist/data/LocalDefinitionMetaProvider.d.ts +22 -0
- package/dist/data/LocalDefinitionMetaProvider.d.ts.map +1 -0
- package/dist/data/LocalDefinitionMetaProvider.js +131 -0
- package/dist/data/LocalDefinitionMetaProvider.js.map +1 -0
- package/dist/data/LocalDefinitionStore.d.ts +33 -0
- package/dist/data/LocalDefinitionStore.d.ts.map +1 -0
- package/dist/data/LocalDefinitionStore.js +274 -0
- package/dist/data/LocalDefinitionStore.js.map +1 -0
- package/dist/data/LocalInviteStore.d.ts +23 -0
- package/dist/data/LocalInviteStore.d.ts.map +1 -0
- package/dist/data/LocalInviteStore.js +98 -0
- package/dist/data/LocalInviteStore.js.map +1 -0
- package/dist/data/LocalOrgStore.d.ts +67 -0
- package/dist/data/LocalOrgStore.d.ts.map +1 -0
- package/dist/data/LocalOrgStore.js +255 -0
- package/dist/data/LocalOrgStore.js.map +1 -0
- package/dist/data/LocalPlatformProjectGrantStore.d.ts +14 -0
- package/dist/data/LocalPlatformProjectGrantStore.d.ts.map +1 -0
- package/dist/data/LocalPlatformProjectGrantStore.js +62 -0
- package/dist/data/LocalPlatformProjectGrantStore.js.map +1 -0
- package/dist/data/LocalProjectStore.d.ts +30 -0
- package/dist/data/LocalProjectStore.d.ts.map +1 -0
- package/dist/data/LocalProjectStore.js +171 -0
- package/dist/data/LocalProjectStore.js.map +1 -0
- package/dist/data/LocalShareLinkStore.d.ts +39 -0
- package/dist/data/LocalShareLinkStore.d.ts.map +1 -0
- package/dist/data/LocalShareLinkStore.js +108 -0
- package/dist/data/LocalShareLinkStore.js.map +1 -0
- package/dist/data/__tests__/LocalDefinitionMetaProvider.test.d.ts +2 -0
- package/dist/data/__tests__/LocalDefinitionMetaProvider.test.d.ts.map +1 -0
- package/dist/data/__tests__/LocalDefinitionMetaProvider.test.js +21 -0
- package/dist/data/__tests__/LocalDefinitionMetaProvider.test.js.map +1 -0
- package/dist/data/__tests__/cascade.test.d.ts +2 -0
- package/dist/data/__tests__/cascade.test.d.ts.map +1 -0
- package/dist/data/__tests__/cascade.test.js +265 -0
- package/dist/data/__tests__/cascade.test.js.map +1 -0
- package/dist/data/__tests__/compute-server-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/compute-server-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/compute-server-conformance.test.js +21 -0
- package/dist/data/__tests__/compute-server-conformance.test.js.map +1 -0
- package/dist/data/__tests__/compute-server-encryption.test.d.ts +2 -0
- package/dist/data/__tests__/compute-server-encryption.test.d.ts.map +1 -0
- package/dist/data/__tests__/compute-server-encryption.test.js +131 -0
- package/dist/data/__tests__/compute-server-encryption.test.js.map +1 -0
- package/dist/data/__tests__/definition-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/definition-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/definition-conformance.test.js +20 -0
- package/dist/data/__tests__/definition-conformance.test.js.map +1 -0
- package/dist/data/__tests__/event-sink-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/event-sink-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/event-sink-conformance.test.js +24 -0
- package/dist/data/__tests__/event-sink-conformance.test.js.map +1 -0
- package/dist/data/__tests__/invite-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/invite-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/invite-conformance.test.js +21 -0
- package/dist/data/__tests__/invite-conformance.test.js.map +1 -0
- package/dist/data/__tests__/org-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/org-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/org-conformance.test.js +36 -0
- package/dist/data/__tests__/org-conformance.test.js.map +1 -0
- package/dist/data/__tests__/platform-project-grant-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/platform-project-grant-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/platform-project-grant-conformance.test.js +20 -0
- package/dist/data/__tests__/platform-project-grant-conformance.test.js.map +1 -0
- package/dist/data/__tests__/project-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/project-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/project-conformance.test.js +53 -0
- package/dist/data/__tests__/project-conformance.test.js.map +1 -0
- package/dist/data/__tests__/rules.test.d.ts +2 -0
- package/dist/data/__tests__/rules.test.d.ts.map +1 -0
- package/dist/data/__tests__/rules.test.js +484 -0
- package/dist/data/__tests__/rules.test.js.map +1 -0
- package/dist/data/__tests__/share-link-conformance.test.d.ts +2 -0
- package/dist/data/__tests__/share-link-conformance.test.d.ts.map +1 -0
- package/dist/data/__tests__/share-link-conformance.test.js +20 -0
- package/dist/data/__tests__/share-link-conformance.test.js.map +1 -0
- package/dist/data/fsJson.d.ts +12 -0
- package/dist/data/fsJson.d.ts.map +1 -0
- package/dist/data/fsJson.js +29 -0
- package/dist/data/fsJson.js.map +1 -0
- package/dist/data/index.d.ts +13 -0
- package/dist/data/index.d.ts.map +1 -0
- package/dist/data/index.js +9 -0
- package/dist/data/index.js.map +1 -0
- package/dist/data/pagination.d.ts +15 -0
- package/dist/data/pagination.d.ts.map +1 -0
- package/dist/data/pagination.js +36 -0
- package/dist/data/pagination.js.map +1 -0
- package/dist/data/secretCrypto.d.ts +23 -0
- package/dist/data/secretCrypto.d.ts.map +1 -0
- package/dist/data/secretCrypto.js +64 -0
- package/dist/data/secretCrypto.js.map +1 -0
- package/dist/data/userData.d.ts +40 -0
- package/dist/data/userData.d.ts.map +1 -0
- package/dist/data/userData.js +84 -0
- package/dist/data/userData.js.map +1 -0
- package/dist/definitions/LocalDefinitionMetaProvider.d.ts +27 -0
- package/dist/definitions/LocalDefinitionMetaProvider.d.ts.map +1 -0
- package/dist/definitions/LocalDefinitionMetaProvider.js +188 -0
- package/dist/definitions/LocalDefinitionMetaProvider.js.map +1 -0
- package/dist/definitions/__tests__/conformance.test.d.ts +2 -0
- package/dist/definitions/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/definitions/__tests__/conformance.test.js +20 -0
- package/dist/definitions/__tests__/conformance.test.js.map +1 -0
- package/dist/definitions/index.d.ts +2 -0
- package/dist/definitions/index.d.ts.map +1 -0
- package/dist/definitions/index.js +2 -0
- package/dist/definitions/index.js.map +1 -0
- package/dist/definitions/providers/filesystem-files.d.ts +24 -0
- package/dist/definitions/providers/filesystem-files.d.ts.map +1 -0
- package/dist/definitions/providers/filesystem-files.js +170 -0
- package/dist/definitions/providers/filesystem-files.js.map +1 -0
- package/dist/definitions/providers/filesystem-meta.d.ts +17 -0
- package/dist/definitions/providers/filesystem-meta.d.ts.map +1 -0
- package/dist/definitions/providers/filesystem-meta.js +216 -0
- package/dist/definitions/providers/filesystem-meta.js.map +1 -0
- package/dist/fsJson.d.ts +12 -0
- package/dist/fsJson.d.ts.map +1 -0
- package/dist/fsJson.js +29 -0
- package/dist/fsJson.js.map +1 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +16 -0
- package/dist/index.js.map +1 -0
- package/dist/invites/LocalInviteProvider.d.ts +24 -0
- package/dist/invites/LocalInviteProvider.d.ts.map +1 -0
- package/dist/invites/LocalInviteProvider.js +89 -0
- package/dist/invites/LocalInviteProvider.js.map +1 -0
- package/dist/invites/__tests__/conformance.test.d.ts +2 -0
- package/dist/invites/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/invites/__tests__/conformance.test.js +21 -0
- package/dist/invites/__tests__/conformance.test.js.map +1 -0
- package/dist/organizations/LocalOrganizationProvider.d.ts +41 -0
- package/dist/organizations/LocalOrganizationProvider.d.ts.map +1 -0
- package/dist/organizations/LocalOrganizationProvider.js +198 -0
- package/dist/organizations/LocalOrganizationProvider.js.map +1 -0
- package/dist/organizations/__tests__/conformance.test.d.ts +2 -0
- package/dist/organizations/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/organizations/__tests__/conformance.test.js +20 -0
- package/dist/organizations/__tests__/conformance.test.js.map +1 -0
- package/dist/organizations/index.d.ts +2 -0
- package/dist/organizations/index.d.ts.map +1 -0
- package/dist/organizations/index.js +2 -0
- package/dist/organizations/index.js.map +1 -0
- package/dist/pagination.d.ts +15 -0
- package/dist/pagination.d.ts.map +1 -0
- package/dist/pagination.js +36 -0
- package/dist/pagination.js.map +1 -0
- package/dist/permissions/LocalPlatformPermissionStore.d.ts +39 -0
- package/dist/permissions/LocalPlatformPermissionStore.d.ts.map +1 -0
- package/dist/permissions/LocalPlatformPermissionStore.js +117 -0
- package/dist/permissions/LocalPlatformPermissionStore.js.map +1 -0
- package/dist/permissions/__tests__/conformance.test.d.ts +2 -0
- package/dist/permissions/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/permissions/__tests__/conformance.test.js +37 -0
- package/dist/permissions/__tests__/conformance.test.js.map +1 -0
- package/dist/permissions/index.d.ts +2 -0
- package/dist/permissions/index.d.ts.map +1 -0
- package/dist/permissions/index.js +2 -0
- package/dist/permissions/index.js.map +1 -0
- package/dist/projects/LocalProjectProvider.d.ts +21 -0
- package/dist/projects/LocalProjectProvider.d.ts.map +1 -0
- package/dist/projects/LocalProjectProvider.js +125 -0
- package/dist/projects/LocalProjectProvider.js.map +1 -0
- package/dist/projects/__tests__/conformance.test.d.ts +2 -0
- package/dist/projects/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/projects/__tests__/conformance.test.js +44 -0
- package/dist/projects/__tests__/conformance.test.js.map +1 -0
- package/dist/projects/index.d.ts +2 -0
- package/dist/projects/index.d.ts.map +1 -0
- package/dist/projects/index.js +2 -0
- package/dist/projects/index.js.map +1 -0
- package/dist/storage/LocalStorageProvider.d.ts +27 -0
- package/dist/storage/LocalStorageProvider.d.ts.map +1 -0
- package/dist/storage/LocalStorageProvider.js +74 -0
- package/dist/storage/LocalStorageProvider.js.map +1 -0
- package/dist/storage/__tests__/conformance.test.d.ts +2 -0
- package/dist/storage/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/storage/__tests__/conformance.test.js +20 -0
- package/dist/storage/__tests__/conformance.test.js.map +1 -0
- package/dist/storage/index.d.ts +2 -0
- package/dist/storage/index.d.ts.map +1 -0
- package/dist/storage/index.js +2 -0
- package/dist/storage/index.js.map +1 -0
- package/dist/userProfile/LocalUserProfileProvider.d.ts +25 -0
- package/dist/userProfile/LocalUserProfileProvider.d.ts.map +1 -0
- package/dist/userProfile/LocalUserProfileProvider.js +110 -0
- package/dist/userProfile/LocalUserProfileProvider.js.map +1 -0
- package/dist/userProfile/__tests__/conformance.test.d.ts +2 -0
- package/dist/userProfile/__tests__/conformance.test.d.ts.map +1 -0
- package/dist/userProfile/__tests__/conformance.test.js +40 -0
- package/dist/userProfile/__tests__/conformance.test.js.map +1 -0
- package/dist/userProfile/index.d.ts +2 -0
- package/dist/userProfile/index.d.ts.map +1 -0
- package/dist/userProfile/index.js +2 -0
- package/dist/userProfile/index.js.map +1 -0
- package/package.json +70 -0
- package/src/README.md +37 -0
- package/src/auth/LocalAuthProvider.ts +165 -0
- package/src/auth/__tests__/conformance.test.ts +40 -0
- package/src/auth/hmac.ts +53 -0
- package/src/auth/index.ts +5 -0
- package/src/auth/users.ts +151 -0
- package/src/data/LocalComputeServerStore.ts +290 -0
- package/src/data/LocalDataProvider.ts +148 -0
- package/src/data/LocalDefinitionStore.ts +369 -0
- package/src/data/LocalInviteStore.ts +117 -0
- package/src/data/LocalOrgStore.ts +356 -0
- package/src/data/LocalPlatformProjectGrantStore.ts +85 -0
- package/src/data/LocalProjectStore.ts +274 -0
- package/src/data/LocalShareLinkStore.ts +138 -0
- package/src/data/__tests__/cascade.test.ts +300 -0
- package/src/data/__tests__/compute-server-conformance.test.ts +26 -0
- package/src/data/__tests__/compute-server-encryption.test.ts +185 -0
- package/src/data/__tests__/definition-conformance.test.ts +23 -0
- package/src/data/__tests__/event-sink-conformance.test.ts +28 -0
- package/src/data/__tests__/invite-conformance.test.ts +24 -0
- package/src/data/__tests__/org-conformance.test.ts +43 -0
- package/src/data/__tests__/platform-project-grant-conformance.test.ts +24 -0
- package/src/data/__tests__/project-conformance.test.ts +64 -0
- package/src/data/__tests__/rules.test.ts +682 -0
- package/src/data/__tests__/share-link-conformance.test.ts +23 -0
- package/src/data/fsJson.ts +28 -0
- package/src/data/index.ts +16 -0
- package/src/data/pagination.ts +48 -0
- package/src/data/secretCrypto.ts +69 -0
- package/src/data/userData.ts +134 -0
- package/src/index.ts +42 -0
- package/src/permissions/LocalPlatformPermissionStore.ts +129 -0
- package/src/permissions/__tests__/conformance.test.ts +40 -0
- package/src/permissions/index.ts +1 -0
- package/src/storage/LocalStorageProvider.ts +78 -0
- package/src/storage/__tests__/conformance.test.ts +23 -0
- package/src/storage/index.ts +1 -0
- package/src/userProfile/LocalUserProfileProvider.ts +135 -0
- package/src/userProfile/__tests__/conformance.test.ts +43 -0
- package/src/userProfile/index.ts +1 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.js","sourceRoot":"","sources":["../../../src/permissions/__tests__/conformance.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzD,OAAO,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACvC,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAClC,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,EAAE,qCAAqC,EAAE,MAAM,2BAA2B,CAAC;AAClF,OAAO,EAAE,4BAA4B,EAAE,MAAM,oCAAoC,CAAC;AAClF,OAAO,EAAE,wBAAwB,EAAE,MAAM,wBAAwB,CAAC;AAClE,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC,QAAQ,CAAC,8BAA8B,EAAE,GAAG,EAAE;IAC7C,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,KAAK,IAAI,EAAE;QACrB,OAAO,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,kBAAkB,CAAC,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,KAAK,IAAI,EAAE;QACpB,MAAM,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC;IAEH,qCAAqC,CAAC;QACrC,IAAI,EAAE,8BAA8B;QACpC,WAAW,EAAE,KAAK,IAAI,EAAE;YACvB,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,gBAAgB,CAAC,CAAC;YAC1D,MAAM,QAAQ,GAAG,wBAAwB,CAAC,YAAY,CAAC,CAAC;YACxD,MAAM,KAAK,GAAG,IAAI,4BAA4B,CAAC,YAAY,CAAC,CAAC;YAC7D,OAAO;gBACN,KAAK;gBACL,QAAQ,EAAE,KAAK,IAAI,EAAE;oBACpB,gEAAgE;oBAChE,6DAA6D;oBAC7D,iCAAiC;oBACjC,MAAM,EAAE,GAAG,UAAU,EAAE,CAAC;oBACxB,MAAM,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBAC1B,OAAO,EAAE,CAAC;gBACX,CAAC;aACD,CAAC;QACH,CAAC;KACD,CAAC,CAAC;AACJ,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/permissions/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,4BAA4B,EAAE,MAAM,mCAAmC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/permissions/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,4BAA4B,EAAE,MAAM,mCAAmC,CAAC"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { IProjectStore, Project, ProjectRole, ProjectMember, RequestContext, ListOptions, Page } from '@selva/platform';
|
|
2
|
+
import type { LocalOrgStoreLoader } from '../organizations/LocalOrganizationProvider.js';
|
|
3
|
+
export declare class LocalProjectProvider implements IProjectStore {
|
|
4
|
+
private readonly loader;
|
|
5
|
+
constructor(loader: LocalOrgStoreLoader);
|
|
6
|
+
listProjects(_ctx: RequestContext, orgId: string, opts?: ListOptions): Promise<Page<Project>>;
|
|
7
|
+
getProject(_ctx: RequestContext, id: string): Promise<Project | null>;
|
|
8
|
+
getProjectBySlug(_ctx: RequestContext, orgId: string, slug: string): Promise<Project | null>;
|
|
9
|
+
createProject(_ctx: RequestContext, project: Project): Promise<void>;
|
|
10
|
+
updateProject(_ctx: RequestContext, id: string, patch: Partial<Pick<Project, 'name' | 'slug' | 'description' | 'visibility'>>): Promise<void>;
|
|
11
|
+
deleteProject(_ctx: RequestContext, id: string): Promise<void>;
|
|
12
|
+
listProjectMembers(_ctx: RequestContext, projectId: string, opts?: ListOptions): Promise<Page<ProjectMember>>;
|
|
13
|
+
getProjectMember(_ctx: RequestContext, projectId: string, userId: string): Promise<ProjectMember | null>;
|
|
14
|
+
addProjectMember(_ctx: RequestContext, member: ProjectMember): Promise<void>;
|
|
15
|
+
updateProjectMemberRole(_ctx: RequestContext, projectId: string, userId: string, role: ProjectRole): Promise<void>;
|
|
16
|
+
removeProjectMember(_ctx: RequestContext, projectId: string, userId: string): Promise<void>;
|
|
17
|
+
canEdit(ctx: RequestContext, projectId: string): Promise<boolean>;
|
|
18
|
+
canManage(ctx: RequestContext, projectId: string): Promise<boolean>;
|
|
19
|
+
canEditProjectSettings(ctx: RequestContext, projectId: string): Promise<boolean>;
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=LocalProjectProvider.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalProjectProvider.d.ts","sourceRoot":"","sources":["../../src/projects/LocalProjectProvider.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACX,aAAa,EACb,OAAO,EACP,WAAW,EACX,aAAa,EACb,cAAc,EACd,WAAW,EACX,IAAI,EACJ,MAAM,iBAAiB,CAAC;AAGzB,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,+CAA+C,CAAC;AAEzF,qBAAa,oBAAqB,YAAW,aAAa;IACzD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAsB;gBAEjC,MAAM,EAAE,mBAAmB;IAMjC,YAAY,CACjB,IAAI,EAAE,cAAc,EACpB,KAAK,EAAE,MAAM,EACb,IAAI,CAAC,EAAE,WAAW,GAChB,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAKnB,UAAU,CAAC,IAAI,EAAE,cAAc,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,GAAG,IAAI,CAAC;IAKrE,gBAAgB,CACrB,IAAI,EAAE,cAAc,EACpB,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,MAAM,GACV,OAAO,CAAC,OAAO,GAAG,IAAI,CAAC;IAKpB,aAAa,CAAC,IAAI,EAAE,cAAc,EAAE,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;IAepE,aAAa,CAClB,IAAI,EAAE,cAAc,EACpB,EAAE,EAAE,MAAM,EACV,KAAK,EAAE,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,aAAa,GAAG,YAAY,CAAC,CAAC,GAC3E,OAAO,CAAC,IAAI,CAAC;IAgBV,aAAa,CAAC,IAAI,EAAE,cAAc,EAAE,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAW9D,kBAAkB,CACvB,IAAI,EAAE,cAAc,EACpB,SAAS,EAAE,MAAM,EACjB,IAAI,CAAC,EAAE,WAAW,GAChB,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAKzB,gBAAgB,CACrB,IAAI,EAAE,cAAc,EACpB,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,MAAM,GACZ,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC;IAK1B,gBAAgB,CAAC,IAAI,EAAE,cAAc,EAAE,MAAM,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC;IAM5E,uBAAuB,CAC5B,IAAI,EAAE,cAAc,EACpB,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,MAAM,EACd,IAAI,EAAE,WAAW,GACf,OAAO,CAAC,IAAI,CAAC;IAQV,mBAAmB,CACxB,IAAI,EAAE,cAAc,EACpB,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,MAAM,GACZ,OAAO,CAAC,IAAI,CAAC;IAWV,OAAO,CAAC,GAAG,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAgBjE,SAAS,CAAC,GAAG,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAOnE,sBAAsB,CAAC,GAAG,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;CAWtF"}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import { ProviderError, hasPermission } from '@selva/platform';
|
|
2
|
+
import { paginate, applyOrder } from '../pagination.js';
|
|
3
|
+
export class LocalProjectProvider {
|
|
4
|
+
loader;
|
|
5
|
+
constructor(loader) {
|
|
6
|
+
this.loader = loader;
|
|
7
|
+
}
|
|
8
|
+
// ── Projects ─────────────────────────────────────────────────────────────────
|
|
9
|
+
async listProjects(_ctx, orgId, opts) {
|
|
10
|
+
const { projects } = await this.loader.get();
|
|
11
|
+
return paginate(applyOrder(projects.filter((p) => p.orgId === orgId), opts), opts);
|
|
12
|
+
}
|
|
13
|
+
async getProject(_ctx, id) {
|
|
14
|
+
const { projects } = await this.loader.get();
|
|
15
|
+
return projects.find((p) => p.id === id) ?? null;
|
|
16
|
+
}
|
|
17
|
+
async getProjectBySlug(_ctx, orgId, slug) {
|
|
18
|
+
const { projects } = await this.loader.get();
|
|
19
|
+
return projects.find((p) => p.orgId === orgId && p.slug === slug) ?? null;
|
|
20
|
+
}
|
|
21
|
+
async createProject(_ctx, project) {
|
|
22
|
+
const store = await this.loader.get();
|
|
23
|
+
if (!store.orgs.some((o) => o.id === project.orgId)) {
|
|
24
|
+
throw new ProviderError(`Org '${project.orgId}' not found`, 404);
|
|
25
|
+
}
|
|
26
|
+
if (store.projects.some((p) => p.id === project.id)) {
|
|
27
|
+
throw new ProviderError(`Project '${project.id}' already exists`, 409);
|
|
28
|
+
}
|
|
29
|
+
if (store.projects.some((p) => p.orgId === project.orgId && p.slug === project.slug)) {
|
|
30
|
+
throw new ProviderError(`Project slug '${project.slug}' already in use`, 409);
|
|
31
|
+
}
|
|
32
|
+
store.projects.push(project);
|
|
33
|
+
await this.loader.write(store);
|
|
34
|
+
}
|
|
35
|
+
async updateProject(_ctx, id, patch) {
|
|
36
|
+
const store = await this.loader.get();
|
|
37
|
+
const idx = store.projects.findIndex((p) => p.id === id);
|
|
38
|
+
if (idx === -1)
|
|
39
|
+
throw new ProviderError(`Project '${id}' not found`, 404);
|
|
40
|
+
if (patch.slug && patch.slug !== store.projects[idx].slug) {
|
|
41
|
+
const orgId = store.projects[idx].orgId;
|
|
42
|
+
if (store.projects.some((p) => p.orgId === orgId && p.slug === patch.slug && p.id !== id)) {
|
|
43
|
+
throw new ProviderError(`Project slug '${patch.slug}' already in use`, 409);
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
store.projects[idx] = { ...store.projects[idx], ...patch, updatedAt: new Date().toISOString() };
|
|
47
|
+
await this.loader.write(store);
|
|
48
|
+
}
|
|
49
|
+
async deleteProject(_ctx, id) {
|
|
50
|
+
const store = await this.loader.get();
|
|
51
|
+
const idx = store.projects.findIndex((p) => p.id === id);
|
|
52
|
+
if (idx === -1)
|
|
53
|
+
throw new ProviderError(`Project '${id}' not found`, 404);
|
|
54
|
+
store.projects.splice(idx, 1);
|
|
55
|
+
store.projectMembers = store.projectMembers.filter((m) => m.projectId !== id);
|
|
56
|
+
await this.loader.write(store);
|
|
57
|
+
}
|
|
58
|
+
// ── Project members ──────────────────────────────────────────────────────────
|
|
59
|
+
async listProjectMembers(_ctx, projectId, opts) {
|
|
60
|
+
const { projectMembers } = await this.loader.get();
|
|
61
|
+
return paginate(projectMembers.filter((m) => m.projectId === projectId), opts);
|
|
62
|
+
}
|
|
63
|
+
async getProjectMember(_ctx, projectId, userId) {
|
|
64
|
+
const { projectMembers } = await this.loader.get();
|
|
65
|
+
return projectMembers.find((m) => m.projectId === projectId && m.userId === userId) ?? null;
|
|
66
|
+
}
|
|
67
|
+
async addProjectMember(_ctx, member) {
|
|
68
|
+
const store = await this.loader.get();
|
|
69
|
+
store.projectMembers.push(member);
|
|
70
|
+
await this.loader.write(store);
|
|
71
|
+
}
|
|
72
|
+
async updateProjectMemberRole(_ctx, projectId, userId, role) {
|
|
73
|
+
const store = await this.loader.get();
|
|
74
|
+
const m = store.projectMembers.find((m) => m.projectId === projectId && m.userId === userId);
|
|
75
|
+
if (!m)
|
|
76
|
+
throw new ProviderError(`Project member '${userId}' not found`, 404);
|
|
77
|
+
m.role = role;
|
|
78
|
+
await this.loader.write(store);
|
|
79
|
+
}
|
|
80
|
+
async removeProjectMember(_ctx, projectId, userId) {
|
|
81
|
+
const store = await this.loader.get();
|
|
82
|
+
store.projectMembers = store.projectMembers.filter((m) => !(m.projectId === projectId && m.userId === userId));
|
|
83
|
+
await this.loader.write(store);
|
|
84
|
+
}
|
|
85
|
+
// ── Access checks ─────────────────────────────────────────────────────────────
|
|
86
|
+
// UI gating only — the mutating methods above are the real security boundary.
|
|
87
|
+
async canEdit(ctx, projectId) {
|
|
88
|
+
if (hasPermission(ctx, 'platform_admin'))
|
|
89
|
+
return true;
|
|
90
|
+
const { projectMembers, projects, orgMembers } = await this.loader.get();
|
|
91
|
+
const member = projectMembers.find((m) => m.projectId === projectId && m.userId === ctx.userId);
|
|
92
|
+
if (member?.role === 'owner' || member?.role === 'editor')
|
|
93
|
+
return true;
|
|
94
|
+
// manage_definitions permission allows editing public projects in orgs where user is a member
|
|
95
|
+
if (hasPermission(ctx, 'manage_definitions')) {
|
|
96
|
+
const project = projects.find((p) => p.id === projectId);
|
|
97
|
+
if (project?.visibility === 'public') {
|
|
98
|
+
const isOrgMember = orgMembers.some((m) => m.orgId === project.orgId && m.userId === ctx.userId);
|
|
99
|
+
return isOrgMember;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return false;
|
|
103
|
+
}
|
|
104
|
+
async canManage(ctx, projectId) {
|
|
105
|
+
if (hasPermission(ctx, 'platform_admin'))
|
|
106
|
+
return true;
|
|
107
|
+
const { projectMembers } = await this.loader.get();
|
|
108
|
+
const member = projectMembers.find((m) => m.projectId === projectId && m.userId === ctx.userId);
|
|
109
|
+
return member?.role === 'owner';
|
|
110
|
+
}
|
|
111
|
+
async canEditProjectSettings(ctx, projectId) {
|
|
112
|
+
if (hasPermission(ctx, 'platform_admin'))
|
|
113
|
+
return true;
|
|
114
|
+
const { projectMembers } = await this.loader.get();
|
|
115
|
+
const member = projectMembers.find((m) => m.projectId === projectId && m.userId === ctx.userId);
|
|
116
|
+
// project owners can always edit
|
|
117
|
+
if (member?.role === 'owner')
|
|
118
|
+
return true;
|
|
119
|
+
// manage_definitions permission allows editing project settings if they're an editor
|
|
120
|
+
if (hasPermission(ctx, 'manage_definitions') && member?.role === 'editor')
|
|
121
|
+
return true;
|
|
122
|
+
return false;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
//# sourceMappingURL=LocalProjectProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalProjectProvider.js","sourceRoot":"","sources":["../../src/projects/LocalProjectProvider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.d.ts","sourceRoot":"","sources":["../../../src/projects/__tests__/conformance.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { describe, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import * as fs from 'node:fs/promises';
|
|
3
|
+
import * as path from 'node:path';
|
|
4
|
+
import * as os from 'node:os';
|
|
5
|
+
import { randomUUID } from 'node:crypto';
|
|
6
|
+
import { runProjectStoreConformance } from '@selva/platform/testing';
|
|
7
|
+
import { ALL_PLATFORM_PERMISSIONS, ALL_ORG_PERMISSIONS } from '@selva/platform';
|
|
8
|
+
import { LocalOrgStoreLoader, LocalOrganizationProvider } from '../../organizations/LocalOrganizationProvider.js';
|
|
9
|
+
import { LocalProjectProvider } from '../LocalProjectProvider.js';
|
|
10
|
+
describe('LocalProjectProvider', () => {
|
|
11
|
+
let tempDir;
|
|
12
|
+
beforeEach(async () => {
|
|
13
|
+
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'selva-test-'));
|
|
14
|
+
});
|
|
15
|
+
afterEach(async () => {
|
|
16
|
+
await fs.rm(tempDir, { recursive: true, force: true });
|
|
17
|
+
});
|
|
18
|
+
runProjectStoreConformance({
|
|
19
|
+
name: 'LocalProjectProvider',
|
|
20
|
+
createStore: async () => {
|
|
21
|
+
const loader = new LocalOrgStoreLoader(tempDir);
|
|
22
|
+
const orgs = new LocalOrganizationProvider(loader);
|
|
23
|
+
const store = new LocalProjectProvider(loader);
|
|
24
|
+
// Explicitly create the host org for the project tests.
|
|
25
|
+
const ownerId = randomUUID();
|
|
26
|
+
const orgId = randomUUID();
|
|
27
|
+
const now = new Date().toISOString();
|
|
28
|
+
await orgs.createOrg({
|
|
29
|
+
userId: ownerId,
|
|
30
|
+
platformPermissions: [...ALL_PLATFORM_PERMISSIONS],
|
|
31
|
+
orgPermissions: [...ALL_ORG_PERMISSIONS]
|
|
32
|
+
}, {
|
|
33
|
+
id: orgId,
|
|
34
|
+
name: 'Test Org',
|
|
35
|
+
slug: 'test',
|
|
36
|
+
ownerId,
|
|
37
|
+
createdAt: now,
|
|
38
|
+
updatedAt: now
|
|
39
|
+
});
|
|
40
|
+
return { store, orgId, ownerId };
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
//# sourceMappingURL=conformance.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.js","sourceRoot":"","sources":["../../../src/projects/__tests__/conformance.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzD,OAAO,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACvC,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAClC,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,0BAA0B,EAAE,MAAM,yBAAyB,CAAC;AACrE,OAAO,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAC;AAChF,OAAO,EACN,mBAAmB,EACnB,yBAAyB,EACzB,MAAM,kDAAkD,CAAC;AAC1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,4BAA4B,CAAC;AAElE,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;IACrC,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,KAAK,IAAI,EAAE;QACrB,OAAO,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,aAAa,CAAC,CAAC,CAAC;IACnE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,KAAK,IAAI,EAAE;QACpB,MAAM,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC;IAEH,0BAA0B,CAAC;QAC1B,IAAI,EAAE,sBAAsB;QAC5B,WAAW,EAAE,KAAK,IAAI,EAAE;YACvB,MAAM,MAAM,GAAG,IAAI,mBAAmB,CAAC,OAAO,CAAC,CAAC;YAChD,MAAM,IAAI,GAAG,IAAI,yBAAyB,CAAC,MAAM,CAAC,CAAC;YACnD,MAAM,KAAK,GAAG,IAAI,oBAAoB,CAAC,MAAM,CAAC,CAAC;YAC/C,wDAAwD;YACxD,MAAM,OAAO,GAAG,UAAU,EAAE,CAAC;YAC7B,MAAM,KAAK,GAAG,UAAU,EAAE,CAAC;YAC3B,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YACrC,MAAM,IAAI,CAAC,SAAS,CACnB;gBACC,MAAM,EAAE,OAAO;gBACf,mBAAmB,EAAE,CAAC,GAAG,wBAAwB,CAAC;gBAClD,cAAc,EAAE,CAAC,GAAG,mBAAmB,CAAC;aACxC,EACD;gBACC,EAAE,EAAE,KAAK;gBACT,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,MAAM;gBACZ,OAAO;gBACP,SAAS,EAAE,GAAG;gBACd,SAAS,EAAE,GAAG;aACd,CACD,CAAC;YACF,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC;QAClC,CAAC;KACD,CAAC,CAAC;AACJ,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/projects/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/projects/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import type { IStorageProvider } from '@selvajs/platform/storage';
|
|
2
|
+
/**
|
|
3
|
+
* Filesystem implementation of IStorageProvider.
|
|
4
|
+
* Files are stored under DATA_PATH/{path}.
|
|
5
|
+
* Images are auto-transcoded to WebP on put() via the shared platform helper
|
|
6
|
+
* (`transcodeImageIfNeeded`) — the same helper the Supabase adapter uses,
|
|
7
|
+
* so both providers produce identical bytes for the same upload.
|
|
8
|
+
*/
|
|
9
|
+
export declare class LocalStorageProvider implements IStorageProvider {
|
|
10
|
+
private readonly basePath;
|
|
11
|
+
private readonly publicUrlBase;
|
|
12
|
+
static fromEnv(env: Record<string, string | undefined>): LocalStorageProvider;
|
|
13
|
+
constructor(basePath: string, publicUrlBase?: string);
|
|
14
|
+
/**
|
|
15
|
+
* Resolve a caller-provided path under basePath, rejecting anything that
|
|
16
|
+
* would escape the root. Last line of defense against traversal — while
|
|
17
|
+
* platform-side helpers (definitionPaths) also assert safe keys, this
|
|
18
|
+
* adapter is reached by any IStorageProvider caller.
|
|
19
|
+
*/
|
|
20
|
+
private resolvePath;
|
|
21
|
+
get(storagePath: string): Promise<Uint8Array | null>;
|
|
22
|
+
put(storagePath: string, data: Uint8Array, contentType?: string): Promise<void>;
|
|
23
|
+
delete(storagePath: string): Promise<void>;
|
|
24
|
+
deletePrefix(prefix: string): Promise<void>;
|
|
25
|
+
getPublicUrl(storagePath: string): string;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=LocalStorageProvider.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalStorageProvider.d.ts","sourceRoot":"","sources":["../../src/storage/LocalStorageProvider.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAElE;;;;;;GAMG;AACH,qBAAa,oBAAqB,YAAW,gBAAgB;IAC5D,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;IAClC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAS;IAEvC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,GAAG,oBAAoB;gBAKjE,QAAQ,EAAE,MAAM,EAAE,aAAa,SAAe;IAK1D;;;;;OAKG;IACH,OAAO,CAAC,WAAW;IASb,GAAG,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC;IAUpD,GAAG,CAAC,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,UAAU,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAU/E,MAAM,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAQ1C,YAAY,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAKjD,YAAY,CAAC,WAAW,EAAE,MAAM,GAAG,MAAM;CAGzC"}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import * as fs from 'node:fs/promises';
|
|
2
|
+
import * as path from 'node:path';
|
|
3
|
+
import { transcodeImageIfNeeded } from '@selvajs/platform/storage';
|
|
4
|
+
/**
|
|
5
|
+
* Filesystem implementation of IStorageProvider.
|
|
6
|
+
* Files are stored under DATA_PATH/{path}.
|
|
7
|
+
* Images are auto-transcoded to WebP on put() via the shared platform helper
|
|
8
|
+
* (`transcodeImageIfNeeded`) — the same helper the Supabase adapter uses,
|
|
9
|
+
* so both providers produce identical bytes for the same upload.
|
|
10
|
+
*/
|
|
11
|
+
export class LocalStorageProvider {
|
|
12
|
+
basePath;
|
|
13
|
+
publicUrlBase;
|
|
14
|
+
static fromEnv(env) {
|
|
15
|
+
if (!env.DATA_PATH)
|
|
16
|
+
throw new Error('Missing required env var: DATA_PATH');
|
|
17
|
+
return new LocalStorageProvider(env.DATA_PATH, '/api/files');
|
|
18
|
+
}
|
|
19
|
+
constructor(basePath, publicUrlBase = '/api/files') {
|
|
20
|
+
this.basePath = basePath;
|
|
21
|
+
this.publicUrlBase = publicUrlBase;
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Resolve a caller-provided path under basePath, rejecting anything that
|
|
25
|
+
* would escape the root. Last line of defense against traversal — while
|
|
26
|
+
* platform-side helpers (definitionPaths) also assert safe keys, this
|
|
27
|
+
* adapter is reached by any IStorageProvider caller.
|
|
28
|
+
*/
|
|
29
|
+
resolvePath(storagePath) {
|
|
30
|
+
const base = path.resolve(this.basePath);
|
|
31
|
+
const full = path.resolve(base, storagePath);
|
|
32
|
+
if (full !== base && !full.startsWith(base + path.sep)) {
|
|
33
|
+
throw new Error(`Path escapes base: ${storagePath}`);
|
|
34
|
+
}
|
|
35
|
+
return full;
|
|
36
|
+
}
|
|
37
|
+
async get(storagePath) {
|
|
38
|
+
try {
|
|
39
|
+
const buffer = await fs.readFile(this.resolvePath(storagePath));
|
|
40
|
+
return new Uint8Array(buffer);
|
|
41
|
+
}
|
|
42
|
+
catch (err) {
|
|
43
|
+
if (err.code === 'ENOENT')
|
|
44
|
+
return null;
|
|
45
|
+
throw err;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
async put(storagePath, data, contentType) {
|
|
49
|
+
// Normalize images through the shared transcoder — rewrites `.png` → `.webp`,
|
|
50
|
+
// caps dimensions, and re-encodes at the platform's canonical quality.
|
|
51
|
+
// Non-images pass through untouched.
|
|
52
|
+
const transcoded = await transcodeImageIfNeeded(data, contentType, storagePath);
|
|
53
|
+
const fullPath = this.resolvePath(transcoded.path);
|
|
54
|
+
await fs.mkdir(path.dirname(fullPath), { recursive: true });
|
|
55
|
+
await fs.writeFile(fullPath, Buffer.from(transcoded.data));
|
|
56
|
+
}
|
|
57
|
+
async delete(storagePath) {
|
|
58
|
+
try {
|
|
59
|
+
await fs.unlink(this.resolvePath(storagePath));
|
|
60
|
+
}
|
|
61
|
+
catch (err) {
|
|
62
|
+
if (err.code !== 'ENOENT')
|
|
63
|
+
throw err;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
async deletePrefix(prefix) {
|
|
67
|
+
const fullPath = this.resolvePath(prefix);
|
|
68
|
+
await fs.rm(fullPath, { recursive: true, force: true });
|
|
69
|
+
}
|
|
70
|
+
getPublicUrl(storagePath) {
|
|
71
|
+
return `${this.publicUrlBase}/${storagePath}`;
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
//# sourceMappingURL=LocalStorageProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalStorageProvider.js","sourceRoot":"","sources":["../../src/storage/LocalStorageProvider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.d.ts","sourceRoot":"","sources":["../../../src/storage/__tests__/conformance.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { describe, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import * as fs from 'node:fs/promises';
|
|
3
|
+
import * as path from 'node:path';
|
|
4
|
+
import * as os from 'node:os';
|
|
5
|
+
import { runStorageProviderConformance } from '@selvajs/platform/testing';
|
|
6
|
+
import { LocalStorageProvider } from '../LocalStorageProvider.js';
|
|
7
|
+
describe('LocalStorageProvider', () => {
|
|
8
|
+
let tempDir;
|
|
9
|
+
beforeEach(async () => {
|
|
10
|
+
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'selva-storage-test-'));
|
|
11
|
+
});
|
|
12
|
+
afterEach(async () => {
|
|
13
|
+
await fs.rm(tempDir, { recursive: true, force: true });
|
|
14
|
+
});
|
|
15
|
+
runStorageProviderConformance({
|
|
16
|
+
name: 'LocalStorageProvider',
|
|
17
|
+
createStorage: () => new LocalStorageProvider(tempDir)
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
//# sourceMappingURL=conformance.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.js","sourceRoot":"","sources":["../../../src/storage/__tests__/conformance.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzD,OAAO,KAAK,EAAE,MAAM,kBAAkB,CAAC;AACvC,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAClC,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,EAAE,6BAA6B,EAAE,MAAM,2BAA2B,CAAC;AAC1E,OAAO,EAAE,oBAAoB,EAAE,MAAM,4BAA4B,CAAC;AAElE,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;IACrC,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,KAAK,IAAI,EAAE;QACrB,OAAO,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,qBAAqB,CAAC,CAAC,CAAC;IAC3E,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,KAAK,IAAI,EAAE;QACpB,MAAM,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC;IAEH,6BAA6B,CAAC;QAC7B,IAAI,EAAE,sBAAsB;QAC5B,aAAa,EAAE,GAAG,EAAE,CAAC,IAAI,oBAAoB,CAAC,OAAO,CAAC;KACtD,CAAC,CAAC;AACJ,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import type { IUserProfileStore, RecentRun, RequestContext, UserManagementResult, UserProfile } from '@selvajs/platform';
|
|
2
|
+
/**
|
|
3
|
+
* Filesystem-backed user-profile store. Reads and writes `user-data.json`,
|
|
4
|
+
* shared with `LocalPlatformPermissionStore`. Identity (email, password
|
|
5
|
+
* hash) lives separately on the auth provider — this store keys exclusively
|
|
6
|
+
* by user ID.
|
|
7
|
+
*
|
|
8
|
+
* `IDataProvider.ensureUser` (called from `hooks.server.ts`) seeds an empty
|
|
9
|
+
* row for every authed user, mirroring Supabase's `handle_new_auth_user`
|
|
10
|
+
* trigger.
|
|
11
|
+
*/
|
|
12
|
+
export declare class LocalUserProfileProvider implements IUserProfileStore {
|
|
13
|
+
private readonly data;
|
|
14
|
+
constructor(userDataFilePath: string);
|
|
15
|
+
static fromEnv(env: Record<string, string | undefined>): LocalUserProfileProvider;
|
|
16
|
+
getProfile(ctx: RequestContext, userId: string): Promise<UserProfile | null>;
|
|
17
|
+
getProfiles(_ctx: RequestContext, userIds: readonly string[]): Promise<UserProfile[]>;
|
|
18
|
+
updateProfile(ctx: RequestContext, userId: string, patch: {
|
|
19
|
+
displayName?: string;
|
|
20
|
+
}): Promise<UserManagementResult>;
|
|
21
|
+
starDefinition(ctx: RequestContext, userId: string, definitionId: string): Promise<UserManagementResult>;
|
|
22
|
+
unstarDefinition(ctx: RequestContext, userId: string, definitionId: string): Promise<UserManagementResult>;
|
|
23
|
+
recordRun(ctx: RequestContext, userId: string, run: RecentRun): Promise<UserManagementResult>;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=LocalUserProfileProvider.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalUserProfileProvider.d.ts","sourceRoot":"","sources":["../../src/userProfile/LocalUserProfileProvider.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACX,iBAAiB,EACjB,SAAS,EACT,cAAc,EACd,oBAAoB,EACpB,WAAW,EACX,MAAM,mBAAmB,CAAC;AAiB3B;;;;;;;;;GASG;AACH,qBAAa,wBAAyB,YAAW,iBAAiB;IACjE,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAqB;gBAE9B,gBAAgB,EAAE,MAAM;IAIpC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC,GAAG,wBAAwB;IAK3E,UAAU,CAAC,GAAG,EAAE,cAAc,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC;IAM5E,WAAW,CAAC,IAAI,EAAE,cAAc,EAAE,OAAO,EAAE,SAAS,MAAM,EAAE,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC;IASrF,aAAa,CAClB,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,EACd,KAAK,EAAE;QAAE,WAAW,CAAC,EAAE,MAAM,CAAA;KAAE,GAC7B,OAAO,CAAC,oBAAoB,CAAC;IAa1B,cAAc,CACnB,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,EACd,YAAY,EAAE,MAAM,GAClB,OAAO,CAAC,oBAAoB,CAAC;IAW1B,gBAAgB,CACrB,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,EACd,YAAY,EAAE,MAAM,GAClB,OAAO,CAAC,oBAAoB,CAAC;IAW1B,SAAS,CACd,GAAG,EAAE,cAAc,EACnB,MAAM,EAAE,MAAM,EACd,GAAG,EAAE,SAAS,GACZ,OAAO,CAAC,oBAAoB,CAAC;CAUhC"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import * as path from 'node:path';
|
|
2
|
+
import { ProviderError, hasPermission } from '@selvajs/platform';
|
|
3
|
+
import { createLocalUserDataStore } from '../data/userData.js';
|
|
4
|
+
function toProfile(u) {
|
|
5
|
+
return {
|
|
6
|
+
userId: u.userId,
|
|
7
|
+
displayName: u.displayName,
|
|
8
|
+
starredDefinitions: u.starredDefinitions ?? [],
|
|
9
|
+
recentRuns: u.recentRuns ?? []
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Filesystem-backed user-profile store. Reads and writes `user-data.json`,
|
|
14
|
+
* shared with `LocalPlatformPermissionStore`. Identity (email, password
|
|
15
|
+
* hash) lives separately on the auth provider — this store keys exclusively
|
|
16
|
+
* by user ID.
|
|
17
|
+
*
|
|
18
|
+
* `IDataProvider.ensureUser` (called from `hooks.server.ts`) seeds an empty
|
|
19
|
+
* row for every authed user, mirroring Supabase's `handle_new_auth_user`
|
|
20
|
+
* trigger.
|
|
21
|
+
*/
|
|
22
|
+
export class LocalUserProfileProvider {
|
|
23
|
+
data;
|
|
24
|
+
constructor(userDataFilePath) {
|
|
25
|
+
this.data = createLocalUserDataStore(userDataFilePath);
|
|
26
|
+
}
|
|
27
|
+
static fromEnv(env) {
|
|
28
|
+
if (!env.DATA_PATH)
|
|
29
|
+
throw new Error('Missing required env var: DATA_PATH');
|
|
30
|
+
return new LocalUserProfileProvider(path.join(env.DATA_PATH, 'user-data.json'));
|
|
31
|
+
}
|
|
32
|
+
async getProfile(ctx, userId) {
|
|
33
|
+
assertCanAccess(ctx, userId);
|
|
34
|
+
const u = await this.data.findById(userId);
|
|
35
|
+
return u ? toProfile(u) : null;
|
|
36
|
+
}
|
|
37
|
+
async getProfiles(_ctx, userIds) {
|
|
38
|
+
// Batch read is read-only and used for display name lookups across the UI;
|
|
39
|
+
// scoping to a single user defeats the purpose. Adapters with stricter
|
|
40
|
+
// requirements should override.
|
|
41
|
+
const all = await this.data.listAll();
|
|
42
|
+
const wanted = new Set(userIds);
|
|
43
|
+
return all.filter((u) => wanted.has(u.userId)).map(toProfile);
|
|
44
|
+
}
|
|
45
|
+
async updateProfile(ctx, userId, patch) {
|
|
46
|
+
assertCanAccess(ctx, userId);
|
|
47
|
+
try {
|
|
48
|
+
if (patch.displayName !== undefined) {
|
|
49
|
+
await this.data.updateDisplayName(userId, patch.displayName);
|
|
50
|
+
}
|
|
51
|
+
return 'ok';
|
|
52
|
+
}
|
|
53
|
+
catch (err) {
|
|
54
|
+
if (err instanceof ProviderError && err.statusCode === 404)
|
|
55
|
+
return 'not_found';
|
|
56
|
+
throw err;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
async starDefinition(ctx, userId, definitionId) {
|
|
60
|
+
assertCanAccess(ctx, userId);
|
|
61
|
+
try {
|
|
62
|
+
await this.data.starDefinition(userId, definitionId);
|
|
63
|
+
return 'ok';
|
|
64
|
+
}
|
|
65
|
+
catch (err) {
|
|
66
|
+
if (err instanceof ProviderError && err.statusCode === 404)
|
|
67
|
+
return 'not_found';
|
|
68
|
+
throw err;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
async unstarDefinition(ctx, userId, definitionId) {
|
|
72
|
+
assertCanAccess(ctx, userId);
|
|
73
|
+
try {
|
|
74
|
+
await this.data.unstarDefinition(userId, definitionId);
|
|
75
|
+
return 'ok';
|
|
76
|
+
}
|
|
77
|
+
catch (err) {
|
|
78
|
+
if (err instanceof ProviderError && err.statusCode === 404)
|
|
79
|
+
return 'not_found';
|
|
80
|
+
throw err;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
async recordRun(ctx, userId, run) {
|
|
84
|
+
assertCanAccess(ctx, userId);
|
|
85
|
+
try {
|
|
86
|
+
await this.data.recordRun(userId, run);
|
|
87
|
+
return 'ok';
|
|
88
|
+
}
|
|
89
|
+
catch (err) {
|
|
90
|
+
if (err instanceof ProviderError && err.statusCode === 404)
|
|
91
|
+
return 'not_found';
|
|
92
|
+
throw err;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Profile reads/writes are scoped to the user themselves. `instance_admin`
|
|
98
|
+
* bypasses for admin tooling. `system: true` (background jobs, signup
|
|
99
|
+
* auto-seed) also passes — those flows have already authorized.
|
|
100
|
+
*/
|
|
101
|
+
function assertCanAccess(ctx, userId) {
|
|
102
|
+
if (ctx.system)
|
|
103
|
+
return;
|
|
104
|
+
if (ctx.userId === userId)
|
|
105
|
+
return;
|
|
106
|
+
if (hasPermission(ctx, 'instance_admin'))
|
|
107
|
+
return;
|
|
108
|
+
throw new ProviderError('Forbidden: cannot access another user’s profile', 403);
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=LocalUserProfileProvider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"LocalUserProfileProvider.js","sourceRoot":"","sources":["../../src/userProfile/LocalUserProfileProvider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"conformance.test.d.ts","sourceRoot":"","sources":["../../../src/userProfile/__tests__/conformance.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { describe, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import * as fs from 'node:fs/promises';
|
|
3
|
+
import * as path from 'node:path';
|
|
4
|
+
import * as os from 'node:os';
|
|
5
|
+
import { runUserProfileStoreConformance } from '@selvajs/platform/testing';
|
|
6
|
+
import { LocalAuthProvider } from '../../auth/LocalAuthProvider.js';
|
|
7
|
+
import { LocalUserProfileProvider } from '../LocalUserProfileProvider.js';
|
|
8
|
+
import { createLocalUserDataStore } from '../../data/userData.js';
|
|
9
|
+
describe('LocalUserProfileProvider', () => {
|
|
10
|
+
let tempDir;
|
|
11
|
+
beforeEach(async () => {
|
|
12
|
+
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'selva-userprofile-test-'));
|
|
13
|
+
});
|
|
14
|
+
afterEach(async () => {
|
|
15
|
+
await fs.rm(tempDir, { recursive: true, force: true });
|
|
16
|
+
});
|
|
17
|
+
runUserProfileStoreConformance({
|
|
18
|
+
name: 'LocalUserProfileProvider',
|
|
19
|
+
createStore: async () => {
|
|
20
|
+
const auth = new LocalAuthProvider({
|
|
21
|
+
hmacSecret: 'test-secret',
|
|
22
|
+
usersFilePath: path.join(tempDir, 'auth-users.json')
|
|
23
|
+
});
|
|
24
|
+
const userDataPath = path.join(tempDir, 'user-data.json');
|
|
25
|
+
const userData = createLocalUserDataStore(userDataPath);
|
|
26
|
+
const store = new LocalUserProfileProvider(userDataPath);
|
|
27
|
+
return {
|
|
28
|
+
store,
|
|
29
|
+
seedUser: async (email) => {
|
|
30
|
+
// Real user creation: auth-users row + user-data row, mirroring
|
|
31
|
+
// what `hooks.server.ts` does in production via `ensureUser`.
|
|
32
|
+
const user = await auth.passwordAuth.createUserWithPassword(email, 'pw');
|
|
33
|
+
await userData.ensure(user.id);
|
|
34
|
+
return user;
|
|
35
|
+
}
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
//# sourceMappingURL=conformance.test.js.map
|