@morscherlab/mint-sdk 1.0.0-rc.8 → 1.0.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/dist/composables/index.d.ts +1 -1
- package/dist/composables/index.js +2 -2
- package/dist/composables/usePluginClient.d.ts +24 -0
- package/dist/{composables-k-hcs7Ze.js → composables-wNt7VtkF.js} +95 -3
- package/dist/{composables-k-hcs7Ze.js.map → composables-wNt7VtkF.js.map} +1 -1
- package/dist/index.js +2 -2
- package/package.json +1 -1
- package/src/__tests__/composables/usePluginClient.test.ts +169 -1
- package/src/composables/index.ts +9 -0
- package/src/composables/pluginEndpointBuilder.ts +32 -4
- package/src/composables/usePluginClient.ts +144 -0
|
@@ -44,4 +44,4 @@ export { getBioTemplateComponentProps, getBioTemplateComponentBindings, toBioTem
|
|
|
44
44
|
export { useBioTemplateWorkspace, type BioTemplateRendererBinding, type BioTemplateWorkspaceBindings, type BioTemplateWorkspaceTarget, type UseBioTemplateWorkspaceReturn, } from './useBioTemplateWorkspace';
|
|
45
45
|
export { useBioTemplatePresetWorkspace, type BioTemplatePresetWorkspaceBindings, type UseBioTemplatePresetWorkspaceOptions, type UseBioTemplatePresetWorkspaceReturn, } from './useBioTemplatePresetWorkspace';
|
|
46
46
|
export { useBioTemplatePackWorkspace, type UseBioTemplatePackWorkspaceOptions, type UseBioTemplatePackWorkspaceReturn, } from './useBioTemplatePackWorkspace';
|
|
47
|
-
export { buildPluginEndpointUrl, createPluginClient, getPluginPageSelectorItems, resolvePluginBaseUrl, usePluginClient, usePluginSettings, useCurrentExperiment, type BuildPluginEndpointUrlOptions, type PluginContract, type PluginEndpointContract, type PluginEndpointDefinition, type PluginHttpMethod, type PluginNavItemContract, type CreatePluginClientOptions, type UseCurrentExperimentOptions, type UseCurrentExperimentReturn, } from './usePluginClient';
|
|
47
|
+
export { buildPluginEndpointUrl, createPluginClient, downloadBlob, downloadPluginEndpoint, getPluginPageSelectorItems, pluginFormDataFromPayload, resolvePluginBaseUrl, uploadPluginEndpoint, usePluginClient, usePluginSettings, useCurrentExperiment, type BuildPluginEndpointUrlOptions, type DownloadBlobOptions, type DownloadPluginEndpointOptions, type PluginContract, type PluginEndpointContract, type PluginEndpointDefinition, type PluginHttpMethod, type PluginNavItemContract, type PluginEndpointRequestOptions, type PluginFormDataPayload, type PluginFormDataValue, type CreatePluginClientOptions, type UseCurrentExperimentOptions, type UseCurrentExperimentReturn, } from './usePluginClient';
|
|
@@ -1,4 +1,4 @@
|
|
|
1
1
|
import { An as controlsToSectionFormSchema, B as toBioTemplateComponentProps, Cn as defineControlModel, Dn as controlValuesToComponentProps, En as controlValuesToComponentBindingsById, Er as useEventListener, Fn as controlsToViewIds, Gn as useSequenceUtils, H as toBioTemplateComponentPropsById, Hn as useConcentrationUnits, In as controlsToViewItems, Jn as useChemicalFormula, L as toBioTemplateComponentBindings, Ln as getDefaultControlView, Mn as controlsToSettingsSchema, Nn as controlsToSidebarPanels, P as getBioTemplateComponentBindings, Pn as controlsToTopBarSettingsConfig, R as toBioTemplateComponentBindingsById, Rn as getFieldRegistryEntry, Sn as useControlWorkspace, Tn as controlValuesToComponentBindings, _n as defineDoseCalculatorControlProps, ar as formatDuration, bn as defineWellPlateDoseComponentBindings, br as useSelectionLimit, cr as fromMinutes, dr as parseTime, er as addMinutes, fr as rangesOverlap, gn as useControlSchema, hn as getControlDefaults, hr as useTimeUtils, ir as findNearestTimeSlotIndex, jn as controlsToSectionFormSchemas, kn as controlsToFormSchema, l as requireBioTemplateControlSchema, lr as generateTimeSlots, mn as defineControls, mr as toMinutes, nr as durationMinutes, o as getBioTemplateControlSchema, or as formatTime, pn as defineControlComponentBindings, pr as snapToSlot, qn as ATOMIC_WEIGHTS, rr as findAvailableSlots, sr as formatTimeSlot, tr as compareTime, ur as isTimeInRange, vn as defineDoseDesignControlModel, xn as defineWellPlateDoseControlProps, yn as defineWellPlateControlProps, yr as useListSelection, zn as getTypeDefault } from "../templates-DSbHJC4v.js";
|
|
2
2
|
import { $ as resolveExperimentCode, B as useExperimentSelector, C as useExpansionSet, G as EXPERIMENT_STATUS_LABELS, H as useDebouncedWatch, I as useWellPlateEditor, J as SORT_OPTIONS, K as EXPERIMENT_STATUS_OPTIONS, L as useDoseCalculator, M as extractSampleOptionsFromDesignData, N as unwrapExperimentDesignData, O as extractSamplesFromDesignData, P as DEFAULT_COLORS, Q as getExperimentStatusVariant, R as APP_EXPERIMENT_KEY, U as useApi, V as useRequestSyncState, W as DATE_PRESET_OPTIONS, X as formatExperimentDate, Y as datePresetToISO, Z as formatExperimentStatus, _ as useScheduleDrag, a as useReagentSeries, at as useToast, c as useBioTemplatePresetWorkspace, ct as useTextSearch, d as getBioTemplateComponentProps, et as usePlatformContext, f as toBioTemplateComponentPropsByComponent, g as useExperimentSave, h as useTemplateCollection, i as generateDilutionSeries, it as useTheme, j as extractSampleNamesFromDesignData, k as useAutoGroup, l as useBioTemplatePackWorkspace, lt as compareSortValues, m as useBioTemplateControls, n as DEFAULT_PRESETS, nt as evaluateCondition, o as useGroupAssignment, ot as candidateMatchesSearch, p as useBioTemplateComponents, q as EXPERIMENT_STATUS_VARIANT_MAP, r as DEFAULT_UNITS, rt as useForm, s as useRackEditor, st as normalizeSearchQuery, t as useProtocolTemplates, tt as useFormBuilder, u as useBioTemplateWorkspace, ut as useSortedItems, v as useExperimentSamples, w as useSampleGroups, y as useExperimentData, z as useAppExperiment } from "../useProtocolTemplates-DwBhEPPU.js";
|
|
3
|
-
import { a as
|
|
4
|
-
export { APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DEFAULT_PRESETS, DEFAULT_UNITS, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, SORT_OPTIONS, addMinutes, buildPluginEndpointUrl, candidateMatchesSearch, compareSortValues, compareTime, controlValuesToComponentBindings, controlValuesToComponentBindingsById, controlValuesToComponentProps, controlsToFormSchema, controlsToSectionFormSchema, controlsToSectionFormSchemas, controlsToSettingsSchema, controlsToSidebarPanels, controlsToTopBarSettingsConfig, controlsToViewIds, controlsToViewItems, createPluginClient, datePresetToISO, defineControlComponentBindings, defineControlModel, defineControls, defineDoseCalculatorControlProps, defineDoseDesignControlModel, defineWellPlateControlProps, defineWellPlateDoseComponentBindings, defineWellPlateDoseControlProps, durationMinutes, evaluateCondition, extractSampleNamesFromDesignData, extractSampleOptionsFromDesignData, extractSamplesFromDesignData, findAvailableSlots, findNearestTimeSlotIndex, formatDuration, formatExperimentDate, formatExperimentStatus, formatTime, formatTimeSlot, fromMinutes, generateDilutionSeries, generateTimeSlots, getBioTemplateComponentBindings, getBioTemplateComponentProps, getBioTemplateControlSchema, getControlDefaults, getDefaultControlView, getExperimentStatusVariant, getFieldRegistryEntry, getPluginPageSelectorItems, getTypeDefault, isTimeInRange, normalizeSearchQuery, parseTime, rangesOverlap, requireBioTemplateControlSchema, resolveExperimentCode, resolvePluginBaseUrl, snapToSlot, toBioTemplateComponentBindings, toBioTemplateComponentBindingsById, toBioTemplateComponentProps, toBioTemplateComponentPropsByComponent, toBioTemplateComponentPropsById, toMinutes, unwrapExperimentDesignData, useApi, useAppExperiment, useAsync, useAsyncBatch, useAuth, useAutoGroup, useBioTemplateComponents, useBioTemplateControls, useBioTemplatePackWorkspace, useBioTemplatePresetWorkspace, useBioTemplateWorkspace, useChemicalFormula, useConcentrationUnits, useControlSchema, useControlWorkspace, useCurrentExperiment, useDebouncedWatch, useDoseCalculator, useEventListener, useExpansionSet, useExperimentData, useExperimentSamples, useExperimentSave, useExperimentSelector, useForm, useFormBuilder, useGroupAssignment, useListSelection, usePasskey, usePlatformContext, usePluginClient, usePluginConfig, usePluginSettings, useProtocolTemplates, useRackEditor, useReagentSeries, useRequestSyncState, useSampleGroups, useScheduleDrag, useSelectionLimit, useSequenceUtils, useSortedItems, useTemplateCollection, useTextSearch, useTheme, useTimeUtils, useToast, useWellPlateEditor };
|
|
3
|
+
import { a as pluginFormDataFromPayload, c as usePluginClient, d as resolvePluginBaseUrl, f as usePluginConfig, g as useAuth, h as usePasskey, i as getPluginPageSelectorItems, l as usePluginSettings, m as useAsyncBatch, n as downloadBlob, o as uploadPluginEndpoint, p as useAsync, r as downloadPluginEndpoint, s as useCurrentExperiment, t as createPluginClient, u as buildPluginEndpointUrl } from "../composables-wNt7VtkF.js";
|
|
4
|
+
export { APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DEFAULT_PRESETS, DEFAULT_UNITS, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, SORT_OPTIONS, addMinutes, buildPluginEndpointUrl, candidateMatchesSearch, compareSortValues, compareTime, controlValuesToComponentBindings, controlValuesToComponentBindingsById, controlValuesToComponentProps, controlsToFormSchema, controlsToSectionFormSchema, controlsToSectionFormSchemas, controlsToSettingsSchema, controlsToSidebarPanels, controlsToTopBarSettingsConfig, controlsToViewIds, controlsToViewItems, createPluginClient, datePresetToISO, defineControlComponentBindings, defineControlModel, defineControls, defineDoseCalculatorControlProps, defineDoseDesignControlModel, defineWellPlateControlProps, defineWellPlateDoseComponentBindings, defineWellPlateDoseControlProps, downloadBlob, downloadPluginEndpoint, durationMinutes, evaluateCondition, extractSampleNamesFromDesignData, extractSampleOptionsFromDesignData, extractSamplesFromDesignData, findAvailableSlots, findNearestTimeSlotIndex, formatDuration, formatExperimentDate, formatExperimentStatus, formatTime, formatTimeSlot, fromMinutes, generateDilutionSeries, generateTimeSlots, getBioTemplateComponentBindings, getBioTemplateComponentProps, getBioTemplateControlSchema, getControlDefaults, getDefaultControlView, getExperimentStatusVariant, getFieldRegistryEntry, getPluginPageSelectorItems, getTypeDefault, isTimeInRange, normalizeSearchQuery, parseTime, pluginFormDataFromPayload, rangesOverlap, requireBioTemplateControlSchema, resolveExperimentCode, resolvePluginBaseUrl, snapToSlot, toBioTemplateComponentBindings, toBioTemplateComponentBindingsById, toBioTemplateComponentProps, toBioTemplateComponentPropsByComponent, toBioTemplateComponentPropsById, toMinutes, unwrapExperimentDesignData, uploadPluginEndpoint, useApi, useAppExperiment, useAsync, useAsyncBatch, useAuth, useAutoGroup, useBioTemplateComponents, useBioTemplateControls, useBioTemplatePackWorkspace, useBioTemplatePresetWorkspace, useBioTemplateWorkspace, useChemicalFormula, useConcentrationUnits, useControlSchema, useControlWorkspace, useCurrentExperiment, useDebouncedWatch, useDoseCalculator, useEventListener, useExpansionSet, useExperimentData, useExperimentSamples, useExperimentSave, useExperimentSelector, useForm, useFormBuilder, useGroupAssignment, useListSelection, usePasskey, usePlatformContext, usePluginClient, usePluginConfig, usePluginSettings, useProtocolTemplates, useRackEditor, useReagentSeries, useRequestSyncState, useSampleGroups, useScheduleDrag, useSelectionLimit, useSequenceUtils, useSortedItems, useTemplateCollection, useTextSearch, useTheme, useTimeUtils, useToast, useWellPlateEditor };
|
|
@@ -69,6 +69,22 @@ export interface BuildPluginEndpointUrlOptions {
|
|
|
69
69
|
/** Return only the endpoint path and query string instead of base URL + path. */
|
|
70
70
|
includeBaseUrl?: boolean;
|
|
71
71
|
}
|
|
72
|
+
export interface PluginEndpointRequestOptions {
|
|
73
|
+
/** Override the generated or platform-injected API base URL. */
|
|
74
|
+
baseUrl?: string;
|
|
75
|
+
}
|
|
76
|
+
export interface DownloadPluginEndpointOptions extends PluginEndpointRequestOptions {
|
|
77
|
+
/** Browser download filename. When omitted, the blob is returned without triggering a download. */
|
|
78
|
+
filename?: string;
|
|
79
|
+
/** Delay before revoking the object URL after a browser download is triggered. */
|
|
80
|
+
revokeObjectUrlDelayMs?: number;
|
|
81
|
+
}
|
|
82
|
+
export interface DownloadBlobOptions {
|
|
83
|
+
/** Delay before revoking the object URL after a browser download is triggered. */
|
|
84
|
+
revokeObjectUrlDelayMs?: number;
|
|
85
|
+
}
|
|
86
|
+
export type PluginFormDataValue = string | number | boolean | Blob | null | undefined | Record<string, unknown>;
|
|
87
|
+
export type PluginFormDataPayload = Record<string, PluginFormDataValue | PluginFormDataValue[]>;
|
|
72
88
|
export interface UseCurrentExperimentOptions {
|
|
73
89
|
apiBaseUrl?: string;
|
|
74
90
|
immediate?: boolean;
|
|
@@ -95,10 +111,18 @@ export interface UseCurrentExperimentReturn<TExperiment = ExperimentSummary> {
|
|
|
95
111
|
}
|
|
96
112
|
export type PluginEndpointCaller = (payload?: unknown) => Promise<unknown>;
|
|
97
113
|
export type GeneratedPluginClient = Record<string, (...args: any[]) => Promise<any>>;
|
|
114
|
+
/** Convert plain upload payloads into FormData using the same encoding across generated plugin clients. */
|
|
115
|
+
export declare function pluginFormDataFromPayload(payload: unknown): FormData;
|
|
116
|
+
/** Trigger a browser download for an already-created Blob. */
|
|
117
|
+
export declare function downloadBlob(blob: Blob, filename: string, options?: DownloadBlobOptions): void;
|
|
98
118
|
/** Convert PluginMetadata.nav_items into AppTopBar pageSelector items for topbar page switches, homepage PluginCards, and fallback views. */
|
|
99
119
|
export declare function getPluginPageSelectorItems(contract: PluginContract): PageSelectorItem[];
|
|
100
120
|
/** Create a typed plugin API client from a generated MINT plugin contract. */
|
|
101
121
|
export declare function createPluginClient<TClient extends object = GeneratedPluginClient>(contract: PluginContract, options: CreatePluginClientOptions): TClient;
|
|
122
|
+
/** Send multipart form data to a generated plugin endpoint while reusing contract URL, query, and auth handling. */
|
|
123
|
+
export declare function uploadPluginEndpoint<TResponse = unknown>(contract: PluginContract, endpoint: PluginEndpointDefinition, payload?: unknown, options?: PluginEndpointRequestOptions): Promise<TResponse>;
|
|
124
|
+
/** Fetch a generated plugin endpoint as a Blob and optionally trigger a browser download. */
|
|
125
|
+
export declare function downloadPluginEndpoint(contract: PluginContract, endpoint: PluginEndpointDefinition, payload?: unknown, options?: DownloadPluginEndpointOptions): Promise<Blob>;
|
|
102
126
|
/** Return a generated plugin client from setup code with a stable typed identity. */
|
|
103
127
|
export declare function usePluginClient<TClient>(client: TClient): TClient;
|
|
104
128
|
/** Read plugin settings exposed through the platform context. */
|
|
@@ -732,6 +732,25 @@ function queryParamsFromPayload(payload, pathParams, queryParams, endpointPath,
|
|
|
732
732
|
function hasKeys(value) {
|
|
733
733
|
return Object.keys(value).length > 0;
|
|
734
734
|
}
|
|
735
|
+
function isPlainPayloadObject(value) {
|
|
736
|
+
return !!value && typeof value === "object" && !Array.isArray(value);
|
|
737
|
+
}
|
|
738
|
+
function hasOwn(value, key) {
|
|
739
|
+
return Object.prototype.hasOwnProperty.call(value, key);
|
|
740
|
+
}
|
|
741
|
+
function normalizeStructuredEndpointPayload(payload) {
|
|
742
|
+
const pathParamPayload = payload.pathParams;
|
|
743
|
+
const queryPayload = payload.query;
|
|
744
|
+
const hasStructuredPathParams = hasOwn(payload, "pathParams") && isPlainPayloadObject(pathParamPayload);
|
|
745
|
+
const hasStructuredQuery = hasOwn(payload, "query") && isPlainPayloadObject(queryPayload);
|
|
746
|
+
if (!hasStructuredPathParams && !hasStructuredQuery) return payload;
|
|
747
|
+
const nextPayload = { ...payload };
|
|
748
|
+
delete nextPayload.pathParams;
|
|
749
|
+
delete nextPayload.query;
|
|
750
|
+
if (hasStructuredPathParams) Object.assign(nextPayload, pathParamPayload);
|
|
751
|
+
if (hasStructuredQuery) Object.assign(nextPayload, queryPayload);
|
|
752
|
+
return nextPayload;
|
|
753
|
+
}
|
|
735
754
|
function requestBodyFromPayload(payload, requestPayload, endpoint) {
|
|
736
755
|
if (!endpoint.hasBody) return void 0;
|
|
737
756
|
const body = endpoint.pathParams?.length || endpoint.queryParams?.length ? requestPayload.body : payload;
|
|
@@ -739,7 +758,7 @@ function requestBodyFromPayload(payload, requestPayload, endpoint) {
|
|
|
739
758
|
return body;
|
|
740
759
|
}
|
|
741
760
|
function pluginEndpointRequestParts(contract, endpoint, payload, explicitBaseUrl) {
|
|
742
|
-
const payloadObject =
|
|
761
|
+
const payloadObject = isPlainPayloadObject(payload) ? normalizeStructuredEndpointPayload(payload) : {};
|
|
743
762
|
const pathParams = endpoint.pathParams ?? [];
|
|
744
763
|
const requestPayload = withDefaultPathParams(payloadObject, pathParams);
|
|
745
764
|
const path = encodePath(endpoint.path, requestPayload, pathParams);
|
|
@@ -782,6 +801,51 @@ function buildPluginEndpointUrl(contract, endpoint, payload, options = {}) {
|
|
|
782
801
|
}
|
|
783
802
|
//#endregion
|
|
784
803
|
//#region src/composables/usePluginClient.ts
|
|
804
|
+
function queryConfig(query) {
|
|
805
|
+
return hasKeys(query) ? { params: query } : void 0;
|
|
806
|
+
}
|
|
807
|
+
function isFormData(value) {
|
|
808
|
+
return typeof FormData !== "undefined" && value instanceof FormData;
|
|
809
|
+
}
|
|
810
|
+
function isBlob(value) {
|
|
811
|
+
return typeof Blob !== "undefined" && value instanceof Blob;
|
|
812
|
+
}
|
|
813
|
+
function appendFormDataValue(formData, key, value) {
|
|
814
|
+
if (value === void 0 || value === null) return;
|
|
815
|
+
if (Array.isArray(value)) {
|
|
816
|
+
for (const item of value) appendFormDataValue(formData, key, item);
|
|
817
|
+
return;
|
|
818
|
+
}
|
|
819
|
+
if (isBlob(value)) {
|
|
820
|
+
formData.append(key, value);
|
|
821
|
+
return;
|
|
822
|
+
}
|
|
823
|
+
if (typeof value === "object") {
|
|
824
|
+
formData.append(key, JSON.stringify(value));
|
|
825
|
+
return;
|
|
826
|
+
}
|
|
827
|
+
formData.append(key, String(value));
|
|
828
|
+
}
|
|
829
|
+
/** Convert plain upload payloads into FormData using the same encoding across generated plugin clients. */
|
|
830
|
+
function pluginFormDataFromPayload(payload) {
|
|
831
|
+
if (isFormData(payload)) return payload;
|
|
832
|
+
if (!payload || typeof payload !== "object" || Array.isArray(payload)) throw new Error("[MINT SDK] Plugin upload endpoints require a FormData or plain object payload.");
|
|
833
|
+
const formData = new FormData();
|
|
834
|
+
for (const [key, value] of Object.entries(payload)) appendFormDataValue(formData, key, value);
|
|
835
|
+
return formData;
|
|
836
|
+
}
|
|
837
|
+
/** Trigger a browser download for an already-created Blob. */
|
|
838
|
+
function downloadBlob(blob, filename, options = {}) {
|
|
839
|
+
if (typeof document === "undefined" || typeof URL === "undefined") throw new Error("[MINT SDK] Cannot trigger a file download outside a browser.");
|
|
840
|
+
const blobUrl = URL.createObjectURL(blob);
|
|
841
|
+
const link = document.createElement("a");
|
|
842
|
+
link.href = blobUrl;
|
|
843
|
+
link.download = filename;
|
|
844
|
+
document.body.appendChild(link);
|
|
845
|
+
link.click();
|
|
846
|
+
document.body.removeChild(link);
|
|
847
|
+
setTimeout(() => URL.revokeObjectURL(blobUrl), options.revokeObjectUrlDelayMs ?? 100);
|
|
848
|
+
}
|
|
785
849
|
function normalizePluginNavPath(path) {
|
|
786
850
|
const raw = path.trim() || "/";
|
|
787
851
|
return (raw.startsWith("/") ? raw : `/${raw}`).replace(/\/+$/, "") || "/";
|
|
@@ -833,6 +897,34 @@ function createPluginClient(contract, options) {
|
|
|
833
897
|
};
|
|
834
898
|
return client;
|
|
835
899
|
}
|
|
900
|
+
/** Send multipart form data to a generated plugin endpoint while reusing contract URL, query, and auth handling. */
|
|
901
|
+
async function uploadPluginEndpoint(contract, endpoint, payload, options = {}) {
|
|
902
|
+
if (![
|
|
903
|
+
"post",
|
|
904
|
+
"put",
|
|
905
|
+
"patch"
|
|
906
|
+
].includes(endpoint.method)) throw new Error(`[MINT SDK] Plugin upload endpoints must use POST, PUT, or PATCH; got ${endpoint.method}`);
|
|
907
|
+
const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl);
|
|
908
|
+
const api = useApi({ baseUrl: parts.baseUrl });
|
|
909
|
+
const formData = pluginFormDataFromPayload(endpoint.hasBody ? requestBodyFromPayload(payload, parts.requestPayload, endpoint) : payload);
|
|
910
|
+
const config = {
|
|
911
|
+
...queryConfig(parts.query) ?? {},
|
|
912
|
+
headers: { "Content-Type": void 0 }
|
|
913
|
+
};
|
|
914
|
+
if (endpoint.method === "post") return api.post(parts.path, formData, config);
|
|
915
|
+
if (endpoint.method === "put") return api.put(parts.path, formData, config);
|
|
916
|
+
return api.patch(parts.path, formData, config);
|
|
917
|
+
}
|
|
918
|
+
/** Fetch a generated plugin endpoint as a Blob and optionally trigger a browser download. */
|
|
919
|
+
async function downloadPluginEndpoint(contract, endpoint, payload, options = {}) {
|
|
920
|
+
const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl);
|
|
921
|
+
const blob = await useApi({ baseUrl: parts.baseUrl }).get(parts.path, {
|
|
922
|
+
...queryConfig(parts.query) ?? {},
|
|
923
|
+
responseType: "blob"
|
|
924
|
+
});
|
|
925
|
+
if (options.filename) downloadBlob(blob, options.filename, { revokeObjectUrlDelayMs: options.revokeObjectUrlDelayMs });
|
|
926
|
+
return blob;
|
|
927
|
+
}
|
|
836
928
|
/** Return a generated plugin client from setup code with a stable typed identity. */
|
|
837
929
|
function usePluginClient(client) {
|
|
838
930
|
return client;
|
|
@@ -900,6 +992,6 @@ function useCurrentExperiment(options = {}) {
|
|
|
900
992
|
};
|
|
901
993
|
}
|
|
902
994
|
//#endregion
|
|
903
|
-
export {
|
|
995
|
+
export { pluginFormDataFromPayload as a, usePluginClient as c, resolvePluginBaseUrl as d, usePluginConfig as f, useAuth as g, usePasskey as h, getPluginPageSelectorItems as i, usePluginSettings as l, useAsyncBatch as m, downloadBlob as n, uploadPluginEndpoint as o, useAsync as p, downloadPluginEndpoint as r, useCurrentExperiment as s, createPluginClient as t, buildPluginEndpointUrl as u };
|
|
904
996
|
|
|
905
|
-
//# sourceMappingURL=composables-
|
|
997
|
+
//# sourceMappingURL=composables-wNt7VtkF.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"composables-k-hcs7Ze.js","names":[],"sources":["../src/composables/useAuth.ts","../src/composables/usePasskey.ts","../src/composables/useAsync.ts","../src/composables/usePluginConfig.ts","../src/composables/pluginEndpointBuilder.ts","../src/composables/usePluginClient.ts"],"sourcesContent":["import axios from 'axios'\nimport { ref, onMounted, onUnmounted, watch, getCurrentInstance, type Ref } from 'vue'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, UpdateProfileRequest } from '../types'\nimport type { RoleInfo } from '../permissions'\n\ninterface UserResponse {\n id: string\n username: string\n shortname: string | null\n email: string | null\n role: string\n role_obj?: RoleInfo | null\n is_active: boolean\n}\n\ninterface RefreshResponse {\n access_token: string\n expires_in: number\n token_type: string\n}\n\n// Token refresh configuration\nconst TOKEN_REFRESH_MARGIN_MS = 5 * 60 * 1000 // Refresh 5 minutes before expiry\nconst TOKEN_REFRESH_CHECK_INTERVAL_MS = 60 * 1000 // Check every minute\n\n/**\n * Authentication composable with automatic token refresh.\n *\n * Features:\n * - Automatic token refresh before expiration\n * - Login/logout/register functionality\n * - Token verification on startup\n * - User profile management\n *\n * @example\n * ```typescript\n * const { login, logout, isAuthenticated, user } = useAuth()\n *\n * // Login\n * const success = await login('username', 'password')\n *\n * // Automatic refresh is enabled by default\n * // Tokens are refreshed 5 minutes before expiration\n * ```\n */\nexport interface UseAuthReturn {\n login: (username: string, password: string) => Promise<boolean>\n logout: () => void\n register: (username: string, password: string, email?: string) => Promise<boolean>\n verifyToken: () => Promise<boolean>\n fetchAuthConfig: () => Promise<AuthConfig>\n initializeAuth: () => Promise<void>\n getCurrentUser: () => Promise<UserInfo | null>\n getAuthHeader: () => Record<string, string>\n updateProfile: (data: { email?: string; shortname?: string; currentPassword?: string; newPassword?: string }) => Promise<{ success: boolean; error?: string }>\n refreshToken: () => Promise<boolean>\n isRefreshing: Ref<boolean>\n}\n\n// Module-level singletons to prevent duplicate refresh requests and timers\n// across multiple useAuth() instances\nlet _refreshPromise: Promise<boolean> | null = null\nlet _refreshTimerId: number | null = null\nlet _mountedConsumerCount = 0\n\n/** @internal Reset module-level state — only for use in tests. */\nexport function _resetAuthStateForTesting(): void {\n _refreshPromise = null\n if (_refreshTimerId !== null) {\n clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n _mountedConsumerCount = 0\n}\n\n/** Manages authentication state with login/logout/register and automatic JWT token refresh. */\nexport function useAuth(): UseAuthReturn {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n const isRefreshing = ref(false)\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function fetchAuthConfig(): Promise<AuthConfig> {\n try {\n const response = await axios.get<{\n auth_required: boolean\n passkey_enabled: boolean\n passkey_registered?: boolean\n registration_enabled?: boolean\n database_mode?: string\n access?: {\n experiment_visibility_mode?: 'open' | 'restricted'\n experimentVisibilityMode?: 'open' | 'restricted'\n }\n }>(`${getApiBaseUrl()}/setup/config/public`)\n\n const config: AuthConfig = {\n authRequired: response.data.auth_required,\n passkeyEnabled: response.data.passkey_enabled,\n passkeyRegistered: response.data.passkey_registered ?? false,\n registrationEnabled: response.data.registration_enabled ?? false,\n databaseMode: response.data.database_mode ?? 'none',\n experimentVisibilityMode:\n response.data.access?.experiment_visibility_mode\n ?? response.data.access?.experimentVisibilityMode\n ?? 'open',\n }\n\n authStore.setAuthConfig(config)\n return config\n } catch (error) {\n console.error('Failed to fetch auth config:', error)\n return {\n authRequired: false,\n passkeyEnabled: false,\n passkeyRegistered: false,\n registrationEnabled: false,\n databaseMode: 'none',\n experimentVisibilityMode: 'open',\n }\n }\n }\n\n async function login(username: string, password: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const response = await axios.post<LoginResponse>(\n `${getApiBaseUrl()}/auth/login`,\n { username, password }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n authStore.setUsername(username)\n\n await getCurrentUser()\n\n // Start auto-refresh after successful login\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Login failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function register(username: string, password: string, email?: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n await axios.post<UserResponse>(\n `${getApiBaseUrl()}/users/register`,\n { username, password, email }\n )\n\n return await login(username, password)\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Registration failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function getCurrentUser(): Promise<UserInfo | null> {\n if (!authStore.token) {\n return null\n }\n\n try {\n const response = await axios.get<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n roleObj: response.data.role_obj ?? null,\n isActive: response.data.is_active,\n }\n\n authStore.setUserInfo(userInfo)\n return userInfo\n } catch {\n return null\n }\n }\n\n async function verifyToken(): Promise<boolean> {\n if (!authStore.token) {\n return false\n }\n\n try {\n const response = await axios.get<TokenVerifyResponse>(\n `${getApiBaseUrl()}/auth/verify`,\n {\n headers: {\n Authorization: `Bearer ${authStore.token}`,\n },\n }\n )\n\n if (response.data.valid && response.data.username) {\n authStore.setUsername(response.data.username)\n return true\n }\n\n authStore.clearToken()\n return false\n } catch {\n authStore.clearToken()\n return false\n }\n }\n\n /**\n * Refresh the authentication token.\n * Called automatically before token expiration.\n * Uses promise caching to prevent concurrent refresh requests.\n */\n async function refreshToken(): Promise<boolean> {\n if (!authStore.token) return false\n if (_refreshPromise) return _refreshPromise\n\n _refreshPromise = (async () => {\n isRefreshing.value = true\n\n try {\n const response = await axios.post<RefreshResponse>(\n `${getApiBaseUrl()}/auth/refresh`,\n {},\n { headers: getAuthHeader() }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n // Reschedule next refresh\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n // If refresh fails, the token may have been revoked\n // Clear auth state and let user re-login\n if (axios.isAxiosError(error) && error.response?.status === 401) {\n console.warn('[Auth] Token refresh failed - session expired')\n authStore.clearToken()\n stopTokenRefresh()\n }\n return false\n } finally {\n isRefreshing.value = false\n _refreshPromise = null\n }\n })()\n\n return _refreshPromise\n }\n\n /**\n * Schedule automatic token refresh before expiration.\n */\n function scheduleTokenRefresh(): void {\n // Clear any existing timer\n stopTokenRefresh()\n\n if (!authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (refreshAt <= now) {\n // Token is already close to expiring or expired, refresh now\n refreshToken()\n return\n }\n\n // Schedule refresh\n const delay = refreshAt - now\n _refreshTimerId = window.setTimeout(() => {\n refreshToken()\n }, delay)\n }\n\n /**\n * Stop automatic token refresh.\n */\n function stopTokenRefresh(): void {\n if (_refreshTimerId !== null) {\n window.clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n }\n\n /**\n * Check if token needs refresh and refresh if necessary.\n * Called periodically as a safety net.\n */\n function checkAndRefreshIfNeeded(): void {\n if (!authStore.token || !authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (now >= refreshAt) {\n refreshToken()\n }\n }\n\n async function initializeAuth(): Promise<void> {\n authStore.initialize()\n await fetchAuthConfig()\n\n if (authStore.token) {\n const valid = await verifyToken()\n if (valid) {\n await getCurrentUser()\n // Start auto-refresh for existing session\n scheduleTokenRefresh()\n }\n }\n }\n\n function logout(): void {\n stopTokenRefresh()\n authStore.logout()\n }\n\n function getAuthHeader(): Record<string, string> {\n if (authStore.token) {\n return { Authorization: `Bearer ${authStore.token}` }\n }\n return {}\n }\n\n async function updateProfile(data: {\n email?: string\n shortname?: string\n currentPassword?: string\n newPassword?: string\n }): Promise<{ success: boolean; error?: string }> {\n if (!authStore.token) {\n return { success: false, error: 'Not authenticated' }\n }\n\n try {\n const requestData: UpdateProfileRequest = {}\n if (data.email !== undefined) requestData.email = data.email\n if (data.shortname !== undefined) requestData.shortname = data.shortname\n if (data.currentPassword) requestData.current_password = data.currentPassword\n if (data.newPassword) requestData.new_password = data.newPassword\n\n const response = await axios.put<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n requestData,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n roleObj: response.data.role_obj ?? null,\n isActive: response.data.is_active,\n }\n authStore.setUserInfo(userInfo)\n\n return { success: true }\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n return { success: false, error: error.response.data.detail || 'Update failed' }\n }\n return { success: false, error: 'Network error. Please try again.' }\n }\n }\n\n // Set up periodic check as safety net (only inside component setup)\n let checkInterval: number | null = null\n\n if (getCurrentInstance()) {\n onMounted(() => {\n _mountedConsumerCount += 1\n checkInterval = window.setInterval(checkAndRefreshIfNeeded, TOKEN_REFRESH_CHECK_INTERVAL_MS)\n })\n\n onUnmounted(() => {\n if (checkInterval !== null) {\n window.clearInterval(checkInterval)\n checkInterval = null\n }\n\n _mountedConsumerCount = Math.max(0, _mountedConsumerCount - 1)\n if (_mountedConsumerCount === 0) {\n stopTokenRefresh()\n }\n })\n\n // Watch for token changes to reschedule refresh\n watch(\n () => authStore.tokenExpires,\n (newExpires) => {\n if (newExpires) {\n scheduleTokenRefresh()\n } else {\n stopTokenRefresh()\n }\n }\n )\n }\n\n return {\n // Core auth methods\n login,\n logout,\n register,\n verifyToken,\n fetchAuthConfig,\n initializeAuth,\n getCurrentUser,\n getAuthHeader,\n updateProfile,\n\n // Token refresh\n refreshToken,\n isRefreshing,\n }\n}\n","import axios from 'axios'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { CredentialInfo } from '../types'\n\n// Lazy-load @simplewebauthn/browser so plugins that never call usePasskey\n// don't need the optional peer dep installed (avoids build failures).\nasync function loadWebAuthn() {\n try {\n return await import('@simplewebauthn/browser')\n } catch {\n throw new Error(\n '@simplewebauthn/browser is required for passkey support. Install it: bun add @simplewebauthn/browser',\n )\n }\n}\n\ninterface PasskeyLoginResponse {\n access_token: string\n token_type: string\n expires_in: number\n}\n\n/** Registers and authenticates passkeys using WebAuthn, lazily loading the browser dependency. */\nexport function usePasskey() {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function isSupported(): Promise<boolean> {\n try {\n const { browserSupportsWebAuthn } = await loadWebAuthn()\n return browserSupportsWebAuthn()\n } catch {\n return false\n }\n }\n\n async function registerPasskey(deviceName?: string): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/register/options`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startRegistration(options)\n\n await axios.post(\n `${getApiBaseUrl()}/auth/passkey/register/verify`,\n {\n credential: JSON.stringify(credential),\n device_name: deviceName,\n },\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: true,\n })\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Passkey registration failed')\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Registration was cancelled or timed out')\n } else if (error.name === 'InvalidStateError') {\n authStore.setError('This authenticator is already registered')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey registration failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function loginWithPasskey(): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/login/options`,\n { withCredentials: true }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startAuthentication(options)\n\n const response = await axios.post<PasskeyLoginResponse>(\n `${getApiBaseUrl()}/auth/passkey/login/verify`,\n { credential: JSON.stringify(credential) },\n { withCredentials: true }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n if (error.response.status === 404) {\n authStore.setError('No passkeys registered. Please login with password first.')\n } else {\n authStore.setError(error.response.data.detail || 'Passkey login failed')\n }\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Authentication was cancelled or timed out')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey login failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function listCredentials(): Promise<CredentialInfo[]> {\n try {\n const response = await axios.get<{ credentials: CredentialInfo[] }>(\n `${getApiBaseUrl()}/auth/passkey/credentials`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n return response.data.credentials\n } catch {\n return []\n }\n }\n\n async function deleteCredential(credentialId: string): Promise<boolean> {\n try {\n await axios.delete(\n `${getApiBaseUrl()}/auth/passkey/credentials/${encodeURIComponent(credentialId)}`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n\n const remaining = await listCredentials()\n if (remaining.length === 0) {\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n }\n\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAllCredentials(): Promise<boolean> {\n try {\n await axios.delete(`${getApiBaseUrl()}/auth/passkey/credentials`, {\n headers: { Authorization: `Bearer ${authStore.token}` },\n })\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSupported,\n registerPasskey,\n loginWithPasskey,\n listCredentials,\n deleteCredential,\n deleteAllCredentials,\n }\n}\n","import { ref, computed, type Ref, type ComputedRef } from 'vue'\n\n/**\n * Error type for async operations.\n */\nexport interface AsyncError {\n message: string\n code?: string\n details?: Record<string, unknown>\n originalError?: unknown\n}\n\n/**\n * State of an async operation.\n */\nexport type AsyncState = 'idle' | 'loading' | 'success' | 'error'\n\n/**\n * Return type for useAsync composable.\n */\nexport interface UseAsyncReturn<T> {\n // Data\n data: Ref<T | null>\n error: Ref<AsyncError | null>\n\n // State\n state: Ref<AsyncState>\n isIdle: ComputedRef<boolean>\n isLoading: ComputedRef<boolean>\n isSuccess: ComputedRef<boolean>\n isError: ComputedRef<boolean>\n\n // Methods\n execute: (...args: unknown[]) => Promise<T | null>\n reset: () => void\n setData: (data: T | null) => void\n setError: (error: AsyncError | null) => void\n}\n\n/**\n * Options for useAsync.\n */\nexport interface UseAsyncOptions<T> {\n // Initial data value\n initialData?: T | null\n\n // Whether to run immediately on creation\n immediate?: boolean\n\n // Arguments for immediate execution\n immediateArgs?: unknown[]\n\n // Transform error into AsyncError\n transformError?: (error: unknown) => AsyncError\n\n // Called on success\n onSuccess?: (data: T) => void\n\n // Called on error\n onError?: (error: AsyncError) => void\n\n // Reset data on new execution\n resetOnExecute?: boolean\n}\n\n/**\n * Default error transformer.\n */\nfunction defaultTransformError(error: unknown): AsyncError {\n if (error instanceof Error) {\n return {\n message: error.message,\n originalError: error,\n }\n }\n\n if (typeof error === 'object' && error !== null) {\n const errorObj = error as Record<string, unknown>\n\n // Handle Axios-like errors\n if ('response' in errorObj && errorObj.response) {\n const response = errorObj.response as Record<string, unknown>\n const data = response.data as Record<string, unknown> | undefined\n\n return {\n message: (data?.detail as string) || (data?.message as string) || 'Request failed',\n code: String(response.status),\n details: data,\n originalError: error,\n }\n }\n\n // Handle generic error objects\n return {\n message: (errorObj.message as string) || 'Unknown error',\n code: errorObj.code as string | undefined,\n details: errorObj,\n originalError: error,\n }\n }\n\n return {\n message: String(error),\n originalError: error,\n }\n}\n\n/**\n * Composable for managing async operation state.\n *\n * Provides standardized loading, error, and success state management\n * for any async function.\n *\n * @param asyncFn - The async function to wrap\n * @param options - Configuration options\n *\n * @example\n * ```typescript\n * // Basic usage\n * const { data, isLoading, error, execute } = useAsync(\n * async (id: string) => {\n * const response = await api.get(`/users/${id}`)\n * return response.data\n * }\n * )\n *\n * // Execute the function\n * await execute('user-123')\n *\n * // In template\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error.message }}</div>\n * <div v-else-if=\"data\">{{ data.name }}</div>\n * ```\n *\n * @example\n * ```typescript\n * // With options\n * const { data, execute } = useAsync(\n * fetchUser,\n * {\n * immediate: true,\n * immediateArgs: ['default-user'],\n * onSuccess: (user) => console.log('Fetched:', user.name),\n * onError: (error) => toast.error(error.message),\n * }\n * )\n * ```\n *\n * @example\n * ```typescript\n * // Form submission\n * const { isLoading, error, execute: submit } = useAsync(\n * async (formData: FormData) => {\n * await api.post('/submit', formData)\n * },\n * {\n * onSuccess: () => {\n * toast.success('Submitted!')\n * router.push('/success')\n * },\n * }\n * )\n *\n * const handleSubmit = () => submit(new FormData(formRef.value))\n * ```\n */\n/** Wraps an async function with reactive loading, error, and success state plus optional auto-execute. */\nexport function useAsync<T>(\n asyncFn: (...args: unknown[]) => Promise<T>,\n options: UseAsyncOptions<T> = {}\n): UseAsyncReturn<T> {\n const {\n initialData = null,\n immediate = false,\n immediateArgs = [],\n transformError = defaultTransformError,\n onSuccess,\n onError,\n resetOnExecute = false,\n } = options\n\n // State\n const data = ref<T | null>(initialData) as Ref<T | null>\n const error = ref<AsyncError | null>(null)\n const state = ref<AsyncState>('idle')\n\n // Computed state helpers\n const isIdle = computed(() => state.value === 'idle')\n const isLoading = computed(() => state.value === 'loading')\n const isSuccess = computed(() => state.value === 'success')\n const isError = computed(() => state.value === 'error')\n\n // Execute the async function\n async function execute(...args: unknown[]): Promise<T | null> {\n state.value = 'loading'\n error.value = null\n\n if (resetOnExecute) {\n data.value = null\n }\n\n try {\n const result = await asyncFn(...args)\n data.value = result\n state.value = 'success'\n onSuccess?.(result)\n return result\n } catch (e) {\n const asyncError = transformError(e)\n error.value = asyncError\n state.value = 'error'\n onError?.(asyncError)\n return null\n }\n }\n\n // Reset to initial state\n function reset(): void {\n data.value = initialData\n error.value = null\n state.value = 'idle'\n }\n\n // Manual data setter\n function setData(newData: T | null): void {\n data.value = newData\n if (newData !== null) {\n state.value = 'success'\n error.value = null\n }\n }\n\n // Manual error setter\n function setError(newError: AsyncError | null): void {\n error.value = newError\n if (newError !== null) {\n state.value = 'error'\n }\n }\n\n // Execute immediately if requested\n if (immediate) {\n execute(...immediateArgs)\n }\n\n return {\n data,\n error,\n state,\n isIdle,\n isLoading,\n isSuccess,\n isError,\n execute,\n reset,\n setData,\n setError,\n }\n}\n\n/**\n * Create a batch of async operations that can be executed in parallel.\n *\n * @example\n * ```typescript\n * const { results, isLoading, execute } = useAsyncBatch([\n * () => fetchUser(userId),\n * () => fetchPosts(userId),\n * () => fetchComments(userId),\n * ])\n *\n * await execute()\n * // results.value = [user, posts, comments]\n * ```\n */\nexport function useAsyncBatch<T extends readonly (() => Promise<unknown>)[]>(\n asyncFns: T\n): {\n results: Ref<{ [K in keyof T]: Awaited<ReturnType<T[K]>> | null }>\n errors: Ref<(AsyncError | null)[]>\n isLoading: Ref<boolean>\n execute: () => Promise<void>\n reset: () => void\n} {\n type Results = { [K in keyof T]: Awaited<ReturnType<T[K]>> | null }\n\n const results = ref<Results>(asyncFns.map(() => null) as unknown as Results) as Ref<Results>\n const errors = ref<(AsyncError | null)[]>(asyncFns.map(() => null))\n const isLoading = ref(false)\n\n async function execute(): Promise<void> {\n isLoading.value = true\n errors.value = asyncFns.map(() => null)\n\n const promises = asyncFns.map(async (fn, index) => {\n try {\n const result = await fn()\n ;(results.value as unknown[])[index] = result\n } catch (e) {\n errors.value[index] = defaultTransformError(e)\n ;(results.value as unknown[])[index] = null\n }\n })\n\n await Promise.all(promises)\n isLoading.value = false\n }\n\n function reset(): void {\n results.value = asyncFns.map(() => null) as unknown as Results\n errors.value = asyncFns.map(() => null)\n isLoading.value = false\n }\n\n return {\n results,\n errors,\n isLoading,\n execute,\n reset,\n }\n}\n","import { ref, computed, onMounted, type Ref, type ComputedRef } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\nimport { useRequestSyncState } from './useRequestSyncState'\n\nexport interface UsePluginConfigReturn {\n /** Editable plugin configuration values. */\n config: Ref<Record<string, unknown>>\n /** Whether the config is currently loading. */\n isLoading: Ref<boolean>\n /** Whether the config is currently saving. */\n isSaving: Ref<boolean>\n /** Error message from the last failed operation, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful load, or null. */\n lastLoadedAt: Ref<Date | null>\n /** Timestamp of the last successful save, or null. */\n lastSavedAt: Ref<Date | null>\n /** Whether local config differs from the last loaded or saved snapshot. */\n isDirty: ComputedRef<boolean>\n /** Load plugin configuration from the platform. */\n load: () => Promise<void>\n /** Save plugin configuration to the platform. */\n save: () => Promise<boolean>\n /** Restore the last loaded or saved configuration snapshot. */\n reset: () => void\n}\n\n/** Loads, saves, and tracks dirty state for a plugin's persistent configuration via the platform API. */\nexport function usePluginConfig(pluginName?: string): UsePluginConfigReturn {\n const api = useApi()\n const { plugin } = usePlatformContext()\n\n const resolvedName = computed(() => pluginName ?? plugin.value?.name ?? '')\n\n const config = ref<Record<string, unknown>>({})\n const savedConfig = ref<Record<string, unknown>>({})\n const request = useRequestSyncState('Plugin config request failed.')\n const isLoading = ref(false)\n const isSaving = ref(false)\n const error = request.error\n const lastLoadedAt = request.lastLoadedAt\n const lastSavedAt = request.lastSavedAt\n\n const isDirty = computed(() => {\n return JSON.stringify(config.value) !== JSON.stringify(savedConfig.value)\n })\n\n async function load(): Promise<void> {\n const name = resolvedName.value\n if (!name) return\n\n isLoading.value = true\n try {\n const response = await request.run(\n () => api.get<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n ),\n { success: 'load', errorMessage: 'Failed to load plugin config' },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n } catch {\n // Error state is handled by request.run().\n } finally {\n isLoading.value = false\n }\n }\n\n async function save(): Promise<boolean> {\n const name = resolvedName.value\n if (!name) return false\n\n isSaving.value = true\n try {\n const response = await request.run(\n () => api.patch<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n { config: config.value },\n ),\n { success: 'save', errorMessage: 'Failed to save plugin config' },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n return true\n } catch {\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n function reset(): void {\n config.value = { ...savedConfig.value }\n request.clearError()\n }\n\n onMounted(() => {\n load()\n })\n\n return {\n config,\n isLoading,\n isSaving,\n error,\n lastLoadedAt,\n lastSavedAt,\n isDirty,\n load,\n save,\n reset,\n }\n}\n","import {\n getInjectedPlatformContext,\n resolveCurrentExperimentId,\n} from './platformContextHelpers'\nimport type {\n BuildPluginEndpointUrlOptions,\n PluginContract,\n PluginEndpointDefinition,\n PluginEndpointParamDefinition,\n} from './usePluginClient'\n\nfunction normalizeApiBaseUrl(baseUrl: string | undefined): string | undefined {\n if (!baseUrl) return undefined\n return baseUrl.length > 1 ? baseUrl.replace(/\\/+$/, '') : baseUrl\n}\n\nfunction normalizeApiPrefix(prefix: string | undefined): string | undefined {\n if (!prefix) return undefined\n if (prefix.startsWith('/api/')) return normalizeApiBaseUrl(prefix)\n if (prefix === '/api') return prefix\n if (prefix.startsWith('/')) return normalizeApiBaseUrl(`/api${prefix}`)\n return normalizeApiBaseUrl(`/api/${prefix}`)\n}\n\n/** Resolve the runtime plugin API base URL from env, explicit options, platform injection, or contract metadata. */\nexport function resolvePluginBaseUrl(contract: PluginContract, explicitBaseUrl?: string): string {\n const envPrefix = (import.meta.env?.VITE_API_PREFIX as string | undefined) || undefined\n if (envPrefix) return normalizeApiBaseUrl(envPrefix) ?? envPrefix\n if (explicitBaseUrl) return normalizeApiBaseUrl(explicitBaseUrl) ?? explicitBaseUrl\n\n const injected = getInjectedPlatformContext()\n const platformPrefix =\n normalizeApiBaseUrl(injected?.plugin?.api_prefix) ||\n normalizeApiPrefix(injected?.plugin?.route_prefix)\n if (platformPrefix) return platformPrefix\n\n return normalizeApiBaseUrl(contract.plugin.apiPrefix) || '/api'\n}\n\nfunction encodePath(path: string, payload: Record<string, unknown>, pathParams: string[]): string {\n let nextPath = path\n for (const param of pathParams) {\n const value = payload[param]\n if (value === undefined || value === null) {\n throw new Error(`[MINT SDK] Missing path parameter '${param}' for plugin endpoint ${path}`)\n }\n nextPath = nextPath.replace(`{${param}}`, encodeURIComponent(String(value)))\n nextPath = nextPath.replace(`:${param}`, encodeURIComponent(String(value)))\n }\n return nextPath\n}\n\nfunction withDefaultPathParams(\n payload: Record<string, unknown>,\n pathParams: string[],\n): Record<string, unknown> {\n if (!pathParams.length) return payload\n const nextPayload = { ...payload }\n for (const param of pathParams) {\n if (nextPayload[param] !== undefined && nextPayload[param] !== null) continue\n if (isExperimentPathParam(param)) {\n const experimentId = resolveCurrentExperimentId()\n if (experimentId !== undefined) {\n nextPayload[param] = experimentId\n }\n }\n }\n return nextPayload\n}\n\nfunction isExperimentPathParam(param: string): boolean {\n return param === 'experimentId' || param === 'experiment_id'\n}\n\nfunction withoutPathParams(payload: Record<string, unknown>, pathParams: string[]) {\n const copy = { ...payload }\n for (const param of pathParams) {\n delete copy[param]\n }\n return copy\n}\n\nfunction queryParamsFromPayload(\n payload: Record<string, unknown>,\n pathParams: string[],\n queryParams?: PluginEndpointParamDefinition[],\n endpointPath?: string,\n includeUnspecifiedParams = false,\n) {\n if (!queryParams?.length) {\n return includeUnspecifiedParams ? withoutPathParams(payload, pathParams) : {}\n }\n\n const query: Record<string, unknown> = {}\n for (const param of queryParams) {\n const name = typeof param === 'string' ? param : param.name\n const fieldName = typeof param === 'string' ? param : param.fieldName ?? param.name\n const required = typeof param !== 'string' && param.required === true\n const value = payload[fieldName]\n if (value === undefined || value === null) {\n if (required) {\n throw new Error(\n `[MINT SDK] Missing query parameter '${fieldName}' for plugin endpoint ${endpointPath ?? ''}`,\n )\n }\n continue\n }\n query[name] = value\n }\n return query\n}\n\nexport function hasKeys(value: Record<string, unknown>): boolean {\n return Object.keys(value).length > 0\n}\n\nexport function requestBodyFromPayload(\n payload: unknown,\n requestPayload: Record<string, unknown>,\n endpoint: PluginEndpointDefinition,\n): unknown {\n if (!endpoint.hasBody) return undefined\n const body = endpoint.pathParams?.length || endpoint.queryParams?.length\n ? requestPayload.body\n : payload\n if (body === undefined || body === null) {\n throw new Error(`[MINT SDK] Missing request body for plugin endpoint ${endpoint.path}`)\n }\n return body\n}\n\nexport function pluginEndpointRequestParts(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n explicitBaseUrl?: string,\n) {\n const payloadObject =\n payload && typeof payload === 'object' && !Array.isArray(payload)\n ? payload as Record<string, unknown>\n : {}\n const pathParams = endpoint.pathParams ?? []\n const requestPayload = withDefaultPathParams(payloadObject, pathParams)\n const path = encodePath(endpoint.path, requestPayload, pathParams)\n const query = queryParamsFromPayload(\n requestPayload,\n pathParams,\n endpoint.queryParams,\n endpoint.path,\n endpoint.method === 'get',\n )\n\n return {\n baseUrl: resolvePluginBaseUrl(contract, explicitBaseUrl),\n path,\n query,\n requestPayload,\n }\n}\n\nfunction appendQueryValue(params: URLSearchParams, key: string, value: unknown): void {\n if (value === undefined || value === null) return\n if (Array.isArray(value)) {\n for (const item of value) {\n appendQueryValue(params, key, item)\n }\n return\n }\n if (typeof value === 'object') {\n params.append(key, JSON.stringify(value))\n return\n }\n params.append(key, String(value))\n}\n\nfunction queryString(query: Record<string, unknown>): string {\n const params = new URLSearchParams()\n for (const [key, value] of Object.entries(query)) {\n appendQueryValue(params, key, value)\n }\n const serialized = params.toString()\n return serialized ? `?${serialized}` : ''\n}\n\nfunction joinBaseUrl(baseUrl: string, path: string): string {\n if (!baseUrl) return path\n const normalizedBase = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl\n const normalizedPath = path.startsWith('/') ? path : `/${path}`\n return `${normalizedBase}${normalizedPath}`\n}\n\n/** Build the concrete URL for a generated plugin endpoint without making a request. */\nexport function buildPluginEndpointUrl(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n options: BuildPluginEndpointUrlOptions = {},\n): string {\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const pathWithQuery = `${parts.path}${queryString(parts.query)}`\n if (options.includeBaseUrl === false) return pathWithQuery\n return joinBaseUrl(parts.baseUrl, pathWithQuery)\n}\n","import { computed, shallowRef, watch, type ComputedRef, type Ref } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\nimport { useRequestSyncState } from './useRequestSyncState'\nimport {\n currentExperimentFromContext,\n getInjectedPlatformContext,\n resolveCurrentExperimentId,\n} from './platformContextHelpers'\nimport {\n hasKeys,\n pluginEndpointRequestParts,\n requestBodyFromPayload,\n} from './pluginEndpointBuilder'\nimport type { ExperimentSummary, PageSelectorItem } from '../types'\n\nexport {\n buildPluginEndpointUrl,\n resolvePluginBaseUrl,\n} from './pluginEndpointBuilder'\n\nexport type PluginHttpMethod = 'get' | 'post' | 'put' | 'patch' | 'delete'\n\nexport interface PluginEndpointContract {\n name: string\n method: PluginHttpMethod\n path: string\n pathParams?: string[]\n queryParams?: PluginEndpointParamContract[]\n requestType?: string | null\n responseType?: string | null\n}\n\nexport interface PluginEndpointParamContract {\n name: string\n fieldName: string\n type?: string\n required?: boolean\n}\n\nexport interface PluginNavItemContract {\n path: string\n label: string\n id?: string\n icon?: string\n description?: string\n requiresAuth?: boolean\n requiresAdmin?: boolean\n requiresFeature?: string\n}\n\nexport interface PluginContract {\n schemaVersion: number\n plugin: {\n name?: string\n packageName?: string\n version?: string\n description?: string\n routesPrefix: string\n apiPrefix: string\n type: 'analysis' | 'experiment-design' | string\n analysisType?: string\n icon?: string\n color?: string\n navItems?: PluginNavItemContract[]\n capabilities?: Record<string, unknown>\n }\n endpoints: PluginEndpointContract[]\n hash: string\n}\n\nexport interface PluginEndpointDefinition {\n method: PluginHttpMethod\n path: string\n pathParams?: string[]\n queryParams?: PluginEndpointParamDefinition[]\n hasBody?: boolean\n}\n\nexport type PluginEndpointParamDefinition = string | {\n name: string\n fieldName?: string\n type?: string\n required?: boolean\n}\n\nexport interface CreatePluginClientOptions {\n endpoints: Record<string, PluginEndpointDefinition>\n baseUrl?: string\n}\n\nexport interface BuildPluginEndpointUrlOptions {\n /** Override the generated or platform-injected API base URL. */\n baseUrl?: string\n /** Return only the endpoint path and query string instead of base URL + path. */\n includeBaseUrl?: boolean\n}\n\nexport interface UseCurrentExperimentOptions {\n apiBaseUrl?: string\n immediate?: boolean\n}\n\nexport interface UseCurrentExperimentReturn<TExperiment = ExperimentSummary> {\n /** Current experiment id resolved from platform injection or URL conventions. */\n experimentId: ComputedRef<number | undefined>\n /** Whether a current experiment id is available. */\n hasExperiment: ComputedRef<boolean>\n /** Current experiment payload, if injected or successfully fetched. */\n experiment: Ref<TExperiment | undefined>\n /** Whether the current experiment is currently loading. */\n isLoading: Ref<boolean>\n /** Error message from the last failed current-experiment fetch, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful current-experiment fetch, or null. */\n lastLoadedAt: Ref<Date | null>\n /** Return the current experiment id or throw a clear SDK error when absent. */\n requireExperimentId: () => number\n /** Fetch a specific experiment id, defaulting to the current experiment id. */\n fetch: (experimentId?: number) => Promise<TExperiment | undefined>\n /** Refetch the currently resolved experiment id. */\n refresh: () => Promise<TExperiment | undefined>\n}\n\nexport type PluginEndpointCaller = (payload?: unknown) => Promise<unknown>\nexport type GeneratedPluginClient = Record<string, (...args: any[]) => Promise<any>>\n\nfunction normalizePluginNavPath(path: string): string {\n const raw = path.trim() || '/'\n const prefixed = raw.startsWith('/') ? raw : `/${raw}`\n return prefixed.replace(/\\/+$/, '') || '/'\n}\n\nfunction pluginPageIdFromPath(path: string, fallbackIndex: number): string {\n const normalizedPath = normalizePluginNavPath(path)\n if (normalizedPath === '/') return 'dashboard'\n return normalizedPath.replace(/^\\/+/, '').replace(/\\/+/g, '-') || `page-${fallbackIndex + 1}`\n}\n\n/** Convert PluginMetadata.nav_items into AppTopBar pageSelector items for topbar page switches, homepage PluginCards, and fallback views. */\nexport function getPluginPageSelectorItems(contract: PluginContract): PageSelectorItem[] {\n const pluginIcon = contract.plugin.icon ?? ''\n const navItems = contract.plugin.navItems ?? []\n const source: PluginNavItemContract[] = navItems.length > 0\n ? navItems\n : [{\n path: '/',\n label: contract.plugin.name ?? 'Dashboard',\n id: 'dashboard',\n icon: pluginIcon || undefined,\n description: contract.plugin.description,\n }]\n\n return source.map((item, index) => ({\n id: item.id || pluginPageIdFromPath(item.path, index),\n label: item.label,\n to: normalizePluginNavPath(item.path),\n icon: item.icon || pluginIcon || undefined,\n hint: item.description || contract.plugin.name,\n }))\n}\n\n/** Create a typed plugin API client from a generated MINT plugin contract. */\nexport function createPluginClient<TClient extends object = GeneratedPluginClient>(\n contract: PluginContract,\n options: CreatePluginClientOptions,\n): TClient {\n const client: Record<string, PluginEndpointCaller> = {}\n\n for (const [name, endpoint] of Object.entries(options.endpoints)) {\n client[name] = async (payload?: unknown) => {\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const api = useApi({ baseUrl: parts.baseUrl })\n const queryConfig = hasKeys(parts.query) ? { params: parts.query } : undefined\n\n if (endpoint.method === 'get') {\n return api.get(parts.path, queryConfig)\n }\n\n if (endpoint.method === 'delete') {\n const body = requestBodyFromPayload(payload, parts.requestPayload, endpoint)\n const config = body === undefined\n ? queryConfig\n : { ...queryConfig, data: body }\n return api.delete(parts.path, config)\n }\n\n const body = requestBodyFromPayload(payload, parts.requestPayload, endpoint)\n\n if (endpoint.method === 'post') return api.post(parts.path, body, queryConfig)\n if (endpoint.method === 'put') return api.put(parts.path, body, queryConfig)\n if (endpoint.method === 'patch') return api.patch(parts.path, body, queryConfig)\n throw new Error(`[MINT SDK] Unsupported plugin endpoint method: ${endpoint.method}`)\n }\n }\n\n return client as TClient\n}\n\n/** Return a generated plugin client from setup code with a stable typed identity. */\nexport function usePluginClient<TClient>(client: TClient): TClient {\n return client\n}\n\n/** Read plugin settings exposed through the platform context. */\nexport function usePluginSettings<TSettings = Record<string, unknown>>() {\n const { plugin } = usePlatformContext()\n const settings = computed(() => plugin.value?.settings as TSettings | undefined)\n return { settings }\n}\n\n/** Read and optionally load the current platform experiment for integrated plugin views. */\nexport function useCurrentExperiment<TExperiment = ExperimentSummary>(\n options: UseCurrentExperimentOptions = {},\n): UseCurrentExperimentReturn<TExperiment> {\n const api = useApi({ baseUrl: options.apiBaseUrl ?? getInjectedPlatformContext()?.platformApiUrl })\n const experiment = shallowRef<TExperiment | undefined>(currentExperimentFromContext<TExperiment>())\n const request = useRequestSyncState('Failed to load current experiment')\n const isLoading = request.loading\n const error = request.error\n const lastLoadedAt = request.lastLoadedAt\n\n const experimentId = computed<number | undefined>(() => {\n return resolveCurrentExperimentId()\n })\n const hasExperiment = computed(() => experimentId.value !== undefined)\n\n function requireExperimentId(): number {\n const id = experimentId.value\n if (id === undefined) {\n throw new Error('[MINT SDK] No current experiment is selected.')\n }\n return id\n }\n\n async function fetchExperiment(id = experimentId.value): Promise<TExperiment | undefined> {\n if (id === undefined) {\n experiment.value = currentExperimentFromContext<TExperiment>()\n return experiment.value\n }\n\n try {\n const result = await request.run(\n () => api.get<TExperiment>(`/experiments/${id}`),\n { success: 'load', errorMessage: 'Failed to load current experiment' },\n )\n experiment.value = result\n return result\n } catch (e) {\n experiment.value = undefined\n return undefined\n }\n }\n\n async function refresh(): Promise<TExperiment | undefined> {\n return fetchExperiment()\n }\n\n watch(\n experimentId,\n (id) => {\n if (options.immediate === false) return\n if (id === undefined) {\n experiment.value = currentExperimentFromContext<TExperiment>()\n return\n }\n void fetchExperiment(id)\n },\n { immediate: options.immediate !== false },\n )\n\n return {\n experimentId,\n hasExperiment,\n experiment,\n isLoading,\n error,\n lastLoadedAt,\n requireExperimentId,\n fetch: fetchExperiment,\n refresh,\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"composables-wNt7VtkF.js","names":[],"sources":["../src/composables/useAuth.ts","../src/composables/usePasskey.ts","../src/composables/useAsync.ts","../src/composables/usePluginConfig.ts","../src/composables/pluginEndpointBuilder.ts","../src/composables/usePluginClient.ts"],"sourcesContent":["import axios from 'axios'\nimport { ref, onMounted, onUnmounted, watch, getCurrentInstance, type Ref } from 'vue'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, UpdateProfileRequest } from '../types'\nimport type { RoleInfo } from '../permissions'\n\ninterface UserResponse {\n id: string\n username: string\n shortname: string | null\n email: string | null\n role: string\n role_obj?: RoleInfo | null\n is_active: boolean\n}\n\ninterface RefreshResponse {\n access_token: string\n expires_in: number\n token_type: string\n}\n\n// Token refresh configuration\nconst TOKEN_REFRESH_MARGIN_MS = 5 * 60 * 1000 // Refresh 5 minutes before expiry\nconst TOKEN_REFRESH_CHECK_INTERVAL_MS = 60 * 1000 // Check every minute\n\n/**\n * Authentication composable with automatic token refresh.\n *\n * Features:\n * - Automatic token refresh before expiration\n * - Login/logout/register functionality\n * - Token verification on startup\n * - User profile management\n *\n * @example\n * ```typescript\n * const { login, logout, isAuthenticated, user } = useAuth()\n *\n * // Login\n * const success = await login('username', 'password')\n *\n * // Automatic refresh is enabled by default\n * // Tokens are refreshed 5 minutes before expiration\n * ```\n */\nexport interface UseAuthReturn {\n login: (username: string, password: string) => Promise<boolean>\n logout: () => void\n register: (username: string, password: string, email?: string) => Promise<boolean>\n verifyToken: () => Promise<boolean>\n fetchAuthConfig: () => Promise<AuthConfig>\n initializeAuth: () => Promise<void>\n getCurrentUser: () => Promise<UserInfo | null>\n getAuthHeader: () => Record<string, string>\n updateProfile: (data: { email?: string; shortname?: string; currentPassword?: string; newPassword?: string }) => Promise<{ success: boolean; error?: string }>\n refreshToken: () => Promise<boolean>\n isRefreshing: Ref<boolean>\n}\n\n// Module-level singletons to prevent duplicate refresh requests and timers\n// across multiple useAuth() instances\nlet _refreshPromise: Promise<boolean> | null = null\nlet _refreshTimerId: number | null = null\nlet _mountedConsumerCount = 0\n\n/** @internal Reset module-level state — only for use in tests. */\nexport function _resetAuthStateForTesting(): void {\n _refreshPromise = null\n if (_refreshTimerId !== null) {\n clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n _mountedConsumerCount = 0\n}\n\n/** Manages authentication state with login/logout/register and automatic JWT token refresh. */\nexport function useAuth(): UseAuthReturn {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n const isRefreshing = ref(false)\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function fetchAuthConfig(): Promise<AuthConfig> {\n try {\n const response = await axios.get<{\n auth_required: boolean\n passkey_enabled: boolean\n passkey_registered?: boolean\n registration_enabled?: boolean\n database_mode?: string\n access?: {\n experiment_visibility_mode?: 'open' | 'restricted'\n experimentVisibilityMode?: 'open' | 'restricted'\n }\n }>(`${getApiBaseUrl()}/setup/config/public`)\n\n const config: AuthConfig = {\n authRequired: response.data.auth_required,\n passkeyEnabled: response.data.passkey_enabled,\n passkeyRegistered: response.data.passkey_registered ?? false,\n registrationEnabled: response.data.registration_enabled ?? false,\n databaseMode: response.data.database_mode ?? 'none',\n experimentVisibilityMode:\n response.data.access?.experiment_visibility_mode\n ?? response.data.access?.experimentVisibilityMode\n ?? 'open',\n }\n\n authStore.setAuthConfig(config)\n return config\n } catch (error) {\n console.error('Failed to fetch auth config:', error)\n return {\n authRequired: false,\n passkeyEnabled: false,\n passkeyRegistered: false,\n registrationEnabled: false,\n databaseMode: 'none',\n experimentVisibilityMode: 'open',\n }\n }\n }\n\n async function login(username: string, password: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const response = await axios.post<LoginResponse>(\n `${getApiBaseUrl()}/auth/login`,\n { username, password }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n authStore.setUsername(username)\n\n await getCurrentUser()\n\n // Start auto-refresh after successful login\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Login failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function register(username: string, password: string, email?: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n await axios.post<UserResponse>(\n `${getApiBaseUrl()}/users/register`,\n { username, password, email }\n )\n\n return await login(username, password)\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Registration failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function getCurrentUser(): Promise<UserInfo | null> {\n if (!authStore.token) {\n return null\n }\n\n try {\n const response = await axios.get<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n roleObj: response.data.role_obj ?? null,\n isActive: response.data.is_active,\n }\n\n authStore.setUserInfo(userInfo)\n return userInfo\n } catch {\n return null\n }\n }\n\n async function verifyToken(): Promise<boolean> {\n if (!authStore.token) {\n return false\n }\n\n try {\n const response = await axios.get<TokenVerifyResponse>(\n `${getApiBaseUrl()}/auth/verify`,\n {\n headers: {\n Authorization: `Bearer ${authStore.token}`,\n },\n }\n )\n\n if (response.data.valid && response.data.username) {\n authStore.setUsername(response.data.username)\n return true\n }\n\n authStore.clearToken()\n return false\n } catch {\n authStore.clearToken()\n return false\n }\n }\n\n /**\n * Refresh the authentication token.\n * Called automatically before token expiration.\n * Uses promise caching to prevent concurrent refresh requests.\n */\n async function refreshToken(): Promise<boolean> {\n if (!authStore.token) return false\n if (_refreshPromise) return _refreshPromise\n\n _refreshPromise = (async () => {\n isRefreshing.value = true\n\n try {\n const response = await axios.post<RefreshResponse>(\n `${getApiBaseUrl()}/auth/refresh`,\n {},\n { headers: getAuthHeader() }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n // Reschedule next refresh\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n // If refresh fails, the token may have been revoked\n // Clear auth state and let user re-login\n if (axios.isAxiosError(error) && error.response?.status === 401) {\n console.warn('[Auth] Token refresh failed - session expired')\n authStore.clearToken()\n stopTokenRefresh()\n }\n return false\n } finally {\n isRefreshing.value = false\n _refreshPromise = null\n }\n })()\n\n return _refreshPromise\n }\n\n /**\n * Schedule automatic token refresh before expiration.\n */\n function scheduleTokenRefresh(): void {\n // Clear any existing timer\n stopTokenRefresh()\n\n if (!authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (refreshAt <= now) {\n // Token is already close to expiring or expired, refresh now\n refreshToken()\n return\n }\n\n // Schedule refresh\n const delay = refreshAt - now\n _refreshTimerId = window.setTimeout(() => {\n refreshToken()\n }, delay)\n }\n\n /**\n * Stop automatic token refresh.\n */\n function stopTokenRefresh(): void {\n if (_refreshTimerId !== null) {\n window.clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n }\n\n /**\n * Check if token needs refresh and refresh if necessary.\n * Called periodically as a safety net.\n */\n function checkAndRefreshIfNeeded(): void {\n if (!authStore.token || !authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (now >= refreshAt) {\n refreshToken()\n }\n }\n\n async function initializeAuth(): Promise<void> {\n authStore.initialize()\n await fetchAuthConfig()\n\n if (authStore.token) {\n const valid = await verifyToken()\n if (valid) {\n await getCurrentUser()\n // Start auto-refresh for existing session\n scheduleTokenRefresh()\n }\n }\n }\n\n function logout(): void {\n stopTokenRefresh()\n authStore.logout()\n }\n\n function getAuthHeader(): Record<string, string> {\n if (authStore.token) {\n return { Authorization: `Bearer ${authStore.token}` }\n }\n return {}\n }\n\n async function updateProfile(data: {\n email?: string\n shortname?: string\n currentPassword?: string\n newPassword?: string\n }): Promise<{ success: boolean; error?: string }> {\n if (!authStore.token) {\n return { success: false, error: 'Not authenticated' }\n }\n\n try {\n const requestData: UpdateProfileRequest = {}\n if (data.email !== undefined) requestData.email = data.email\n if (data.shortname !== undefined) requestData.shortname = data.shortname\n if (data.currentPassword) requestData.current_password = data.currentPassword\n if (data.newPassword) requestData.new_password = data.newPassword\n\n const response = await axios.put<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n requestData,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n roleObj: response.data.role_obj ?? null,\n isActive: response.data.is_active,\n }\n authStore.setUserInfo(userInfo)\n\n return { success: true }\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n return { success: false, error: error.response.data.detail || 'Update failed' }\n }\n return { success: false, error: 'Network error. Please try again.' }\n }\n }\n\n // Set up periodic check as safety net (only inside component setup)\n let checkInterval: number | null = null\n\n if (getCurrentInstance()) {\n onMounted(() => {\n _mountedConsumerCount += 1\n checkInterval = window.setInterval(checkAndRefreshIfNeeded, TOKEN_REFRESH_CHECK_INTERVAL_MS)\n })\n\n onUnmounted(() => {\n if (checkInterval !== null) {\n window.clearInterval(checkInterval)\n checkInterval = null\n }\n\n _mountedConsumerCount = Math.max(0, _mountedConsumerCount - 1)\n if (_mountedConsumerCount === 0) {\n stopTokenRefresh()\n }\n })\n\n // Watch for token changes to reschedule refresh\n watch(\n () => authStore.tokenExpires,\n (newExpires) => {\n if (newExpires) {\n scheduleTokenRefresh()\n } else {\n stopTokenRefresh()\n }\n }\n )\n }\n\n return {\n // Core auth methods\n login,\n logout,\n register,\n verifyToken,\n fetchAuthConfig,\n initializeAuth,\n getCurrentUser,\n getAuthHeader,\n updateProfile,\n\n // Token refresh\n refreshToken,\n isRefreshing,\n }\n}\n","import axios from 'axios'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { CredentialInfo } from '../types'\n\n// Lazy-load @simplewebauthn/browser so plugins that never call usePasskey\n// don't need the optional peer dep installed (avoids build failures).\nasync function loadWebAuthn() {\n try {\n return await import('@simplewebauthn/browser')\n } catch {\n throw new Error(\n '@simplewebauthn/browser is required for passkey support. Install it: bun add @simplewebauthn/browser',\n )\n }\n}\n\ninterface PasskeyLoginResponse {\n access_token: string\n token_type: string\n expires_in: number\n}\n\n/** Registers and authenticates passkeys using WebAuthn, lazily loading the browser dependency. */\nexport function usePasskey() {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function isSupported(): Promise<boolean> {\n try {\n const { browserSupportsWebAuthn } = await loadWebAuthn()\n return browserSupportsWebAuthn()\n } catch {\n return false\n }\n }\n\n async function registerPasskey(deviceName?: string): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/register/options`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startRegistration(options)\n\n await axios.post(\n `${getApiBaseUrl()}/auth/passkey/register/verify`,\n {\n credential: JSON.stringify(credential),\n device_name: deviceName,\n },\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: true,\n })\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Passkey registration failed')\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Registration was cancelled or timed out')\n } else if (error.name === 'InvalidStateError') {\n authStore.setError('This authenticator is already registered')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey registration failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function loginWithPasskey(): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/login/options`,\n { withCredentials: true }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startAuthentication(options)\n\n const response = await axios.post<PasskeyLoginResponse>(\n `${getApiBaseUrl()}/auth/passkey/login/verify`,\n { credential: JSON.stringify(credential) },\n { withCredentials: true }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n if (error.response.status === 404) {\n authStore.setError('No passkeys registered. Please login with password first.')\n } else {\n authStore.setError(error.response.data.detail || 'Passkey login failed')\n }\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Authentication was cancelled or timed out')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey login failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function listCredentials(): Promise<CredentialInfo[]> {\n try {\n const response = await axios.get<{ credentials: CredentialInfo[] }>(\n `${getApiBaseUrl()}/auth/passkey/credentials`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n return response.data.credentials\n } catch {\n return []\n }\n }\n\n async function deleteCredential(credentialId: string): Promise<boolean> {\n try {\n await axios.delete(\n `${getApiBaseUrl()}/auth/passkey/credentials/${encodeURIComponent(credentialId)}`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n\n const remaining = await listCredentials()\n if (remaining.length === 0) {\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n }\n\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAllCredentials(): Promise<boolean> {\n try {\n await axios.delete(`${getApiBaseUrl()}/auth/passkey/credentials`, {\n headers: { Authorization: `Bearer ${authStore.token}` },\n })\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSupported,\n registerPasskey,\n loginWithPasskey,\n listCredentials,\n deleteCredential,\n deleteAllCredentials,\n }\n}\n","import { ref, computed, type Ref, type ComputedRef } from 'vue'\n\n/**\n * Error type for async operations.\n */\nexport interface AsyncError {\n message: string\n code?: string\n details?: Record<string, unknown>\n originalError?: unknown\n}\n\n/**\n * State of an async operation.\n */\nexport type AsyncState = 'idle' | 'loading' | 'success' | 'error'\n\n/**\n * Return type for useAsync composable.\n */\nexport interface UseAsyncReturn<T> {\n // Data\n data: Ref<T | null>\n error: Ref<AsyncError | null>\n\n // State\n state: Ref<AsyncState>\n isIdle: ComputedRef<boolean>\n isLoading: ComputedRef<boolean>\n isSuccess: ComputedRef<boolean>\n isError: ComputedRef<boolean>\n\n // Methods\n execute: (...args: unknown[]) => Promise<T | null>\n reset: () => void\n setData: (data: T | null) => void\n setError: (error: AsyncError | null) => void\n}\n\n/**\n * Options for useAsync.\n */\nexport interface UseAsyncOptions<T> {\n // Initial data value\n initialData?: T | null\n\n // Whether to run immediately on creation\n immediate?: boolean\n\n // Arguments for immediate execution\n immediateArgs?: unknown[]\n\n // Transform error into AsyncError\n transformError?: (error: unknown) => AsyncError\n\n // Called on success\n onSuccess?: (data: T) => void\n\n // Called on error\n onError?: (error: AsyncError) => void\n\n // Reset data on new execution\n resetOnExecute?: boolean\n}\n\n/**\n * Default error transformer.\n */\nfunction defaultTransformError(error: unknown): AsyncError {\n if (error instanceof Error) {\n return {\n message: error.message,\n originalError: error,\n }\n }\n\n if (typeof error === 'object' && error !== null) {\n const errorObj = error as Record<string, unknown>\n\n // Handle Axios-like errors\n if ('response' in errorObj && errorObj.response) {\n const response = errorObj.response as Record<string, unknown>\n const data = response.data as Record<string, unknown> | undefined\n\n return {\n message: (data?.detail as string) || (data?.message as string) || 'Request failed',\n code: String(response.status),\n details: data,\n originalError: error,\n }\n }\n\n // Handle generic error objects\n return {\n message: (errorObj.message as string) || 'Unknown error',\n code: errorObj.code as string | undefined,\n details: errorObj,\n originalError: error,\n }\n }\n\n return {\n message: String(error),\n originalError: error,\n }\n}\n\n/**\n * Composable for managing async operation state.\n *\n * Provides standardized loading, error, and success state management\n * for any async function.\n *\n * @param asyncFn - The async function to wrap\n * @param options - Configuration options\n *\n * @example\n * ```typescript\n * // Basic usage\n * const { data, isLoading, error, execute } = useAsync(\n * async (id: string) => {\n * const response = await api.get(`/users/${id}`)\n * return response.data\n * }\n * )\n *\n * // Execute the function\n * await execute('user-123')\n *\n * // In template\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error.message }}</div>\n * <div v-else-if=\"data\">{{ data.name }}</div>\n * ```\n *\n * @example\n * ```typescript\n * // With options\n * const { data, execute } = useAsync(\n * fetchUser,\n * {\n * immediate: true,\n * immediateArgs: ['default-user'],\n * onSuccess: (user) => console.log('Fetched:', user.name),\n * onError: (error) => toast.error(error.message),\n * }\n * )\n * ```\n *\n * @example\n * ```typescript\n * // Form submission\n * const { isLoading, error, execute: submit } = useAsync(\n * async (formData: FormData) => {\n * await api.post('/submit', formData)\n * },\n * {\n * onSuccess: () => {\n * toast.success('Submitted!')\n * router.push('/success')\n * },\n * }\n * )\n *\n * const handleSubmit = () => submit(new FormData(formRef.value))\n * ```\n */\n/** Wraps an async function with reactive loading, error, and success state plus optional auto-execute. */\nexport function useAsync<T>(\n asyncFn: (...args: unknown[]) => Promise<T>,\n options: UseAsyncOptions<T> = {}\n): UseAsyncReturn<T> {\n const {\n initialData = null,\n immediate = false,\n immediateArgs = [],\n transformError = defaultTransformError,\n onSuccess,\n onError,\n resetOnExecute = false,\n } = options\n\n // State\n const data = ref<T | null>(initialData) as Ref<T | null>\n const error = ref<AsyncError | null>(null)\n const state = ref<AsyncState>('idle')\n\n // Computed state helpers\n const isIdle = computed(() => state.value === 'idle')\n const isLoading = computed(() => state.value === 'loading')\n const isSuccess = computed(() => state.value === 'success')\n const isError = computed(() => state.value === 'error')\n\n // Execute the async function\n async function execute(...args: unknown[]): Promise<T | null> {\n state.value = 'loading'\n error.value = null\n\n if (resetOnExecute) {\n data.value = null\n }\n\n try {\n const result = await asyncFn(...args)\n data.value = result\n state.value = 'success'\n onSuccess?.(result)\n return result\n } catch (e) {\n const asyncError = transformError(e)\n error.value = asyncError\n state.value = 'error'\n onError?.(asyncError)\n return null\n }\n }\n\n // Reset to initial state\n function reset(): void {\n data.value = initialData\n error.value = null\n state.value = 'idle'\n }\n\n // Manual data setter\n function setData(newData: T | null): void {\n data.value = newData\n if (newData !== null) {\n state.value = 'success'\n error.value = null\n }\n }\n\n // Manual error setter\n function setError(newError: AsyncError | null): void {\n error.value = newError\n if (newError !== null) {\n state.value = 'error'\n }\n }\n\n // Execute immediately if requested\n if (immediate) {\n execute(...immediateArgs)\n }\n\n return {\n data,\n error,\n state,\n isIdle,\n isLoading,\n isSuccess,\n isError,\n execute,\n reset,\n setData,\n setError,\n }\n}\n\n/**\n * Create a batch of async operations that can be executed in parallel.\n *\n * @example\n * ```typescript\n * const { results, isLoading, execute } = useAsyncBatch([\n * () => fetchUser(userId),\n * () => fetchPosts(userId),\n * () => fetchComments(userId),\n * ])\n *\n * await execute()\n * // results.value = [user, posts, comments]\n * ```\n */\nexport function useAsyncBatch<T extends readonly (() => Promise<unknown>)[]>(\n asyncFns: T\n): {\n results: Ref<{ [K in keyof T]: Awaited<ReturnType<T[K]>> | null }>\n errors: Ref<(AsyncError | null)[]>\n isLoading: Ref<boolean>\n execute: () => Promise<void>\n reset: () => void\n} {\n type Results = { [K in keyof T]: Awaited<ReturnType<T[K]>> | null }\n\n const results = ref<Results>(asyncFns.map(() => null) as unknown as Results) as Ref<Results>\n const errors = ref<(AsyncError | null)[]>(asyncFns.map(() => null))\n const isLoading = ref(false)\n\n async function execute(): Promise<void> {\n isLoading.value = true\n errors.value = asyncFns.map(() => null)\n\n const promises = asyncFns.map(async (fn, index) => {\n try {\n const result = await fn()\n ;(results.value as unknown[])[index] = result\n } catch (e) {\n errors.value[index] = defaultTransformError(e)\n ;(results.value as unknown[])[index] = null\n }\n })\n\n await Promise.all(promises)\n isLoading.value = false\n }\n\n function reset(): void {\n results.value = asyncFns.map(() => null) as unknown as Results\n errors.value = asyncFns.map(() => null)\n isLoading.value = false\n }\n\n return {\n results,\n errors,\n isLoading,\n execute,\n reset,\n }\n}\n","import { ref, computed, onMounted, type Ref, type ComputedRef } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\nimport { useRequestSyncState } from './useRequestSyncState'\n\nexport interface UsePluginConfigReturn {\n /** Editable plugin configuration values. */\n config: Ref<Record<string, unknown>>\n /** Whether the config is currently loading. */\n isLoading: Ref<boolean>\n /** Whether the config is currently saving. */\n isSaving: Ref<boolean>\n /** Error message from the last failed operation, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful load, or null. */\n lastLoadedAt: Ref<Date | null>\n /** Timestamp of the last successful save, or null. */\n lastSavedAt: Ref<Date | null>\n /** Whether local config differs from the last loaded or saved snapshot. */\n isDirty: ComputedRef<boolean>\n /** Load plugin configuration from the platform. */\n load: () => Promise<void>\n /** Save plugin configuration to the platform. */\n save: () => Promise<boolean>\n /** Restore the last loaded or saved configuration snapshot. */\n reset: () => void\n}\n\n/** Loads, saves, and tracks dirty state for a plugin's persistent configuration via the platform API. */\nexport function usePluginConfig(pluginName?: string): UsePluginConfigReturn {\n const api = useApi()\n const { plugin } = usePlatformContext()\n\n const resolvedName = computed(() => pluginName ?? plugin.value?.name ?? '')\n\n const config = ref<Record<string, unknown>>({})\n const savedConfig = ref<Record<string, unknown>>({})\n const request = useRequestSyncState('Plugin config request failed.')\n const isLoading = ref(false)\n const isSaving = ref(false)\n const error = request.error\n const lastLoadedAt = request.lastLoadedAt\n const lastSavedAt = request.lastSavedAt\n\n const isDirty = computed(() => {\n return JSON.stringify(config.value) !== JSON.stringify(savedConfig.value)\n })\n\n async function load(): Promise<void> {\n const name = resolvedName.value\n if (!name) return\n\n isLoading.value = true\n try {\n const response = await request.run(\n () => api.get<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n ),\n { success: 'load', errorMessage: 'Failed to load plugin config' },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n } catch {\n // Error state is handled by request.run().\n } finally {\n isLoading.value = false\n }\n }\n\n async function save(): Promise<boolean> {\n const name = resolvedName.value\n if (!name) return false\n\n isSaving.value = true\n try {\n const response = await request.run(\n () => api.patch<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n { config: config.value },\n ),\n { success: 'save', errorMessage: 'Failed to save plugin config' },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n return true\n } catch {\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n function reset(): void {\n config.value = { ...savedConfig.value }\n request.clearError()\n }\n\n onMounted(() => {\n load()\n })\n\n return {\n config,\n isLoading,\n isSaving,\n error,\n lastLoadedAt,\n lastSavedAt,\n isDirty,\n load,\n save,\n reset,\n }\n}\n","import {\n getInjectedPlatformContext,\n resolveCurrentExperimentId,\n} from './platformContextHelpers'\nimport type {\n BuildPluginEndpointUrlOptions,\n PluginContract,\n PluginEndpointDefinition,\n PluginEndpointParamDefinition,\n} from './usePluginClient'\n\nfunction normalizeApiBaseUrl(baseUrl: string | undefined): string | undefined {\n if (!baseUrl) return undefined\n return baseUrl.length > 1 ? baseUrl.replace(/\\/+$/, '') : baseUrl\n}\n\nfunction normalizeApiPrefix(prefix: string | undefined): string | undefined {\n if (!prefix) return undefined\n if (prefix.startsWith('/api/')) return normalizeApiBaseUrl(prefix)\n if (prefix === '/api') return prefix\n if (prefix.startsWith('/')) return normalizeApiBaseUrl(`/api${prefix}`)\n return normalizeApiBaseUrl(`/api/${prefix}`)\n}\n\n/** Resolve the runtime plugin API base URL from env, explicit options, platform injection, or contract metadata. */\nexport function resolvePluginBaseUrl(contract: PluginContract, explicitBaseUrl?: string): string {\n const envPrefix = (import.meta.env?.VITE_API_PREFIX as string | undefined) || undefined\n if (envPrefix) return normalizeApiBaseUrl(envPrefix) ?? envPrefix\n if (explicitBaseUrl) return normalizeApiBaseUrl(explicitBaseUrl) ?? explicitBaseUrl\n\n const injected = getInjectedPlatformContext()\n const platformPrefix =\n normalizeApiBaseUrl(injected?.plugin?.api_prefix) ||\n normalizeApiPrefix(injected?.plugin?.route_prefix)\n if (platformPrefix) return platformPrefix\n\n return normalizeApiBaseUrl(contract.plugin.apiPrefix) || '/api'\n}\n\nfunction encodePath(path: string, payload: Record<string, unknown>, pathParams: string[]): string {\n let nextPath = path\n for (const param of pathParams) {\n const value = payload[param]\n if (value === undefined || value === null) {\n throw new Error(`[MINT SDK] Missing path parameter '${param}' for plugin endpoint ${path}`)\n }\n nextPath = nextPath.replace(`{${param}}`, encodeURIComponent(String(value)))\n nextPath = nextPath.replace(`:${param}`, encodeURIComponent(String(value)))\n }\n return nextPath\n}\n\nfunction withDefaultPathParams(\n payload: Record<string, unknown>,\n pathParams: string[],\n): Record<string, unknown> {\n if (!pathParams.length) return payload\n const nextPayload = { ...payload }\n for (const param of pathParams) {\n if (nextPayload[param] !== undefined && nextPayload[param] !== null) continue\n if (isExperimentPathParam(param)) {\n const experimentId = resolveCurrentExperimentId()\n if (experimentId !== undefined) {\n nextPayload[param] = experimentId\n }\n }\n }\n return nextPayload\n}\n\nfunction isExperimentPathParam(param: string): boolean {\n return param === 'experimentId' || param === 'experiment_id'\n}\n\nfunction withoutPathParams(payload: Record<string, unknown>, pathParams: string[]) {\n const copy = { ...payload }\n for (const param of pathParams) {\n delete copy[param]\n }\n return copy\n}\n\nfunction queryParamsFromPayload(\n payload: Record<string, unknown>,\n pathParams: string[],\n queryParams?: PluginEndpointParamDefinition[],\n endpointPath?: string,\n includeUnspecifiedParams = false,\n) {\n if (!queryParams?.length) {\n return includeUnspecifiedParams ? withoutPathParams(payload, pathParams) : {}\n }\n\n const query: Record<string, unknown> = {}\n for (const param of queryParams) {\n const name = typeof param === 'string' ? param : param.name\n const fieldName = typeof param === 'string' ? param : param.fieldName ?? param.name\n const required = typeof param !== 'string' && param.required === true\n const value = payload[fieldName]\n if (value === undefined || value === null) {\n if (required) {\n throw new Error(\n `[MINT SDK] Missing query parameter '${fieldName}' for plugin endpoint ${endpointPath ?? ''}`,\n )\n }\n continue\n }\n query[name] = value\n }\n return query\n}\n\nexport function hasKeys(value: Record<string, unknown>): boolean {\n return Object.keys(value).length > 0\n}\n\nfunction isPlainPayloadObject(value: unknown): value is Record<string, unknown> {\n return !!value && typeof value === 'object' && !Array.isArray(value)\n}\n\nfunction hasOwn(value: Record<string, unknown>, key: string): boolean {\n return Object.prototype.hasOwnProperty.call(value, key)\n}\n\nfunction normalizeStructuredEndpointPayload(payload: Record<string, unknown>): Record<string, unknown> {\n const pathParamPayload = payload.pathParams\n const queryPayload = payload.query\n const hasStructuredPathParams = hasOwn(payload, 'pathParams') && isPlainPayloadObject(pathParamPayload)\n const hasStructuredQuery = hasOwn(payload, 'query') && isPlainPayloadObject(queryPayload)\n if (!hasStructuredPathParams && !hasStructuredQuery) return payload\n\n const nextPayload = { ...payload }\n delete nextPayload.pathParams\n delete nextPayload.query\n\n if (hasStructuredPathParams) {\n Object.assign(nextPayload, pathParamPayload)\n }\n if (hasStructuredQuery) {\n Object.assign(nextPayload, queryPayload)\n }\n\n return nextPayload\n}\n\nexport function requestBodyFromPayload(\n payload: unknown,\n requestPayload: Record<string, unknown>,\n endpoint: PluginEndpointDefinition,\n): unknown {\n if (!endpoint.hasBody) return undefined\n const body = endpoint.pathParams?.length || endpoint.queryParams?.length\n ? requestPayload.body\n : payload\n if (body === undefined || body === null) {\n throw new Error(`[MINT SDK] Missing request body for plugin endpoint ${endpoint.path}`)\n }\n return body\n}\n\nexport function pluginEndpointRequestParts(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n explicitBaseUrl?: string,\n) {\n const payloadObject = isPlainPayloadObject(payload)\n ? normalizeStructuredEndpointPayload(payload)\n : {}\n const pathParams = endpoint.pathParams ?? []\n const requestPayload = withDefaultPathParams(payloadObject, pathParams)\n const path = encodePath(endpoint.path, requestPayload, pathParams)\n const query = queryParamsFromPayload(\n requestPayload,\n pathParams,\n endpoint.queryParams,\n endpoint.path,\n endpoint.method === 'get',\n )\n\n return {\n baseUrl: resolvePluginBaseUrl(contract, explicitBaseUrl),\n path,\n query,\n requestPayload,\n }\n}\n\nfunction appendQueryValue(params: URLSearchParams, key: string, value: unknown): void {\n if (value === undefined || value === null) return\n if (Array.isArray(value)) {\n for (const item of value) {\n appendQueryValue(params, key, item)\n }\n return\n }\n if (typeof value === 'object') {\n params.append(key, JSON.stringify(value))\n return\n }\n params.append(key, String(value))\n}\n\nfunction queryString(query: Record<string, unknown>): string {\n const params = new URLSearchParams()\n for (const [key, value] of Object.entries(query)) {\n appendQueryValue(params, key, value)\n }\n const serialized = params.toString()\n return serialized ? `?${serialized}` : ''\n}\n\nfunction joinBaseUrl(baseUrl: string, path: string): string {\n if (!baseUrl) return path\n const normalizedBase = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl\n const normalizedPath = path.startsWith('/') ? path : `/${path}`\n return `${normalizedBase}${normalizedPath}`\n}\n\n/** Build the concrete URL for a generated plugin endpoint without making a request. */\nexport function buildPluginEndpointUrl(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n options: BuildPluginEndpointUrlOptions = {},\n): string {\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const pathWithQuery = `${parts.path}${queryString(parts.query)}`\n if (options.includeBaseUrl === false) return pathWithQuery\n return joinBaseUrl(parts.baseUrl, pathWithQuery)\n}\n","import { computed, shallowRef, watch, type ComputedRef, type Ref } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\nimport { useRequestSyncState } from './useRequestSyncState'\nimport {\n currentExperimentFromContext,\n getInjectedPlatformContext,\n resolveCurrentExperimentId,\n} from './platformContextHelpers'\nimport {\n hasKeys,\n pluginEndpointRequestParts,\n requestBodyFromPayload,\n} from './pluginEndpointBuilder'\nimport type { ExperimentSummary, PageSelectorItem } from '../types'\n\nexport {\n buildPluginEndpointUrl,\n resolvePluginBaseUrl,\n} from './pluginEndpointBuilder'\n\nexport type PluginHttpMethod = 'get' | 'post' | 'put' | 'patch' | 'delete'\n\nexport interface PluginEndpointContract {\n name: string\n method: PluginHttpMethod\n path: string\n pathParams?: string[]\n queryParams?: PluginEndpointParamContract[]\n requestType?: string | null\n responseType?: string | null\n}\n\nexport interface PluginEndpointParamContract {\n name: string\n fieldName: string\n type?: string\n required?: boolean\n}\n\nexport interface PluginNavItemContract {\n path: string\n label: string\n id?: string\n icon?: string\n description?: string\n requiresAuth?: boolean\n requiresAdmin?: boolean\n requiresFeature?: string\n}\n\nexport interface PluginContract {\n schemaVersion: number\n plugin: {\n name?: string\n packageName?: string\n version?: string\n description?: string\n routesPrefix: string\n apiPrefix: string\n type: 'analysis' | 'experiment-design' | string\n analysisType?: string\n icon?: string\n color?: string\n navItems?: PluginNavItemContract[]\n capabilities?: Record<string, unknown>\n }\n endpoints: PluginEndpointContract[]\n hash: string\n}\n\nexport interface PluginEndpointDefinition {\n method: PluginHttpMethod\n path: string\n pathParams?: string[]\n queryParams?: PluginEndpointParamDefinition[]\n hasBody?: boolean\n}\n\nexport type PluginEndpointParamDefinition = string | {\n name: string\n fieldName?: string\n type?: string\n required?: boolean\n}\n\nexport interface CreatePluginClientOptions {\n endpoints: Record<string, PluginEndpointDefinition>\n baseUrl?: string\n}\n\nexport interface BuildPluginEndpointUrlOptions {\n /** Override the generated or platform-injected API base URL. */\n baseUrl?: string\n /** Return only the endpoint path and query string instead of base URL + path. */\n includeBaseUrl?: boolean\n}\n\nexport interface PluginEndpointRequestOptions {\n /** Override the generated or platform-injected API base URL. */\n baseUrl?: string\n}\n\nexport interface DownloadPluginEndpointOptions extends PluginEndpointRequestOptions {\n /** Browser download filename. When omitted, the blob is returned without triggering a download. */\n filename?: string\n /** Delay before revoking the object URL after a browser download is triggered. */\n revokeObjectUrlDelayMs?: number\n}\n\nexport interface DownloadBlobOptions {\n /** Delay before revoking the object URL after a browser download is triggered. */\n revokeObjectUrlDelayMs?: number\n}\n\nexport type PluginFormDataValue =\n | string\n | number\n | boolean\n | Blob\n | null\n | undefined\n | Record<string, unknown>\n\nexport type PluginFormDataPayload = Record<string, PluginFormDataValue | PluginFormDataValue[]>\n\nexport interface UseCurrentExperimentOptions {\n apiBaseUrl?: string\n immediate?: boolean\n}\n\nexport interface UseCurrentExperimentReturn<TExperiment = ExperimentSummary> {\n /** Current experiment id resolved from platform injection or URL conventions. */\n experimentId: ComputedRef<number | undefined>\n /** Whether a current experiment id is available. */\n hasExperiment: ComputedRef<boolean>\n /** Current experiment payload, if injected or successfully fetched. */\n experiment: Ref<TExperiment | undefined>\n /** Whether the current experiment is currently loading. */\n isLoading: Ref<boolean>\n /** Error message from the last failed current-experiment fetch, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful current-experiment fetch, or null. */\n lastLoadedAt: Ref<Date | null>\n /** Return the current experiment id or throw a clear SDK error when absent. */\n requireExperimentId: () => number\n /** Fetch a specific experiment id, defaulting to the current experiment id. */\n fetch: (experimentId?: number) => Promise<TExperiment | undefined>\n /** Refetch the currently resolved experiment id. */\n refresh: () => Promise<TExperiment | undefined>\n}\n\nexport type PluginEndpointCaller = (payload?: unknown) => Promise<unknown>\nexport type GeneratedPluginClient = Record<string, (...args: any[]) => Promise<any>>\n\nfunction queryConfig(query: Record<string, unknown>) {\n return hasKeys(query) ? { params: query } : undefined\n}\n\nfunction isFormData(value: unknown): value is FormData {\n return typeof FormData !== 'undefined' && value instanceof FormData\n}\n\nfunction isBlob(value: unknown): value is Blob {\n return typeof Blob !== 'undefined' && value instanceof Blob\n}\n\nfunction appendFormDataValue(formData: FormData, key: string, value: PluginFormDataValue | PluginFormDataValue[]): void {\n if (value === undefined || value === null) return\n\n if (Array.isArray(value)) {\n for (const item of value) {\n appendFormDataValue(formData, key, item)\n }\n return\n }\n\n if (isBlob(value)) {\n formData.append(key, value)\n return\n }\n\n if (typeof value === 'object') {\n formData.append(key, JSON.stringify(value))\n return\n }\n\n formData.append(key, String(value))\n}\n\n/** Convert plain upload payloads into FormData using the same encoding across generated plugin clients. */\nexport function pluginFormDataFromPayload(payload: unknown): FormData {\n if (isFormData(payload)) return payload\n if (!payload || typeof payload !== 'object' || Array.isArray(payload)) {\n throw new Error('[MINT SDK] Plugin upload endpoints require a FormData or plain object payload.')\n }\n\n const formData = new FormData()\n for (const [key, value] of Object.entries(payload as PluginFormDataPayload)) {\n appendFormDataValue(formData, key, value)\n }\n return formData\n}\n\n/** Trigger a browser download for an already-created Blob. */\nexport function downloadBlob(blob: Blob, filename: string, options: DownloadBlobOptions = {}): void {\n if (typeof document === 'undefined' || typeof URL === 'undefined') {\n throw new Error('[MINT SDK] Cannot trigger a file download outside a browser.')\n }\n\n const blobUrl = URL.createObjectURL(blob)\n const link = document.createElement('a')\n link.href = blobUrl\n link.download = filename\n document.body.appendChild(link)\n link.click()\n document.body.removeChild(link)\n setTimeout(() => URL.revokeObjectURL(blobUrl), options.revokeObjectUrlDelayMs ?? 100)\n}\n\nfunction normalizePluginNavPath(path: string): string {\n const raw = path.trim() || '/'\n const prefixed = raw.startsWith('/') ? raw : `/${raw}`\n return prefixed.replace(/\\/+$/, '') || '/'\n}\n\nfunction pluginPageIdFromPath(path: string, fallbackIndex: number): string {\n const normalizedPath = normalizePluginNavPath(path)\n if (normalizedPath === '/') return 'dashboard'\n return normalizedPath.replace(/^\\/+/, '').replace(/\\/+/g, '-') || `page-${fallbackIndex + 1}`\n}\n\n/** Convert PluginMetadata.nav_items into AppTopBar pageSelector items for topbar page switches, homepage PluginCards, and fallback views. */\nexport function getPluginPageSelectorItems(contract: PluginContract): PageSelectorItem[] {\n const pluginIcon = contract.plugin.icon ?? ''\n const navItems = contract.plugin.navItems ?? []\n const source: PluginNavItemContract[] = navItems.length > 0\n ? navItems\n : [{\n path: '/',\n label: contract.plugin.name ?? 'Dashboard',\n id: 'dashboard',\n icon: pluginIcon || undefined,\n description: contract.plugin.description,\n }]\n\n return source.map((item, index) => ({\n id: item.id || pluginPageIdFromPath(item.path, index),\n label: item.label,\n to: normalizePluginNavPath(item.path),\n icon: item.icon || pluginIcon || undefined,\n hint: item.description || contract.plugin.name,\n }))\n}\n\n/** Create a typed plugin API client from a generated MINT plugin contract. */\nexport function createPluginClient<TClient extends object = GeneratedPluginClient>(\n contract: PluginContract,\n options: CreatePluginClientOptions,\n): TClient {\n const client: Record<string, PluginEndpointCaller> = {}\n\n for (const [name, endpoint] of Object.entries(options.endpoints)) {\n client[name] = async (payload?: unknown) => {\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const api = useApi({ baseUrl: parts.baseUrl })\n const queryConfig = hasKeys(parts.query) ? { params: parts.query } : undefined\n\n if (endpoint.method === 'get') {\n return api.get(parts.path, queryConfig)\n }\n\n if (endpoint.method === 'delete') {\n const body = requestBodyFromPayload(payload, parts.requestPayload, endpoint)\n const config = body === undefined\n ? queryConfig\n : { ...queryConfig, data: body }\n return api.delete(parts.path, config)\n }\n\n const body = requestBodyFromPayload(payload, parts.requestPayload, endpoint)\n\n if (endpoint.method === 'post') return api.post(parts.path, body, queryConfig)\n if (endpoint.method === 'put') return api.put(parts.path, body, queryConfig)\n if (endpoint.method === 'patch') return api.patch(parts.path, body, queryConfig)\n throw new Error(`[MINT SDK] Unsupported plugin endpoint method: ${endpoint.method}`)\n }\n }\n\n return client as TClient\n}\n\n/** Send multipart form data to a generated plugin endpoint while reusing contract URL, query, and auth handling. */\nexport async function uploadPluginEndpoint<TResponse = unknown>(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n options: PluginEndpointRequestOptions = {},\n): Promise<TResponse> {\n if (!['post', 'put', 'patch'].includes(endpoint.method)) {\n throw new Error(`[MINT SDK] Plugin upload endpoints must use POST, PUT, or PATCH; got ${endpoint.method}`)\n }\n\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const api = useApi({ baseUrl: parts.baseUrl })\n const body = endpoint.hasBody\n ? requestBodyFromPayload(payload, parts.requestPayload, endpoint)\n : payload\n const formData = pluginFormDataFromPayload(body)\n const config = {\n ...(queryConfig(parts.query) ?? {}),\n // Let Axios/browser set multipart boundaries.\n headers: { 'Content-Type': undefined },\n }\n\n if (endpoint.method === 'post') return api.post<TResponse>(parts.path, formData, config)\n if (endpoint.method === 'put') return api.put<TResponse>(parts.path, formData, config)\n return api.patch<TResponse>(parts.path, formData, config)\n}\n\n/** Fetch a generated plugin endpoint as a Blob and optionally trigger a browser download. */\nexport async function downloadPluginEndpoint(\n contract: PluginContract,\n endpoint: PluginEndpointDefinition,\n payload?: unknown,\n options: DownloadPluginEndpointOptions = {},\n): Promise<Blob> {\n const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)\n const api = useApi({ baseUrl: parts.baseUrl })\n const blob = await api.get<Blob>(parts.path, {\n ...(queryConfig(parts.query) ?? {}),\n responseType: 'blob',\n })\n\n if (options.filename) {\n downloadBlob(blob, options.filename, {\n revokeObjectUrlDelayMs: options.revokeObjectUrlDelayMs,\n })\n }\n\n return blob\n}\n\n/** Return a generated plugin client from setup code with a stable typed identity. */\nexport function usePluginClient<TClient>(client: TClient): TClient {\n return client\n}\n\n/** Read plugin settings exposed through the platform context. */\nexport function usePluginSettings<TSettings = Record<string, unknown>>() {\n const { plugin } = usePlatformContext()\n const settings = computed(() => plugin.value?.settings as TSettings | undefined)\n return { settings }\n}\n\n/** Read and optionally load the current platform experiment for integrated plugin views. */\nexport function useCurrentExperiment<TExperiment = ExperimentSummary>(\n options: UseCurrentExperimentOptions = {},\n): UseCurrentExperimentReturn<TExperiment> {\n const api = useApi({ baseUrl: options.apiBaseUrl ?? getInjectedPlatformContext()?.platformApiUrl })\n const experiment = shallowRef<TExperiment | undefined>(currentExperimentFromContext<TExperiment>())\n const request = useRequestSyncState('Failed to load current experiment')\n const isLoading = request.loading\n const error = request.error\n const lastLoadedAt = request.lastLoadedAt\n\n const experimentId = computed<number | undefined>(() => {\n return resolveCurrentExperimentId()\n })\n const hasExperiment = computed(() => experimentId.value !== undefined)\n\n function requireExperimentId(): number {\n const id = experimentId.value\n if (id === undefined) {\n throw new Error('[MINT SDK] No current experiment is selected.')\n }\n return id\n }\n\n async function fetchExperiment(id = experimentId.value): Promise<TExperiment | undefined> {\n if (id === undefined) {\n experiment.value = currentExperimentFromContext<TExperiment>()\n return experiment.value\n }\n\n try {\n const result = await request.run(\n () => api.get<TExperiment>(`/experiments/${id}`),\n { success: 'load', errorMessage: 'Failed to load current experiment' },\n )\n experiment.value = result\n return result\n } catch (e) {\n experiment.value = undefined\n return undefined\n }\n }\n\n async function refresh(): Promise<TExperiment | undefined> {\n return fetchExperiment()\n }\n\n watch(\n experimentId,\n (id) => {\n if (options.immediate === false) return\n if (id === undefined) {\n experiment.value = currentExperimentFromContext<TExperiment>()\n return\n }\n void fetchExperiment(id)\n },\n { immediate: options.immediate !== false },\n )\n\n return {\n experimentId,\n hasExperiment,\n experiment,\n isLoading,\n error,\n lastLoadedAt,\n requireExperimentId,\n fetch: fetchExperiment,\n refresh,\n }\n}\n"],"mappings":"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"}
|
package/dist/index.js
CHANGED
|
@@ -3,7 +3,7 @@ import { $ as toTimeCourseRows, $n as TimePicker_default, $t as listBioTemplateP
|
|
|
3
3
|
import { $ as resolveExperimentCode, B as useExperimentSelector, C as useExpansionSet, D as hslToHex, E as hexToHsl, G as EXPERIMENT_STATUS_LABELS, H as useDebouncedWatch, I as useWellPlateEditor, J as SORT_OPTIONS, K as EXPERIMENT_STATUS_OPTIONS, L as useDoseCalculator, M as extractSampleOptionsFromDesignData, N as unwrapExperimentDesignData, O as extractSamplesFromDesignData, P as DEFAULT_COLORS, Q as getExperimentStatusVariant, R as APP_EXPERIMENT_KEY, T as deriveShade, U as useApi, V as useRequestSyncState, W as DATE_PRESET_OPTIONS, X as formatExperimentDate, Y as datePresetToISO, Z as formatExperimentStatus, _ as useScheduleDrag, a as useReagentSeries, at as useToast, c as useBioTemplatePresetWorkspace, ct as useTextSearch, d as getBioTemplateComponentProps, et as usePlatformContext, f as toBioTemplateComponentPropsByComponent, g as useExperimentSave, h as useTemplateCollection, i as generateDilutionSeries, it as useTheme, j as extractSampleNamesFromDesignData, k as useAutoGroup, l as useBioTemplatePackWorkspace, lt as compareSortValues, m as useBioTemplateControls, n as DEFAULT_PRESETS, nt as evaluateCondition, o as useGroupAssignment, ot as candidateMatchesSearch, p as useBioTemplateComponents, q as EXPERIMENT_STATUS_VARIANT_MAP, r as DEFAULT_UNITS, rt as useForm, s as useRackEditor, st as normalizeSearchQuery, t as useProtocolTemplates, tt as useFormBuilder, u as useBioTemplateWorkspace, ut as useSortedItems, v as useExperimentSamples, w as useSampleGroups, y as useExperimentData, z as useAppExperiment } from "./useProtocolTemplates-DwBhEPPU.js";
|
|
4
4
|
import { a as canAccessByPolicy, c as getRoleInfo, d as hasAnyPermission, f as isAdminRole, g as useSettingsStore, h as colorPalettes, i as canAccessAdmin, l as getUserPermissions, m as normalizeAccessPolicy, n as ADMIN_PANEL_PERMISSIONS, o as canAccessPlugin, p as isAdminUser, r as ADMIN_ROLE, s as getAccessAudience, t as useAuthStore, u as hasAllPermissions } from "./auth-BulIv_km.js";
|
|
5
5
|
import { MINTSdk } from "./install.js";
|
|
6
|
-
import { a as
|
|
6
|
+
import { a as pluginFormDataFromPayload, c as usePluginClient, d as resolvePluginBaseUrl, f as usePluginConfig, g as useAuth, h as usePasskey, i as getPluginPageSelectorItems, l as usePluginSettings, m as useAsyncBatch, n as downloadBlob, o as uploadPluginEndpoint, p as useAsync, r as downloadPluginEndpoint, s as useCurrentExperiment, t as createPluginClient, u as buildPluginEndpointUrl } from "./composables-wNt7VtkF.js";
|
|
7
7
|
import "./stores/index.js";
|
|
8
8
|
//#region src/utils/rack.ts
|
|
9
9
|
var LCMS_DEFAULT_CONTROL_POSITIONS = {
|
|
@@ -165,6 +165,6 @@ function comparePlateCells(a, b) {
|
|
|
165
165
|
return rowDiff === 0 ? a.column - b.column : rowDiff;
|
|
166
166
|
}
|
|
167
167
|
//#endregion
|
|
168
|
-
export { ADMIN_PANEL_PERMISSIONS, ADMIN_ROLE, APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, AlertBox_default as AlertBox, AppAvatarMenu_default as AppAvatarMenu, AppContainer_default as AppContainer, AppLayout_default as AppLayout, AppPluginSwitcher_default as AppPluginSwitcher, AppSidebar_default as AppSidebar, AppToastContainer_default as AppToastContainer, AppTopBar_default as AppTopBar, AuditTrail_default as AuditTrail, AutoGroupModal_default as AutoGroupModal, Avatar_default as Avatar, BaseButton_default as BaseButton, BaseCheckbox_default as BaseCheckbox, BaseInput_default as BaseInput, BaseModal_default as BaseModal, BasePill_default as BasePill, BaseRadioGroup_default as BaseRadioGroup, BaseSelect_default as BaseSelect, BaseSlider_default as BaseSlider, BaseTabs_default as BaseTabs, BaseTextarea_default as BaseTextarea, BaseToggle_default as BaseToggle, BatchProgressList_default as BatchProgressList, BioTemplateExperimentWorkspaceView_default as BioTemplateExperimentWorkspaceView, BioTemplatePackWorkspaceView_default as BioTemplatePackWorkspaceView, BioTemplatePresetWorkspaceView_default as BioTemplatePresetWorkspaceView, BioTemplateRenderer_default as BioTemplateRenderer, Breadcrumb_default as Breadcrumb, Calendar_default as Calendar, ChartContainer_default as ChartContainer, ChemicalFormula_default as ChemicalFormula, CollapsibleCard_default as CollapsibleCard, ColorSlider_default as ColorSlider, ComponentBindingRenderer_default as ComponentBindingRenderer, ConcentrationInput_default as ConcentrationInput, ConfirmDialog_default as ConfirmDialog, ControlWorkspaceView_default as ControlWorkspaceView, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DEFAULT_LCMS_SEQUENCE_COLUMNS, DEFAULT_PRESETS, DEFAULT_UNITS, DataFrame_default as DataFrame, DatePicker_default as DatePicker, DateTimePicker_default as DateTimePicker, Divider_default as Divider, DoseCalculator_default as DoseCalculator, DoseDesignWorkspaceView_default as DoseDesignWorkspaceView, DropdownButton_default as DropdownButton, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, EmptyState_default as EmptyState, ExperimentCodeBadge_default as ExperimentCodeBadge, ExperimentDataViewer_default as ExperimentDataViewer, ExperimentPopover_default as ExperimentPopover, ExperimentSelectorModal_default as ExperimentSelectorModal, ExperimentTimeline_default as ExperimentTimeline, FileUploader_default as FileUploader, FitPanel_default as FitPanel, FormActions_default as FormActions, FormBuilder_default as FormBuilder, FormField_default as FormField, FormulaInput_default as FormulaInput, GroupAssigner_default as GroupAssigner, IconButton_default as IconButton, InstrumentAlertLog_default as InstrumentAlertLog, InstrumentStateBadge_default as InstrumentStateBadge, InstrumentStatusCard_default as InstrumentStatusCard, LCMS_DEFAULT_CONTROL_POSITIONS, LcmsSequenceTable_default as LcmsSequenceTable, LoadingSpinner_default as LoadingSpinner, MINTSdk, MINTSdk as default, MoleculeInput_default as MoleculeInput, MultiSelect_default as MultiSelect, NumberInput_default as NumberInput, PlateMapEditor_default as PlateMapEditor, PluginIcon_default as PluginIcon, PluginWorkspaceView_default as PluginWorkspaceView, ProgressBar_default as ProgressBar, ProtocolStepEditor_default as ProtocolStepEditor, RackEditor_default as RackEditor, ReagentEditor_default as ReagentEditor, ReagentList_default as ReagentList, ResourceCard_default as ResourceCard, SORT_OPTIONS, SampleHierarchyTree_default as SampleHierarchyTree, SampleLegend_default as SampleLegend, SampleSelector_default as SampleSelector, ScheduleCalendar_default as ScheduleCalendar, ScientificNumber_default as ScientificNumber, SegmentedControl_default as SegmentedControl, SequenceInput_default as SequenceInput, SequenceProgressBar_default as SequenceProgressBar, SettingsModal_default as SettingsModal, Skeleton_default as Skeleton, StatusIndicator_default as StatusIndicator, StepWizard_default as StepWizard, TEMPLATE_COLLECTION_KEY, TagsInput_default as TagsInput, ThemeToggle_default as ThemeToggle, TimePicker_default as TimePicker, TimeRangeInput_default as TimeRangeInput, Tooltip_default as Tooltip, UnitInput_default as UnitInput, WellPlate_default as WellPlate, addMinutes, assertTemplateEnvelope, basenameFromWindowsPath, bioTemplateCatalog, bioTemplateControlsToFormSchema, bioTemplateControlsToSectionFormSchemas, bioTemplateControlsToSidebarPanels, bioTemplatePacks, bioTemplatePresetControlValuesToOptions, bioTemplatePresets, buildPluginEndpointUrl, canAccessAdmin, canAccessByPolicy, canAccessPlugin, candidateMatchesSearch, colorPalettes, compareSortValues, compareTime, controlValuesToComponentBindings, controlValuesToComponentBindingsById, controlValuesToComponentProps, controlsToFormSchema, controlsToSectionFormSchema, controlsToSectionFormSchemas, controlsToSettingsSchema, controlsToSidebarPanels, controlsToTopBarSettingsConfig, controlsToViewIds, controlsToViewItems, createAssayMatrixTemplate, createBioTemplateControlToolkit, createBioTemplatePackCollection, createBioTemplatePresetCollection, createBioTemplatePresetCollectionFromControls, createCalibrationCurveTemplate, createDefaultBioTemplate, createDoseResponseTemplate, createElisaAssayCollection, createFlowCytometryAssayCollection, createFlowCytometryPanelTemplate, createInstrumentRunTemplate, createLcmsBatchCollection, createLcmsControlWellEditData, createPlateMapTemplate, createPluginClient, createProtocolStepsTemplate, createQpcrExpressionCollection, createQpcrPlateTemplate, createReagentListTemplate, createSamplePrepTemplate, createSampleSheetTemplate, createTargetedMetabolomicsCollection, createTemplateCollection, createTemplateEnvelope, createTimeCourseTemplate, createWellPlateScreenCollection, createWesternBlotAssayCollection, datePresetToISO, defineControlComponentBindings, defineControlModel, defineControls, defineDoseCalculatorControlProps, defineDoseDesignControlModel, defineWellPlateControlProps, defineWellPlateDoseComponentBindings, defineWellPlateDoseControlProps, deriveShade, durationMinutes, ensureTemplateEnvelope, ensureTemplateFromCollection, estimateSequenceFinishDate, estimateSequenceRemainingSeconds, evaluateCondition, extractLcmsCommonPrefix, extractLcmsSampleName, extractSampleNamesFromDesignData, extractSampleOptionsFromDesignData, extractSamplesFromDesignData, extractTemplateCollection, findAvailableSlots, findNearestTimeSlotIndex, formatDuration, formatExperimentDate, formatExperimentStatus, formatSequenceEta, formatSequenceRemaining, formatTime, formatTimeSlot, fromMinutes, generateDilutionSeries, generateTimeSlots, getAccessAudience, getBioTemplateComponentBindings, getBioTemplateComponentProps, getBioTemplateControlDefaults, getBioTemplateControlSchema, getBioTemplateInfo, getBioTemplatePackInfo, getBioTemplatePresetInfo, getControlDefaults, getDefaultControlView, getExperimentStatusVariant, getFieldRegistryEntry, getLcmsDefaultControlWellId, getPluginPageSelectorItems, getRoleInfo, getTemplateData, getTypeDefault, getUserPermissions, hasAllPermissions, hasAnyPermission, hexToHsl, hslToHex, inferLcmsPlateTypeFromWellIds, isAdminRole, isAdminUser, isTimeInRange, lcmsPlateCellsToRack, lcmsPlateCellsToRacks, lcmsPlateTypeToRackFormat, lcmsWellId, lcmsWellIdFromPosition, listBioTemplateCatalog, listBioTemplateComponentBindings, listBioTemplateControlSchemas, listBioTemplatePacks, listBioTemplatePresetControlSchemas, listBioTemplatePresets, normalizeAccessPolicy, normalizeSearchQuery, parseLcmsWellId, parseTime, rackFormatToLcmsPlateType, rackToLcmsPlateCells, racksToLcmsPlateCells, rangesOverlap, reconstructLcmsPlateCellsFromSequenceItems, requireBioTemplateControlSchema, resolveExperimentCode, resolvePluginBaseUrl, searchBioTemplateCatalog, searchBioTemplatePacks, searchBioTemplatePresets, sequenceProgressPercent, sequenceSamplesRemaining, snapToSlot, toAssayMatrixColumns, toAssayMatrixDataFrame, toAssayMatrixRows, toAssayMatrixSampleOptions, toBioTemplateComponentBindings, toBioTemplateComponentBindingsById, toBioTemplateComponentImports, toBioTemplateComponentProps, toBioTemplateComponentPropsByComponent, toBioTemplateComponentPropsById, toBioTemplateComponentSnippets, toBioTemplateComponentUsage, toCalibrationCurveColumns, toCalibrationCurveDataFrame, toCalibrationCurveRows, toDoseConditions, toDoseLayoutState, toFlowPanelColumns, toFlowPanelDataFrame, toFlowPanelRows, toInstrumentRunColumns, toInstrumentRunDataFrame, toInstrumentRunRows, toInstrumentRunScheduleEvents, toInstrumentRunSteps, toMinutes, toPlateMapEditorState, toProtocolColumns, toProtocolDataFrame, toProtocolRows, toProtocolSteps, toQpcrColumns, toQpcrDataFrame, toQpcrRows, toQpcrWellPlateWells, toReagentColumns, toReagentDataFrame, toReagentListItems, toReagentRows, toSampleColumns, toSampleDataFrame, toSampleOptions, toSamplePrepColumns, toSamplePrepDataFrame, toSamplePrepRows, toSampleRows, toTemplateDataFrame, toTimeCourseColumns, toTimeCourseDataFrame, toTimeCourseRows, toTimeCourseSteps, toWellMapArray, toWellPlateWells, unwrapExperimentDesignData, useApi, useAppExperiment, useAsync, useAsyncBatch, useAuth, useAuthStore, useAutoGroup, useBioTemplateComponents, useBioTemplateControls, useBioTemplatePackWorkspace, useBioTemplatePresetWorkspace, useBioTemplateWorkspace, useChemicalFormula, useConcentrationUnits, useControlSchema, useControlWorkspace, useCurrentExperiment, useDebouncedWatch, useDoseCalculator, useEventListener, useExpansionSet, useExperimentData, useExperimentSamples, useExperimentSave, useExperimentSelector, useForm, useFormBuilder, useGroupAssignment, useListSelection, usePasskey, usePlatformContext, usePluginClient, usePluginConfig, usePluginSettings, useProtocolTemplates, useRackEditor, useReagentSeries, useRequestSyncState, useSampleGroups, useScheduleDrag, useSelectionLimit, useSequenceUtils, useSettingsStore, useSortedItems, useTemplateCollection, useTextSearch, useTheme, useTimeUtils, useToast, useWellPlateEditor, validateAssayMatrixData, validateCalibrationCurveData, validateDoseResponseData, validateFlowCytometryPanelData, validateInstrumentRunData, validatePlateMapData, validateProtocolStepsData, validateQpcrPlateData, validateReagentListData, validateSamplePrepData, validateSampleSheetData, validateTimeCourseData };
|
|
168
|
+
export { ADMIN_PANEL_PERMISSIONS, ADMIN_ROLE, APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, AlertBox_default as AlertBox, AppAvatarMenu_default as AppAvatarMenu, AppContainer_default as AppContainer, AppLayout_default as AppLayout, AppPluginSwitcher_default as AppPluginSwitcher, AppSidebar_default as AppSidebar, AppToastContainer_default as AppToastContainer, AppTopBar_default as AppTopBar, AuditTrail_default as AuditTrail, AutoGroupModal_default as AutoGroupModal, Avatar_default as Avatar, BaseButton_default as BaseButton, BaseCheckbox_default as BaseCheckbox, BaseInput_default as BaseInput, BaseModal_default as BaseModal, BasePill_default as BasePill, BaseRadioGroup_default as BaseRadioGroup, BaseSelect_default as BaseSelect, BaseSlider_default as BaseSlider, BaseTabs_default as BaseTabs, BaseTextarea_default as BaseTextarea, BaseToggle_default as BaseToggle, BatchProgressList_default as BatchProgressList, BioTemplateExperimentWorkspaceView_default as BioTemplateExperimentWorkspaceView, BioTemplatePackWorkspaceView_default as BioTemplatePackWorkspaceView, BioTemplatePresetWorkspaceView_default as BioTemplatePresetWorkspaceView, BioTemplateRenderer_default as BioTemplateRenderer, Breadcrumb_default as Breadcrumb, Calendar_default as Calendar, ChartContainer_default as ChartContainer, ChemicalFormula_default as ChemicalFormula, CollapsibleCard_default as CollapsibleCard, ColorSlider_default as ColorSlider, ComponentBindingRenderer_default as ComponentBindingRenderer, ConcentrationInput_default as ConcentrationInput, ConfirmDialog_default as ConfirmDialog, ControlWorkspaceView_default as ControlWorkspaceView, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DEFAULT_LCMS_SEQUENCE_COLUMNS, DEFAULT_PRESETS, DEFAULT_UNITS, DataFrame_default as DataFrame, DatePicker_default as DatePicker, DateTimePicker_default as DateTimePicker, Divider_default as Divider, DoseCalculator_default as DoseCalculator, DoseDesignWorkspaceView_default as DoseDesignWorkspaceView, DropdownButton_default as DropdownButton, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, EmptyState_default as EmptyState, ExperimentCodeBadge_default as ExperimentCodeBadge, ExperimentDataViewer_default as ExperimentDataViewer, ExperimentPopover_default as ExperimentPopover, ExperimentSelectorModal_default as ExperimentSelectorModal, ExperimentTimeline_default as ExperimentTimeline, FileUploader_default as FileUploader, FitPanel_default as FitPanel, FormActions_default as FormActions, FormBuilder_default as FormBuilder, FormField_default as FormField, FormulaInput_default as FormulaInput, GroupAssigner_default as GroupAssigner, IconButton_default as IconButton, InstrumentAlertLog_default as InstrumentAlertLog, InstrumentStateBadge_default as InstrumentStateBadge, InstrumentStatusCard_default as InstrumentStatusCard, LCMS_DEFAULT_CONTROL_POSITIONS, LcmsSequenceTable_default as LcmsSequenceTable, LoadingSpinner_default as LoadingSpinner, MINTSdk, MINTSdk as default, MoleculeInput_default as MoleculeInput, MultiSelect_default as MultiSelect, NumberInput_default as NumberInput, PlateMapEditor_default as PlateMapEditor, PluginIcon_default as PluginIcon, PluginWorkspaceView_default as PluginWorkspaceView, ProgressBar_default as ProgressBar, ProtocolStepEditor_default as ProtocolStepEditor, RackEditor_default as RackEditor, ReagentEditor_default as ReagentEditor, ReagentList_default as ReagentList, ResourceCard_default as ResourceCard, SORT_OPTIONS, SampleHierarchyTree_default as SampleHierarchyTree, SampleLegend_default as SampleLegend, SampleSelector_default as SampleSelector, ScheduleCalendar_default as ScheduleCalendar, ScientificNumber_default as ScientificNumber, SegmentedControl_default as SegmentedControl, SequenceInput_default as SequenceInput, SequenceProgressBar_default as SequenceProgressBar, SettingsModal_default as SettingsModal, Skeleton_default as Skeleton, StatusIndicator_default as StatusIndicator, StepWizard_default as StepWizard, TEMPLATE_COLLECTION_KEY, TagsInput_default as TagsInput, ThemeToggle_default as ThemeToggle, TimePicker_default as TimePicker, TimeRangeInput_default as TimeRangeInput, Tooltip_default as Tooltip, UnitInput_default as UnitInput, WellPlate_default as WellPlate, addMinutes, assertTemplateEnvelope, basenameFromWindowsPath, bioTemplateCatalog, bioTemplateControlsToFormSchema, bioTemplateControlsToSectionFormSchemas, bioTemplateControlsToSidebarPanels, bioTemplatePacks, bioTemplatePresetControlValuesToOptions, bioTemplatePresets, buildPluginEndpointUrl, canAccessAdmin, canAccessByPolicy, canAccessPlugin, candidateMatchesSearch, colorPalettes, compareSortValues, compareTime, controlValuesToComponentBindings, controlValuesToComponentBindingsById, controlValuesToComponentProps, controlsToFormSchema, controlsToSectionFormSchema, controlsToSectionFormSchemas, controlsToSettingsSchema, controlsToSidebarPanels, controlsToTopBarSettingsConfig, controlsToViewIds, controlsToViewItems, createAssayMatrixTemplate, createBioTemplateControlToolkit, createBioTemplatePackCollection, createBioTemplatePresetCollection, createBioTemplatePresetCollectionFromControls, createCalibrationCurveTemplate, createDefaultBioTemplate, createDoseResponseTemplate, createElisaAssayCollection, createFlowCytometryAssayCollection, createFlowCytometryPanelTemplate, createInstrumentRunTemplate, createLcmsBatchCollection, createLcmsControlWellEditData, createPlateMapTemplate, createPluginClient, createProtocolStepsTemplate, createQpcrExpressionCollection, createQpcrPlateTemplate, createReagentListTemplate, createSamplePrepTemplate, createSampleSheetTemplate, createTargetedMetabolomicsCollection, createTemplateCollection, createTemplateEnvelope, createTimeCourseTemplate, createWellPlateScreenCollection, createWesternBlotAssayCollection, datePresetToISO, defineControlComponentBindings, defineControlModel, defineControls, defineDoseCalculatorControlProps, defineDoseDesignControlModel, defineWellPlateControlProps, defineWellPlateDoseComponentBindings, defineWellPlateDoseControlProps, deriveShade, downloadBlob, downloadPluginEndpoint, durationMinutes, ensureTemplateEnvelope, ensureTemplateFromCollection, estimateSequenceFinishDate, estimateSequenceRemainingSeconds, evaluateCondition, extractLcmsCommonPrefix, extractLcmsSampleName, extractSampleNamesFromDesignData, extractSampleOptionsFromDesignData, extractSamplesFromDesignData, extractTemplateCollection, findAvailableSlots, findNearestTimeSlotIndex, formatDuration, formatExperimentDate, formatExperimentStatus, formatSequenceEta, formatSequenceRemaining, formatTime, formatTimeSlot, fromMinutes, generateDilutionSeries, generateTimeSlots, getAccessAudience, getBioTemplateComponentBindings, getBioTemplateComponentProps, getBioTemplateControlDefaults, getBioTemplateControlSchema, getBioTemplateInfo, getBioTemplatePackInfo, getBioTemplatePresetInfo, getControlDefaults, getDefaultControlView, getExperimentStatusVariant, getFieldRegistryEntry, getLcmsDefaultControlWellId, getPluginPageSelectorItems, getRoleInfo, getTemplateData, getTypeDefault, getUserPermissions, hasAllPermissions, hasAnyPermission, hexToHsl, hslToHex, inferLcmsPlateTypeFromWellIds, isAdminRole, isAdminUser, isTimeInRange, lcmsPlateCellsToRack, lcmsPlateCellsToRacks, lcmsPlateTypeToRackFormat, lcmsWellId, lcmsWellIdFromPosition, listBioTemplateCatalog, listBioTemplateComponentBindings, listBioTemplateControlSchemas, listBioTemplatePacks, listBioTemplatePresetControlSchemas, listBioTemplatePresets, normalizeAccessPolicy, normalizeSearchQuery, parseLcmsWellId, parseTime, pluginFormDataFromPayload, rackFormatToLcmsPlateType, rackToLcmsPlateCells, racksToLcmsPlateCells, rangesOverlap, reconstructLcmsPlateCellsFromSequenceItems, requireBioTemplateControlSchema, resolveExperimentCode, resolvePluginBaseUrl, searchBioTemplateCatalog, searchBioTemplatePacks, searchBioTemplatePresets, sequenceProgressPercent, sequenceSamplesRemaining, snapToSlot, toAssayMatrixColumns, toAssayMatrixDataFrame, toAssayMatrixRows, toAssayMatrixSampleOptions, toBioTemplateComponentBindings, toBioTemplateComponentBindingsById, toBioTemplateComponentImports, toBioTemplateComponentProps, toBioTemplateComponentPropsByComponent, toBioTemplateComponentPropsById, toBioTemplateComponentSnippets, toBioTemplateComponentUsage, toCalibrationCurveColumns, toCalibrationCurveDataFrame, toCalibrationCurveRows, toDoseConditions, toDoseLayoutState, toFlowPanelColumns, toFlowPanelDataFrame, toFlowPanelRows, toInstrumentRunColumns, toInstrumentRunDataFrame, toInstrumentRunRows, toInstrumentRunScheduleEvents, toInstrumentRunSteps, toMinutes, toPlateMapEditorState, toProtocolColumns, toProtocolDataFrame, toProtocolRows, toProtocolSteps, toQpcrColumns, toQpcrDataFrame, toQpcrRows, toQpcrWellPlateWells, toReagentColumns, toReagentDataFrame, toReagentListItems, toReagentRows, toSampleColumns, toSampleDataFrame, toSampleOptions, toSamplePrepColumns, toSamplePrepDataFrame, toSamplePrepRows, toSampleRows, toTemplateDataFrame, toTimeCourseColumns, toTimeCourseDataFrame, toTimeCourseRows, toTimeCourseSteps, toWellMapArray, toWellPlateWells, unwrapExperimentDesignData, uploadPluginEndpoint, useApi, useAppExperiment, useAsync, useAsyncBatch, useAuth, useAuthStore, useAutoGroup, useBioTemplateComponents, useBioTemplateControls, useBioTemplatePackWorkspace, useBioTemplatePresetWorkspace, useBioTemplateWorkspace, useChemicalFormula, useConcentrationUnits, useControlSchema, useControlWorkspace, useCurrentExperiment, useDebouncedWatch, useDoseCalculator, useEventListener, useExpansionSet, useExperimentData, useExperimentSamples, useExperimentSave, useExperimentSelector, useForm, useFormBuilder, useGroupAssignment, useListSelection, usePasskey, usePlatformContext, usePluginClient, usePluginConfig, usePluginSettings, useProtocolTemplates, useRackEditor, useReagentSeries, useRequestSyncState, useSampleGroups, useScheduleDrag, useSelectionLimit, useSequenceUtils, useSettingsStore, useSortedItems, useTemplateCollection, useTextSearch, useTheme, useTimeUtils, useToast, useWellPlateEditor, validateAssayMatrixData, validateCalibrationCurveData, validateDoseResponseData, validateFlowCytometryPanelData, validateInstrumentRunData, validatePlateMapData, validateProtocolStepsData, validateQpcrPlateData, validateReagentListData, validateSamplePrepData, validateSampleSheetData, validateTimeCourseData };
|
|
169
169
|
|
|
170
170
|
//# sourceMappingURL=index.js.map
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@morscherlab/mint-sdk",
|
|
3
|
-
"version": "1.0.0
|
|
3
|
+
"version": "1.0.0",
|
|
4
4
|
"description": "MINT Platform SDK — Vue 3 components, composables, and types for plugin development. MINT = Mass-spec INtegrated Toolkit.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -5,8 +5,12 @@ import axios, { type AxiosRequestConfig } from 'axios'
|
|
|
5
5
|
import {
|
|
6
6
|
buildPluginEndpointUrl,
|
|
7
7
|
createPluginClient,
|
|
8
|
+
downloadBlob,
|
|
9
|
+
downloadPluginEndpoint,
|
|
8
10
|
getPluginPageSelectorItems,
|
|
11
|
+
pluginFormDataFromPayload,
|
|
9
12
|
resolvePluginBaseUrl,
|
|
13
|
+
uploadPluginEndpoint,
|
|
10
14
|
useCurrentExperiment,
|
|
11
15
|
usePluginClient,
|
|
12
16
|
type PluginContract,
|
|
@@ -27,12 +31,14 @@ describe('usePluginClient', () => {
|
|
|
27
31
|
let requestConfigs: AxiosRequestConfig[] = []
|
|
28
32
|
let requestBodies: unknown[] = []
|
|
29
33
|
let nextGetError: Error | null = null
|
|
34
|
+
let nextGetData: unknown
|
|
30
35
|
|
|
31
36
|
beforeEach(() => {
|
|
32
37
|
setActivePinia(createPinia())
|
|
33
38
|
requestConfigs = []
|
|
34
39
|
requestBodies = []
|
|
35
40
|
nextGetError = null
|
|
41
|
+
nextGetData = undefined
|
|
36
42
|
vi.spyOn(axios.Axios.prototype, 'get').mockImplementation(async function (
|
|
37
43
|
this: unknown,
|
|
38
44
|
url: string,
|
|
@@ -40,7 +46,7 @@ describe('usePluginClient', () => {
|
|
|
40
46
|
) {
|
|
41
47
|
requestConfigs.push({ url, ...config })
|
|
42
48
|
if (nextGetError) throw nextGetError
|
|
43
|
-
return { data: { called: url } }
|
|
49
|
+
return { data: nextGetData ?? { called: url } }
|
|
44
50
|
})
|
|
45
51
|
vi.spyOn(axios.Axios.prototype, 'post').mockImplementation(async function (
|
|
46
52
|
this: unknown,
|
|
@@ -64,6 +70,7 @@ describe('usePluginClient', () => {
|
|
|
64
70
|
})
|
|
65
71
|
|
|
66
72
|
afterEach(() => {
|
|
73
|
+
vi.useRealTimers()
|
|
67
74
|
vi.restoreAllMocks()
|
|
68
75
|
vi.unstubAllEnvs()
|
|
69
76
|
window.history.replaceState({}, '', '/')
|
|
@@ -179,6 +186,134 @@ describe('usePluginClient', () => {
|
|
|
179
186
|
expect(requestConfigs).toHaveLength(0)
|
|
180
187
|
})
|
|
181
188
|
|
|
189
|
+
it('builds concrete endpoint URLs from structured generated payloads', () => {
|
|
190
|
+
const url = buildPluginEndpointUrl(
|
|
191
|
+
contract,
|
|
192
|
+
{
|
|
193
|
+
method: 'post',
|
|
194
|
+
path: '/analyze/{experimentId}',
|
|
195
|
+
pathParams: ['experimentId'],
|
|
196
|
+
queryParams: [
|
|
197
|
+
{ name: 'dry_run', fieldName: 'dryRun', type: 'boolean', required: false },
|
|
198
|
+
],
|
|
199
|
+
hasBody: true,
|
|
200
|
+
},
|
|
201
|
+
{
|
|
202
|
+
pathParams: { experimentId: 42 },
|
|
203
|
+
query: { dryRun: true },
|
|
204
|
+
body: { threshold: 0.5 },
|
|
205
|
+
},
|
|
206
|
+
)
|
|
207
|
+
|
|
208
|
+
expect(url).toBe('/api/drp/analyze/42?dry_run=true')
|
|
209
|
+
expect(requestConfigs).toHaveLength(0)
|
|
210
|
+
})
|
|
211
|
+
|
|
212
|
+
it('preserves flat scalar query fields when no structured query object is used', () => {
|
|
213
|
+
const url = buildPluginEndpointUrl(
|
|
214
|
+
contract,
|
|
215
|
+
{
|
|
216
|
+
method: 'get',
|
|
217
|
+
path: '/search',
|
|
218
|
+
},
|
|
219
|
+
{ query: 'glucose' },
|
|
220
|
+
{ includeBaseUrl: false },
|
|
221
|
+
)
|
|
222
|
+
|
|
223
|
+
expect(url).toBe('/search?query=glucose')
|
|
224
|
+
})
|
|
225
|
+
|
|
226
|
+
it('encodes generated plugin upload endpoint payloads as multipart form data', async () => {
|
|
227
|
+
const response = await uploadPluginEndpoint<{ ok: boolean }>(
|
|
228
|
+
contract,
|
|
229
|
+
{
|
|
230
|
+
method: 'post',
|
|
231
|
+
path: '/schedule/import',
|
|
232
|
+
queryParams: [
|
|
233
|
+
{ name: 'instrument_id', fieldName: 'instrumentId', type: 'string', required: true },
|
|
234
|
+
{ name: 'mode', fieldName: 'mode', type: 'string', required: false },
|
|
235
|
+
],
|
|
236
|
+
hasBody: true,
|
|
237
|
+
},
|
|
238
|
+
{
|
|
239
|
+
query: {
|
|
240
|
+
instrumentId: 'orbitrap',
|
|
241
|
+
mode: 'replace',
|
|
242
|
+
},
|
|
243
|
+
body: {
|
|
244
|
+
file: new Blob(['csv']),
|
|
245
|
+
metadata: { source: 'test' },
|
|
246
|
+
dryRun: false,
|
|
247
|
+
},
|
|
248
|
+
},
|
|
249
|
+
)
|
|
250
|
+
|
|
251
|
+
expect(response).toEqual({ called: '/schedule/import' })
|
|
252
|
+
expect(requestConfigs[0]!.baseURL).toBe('/api/drp')
|
|
253
|
+
expect(requestConfigs[0]!.url).toBe('/schedule/import')
|
|
254
|
+
expect(requestConfigs[0]!.params).toEqual({ instrument_id: 'orbitrap', mode: 'replace' })
|
|
255
|
+
expect(requestConfigs[0]!.headers).toEqual({ 'Content-Type': undefined })
|
|
256
|
+
expect(requestBodies[0]).toBeInstanceOf(FormData)
|
|
257
|
+
const form = requestBodies[0] as FormData
|
|
258
|
+
expect(form.get('file')).toBeInstanceOf(Blob)
|
|
259
|
+
expect(form.get('metadata')).toBe('{"source":"test"}')
|
|
260
|
+
expect(form.get('dryRun')).toBe('false')
|
|
261
|
+
})
|
|
262
|
+
|
|
263
|
+
it('builds multipart form data from repeated scalar values', () => {
|
|
264
|
+
const form = pluginFormDataFromPayload({
|
|
265
|
+
tags: ['qc', 'plate'],
|
|
266
|
+
threshold: 0.5,
|
|
267
|
+
enabled: true,
|
|
268
|
+
omitted: undefined,
|
|
269
|
+
})
|
|
270
|
+
|
|
271
|
+
expect(form.getAll('tags')).toEqual(['qc', 'plate'])
|
|
272
|
+
expect(form.get('threshold')).toBe('0.5')
|
|
273
|
+
expect(form.get('enabled')).toBe('true')
|
|
274
|
+
expect(form.has('omitted')).toBe(false)
|
|
275
|
+
})
|
|
276
|
+
|
|
277
|
+
it('downloads generated plugin endpoint blobs with contract query handling', async () => {
|
|
278
|
+
const expectedBlob = new Blob(['a,b'])
|
|
279
|
+
nextGetData = expectedBlob
|
|
280
|
+
|
|
281
|
+
const blob = await downloadPluginEndpoint(
|
|
282
|
+
contract,
|
|
283
|
+
{
|
|
284
|
+
method: 'get',
|
|
285
|
+
path: '/schedule/export',
|
|
286
|
+
queryParams: [
|
|
287
|
+
{ name: 'instrument_id', fieldName: 'instrumentId', type: 'string', required: true },
|
|
288
|
+
{ name: 'start', fieldName: 'start', type: 'string', required: true },
|
|
289
|
+
],
|
|
290
|
+
},
|
|
291
|
+
{ query: { instrumentId: 'orbitrap', start: '2026-05-27' } },
|
|
292
|
+
)
|
|
293
|
+
|
|
294
|
+
expect(blob).toBe(expectedBlob)
|
|
295
|
+
expect(requestConfigs[0]!.baseURL).toBe('/api/drp')
|
|
296
|
+
expect(requestConfigs[0]!.url).toBe('/schedule/export')
|
|
297
|
+
expect(requestConfigs[0]!.params).toEqual({ instrument_id: 'orbitrap', start: '2026-05-27' })
|
|
298
|
+
expect(requestConfigs[0]!.responseType).toBe('blob')
|
|
299
|
+
})
|
|
300
|
+
|
|
301
|
+
it('triggers browser downloads for existing blobs', () => {
|
|
302
|
+
vi.useFakeTimers()
|
|
303
|
+
const createObjectURL = vi.spyOn(URL, 'createObjectURL').mockReturnValue('blob:test')
|
|
304
|
+
const revokeObjectURL = vi.spyOn(URL, 'revokeObjectURL').mockImplementation(() => {})
|
|
305
|
+
const click = vi.spyOn(HTMLAnchorElement.prototype, 'click').mockImplementation(() => {})
|
|
306
|
+
|
|
307
|
+
downloadBlob(new Blob(['csv']), 'export.csv', { revokeObjectUrlDelayMs: 0 })
|
|
308
|
+
|
|
309
|
+
expect(createObjectURL).toHaveBeenCalledOnce()
|
|
310
|
+
expect(click).toHaveBeenCalledOnce()
|
|
311
|
+
expect(revokeObjectURL).not.toHaveBeenCalled()
|
|
312
|
+
vi.runOnlyPendingTimers()
|
|
313
|
+
expect(revokeObjectURL).toHaveBeenCalledWith('blob:test')
|
|
314
|
+
vi.useRealTimers()
|
|
315
|
+
})
|
|
316
|
+
|
|
182
317
|
it('maps plugin contract nav items to AppTopBar page selector metadata', () => {
|
|
183
318
|
const pageSelectorItems = getPluginPageSelectorItems({
|
|
184
319
|
...contract,
|
|
@@ -355,6 +490,39 @@ describe('usePluginClient', () => {
|
|
|
355
490
|
expect(requestBodies[0]).toEqual({ threshold: 0.5 })
|
|
356
491
|
})
|
|
357
492
|
|
|
493
|
+
it('sends structured generated payloads with path params, query params, and bodies', async () => {
|
|
494
|
+
type Client = {
|
|
495
|
+
analyze(params: {
|
|
496
|
+
pathParams: { experimentId: number }
|
|
497
|
+
query?: { dryRun?: boolean }
|
|
498
|
+
body: { threshold: number }
|
|
499
|
+
}): Promise<unknown>
|
|
500
|
+
}
|
|
501
|
+
const client = createPluginClient<Client>(contract, {
|
|
502
|
+
endpoints: {
|
|
503
|
+
analyze: {
|
|
504
|
+
method: 'post',
|
|
505
|
+
path: '/analyze/{experimentId}',
|
|
506
|
+
pathParams: ['experimentId'],
|
|
507
|
+
queryParams: [
|
|
508
|
+
{ name: 'dry_run', fieldName: 'dryRun', type: 'boolean', required: false },
|
|
509
|
+
],
|
|
510
|
+
hasBody: true,
|
|
511
|
+
},
|
|
512
|
+
},
|
|
513
|
+
})
|
|
514
|
+
|
|
515
|
+
await client.analyze({
|
|
516
|
+
pathParams: { experimentId: 42 },
|
|
517
|
+
query: { dryRun: true },
|
|
518
|
+
body: { threshold: 0.5 },
|
|
519
|
+
})
|
|
520
|
+
|
|
521
|
+
expect(requestConfigs[0]!.url).toBe('/analyze/42')
|
|
522
|
+
expect(requestConfigs[0]!.params).toEqual({ dry_run: true })
|
|
523
|
+
expect(requestBodies[0]).toEqual({ threshold: 0.5 })
|
|
524
|
+
})
|
|
525
|
+
|
|
358
526
|
it('throws a clear error before requests with missing request bodies', async () => {
|
|
359
527
|
type Client = {
|
|
360
528
|
analyze(params: { experimentId: number; body?: { threshold: number } }): Promise<unknown>
|
package/src/composables/index.ts
CHANGED
|
@@ -363,17 +363,26 @@ export {
|
|
|
363
363
|
export {
|
|
364
364
|
buildPluginEndpointUrl,
|
|
365
365
|
createPluginClient,
|
|
366
|
+
downloadBlob,
|
|
367
|
+
downloadPluginEndpoint,
|
|
366
368
|
getPluginPageSelectorItems,
|
|
369
|
+
pluginFormDataFromPayload,
|
|
367
370
|
resolvePluginBaseUrl,
|
|
371
|
+
uploadPluginEndpoint,
|
|
368
372
|
usePluginClient,
|
|
369
373
|
usePluginSettings,
|
|
370
374
|
useCurrentExperiment,
|
|
371
375
|
type BuildPluginEndpointUrlOptions,
|
|
376
|
+
type DownloadBlobOptions,
|
|
377
|
+
type DownloadPluginEndpointOptions,
|
|
372
378
|
type PluginContract,
|
|
373
379
|
type PluginEndpointContract,
|
|
374
380
|
type PluginEndpointDefinition,
|
|
375
381
|
type PluginHttpMethod,
|
|
376
382
|
type PluginNavItemContract,
|
|
383
|
+
type PluginEndpointRequestOptions,
|
|
384
|
+
type PluginFormDataPayload,
|
|
385
|
+
type PluginFormDataValue,
|
|
377
386
|
type CreatePluginClientOptions,
|
|
378
387
|
type UseCurrentExperimentOptions,
|
|
379
388
|
type UseCurrentExperimentReturn,
|
|
@@ -114,6 +114,35 @@ export function hasKeys(value: Record<string, unknown>): boolean {
|
|
|
114
114
|
return Object.keys(value).length > 0
|
|
115
115
|
}
|
|
116
116
|
|
|
117
|
+
function isPlainPayloadObject(value: unknown): value is Record<string, unknown> {
|
|
118
|
+
return !!value && typeof value === 'object' && !Array.isArray(value)
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function hasOwn(value: Record<string, unknown>, key: string): boolean {
|
|
122
|
+
return Object.prototype.hasOwnProperty.call(value, key)
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
function normalizeStructuredEndpointPayload(payload: Record<string, unknown>): Record<string, unknown> {
|
|
126
|
+
const pathParamPayload = payload.pathParams
|
|
127
|
+
const queryPayload = payload.query
|
|
128
|
+
const hasStructuredPathParams = hasOwn(payload, 'pathParams') && isPlainPayloadObject(pathParamPayload)
|
|
129
|
+
const hasStructuredQuery = hasOwn(payload, 'query') && isPlainPayloadObject(queryPayload)
|
|
130
|
+
if (!hasStructuredPathParams && !hasStructuredQuery) return payload
|
|
131
|
+
|
|
132
|
+
const nextPayload = { ...payload }
|
|
133
|
+
delete nextPayload.pathParams
|
|
134
|
+
delete nextPayload.query
|
|
135
|
+
|
|
136
|
+
if (hasStructuredPathParams) {
|
|
137
|
+
Object.assign(nextPayload, pathParamPayload)
|
|
138
|
+
}
|
|
139
|
+
if (hasStructuredQuery) {
|
|
140
|
+
Object.assign(nextPayload, queryPayload)
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
return nextPayload
|
|
144
|
+
}
|
|
145
|
+
|
|
117
146
|
export function requestBodyFromPayload(
|
|
118
147
|
payload: unknown,
|
|
119
148
|
requestPayload: Record<string, unknown>,
|
|
@@ -135,10 +164,9 @@ export function pluginEndpointRequestParts(
|
|
|
135
164
|
payload?: unknown,
|
|
136
165
|
explicitBaseUrl?: string,
|
|
137
166
|
) {
|
|
138
|
-
const payloadObject =
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
: {}
|
|
167
|
+
const payloadObject = isPlainPayloadObject(payload)
|
|
168
|
+
? normalizeStructuredEndpointPayload(payload)
|
|
169
|
+
: {}
|
|
142
170
|
const pathParams = endpoint.pathParams ?? []
|
|
143
171
|
const requestPayload = withDefaultPathParams(payloadObject, pathParams)
|
|
144
172
|
const path = encodePath(endpoint.path, requestPayload, pathParams)
|
|
@@ -96,6 +96,34 @@ export interface BuildPluginEndpointUrlOptions {
|
|
|
96
96
|
includeBaseUrl?: boolean
|
|
97
97
|
}
|
|
98
98
|
|
|
99
|
+
export interface PluginEndpointRequestOptions {
|
|
100
|
+
/** Override the generated or platform-injected API base URL. */
|
|
101
|
+
baseUrl?: string
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
export interface DownloadPluginEndpointOptions extends PluginEndpointRequestOptions {
|
|
105
|
+
/** Browser download filename. When omitted, the blob is returned without triggering a download. */
|
|
106
|
+
filename?: string
|
|
107
|
+
/** Delay before revoking the object URL after a browser download is triggered. */
|
|
108
|
+
revokeObjectUrlDelayMs?: number
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export interface DownloadBlobOptions {
|
|
112
|
+
/** Delay before revoking the object URL after a browser download is triggered. */
|
|
113
|
+
revokeObjectUrlDelayMs?: number
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
export type PluginFormDataValue =
|
|
117
|
+
| string
|
|
118
|
+
| number
|
|
119
|
+
| boolean
|
|
120
|
+
| Blob
|
|
121
|
+
| null
|
|
122
|
+
| undefined
|
|
123
|
+
| Record<string, unknown>
|
|
124
|
+
|
|
125
|
+
export type PluginFormDataPayload = Record<string, PluginFormDataValue | PluginFormDataValue[]>
|
|
126
|
+
|
|
99
127
|
export interface UseCurrentExperimentOptions {
|
|
100
128
|
apiBaseUrl?: string
|
|
101
129
|
immediate?: boolean
|
|
@@ -125,6 +153,71 @@ export interface UseCurrentExperimentReturn<TExperiment = ExperimentSummary> {
|
|
|
125
153
|
export type PluginEndpointCaller = (payload?: unknown) => Promise<unknown>
|
|
126
154
|
export type GeneratedPluginClient = Record<string, (...args: any[]) => Promise<any>>
|
|
127
155
|
|
|
156
|
+
function queryConfig(query: Record<string, unknown>) {
|
|
157
|
+
return hasKeys(query) ? { params: query } : undefined
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
function isFormData(value: unknown): value is FormData {
|
|
161
|
+
return typeof FormData !== 'undefined' && value instanceof FormData
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
function isBlob(value: unknown): value is Blob {
|
|
165
|
+
return typeof Blob !== 'undefined' && value instanceof Blob
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function appendFormDataValue(formData: FormData, key: string, value: PluginFormDataValue | PluginFormDataValue[]): void {
|
|
169
|
+
if (value === undefined || value === null) return
|
|
170
|
+
|
|
171
|
+
if (Array.isArray(value)) {
|
|
172
|
+
for (const item of value) {
|
|
173
|
+
appendFormDataValue(formData, key, item)
|
|
174
|
+
}
|
|
175
|
+
return
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
if (isBlob(value)) {
|
|
179
|
+
formData.append(key, value)
|
|
180
|
+
return
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
if (typeof value === 'object') {
|
|
184
|
+
formData.append(key, JSON.stringify(value))
|
|
185
|
+
return
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
formData.append(key, String(value))
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
/** Convert plain upload payloads into FormData using the same encoding across generated plugin clients. */
|
|
192
|
+
export function pluginFormDataFromPayload(payload: unknown): FormData {
|
|
193
|
+
if (isFormData(payload)) return payload
|
|
194
|
+
if (!payload || typeof payload !== 'object' || Array.isArray(payload)) {
|
|
195
|
+
throw new Error('[MINT SDK] Plugin upload endpoints require a FormData or plain object payload.')
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
const formData = new FormData()
|
|
199
|
+
for (const [key, value] of Object.entries(payload as PluginFormDataPayload)) {
|
|
200
|
+
appendFormDataValue(formData, key, value)
|
|
201
|
+
}
|
|
202
|
+
return formData
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
/** Trigger a browser download for an already-created Blob. */
|
|
206
|
+
export function downloadBlob(blob: Blob, filename: string, options: DownloadBlobOptions = {}): void {
|
|
207
|
+
if (typeof document === 'undefined' || typeof URL === 'undefined') {
|
|
208
|
+
throw new Error('[MINT SDK] Cannot trigger a file download outside a browser.')
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
const blobUrl = URL.createObjectURL(blob)
|
|
212
|
+
const link = document.createElement('a')
|
|
213
|
+
link.href = blobUrl
|
|
214
|
+
link.download = filename
|
|
215
|
+
document.body.appendChild(link)
|
|
216
|
+
link.click()
|
|
217
|
+
document.body.removeChild(link)
|
|
218
|
+
setTimeout(() => URL.revokeObjectURL(blobUrl), options.revokeObjectUrlDelayMs ?? 100)
|
|
219
|
+
}
|
|
220
|
+
|
|
128
221
|
function normalizePluginNavPath(path: string): string {
|
|
129
222
|
const raw = path.trim() || '/'
|
|
130
223
|
const prefixed = raw.startsWith('/') ? raw : `/${raw}`
|
|
@@ -197,6 +290,57 @@ export function createPluginClient<TClient extends object = GeneratedPluginClien
|
|
|
197
290
|
return client as TClient
|
|
198
291
|
}
|
|
199
292
|
|
|
293
|
+
/** Send multipart form data to a generated plugin endpoint while reusing contract URL, query, and auth handling. */
|
|
294
|
+
export async function uploadPluginEndpoint<TResponse = unknown>(
|
|
295
|
+
contract: PluginContract,
|
|
296
|
+
endpoint: PluginEndpointDefinition,
|
|
297
|
+
payload?: unknown,
|
|
298
|
+
options: PluginEndpointRequestOptions = {},
|
|
299
|
+
): Promise<TResponse> {
|
|
300
|
+
if (!['post', 'put', 'patch'].includes(endpoint.method)) {
|
|
301
|
+
throw new Error(`[MINT SDK] Plugin upload endpoints must use POST, PUT, or PATCH; got ${endpoint.method}`)
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)
|
|
305
|
+
const api = useApi({ baseUrl: parts.baseUrl })
|
|
306
|
+
const body = endpoint.hasBody
|
|
307
|
+
? requestBodyFromPayload(payload, parts.requestPayload, endpoint)
|
|
308
|
+
: payload
|
|
309
|
+
const formData = pluginFormDataFromPayload(body)
|
|
310
|
+
const config = {
|
|
311
|
+
...(queryConfig(parts.query) ?? {}),
|
|
312
|
+
// Let Axios/browser set multipart boundaries.
|
|
313
|
+
headers: { 'Content-Type': undefined },
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
if (endpoint.method === 'post') return api.post<TResponse>(parts.path, formData, config)
|
|
317
|
+
if (endpoint.method === 'put') return api.put<TResponse>(parts.path, formData, config)
|
|
318
|
+
return api.patch<TResponse>(parts.path, formData, config)
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
/** Fetch a generated plugin endpoint as a Blob and optionally trigger a browser download. */
|
|
322
|
+
export async function downloadPluginEndpoint(
|
|
323
|
+
contract: PluginContract,
|
|
324
|
+
endpoint: PluginEndpointDefinition,
|
|
325
|
+
payload?: unknown,
|
|
326
|
+
options: DownloadPluginEndpointOptions = {},
|
|
327
|
+
): Promise<Blob> {
|
|
328
|
+
const parts = pluginEndpointRequestParts(contract, endpoint, payload, options.baseUrl)
|
|
329
|
+
const api = useApi({ baseUrl: parts.baseUrl })
|
|
330
|
+
const blob = await api.get<Blob>(parts.path, {
|
|
331
|
+
...(queryConfig(parts.query) ?? {}),
|
|
332
|
+
responseType: 'blob',
|
|
333
|
+
})
|
|
334
|
+
|
|
335
|
+
if (options.filename) {
|
|
336
|
+
downloadBlob(blob, options.filename, {
|
|
337
|
+
revokeObjectUrlDelayMs: options.revokeObjectUrlDelayMs,
|
|
338
|
+
})
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
return blob
|
|
342
|
+
}
|
|
343
|
+
|
|
200
344
|
/** Return a generated plugin client from setup code with a stable typed identity. */
|
|
201
345
|
export function usePluginClient<TClient>(client: TClient): TClient {
|
|
202
346
|
return client
|