1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-11-23 22:36:32 +02:00

Cli: Support managing shared notebooks (#12637)

This commit is contained in:
Henry Heino
2025-07-01 14:47:03 -07:00
committed by GitHub
parent 41553eb963
commit 901fe73c08
18 changed files with 599 additions and 23 deletions

View File

@@ -3,20 +3,33 @@ import reducer, { State, defaultState } from '../../reducer';
import ShareService from '../../services/share/ShareService';
import { encryptionService } from '../test-utils';
import JoplinServerApi, { ExecOptions } from '../../JoplinServerApi';
import { ShareInvitation, StateShare } from '../../services/share/reducer';
import { ShareInvitation, StateShare, StateShareUser } from '../../services/share/reducer';
const testReducer = (state = defaultState, action: unknown) => {
return reducer(state, action);
};
type Query = Record<string, unknown>;
type OnShareGetListener = (query: Query)=> Promise<{ items: Partial<StateShare>[] }>;
type OnSharePostListener = (query: Query)=> Promise<{ id: string }>;
type OnInvitationGetListener = (query: Query)=> Promise<{ items: Partial<ShareInvitation>[] }>;
interface ShareStateResponse {
items: Partial<StateShare>[];
}
interface ShareInvitationResponse {
items: Partial<ShareInvitation>[];
}
interface ShareUsersResponse {
items: Partial<StateShareUser>[];
}
type Json = Record<string, unknown>;
type OnShareGetListener = (query: Json)=> Promise<ShareStateResponse>;
type OnSharePostListener = (query: Json)=> Promise<{ id: string }>;
type OnInvitationGetListener = (query: Json)=> Promise<ShareInvitationResponse>;
type OnShareUsersGetListener = (shareId: string)=> Promise<ShareUsersResponse>;
type OnShareUsersPostListener = (shareId: string, body: Json)=> Promise<void>;
type OnApiExecListener = (
method: string,
path: string,
query: Query,
query: Json,
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Needs to interface with old code from before the rule was applied
body: any,
headers: Record<string, unknown>,
@@ -27,6 +40,8 @@ export type ApiMock = {
getShares: OnShareGetListener;
postShares: OnSharePostListener;
getShareInvitations: OnInvitationGetListener;
getShareUsers?: OnShareUsersGetListener;
postShareUsers?: OnShareUsersPostListener;
onUnhandled?: OnApiExecListener;
onExec?: undefined;
@@ -37,6 +52,8 @@ export type ApiMock = {
getShareInvitations?: undefined;
getShares?: undefined;
postShares?: undefined;
getShareUsers?: undefined;
postShareUsers?: undefined;
};
// Initializes a share service with mocks
@@ -57,6 +74,16 @@ const mockShareService = (apiCallHandler: ApiMock, service?: ShareService, store
return apiCallHandler.getShareInvitations(query);
}
const shareUsersMatch = path.match(/^api\/shares\/([^/]+)\/users$/);
const shareId = shareUsersMatch?.[1];
if (shareId) {
if (method === 'GET' && apiCallHandler.getShareUsers) {
return apiCallHandler.getShareUsers(shareId);
}
if (method === 'POST' && apiCallHandler.postShareUsers) {
return apiCallHandler.postShareUsers(shareId, body);
}
}
if (apiCallHandler.onUnhandled) {
return apiCallHandler.onUnhandled(method, path, query, body, headers, options);