1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-08 13:06:15 +02:00
joplin/packages/lib/services/joplinCloudUtils.ts

125 lines
3.9 KiB
TypeScript
Raw Normal View History

import { Reducer } from 'react';
import Setting from '../models/Setting';
import { ApplicationPlatform, ApplicationType } from '../types';
import shim from '../shim';
import { _ } from '../locale';
import eventManager, { EventName } from '../eventManager';
import { reg } from '../registry';
import SyncTargetRegistry from '../SyncTargetRegistry';
type ActionType = 'LINK_USED' | 'COMPLETED' | 'ERROR';
type Action = {
type: ActionType;
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
payload?: any;
};
type DefaultState = {
className: 'text' | 'bold';
message: ()=> string;
next: ActionType;
active: ActionType | 'INITIAL';
errorMessage?: string;
};
export const defaultState: DefaultState = {
className: 'text',
message: ()=> _('Waiting for authorisation...'),
next: 'LINK_USED',
active: 'INITIAL',
};
export const reducer: Reducer<DefaultState, Action> = (state: DefaultState, action: Action) => {
switch (action.type) {
case 'LINK_USED': {
return {
className: 'text',
message: () => _('If you have already authorised, please wait for the application to sync to Joplin Cloud.'),
next: 'COMPLETED',
active: 'LINK_USED',
};
}
case 'COMPLETED': {
return {
className: 'bold',
message: () => _('You are logged in into Joplin Cloud, you can leave this screen now.'),
active: 'COMPLETED',
next: 'COMPLETED',
};
}
case 'ERROR': {
return {
className: 'text',
2024-03-14 11:52:40 +02:00
message: () => _('You were unable to connect to Joplin Cloud. Please check your credentials and try again. Error:'),
active: 'ERROR',
next: 'COMPLETED',
errorMessage: action.payload,
};
}
default: {
return state;
}
}
};
export const getApplicationInformation = async () => {
const platformName = await shim.platformName();
switch (platformName) {
case 'ios':
return { type: ApplicationType.Mobile, platform: ApplicationPlatform.Ios };
case 'android':
return { type: ApplicationType.Mobile, platform: ApplicationPlatform.Android };
case 'darwin':
return { type: ApplicationType.Desktop, platform: ApplicationPlatform.MacOs };
case 'win32':
return { type: ApplicationType.Desktop, platform: ApplicationPlatform.Windows };
case 'linux':
return { type: ApplicationType.Desktop, platform: ApplicationPlatform.Linux };
default:
return { type: ApplicationType.Unknown, platform: ApplicationPlatform.Unknown };
}
};
export const generateApplicationConfirmUrl = async (confirmUrl: string) => {
const applicationInfo = await getApplicationInformation();
const searchParams = new URLSearchParams();
searchParams.append('platform', applicationInfo.platform.toString());
searchParams.append('type', applicationInfo.type.toString());
searchParams.append('version', shim.appVersion());
return `${confirmUrl}?${searchParams.toString()}`;
};
// We have isWaitingResponse inside the function to avoid any state from lingering
// after an error occurs. E.g.: if the function would throw an error while isWaitingResponse
// was set to true the next time we call the function the value would still be true.
// The closure function prevents that.
export const checkIfLoginWasSuccessful = async (applicationsUrl: string) => {
let isWaitingResponse = false;
const performLoginRequest = async () => {
if (isWaitingResponse) return undefined;
isWaitingResponse = true;
const response = await fetch(applicationsUrl);
const jsonBody = await response.json();
if (!response.ok || jsonBody.status !== 'finished') {
isWaitingResponse = false;
return undefined;
}
Setting.setValue('sync.10.username', jsonBody.id);
Setting.setValue('sync.10.password', jsonBody.password);
Setting.setValue('sync.target', SyncTargetRegistry.nameToId('joplinCloud'));
const fileApi = await reg.syncTarget().fileApi();
await fileApi.driver().api().loadSession();
eventManager.emit(EventName.SessionEstablished);
return { success: true };
};
return performLoginRequest();
};