2022-04-16 15:04:06 +01:00
|
|
|
import { rtrimSlashes } from '../../path-utils';
|
2022-04-11 16:49:32 +01:00
|
|
|
import shim from '../../shim';
|
2022-04-16 15:04:06 +01:00
|
|
|
import { CurrentProfileVersion, defaultProfile, defaultProfileConfig, DefaultProfileId, Profile, ProfileConfig } from './types';
|
2022-04-11 16:49:32 +01:00
|
|
|
import { customAlphabet } from 'nanoid/non-secure';
|
2023-01-10 12:08:13 +00:00
|
|
|
import { _ } from '../../locale';
|
2022-04-11 16:49:32 +01:00
|
|
|
|
2022-04-16 15:04:06 +01:00
|
|
|
export const migrateProfileConfig = (profileConfig: any, toVersion: number): ProfileConfig => {
|
|
|
|
let version = 2;
|
|
|
|
|
|
|
|
while (profileConfig.version < toVersion) {
|
|
|
|
if (profileConfig.version === 1) {
|
|
|
|
for (const profile of profileConfig.profiles) {
|
|
|
|
if (profile.path === '.') {
|
|
|
|
profile.id = DefaultProfileId;
|
|
|
|
} else {
|
|
|
|
profile.id = profile.path.split('-').pop();
|
|
|
|
}
|
|
|
|
delete profile.path;
|
|
|
|
}
|
|
|
|
|
|
|
|
const currentProfile = profileConfig.profiles[profileConfig.currentProfile];
|
|
|
|
profileConfig.currentProfileId = currentProfile.id;
|
|
|
|
delete profileConfig.currentProfile;
|
|
|
|
}
|
|
|
|
|
|
|
|
profileConfig.version = version;
|
|
|
|
version++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return profileConfig;
|
|
|
|
};
|
|
|
|
|
2022-04-11 16:49:32 +01:00
|
|
|
export const loadProfileConfig = async (profileConfigPath: string): Promise<ProfileConfig> => {
|
|
|
|
if (!(await shim.fsDriver().exists(profileConfigPath))) {
|
|
|
|
return defaultProfileConfig();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const configContent = await shim.fsDriver().readFile(profileConfigPath, 'utf8');
|
2022-04-16 15:04:06 +01:00
|
|
|
let parsed = JSON.parse(configContent) as ProfileConfig;
|
2022-04-11 16:49:32 +01:00
|
|
|
if (!parsed.profiles || !parsed.profiles.length) throw new Error(`Profile config should contain at least one profile: ${profileConfigPath}`);
|
|
|
|
|
2022-04-16 15:04:06 +01:00
|
|
|
parsed = migrateProfileConfig(parsed, CurrentProfileVersion);
|
|
|
|
|
2022-04-11 16:49:32 +01:00
|
|
|
const output: ProfileConfig = {
|
|
|
|
...defaultProfileConfig(),
|
|
|
|
...parsed,
|
|
|
|
};
|
|
|
|
|
|
|
|
for (let i = 0; i < output.profiles.length; i++) {
|
|
|
|
output.profiles[i] = {
|
|
|
|
...defaultProfile(),
|
|
|
|
...output.profiles[i],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-04-16 15:04:06 +01:00
|
|
|
if (!output.profiles.find(p => p.id === output.currentProfileId)) throw new Error(`Current profile ID is invalid: ${output.currentProfileId}`);
|
2022-04-11 16:49:32 +01:00
|
|
|
return output;
|
|
|
|
} catch (error) {
|
|
|
|
error.message = `Could not parse profile configuration: ${profileConfigPath}: ${error.message}`;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const saveProfileConfig = async (profileConfigPath: string, config: ProfileConfig) => {
|
|
|
|
await shim.fsDriver().writeFile(profileConfigPath, JSON.stringify(config, null, '\t'), 'utf8');
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getCurrentProfile = (config: ProfileConfig): Profile => {
|
2022-04-16 15:04:06 +01:00
|
|
|
return config.profiles.find(p => p.id === config.currentProfileId);
|
2022-04-11 16:49:32 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
export const getProfileFullPath = (profile: Profile, rootProfilePath: string): string => {
|
2022-04-16 15:04:06 +01:00
|
|
|
const folderName = profile.id === DefaultProfileId ? '' : `/profile-${profile.id}`;
|
|
|
|
return `${rtrimSlashes(rootProfilePath)}${folderName}`;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const isSubProfile = (profile: Profile): boolean => {
|
|
|
|
return profile.id !== DefaultProfileId;
|
2022-04-11 16:49:32 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const profileIdGenerator = customAlphabet('0123456789abcdefghijklmnopqrstuvwxyz', 8);
|
|
|
|
|
|
|
|
export const createNewProfile = (config: ProfileConfig, profileName: string) => {
|
2022-04-16 15:04:06 +01:00
|
|
|
const newConfig: ProfileConfig = {
|
2022-04-11 16:49:32 +01:00
|
|
|
...config,
|
|
|
|
profiles: config.profiles.slice(),
|
|
|
|
};
|
|
|
|
|
2022-04-16 15:04:06 +01:00
|
|
|
const newProfile: Profile = {
|
2022-04-11 16:49:32 +01:00
|
|
|
name: profileName,
|
2022-04-16 15:04:06 +01:00
|
|
|
id: profileIdGenerator(),
|
|
|
|
};
|
|
|
|
|
|
|
|
newConfig.profiles.push(newProfile);
|
|
|
|
|
|
|
|
return {
|
|
|
|
newConfig: newConfig,
|
|
|
|
newProfile: newProfile,
|
|
|
|
};
|
|
|
|
};
|
2022-04-11 16:49:32 +01:00
|
|
|
|
2023-01-10 12:08:13 +00:00
|
|
|
export const deleteProfileById = (config: ProfileConfig, profileId: string): ProfileConfig => {
|
|
|
|
if (profileId === DefaultProfileId) throw new Error(_('The default profile cannot be deleted'));
|
|
|
|
if (profileId === config.currentProfileId) throw new Error(_('The active profile cannot be deleted. Switch to a different profile and try again.'));
|
|
|
|
|
|
|
|
const newProfiles = config.profiles.filter(p => p.id !== profileId);
|
|
|
|
return {
|
|
|
|
...config,
|
|
|
|
profiles: newProfiles,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-04-16 15:04:06 +01:00
|
|
|
export const profileIdByIndex = (config: ProfileConfig, index: number): string => {
|
|
|
|
return config.profiles[index].id;
|
2022-04-11 16:49:32 +01:00
|
|
|
};
|