mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2025-04-05 14:41:07 +02:00
Implement move extensions
This commit is contained in:
parent
c33649753b
commit
83965fb611
@ -465,7 +465,7 @@ async function connectToApi(baseUrl) {
|
|||||||
|
|
||||||
function updateStatus(success) {
|
function updateStatus(success) {
|
||||||
connectedToApi = success;
|
connectedToApi = success;
|
||||||
const _text = success ? 'Connected to API' : 'Could not connect to API';
|
const _text = success ? t`Connected to API` : t`Could not connect to API`;
|
||||||
const _class = success ? 'success' : 'failure';
|
const _class = success ? 'success' : 'failure';
|
||||||
$('#extensions_status').text(_text);
|
$('#extensions_status').text(_text);
|
||||||
$('#extensions_status').attr('class', _class);
|
$('#extensions_status').attr('class', _class);
|
||||||
@ -723,7 +723,7 @@ async function showExtensionsDetails() {
|
|||||||
}
|
}
|
||||||
if (stateChanged) {
|
if (stateChanged) {
|
||||||
waitingForSave = true;
|
waitingForSave = true;
|
||||||
const toast = toastr.info('The page will be reloaded shortly...', 'Extensions state changed');
|
const toast = toastr.info(t`The page will be reloaded shortly...`, t`Extensions state changed`);
|
||||||
await saveSettings();
|
await saveSettings();
|
||||||
toastr.clear(toast);
|
toastr.clear(toast);
|
||||||
waitingForSave = false;
|
waitingForSave = false;
|
||||||
@ -735,7 +735,7 @@ async function showExtensionsDetails() {
|
|||||||
popupPromise = popup.show();
|
popupPromise = popup.show();
|
||||||
checkForUpdatesManual(abortController.signal).finally(() => htmlLoading.remove());
|
checkForUpdatesManual(abortController.signal).finally(() => htmlLoading.remove());
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
toastr.error('Error loading extensions. See browser console for details.');
|
toastr.error(t`Error loading extensions. See browser console for details.`);
|
||||||
console.error(error);
|
console.error(error);
|
||||||
}
|
}
|
||||||
if (popupPromise) {
|
if (popupPromise) {
|
||||||
@ -817,7 +817,7 @@ async function onDeleteClick() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// use callPopup to create a popup for the user to confirm before delete
|
// use callPopup to create a popup for the user to confirm before delete
|
||||||
const confirmation = await callGenericPopup(`Are you sure you want to delete ${extensionName}?`, POPUP_TYPE.CONFIRM, '', {});
|
const confirmation = await callGenericPopup(t`Are you sure you want to delete ${extensionName}?`, POPUP_TYPE.CONFIRM, '', {});
|
||||||
if (confirmation === POPUP_RESULT.AFFIRMATIVE) {
|
if (confirmation === POPUP_RESULT.AFFIRMATIVE) {
|
||||||
await deleteExtension(extensionName);
|
await deleteExtension(extensionName);
|
||||||
}
|
}
|
||||||
@ -832,7 +832,57 @@ async function onMoveClick() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toastr.info('Not implemented yet');
|
const source = getExtensionType(extensionName);
|
||||||
|
const destination = source === 'global' ? 'local' : 'global';
|
||||||
|
|
||||||
|
const confirmationHeader = t`Move extension`;
|
||||||
|
const confirmationText = source == 'global'
|
||||||
|
? t`Are you sure you want to move ${extensionName} to your local extensions? This will make it available only for you.`
|
||||||
|
: t`Are you sure you want to move ${extensionName} to the global extensions? This will make it available for all users.`;
|
||||||
|
|
||||||
|
const confirmation = await Popup.show.confirm(confirmationHeader, confirmationText);
|
||||||
|
|
||||||
|
if (!confirmation) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$(this).find('i').addClass('fa-spin');
|
||||||
|
await moveExtension(extensionName, source, destination);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Moves an extension via the API.
|
||||||
|
* @param {string} extensionName Extension name
|
||||||
|
* @param {string} source Source type
|
||||||
|
* @param {string} destination Destination type
|
||||||
|
* @returns {Promise<void>}
|
||||||
|
*/
|
||||||
|
async function moveExtension(extensionName, source, destination) {
|
||||||
|
try {
|
||||||
|
const result = await fetch('/api/extensions/move', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: getRequestHeaders(),
|
||||||
|
body: JSON.stringify({
|
||||||
|
extensionName,
|
||||||
|
source,
|
||||||
|
destination,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!result.ok) {
|
||||||
|
const text = await result.text();
|
||||||
|
toastr.error(text || result.statusText, t`Extension move failed`, { timeOut: 5000 });
|
||||||
|
console.error('Extension move failed', result.status, result.statusText, text);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toastr.success(t`Extension ${extensionName} moved.`);
|
||||||
|
await loadExtensionSettings({}, false, false);
|
||||||
|
await Popup.util.popups.find(popup => popup.content.querySelector('.extensions_info'))?.completeCancelled();
|
||||||
|
showExtensionsDetails();
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error:', error);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -853,7 +903,7 @@ export async function deleteExtension(extensionName) {
|
|||||||
console.error('Error:', error);
|
console.error('Error:', error);
|
||||||
}
|
}
|
||||||
|
|
||||||
toastr.success(`Extension ${extensionName} deleted`);
|
toastr.success(t`Extension ${extensionName} deleted`);
|
||||||
showExtensionsDetails();
|
showExtensionsDetails();
|
||||||
// reload the page to remove the extension from the list
|
// reload the page to remove the extension from the list
|
||||||
location.reload();
|
location.reload();
|
||||||
@ -896,7 +946,7 @@ async function getExtensionVersion(extensionName, abortSignal) {
|
|||||||
export async function installExtension(url, global) {
|
export async function installExtension(url, global) {
|
||||||
console.debug('Extension installation started', url);
|
console.debug('Extension installation started', url);
|
||||||
|
|
||||||
toastr.info('Please wait...', 'Installing extension');
|
toastr.info(t`Please wait...`, t`Installing extension`);
|
||||||
|
|
||||||
const request = await fetch('/api/extensions/install', {
|
const request = await fetch('/api/extensions/install', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
@ -909,7 +959,7 @@ export async function installExtension(url, global) {
|
|||||||
|
|
||||||
if (!request.ok) {
|
if (!request.ok) {
|
||||||
const text = await request.text();
|
const text = await request.text();
|
||||||
toastr.warning(text || request.statusText, 'Extension installation failed', { timeOut: 5000 });
|
toastr.warning(text || request.statusText, t`Extension installation failed`, { timeOut: 5000 });
|
||||||
console.error('Extension installation failed', request.status, request.statusText, text);
|
console.error('Extension installation failed', request.status, request.statusText, text);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -80,6 +80,7 @@ router.post('/install', jsonParser, async (request, response) => {
|
|||||||
const { url, global } = request.body;
|
const { url, global } = request.body;
|
||||||
|
|
||||||
if (global && !request.user.profile.admin) {
|
if (global && !request.user.profile.admin) {
|
||||||
|
console.warn(`User ${request.user.profile.handle} does not have permission to install global extensions.`);
|
||||||
return response.status(403).send('Forbidden: No permission to install global extensions.');
|
return response.status(403).send('Forbidden: No permission to install global extensions.');
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -123,6 +124,7 @@ router.post('/update', jsonParser, async (request, response) => {
|
|||||||
const { extensionName, global } = request.body;
|
const { extensionName, global } = request.body;
|
||||||
|
|
||||||
if (global && !request.user.profile.admin) {
|
if (global && !request.user.profile.admin) {
|
||||||
|
console.warn(`User ${request.user.profile.handle} does not have permission to update global extensions.`);
|
||||||
return response.status(403).send('Forbidden: No permission to update global extensions.');
|
return response.status(403).send('Forbidden: No permission to update global extensions.');
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -153,6 +155,50 @@ router.post('/update', jsonParser, async (request, response) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
router.post('/move', jsonParser, async (request, response) => {
|
||||||
|
try {
|
||||||
|
const { extensionName, source, destination } = request.body;
|
||||||
|
|
||||||
|
if (!extensionName || !source || !destination) {
|
||||||
|
return response.status(400).send('Bad Request. Not all required parameters are provided.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!request.user.profile.admin) {
|
||||||
|
console.warn(`User ${request.user.profile.handle} does not have permission to move extensions.`);
|
||||||
|
return response.status(403).send('Forbidden: No permission to move extensions.');
|
||||||
|
}
|
||||||
|
|
||||||
|
const sourceDirectory = source === 'global' ? PUBLIC_DIRECTORIES.globalExtensions : request.user.directories.extensions;
|
||||||
|
const destinationDirectory = destination === 'global' ? PUBLIC_DIRECTORIES.globalExtensions : request.user.directories.extensions;
|
||||||
|
const sourcePath = path.join(sourceDirectory, sanitize(extensionName));
|
||||||
|
const destinationPath = path.join(destinationDirectory, sanitize(extensionName));
|
||||||
|
|
||||||
|
if (!fs.existsSync(sourcePath) || !fs.statSync(sourcePath).isDirectory()) {
|
||||||
|
console.error(`Source directory does not exist at ${sourcePath}`);
|
||||||
|
return response.status(404).send('Source directory does not exist.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fs.existsSync(destinationPath)) {
|
||||||
|
console.error(`Destination directory already exists at ${destinationPath}`);
|
||||||
|
return response.status(409).send('Destination directory already exists.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (source === destination) {
|
||||||
|
console.error('Source and destination directories are the same');
|
||||||
|
return response.status(409).send('Source and destination directories are the same.');
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.cpSync(sourcePath, destinationPath, { recursive: true, force: true });
|
||||||
|
fs.rmSync(sourcePath, { recursive: true, force: true });
|
||||||
|
console.log(`Extension has been moved from ${sourcePath} to ${destinationPath}`);
|
||||||
|
|
||||||
|
return response.sendStatus(204);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('Moving extension failed', error);
|
||||||
|
return response.status(500).send('Internal Server Error. Try again later.');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* HTTP POST handler function to get the current git commit hash and branch name for a given extension.
|
* HTTP POST handler function to get the current git commit hash and branch name for a given extension.
|
||||||
* It checks whether the repository is up-to-date with the remote, and returns the status along with
|
* It checks whether the repository is up-to-date with the remote, and returns the status along with
|
||||||
@ -211,6 +257,7 @@ router.post('/delete', jsonParser, async (request, response) => {
|
|||||||
const { extensionName, global } = request.body;
|
const { extensionName, global } = request.body;
|
||||||
|
|
||||||
if (global && !request.user.profile.admin) {
|
if (global && !request.user.profile.admin) {
|
||||||
|
console.warn(`User ${request.user.profile.handle} does not have permission to delete global extensions.`);
|
||||||
return response.status(403).send('Forbidden: No permission to delete global extensions.');
|
return response.status(403).send('Forbidden: No permission to delete global extensions.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user