1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-01 08:31:59 +00:00

feat(web,server): logout all devices (#2415)

* feat: logout all devices

* chore: regenerate openapi

* chore: add test

* chore: logout vs log out
This commit is contained in:
Jason Rasmussen 2023-05-09 15:34:17 -04:00 committed by GitHub
parent c956eee919
commit a808b9403e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
11 changed files with 151 additions and 4 deletions

BIN
mobile/openapi/README.md generated

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View file

@ -52,6 +52,12 @@ export class AuthController {
return this.service.getDevices(authUser); return this.service.getDevices(authUser);
} }
@Authenticated()
@Delete('devices')
logoutAuthDevices(@GetAuthUser() authUser: AuthUserDto): Promise<void> {
return this.service.logoutDevices(authUser);
}
@Authenticated() @Authenticated()
@Delete('devices/:id') @Delete('devices/:id')
logoutAuthDevice(@GetAuthUser() authUser: AuthUserDto, @Param() { id }: UUIDParamDto): Promise<void> { logoutAuthDevice(@GetAuthUser() authUser: AuthUserDto, @Param() { id }: UUIDParamDto): Promise<void> {

View file

@ -393,6 +393,29 @@
"api_key": [] "api_key": []
} }
] ]
},
"delete": {
"operationId": "logoutAuthDevices",
"parameters": [],
"responses": {
"200": {
"description": ""
}
},
"tags": [
"Authentication"
],
"security": [
{
"bearer": []
},
{
"cookie": []
},
{
"api_key": []
}
]
} }
}, },
"/auth/devices/{id}": { "/auth/devices/{id}": {

View file

@ -357,6 +357,18 @@ describe('AuthService', () => {
}); });
}); });
describe('logoutDevices', () => {
it('should logout all devices', async () => {
userTokenMock.getAll.mockResolvedValue([userTokenEntityStub.inactiveToken, userTokenEntityStub.userToken]);
await sut.logoutDevices(authStub.user1);
expect(userTokenMock.getAll).toHaveBeenCalledWith(authStub.user1.id);
expect(userTokenMock.delete).toHaveBeenCalledWith(authStub.user1.id, 'not_active');
expect(userTokenMock.delete).not.toHaveBeenCalledWith(authStub.user1.id, 'token-id');
});
});
describe('logoutDevice', () => { describe('logoutDevice', () => {
it('should logout the device', async () => { it('should logout the device', async () => {
await sut.logoutDevice(authStub.user1, 'token-1'); await sut.logoutDevice(authStub.user1, 'token-1');

View file

@ -163,6 +163,16 @@ export class AuthService {
await this.userTokenCore.delete(authUser.id, deviceId); await this.userTokenCore.delete(authUser.id, deviceId);
} }
async logoutDevices(authUser: AuthUserDto): Promise<void> {
const devices = await this.userTokenCore.getAll(authUser.id);
for (const device of devices) {
if (device.id === authUser.accessTokenId) {
continue;
}
await this.userTokenCore.delete(authUser.id, device.id);
}
}
private getBearerToken(headers: IncomingHttpHeaders): string | null { private getBearerToken(headers: IncomingHttpHeaders): string | null {
const [type, token] = (headers.authorization || '').split(' '); const [type, token] = (headers.authorization || '').split(' ');
if (type.toLowerCase() === 'bearer') { if (type.toLowerCase() === 'bearer') {

View file

@ -6299,6 +6299,44 @@ export const AuthenticationApiAxiosParamCreator = function (configuration?: Conf
setSearchParams(localVarUrlObj, localVarQueryParameter);
let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {};
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
return {
url: toPathString(localVarUrlObj),
options: localVarRequestOptions,
};
},
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
logoutAuthDevices: async (options: AxiosRequestConfig = {}): Promise<RequestArgs> => {
const localVarPath = `/auth/devices`;
// use dummy base URL string because the URL constructor only accepts absolute URLs.
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
let baseOptions;
if (configuration) {
baseOptions = configuration.baseOptions;
}
const localVarRequestOptions = { method: 'DELETE', ...baseOptions, ...options};
const localVarHeaderParameter = {} as any;
const localVarQueryParameter = {} as any;
// authentication cookie required
// authentication api_key required
await setApiKeyToObject(localVarHeaderParameter, "x-api-key", configuration)
// authentication bearer required
// http bearer authentication required
await setBearerAuthToObject(localVarHeaderParameter, configuration)
setSearchParams(localVarUrlObj, localVarQueryParameter); setSearchParams(localVarUrlObj, localVarQueryParameter);
let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {}; let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {};
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers}; localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
@ -6414,6 +6452,15 @@ export const AuthenticationApiFp = function(configuration?: Configuration) {
const localVarAxiosArgs = await localVarAxiosParamCreator.logoutAuthDevice(id, options); const localVarAxiosArgs = await localVarAxiosParamCreator.logoutAuthDevice(id, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
async logoutAuthDevices(options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<void>> {
const localVarAxiosArgs = await localVarAxiosParamCreator.logoutAuthDevices(options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
},
/** /**
* *
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
@ -6485,6 +6532,14 @@ export const AuthenticationApiFactory = function (configuration?: Configuration,
logoutAuthDevice(id: string, options?: any): AxiosPromise<void> { logoutAuthDevice(id: string, options?: any): AxiosPromise<void> {
return localVarFp.logoutAuthDevice(id, options).then((request) => request(axios, basePath)); return localVarFp.logoutAuthDevice(id, options).then((request) => request(axios, basePath));
}, },
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
logoutAuthDevices(options?: any): AxiosPromise<void> {
return localVarFp.logoutAuthDevices(options).then((request) => request(axios, basePath));
},
/** /**
* *
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
@ -6567,6 +6622,16 @@ export class AuthenticationApi extends BaseAPI {
return AuthenticationApiFp(this.configuration).logoutAuthDevice(id, options).then((request) => request(this.axios, this.basePath)); return AuthenticationApiFp(this.configuration).logoutAuthDevice(id, options).then((request) => request(this.axios, this.basePath));
} }
/**
*
* @param {*} [options] Override http request option.
* @throws {RequiredError}
* @memberof AuthenticationApi
*/
public logoutAuthDevices(options?: AxiosRequestConfig) {
return AuthenticationApiFp(this.configuration).logoutAuthDevices(options).then((request) => request(this.axios, this.basePath));
}
/** /**
* *
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.

View file

@ -62,7 +62,7 @@
<button <button
on:click={() => dispatcher('delete')} on:click={() => dispatcher('delete')}
class="bg-immich-primary dark:bg-immich-dark-primary text-gray-100 dark:text-gray-700 rounded-full p-3 transition-all duration-150 hover:bg-immich-primary/75" class="bg-immich-primary dark:bg-immich-dark-primary text-gray-100 dark:text-gray-700 rounded-full p-3 transition-all duration-150 hover:bg-immich-primary/75"
title="Logout" title="Log out"
> >
<TrashCanOutline size="16" /> <TrashCanOutline size="16" />
</button> </button>

View file

@ -2,6 +2,7 @@
import { api, AuthDeviceResponseDto } from '@api'; import { api, AuthDeviceResponseDto } from '@api';
import { onMount } from 'svelte'; import { onMount } from 'svelte';
import { handleError } from '../../utils/handle-error'; import { handleError } from '../../utils/handle-error';
import Button from '../elements/buttons/button.svelte';
import ConfirmDialogue from '../shared-components/confirm-dialogue.svelte'; import ConfirmDialogue from '../shared-components/confirm-dialogue.svelte';
import { import {
notificationController, notificationController,
@ -11,6 +12,7 @@
let devices: AuthDeviceResponseDto[] = []; let devices: AuthDeviceResponseDto[] = [];
let deleteDevice: AuthDeviceResponseDto | null = null; let deleteDevice: AuthDeviceResponseDto | null = null;
let deleteAll = false;
const refresh = () => api.authenticationApi.getAuthDevices().then(({ data }) => (devices = data)); const refresh = () => api.authenticationApi.getAuthDevices().then(({ data }) => (devices = data));
@ -30,22 +32,45 @@
await api.authenticationApi.logoutAuthDevice(deleteDevice.id); await api.authenticationApi.logoutAuthDevice(deleteDevice.id);
notificationController.show({ message: `Logged out device`, type: NotificationType.Info }); notificationController.show({ message: `Logged out device`, type: NotificationType.Info });
} catch (error) { } catch (error) {
handleError(error, 'Unable to logout device'); handleError(error, 'Unable to log out device');
} finally { } finally {
await refresh(); await refresh();
deleteDevice = null; deleteDevice = null;
} }
}; };
const handleDeleteAll = async () => {
try {
await api.authenticationApi.logoutAuthDevices();
notificationController.show({
message: `Logged out all devices`,
type: NotificationType.Info
});
} catch (error) {
handleError(error, 'Unable to log out all devices');
} finally {
await refresh();
deleteAll = false;
}
};
</script> </script>
{#if deleteDevice} {#if deleteDevice}
<ConfirmDialogue <ConfirmDialogue
prompt="Are you sure you want to logout this device?" prompt="Are you sure you want to log out this device?"
on:confirm={() => handleDelete()} on:confirm={() => handleDelete()}
on:cancel={() => (deleteDevice = null)} on:cancel={() => (deleteDevice = null)}
/> />
{/if} {/if}
{#if deleteAll}
<ConfirmDialogue
prompt="Are you sure you want to log out all devices?"
on:confirm={() => handleDeleteAll()}
on:cancel={() => (deleteAll = false)}
/>
{/if}
<section class="my-4"> <section class="my-4">
{#if currentDevice} {#if currentDevice}
<div class="mb-6"> <div class="mb-6">
@ -56,7 +81,7 @@
</div> </div>
{/if} {/if}
{#if otherDevices.length > 0} {#if otherDevices.length > 0}
<div> <div class="mb-6">
<h3 class="font-medium text-xs mb-2 text-immich-primary dark:text-immich-dark-primary"> <h3 class="font-medium text-xs mb-2 text-immich-primary dark:text-immich-dark-primary">
OTHER DEVICES OTHER DEVICES
</h3> </h3>
@ -67,5 +92,11 @@
{/if} {/if}
{/each} {/each}
</div> </div>
<h3 class="font-medium text-xs mb-2 text-immich-primary dark:text-immich-dark-primary">
LOG OUT ALL DEVICES
</h3>
<div class="flex justify-end">
<Button color="red" size="sm" on:click={() => (deleteAll = true)}>Log Out All Devices</Button>
</div>
{/if} {/if}
</section> </section>