mirror of
https://github.com/hay-kot/homebox.git
synced 2024-11-26 02:25:41 +00:00
api refactor / registration tests
This commit is contained in:
parent
a67316a889
commit
505099ee26
12 changed files with 120 additions and 33 deletions
|
@ -1,5 +1,5 @@
|
||||||
import { PublicApi } from "~~/lib/api/public";
|
import { PublicApi } from "~~/lib/api/public";
|
||||||
import { UserApi } from "~~/lib/api/user";
|
import { UserClient } from "~~/lib/api/user";
|
||||||
import { Requests } from "~~/lib/requests";
|
import { Requests } from "~~/lib/requests";
|
||||||
import { useAuthStore } from "~~/stores/auth";
|
import { useAuthStore } from "~~/stores/auth";
|
||||||
|
|
||||||
|
@ -28,7 +28,7 @@ export function usePublicApi(): PublicApi {
|
||||||
return new PublicApi(requests);
|
return new PublicApi(requests);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useUserApi(): UserApi {
|
export function useUserApi(): UserClient {
|
||||||
const authStore = useAuthStore();
|
const authStore = useAuthStore();
|
||||||
|
|
||||||
const requests = new Requests("", () => authStore.token, {});
|
const requests = new Requests("", () => authStore.token, {});
|
||||||
|
@ -43,5 +43,5 @@ export function useUserApi(): UserApi {
|
||||||
requests.addResponseInterceptor(observer.handler);
|
requests.addResponseInterceptor(observer.handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new UserApi(requests);
|
return new UserClient(requests);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,17 @@
|
||||||
import { describe, test, expect } from "vitest";
|
import { describe, test, expect } from "vitest";
|
||||||
import { client, userClient } from "./test-utils";
|
import { faker } from "@faker-js/faker";
|
||||||
|
import { UserRegistration } from "../types/data-contracts";
|
||||||
|
import { client, sharedUserClient, userClient } from "./test-utils";
|
||||||
|
|
||||||
|
function userFactory(): UserRegistration {
|
||||||
|
return {
|
||||||
|
email: faker.internet.email(),
|
||||||
|
password: faker.internet.password(),
|
||||||
|
name: faker.name.firstName(),
|
||||||
|
groupName: faker.animal.cat(),
|
||||||
|
token: "",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
describe("[GET] /api/v1/status", () => {
|
describe("[GET] /api/v1/status", () => {
|
||||||
test("server should respond", async () => {
|
test("server should respond", async () => {
|
||||||
|
@ -10,14 +22,9 @@ describe("[GET] /api/v1/status", () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("first time user workflow (register, login)", () => {
|
describe("first time user workflow (register, login, join group)", () => {
|
||||||
const api = client();
|
const api = client();
|
||||||
const userData = {
|
const userData = userFactory();
|
||||||
groupName: "test-group",
|
|
||||||
email: "test-user@email.com",
|
|
||||||
name: "test-user",
|
|
||||||
password: "test-password",
|
|
||||||
};
|
|
||||||
|
|
||||||
test("user should be able to register", async () => {
|
test("user should be able to register", async () => {
|
||||||
const { response } = await api.register(userData);
|
const { response } = await api.register(userData);
|
||||||
|
@ -32,8 +39,47 @@ describe("first time user workflow (register, login)", () => {
|
||||||
// Cleanup
|
// Cleanup
|
||||||
const userApi = userClient(data.token);
|
const userApi = userClient(data.token);
|
||||||
{
|
{
|
||||||
const { response } = await userApi.deleteAccount();
|
const { response } = await userApi.user.delete();
|
||||||
expect(response.status).toBe(204);
|
expect(response.status).toBe(204);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("user should be able to join create join token and have user signup", async () => {
|
||||||
|
// Setup User 1 Token
|
||||||
|
|
||||||
|
const client = await sharedUserClient();
|
||||||
|
const { data: user1 } = await client.user.self();
|
||||||
|
|
||||||
|
const { response, data } = await client.group.createInvitation({
|
||||||
|
expiresAt: new Date(Date.now() + 1000 * 60 * 60 * 24 * 7),
|
||||||
|
uses: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.status).toBe(201);
|
||||||
|
expect(data.token).toBeTruthy();
|
||||||
|
|
||||||
|
// Create User 2 with token
|
||||||
|
|
||||||
|
const duplicateUser = userFactory();
|
||||||
|
duplicateUser.token = data.token;
|
||||||
|
|
||||||
|
const { response: registerResp } = await api.register(duplicateUser);
|
||||||
|
expect(registerResp.status).toBe(204);
|
||||||
|
|
||||||
|
const { response: loginResp, data: loginData } = await api.login(duplicateUser.email, duplicateUser.password);
|
||||||
|
expect(loginResp.status).toBe(200);
|
||||||
|
|
||||||
|
// Get Self and Assert
|
||||||
|
|
||||||
|
const client2 = userClient(loginData.token);
|
||||||
|
|
||||||
|
const { data: user2 } = await client2.user.self();
|
||||||
|
|
||||||
|
user2.item.groupName = user1.item.groupName;
|
||||||
|
|
||||||
|
// Cleanup User 2
|
||||||
|
|
||||||
|
const { response: deleteResp } = await client2.user.delete();
|
||||||
|
expect(deleteResp.status).toBe(204);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { Requests } from "../../requests";
|
||||||
import { overrideParts } from "../base/urls";
|
import { overrideParts } from "../base/urls";
|
||||||
import { PublicApi } from "../public";
|
import { PublicApi } from "../public";
|
||||||
import * as config from "../../../test/config";
|
import * as config from "../../../test/config";
|
||||||
import { UserApi } from "../user";
|
import { UserClient } from "../user";
|
||||||
|
|
||||||
export function client() {
|
export function client() {
|
||||||
overrideParts(config.BASE_URL, "/api/v1");
|
overrideParts(config.BASE_URL, "/api/v1");
|
||||||
|
@ -14,7 +14,7 @@ export function client() {
|
||||||
export function userClient(token: string) {
|
export function userClient(token: string) {
|
||||||
overrideParts(config.BASE_URL, "/api/v1");
|
overrideParts(config.BASE_URL, "/api/v1");
|
||||||
const requests = new Requests("", token);
|
const requests = new Requests("", token);
|
||||||
return new UserApi(requests);
|
return new UserClient(requests);
|
||||||
}
|
}
|
||||||
|
|
||||||
const cache = {
|
const cache = {
|
||||||
|
@ -25,7 +25,7 @@ const cache = {
|
||||||
* Shared UserApi token for tests where the creation of a user is _not_ import
|
* Shared UserApi token for tests where the creation of a user is _not_ import
|
||||||
* to the test. This is useful for tests that are testing the user API itself.
|
* to the test. This is useful for tests that are testing the user API itself.
|
||||||
*/
|
*/
|
||||||
export async function sharedUserClient(): Promise<UserApi> {
|
export async function sharedUserClient(): Promise<UserClient> {
|
||||||
if (cache.token) {
|
if (cache.token) {
|
||||||
return userClient(cache.token);
|
return userClient(cache.token);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { describe, test, expect } from "vitest";
|
import { describe, test, expect } from "vitest";
|
||||||
import { LocationOut } from "../../types/data-contracts";
|
import { LocationOut } from "../../types/data-contracts";
|
||||||
import { AttachmentTypes } from "../../types/non-generated";
|
import { AttachmentTypes } from "../../types/non-generated";
|
||||||
import { UserApi } from "../../user";
|
import { UserClient } from "../../user";
|
||||||
import { sharedUserClient } from "../test-utils";
|
import { sharedUserClient } from "../test-utils";
|
||||||
|
|
||||||
describe("user should be able to create an item and add an attachment", () => {
|
describe("user should be able to create an item and add an attachment", () => {
|
||||||
|
@ -10,7 +10,7 @@ describe("user should be able to create an item and add an attachment", () => {
|
||||||
* useLocatio sets up a location resource for testing, and returns a function
|
* useLocatio sets up a location resource for testing, and returns a function
|
||||||
* that can be used to delete the location from the backend server.
|
* that can be used to delete the location from the backend server.
|
||||||
*/
|
*/
|
||||||
async function useLocation(api: UserApi): Promise<[LocationOut, () => Promise<void>]> {
|
async function useLocation(api: UserClient): Promise<[LocationOut, () => Promise<void>]> {
|
||||||
const { response, data } = await api.locations.create({
|
const { response, data } = await api.locations.create({
|
||||||
name: `__test__.location.name_${increment}`,
|
name: `__test__.location.name_${increment}`,
|
||||||
description: `__test__.location.description_${increment}`,
|
description: `__test__.location.description_${increment}`,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { describe, expect, test } from "vitest";
|
import { describe, expect, test } from "vitest";
|
||||||
import { LabelOut } from "../../types/data-contracts";
|
import { LabelOut } from "../../types/data-contracts";
|
||||||
import { UserApi } from "../../user";
|
import { UserClient } from "../../user";
|
||||||
import { sharedUserClient } from "../test-utils";
|
import { sharedUserClient } from "../test-utils";
|
||||||
|
|
||||||
describe("locations lifecycle (create, update, delete)", () => {
|
describe("locations lifecycle (create, update, delete)", () => {
|
||||||
|
@ -10,7 +10,7 @@ describe("locations lifecycle (create, update, delete)", () => {
|
||||||
* useLabel sets up a label resource for testing, and returns a function
|
* useLabel sets up a label resource for testing, and returns a function
|
||||||
* that can be used to delete the label from the backend server.
|
* that can be used to delete the label from the backend server.
|
||||||
*/
|
*/
|
||||||
async function useLabel(api: UserApi): Promise<[LabelOut, () => Promise<void>]> {
|
async function useLabel(api: UserClient): Promise<[LabelOut, () => Promise<void>]> {
|
||||||
const { response, data } = await api.labels.create({
|
const { response, data } = await api.labels.create({
|
||||||
name: `__test__.label.name_${increment}`,
|
name: `__test__.label.name_${increment}`,
|
||||||
description: `__test__.label.description_${increment}`,
|
description: `__test__.label.description_${increment}`,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { describe, expect, test } from "vitest";
|
import { describe, expect, test } from "vitest";
|
||||||
import { LocationOut } from "../../types/data-contracts";
|
import { LocationOut } from "../../types/data-contracts";
|
||||||
import { UserApi } from "../../user";
|
import { UserClient } from "../../user";
|
||||||
import { sharedUserClient } from "../test-utils";
|
import { sharedUserClient } from "../test-utils";
|
||||||
|
|
||||||
describe("locations lifecycle (create, update, delete)", () => {
|
describe("locations lifecycle (create, update, delete)", () => {
|
||||||
|
@ -10,7 +10,7 @@ describe("locations lifecycle (create, update, delete)", () => {
|
||||||
* useLocatio sets up a location resource for testing, and returns a function
|
* useLocatio sets up a location resource for testing, and returns a function
|
||||||
* that can be used to delete the location from the backend server.
|
* that can be used to delete the location from the backend server.
|
||||||
*/
|
*/
|
||||||
async function useLocation(api: UserApi): Promise<[LocationOut, () => Promise<void>]> {
|
async function useLocation(api: UserClient): Promise<[LocationOut, () => Promise<void>]> {
|
||||||
const { response, data } = await api.locations.create({
|
const { response, data } = await api.locations.create({
|
||||||
name: `__test__.location.name_${increment}`,
|
name: `__test__.location.name_${increment}`,
|
||||||
description: `__test__.location.description_${increment}`,
|
description: `__test__.location.description_${increment}`,
|
||||||
|
|
11
frontend/lib/api/classes/group.ts
Normal file
11
frontend/lib/api/classes/group.ts
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
import { BaseAPI, route } from "../base";
|
||||||
|
import { GroupInvitation, GroupInvitationCreate } from "../types/data-contracts";
|
||||||
|
|
||||||
|
export class GroupApi extends BaseAPI {
|
||||||
|
createInvitation(data: GroupInvitationCreate) {
|
||||||
|
return this.http.post<GroupInvitationCreate, GroupInvitation>({
|
||||||
|
url: route("/groups/invitations"),
|
||||||
|
body: data,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
17
frontend/lib/api/classes/users.ts
Normal file
17
frontend/lib/api/classes/users.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
import { BaseAPI, route } from "../base";
|
||||||
|
import { UserOut } from "../types/data-contracts";
|
||||||
|
import { Result } from "../types/non-generated";
|
||||||
|
|
||||||
|
export class UserApi extends BaseAPI {
|
||||||
|
public self() {
|
||||||
|
return this.http.get<Result<UserOut>>({ url: route("/users/self") });
|
||||||
|
}
|
||||||
|
|
||||||
|
public logout() {
|
||||||
|
return this.http.post<object, void>({ url: route("/users/logout") });
|
||||||
|
}
|
||||||
|
|
||||||
|
public delete() {
|
||||||
|
return this.http.delete<void>({ url: route("/users/self") });
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,37 +1,42 @@
|
||||||
import { BaseAPI, route } from "./base";
|
import { BaseAPI } from "./base";
|
||||||
import { ItemsApi } from "./classes/items";
|
import { ItemsApi } from "./classes/items";
|
||||||
import { LabelsApi } from "./classes/labels";
|
import { LabelsApi } from "./classes/labels";
|
||||||
import { LocationsApi } from "./classes/locations";
|
import { LocationsApi } from "./classes/locations";
|
||||||
import { UserOut } from "./types/data-contracts";
|
import { GroupApi } from "./classes/group";
|
||||||
|
import { UserApi } from "./classes/users";
|
||||||
import { Requests } from "~~/lib/requests";
|
import { Requests } from "~~/lib/requests";
|
||||||
|
|
||||||
export type Result<T> = {
|
export class UserClient extends BaseAPI {
|
||||||
item: T;
|
|
||||||
};
|
|
||||||
|
|
||||||
export class UserApi extends BaseAPI {
|
|
||||||
locations: LocationsApi;
|
locations: LocationsApi;
|
||||||
labels: LabelsApi;
|
labels: LabelsApi;
|
||||||
items: ItemsApi;
|
items: ItemsApi;
|
||||||
|
group: GroupApi;
|
||||||
|
user: UserApi;
|
||||||
|
|
||||||
constructor(requests: Requests) {
|
constructor(requests: Requests) {
|
||||||
super(requests);
|
super(requests);
|
||||||
|
|
||||||
this.locations = new LocationsApi(requests);
|
this.locations = new LocationsApi(requests);
|
||||||
this.labels = new LabelsApi(requests);
|
this.labels = new LabelsApi(requests);
|
||||||
this.items = new ItemsApi(requests);
|
this.items = new ItemsApi(requests);
|
||||||
|
this.group = new GroupApi(requests);
|
||||||
|
this.user = new UserApi(requests);
|
||||||
|
|
||||||
Object.freeze(this);
|
Object.freeze(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @deprecated use this.user.self() */
|
||||||
public self() {
|
public self() {
|
||||||
return this.http.get<Result<UserOut>>({ url: route("/users/self") });
|
return this.user.self();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @deprecated use this.user.logout() */
|
||||||
public logout() {
|
public logout() {
|
||||||
return this.http.post<object, void>({ url: route("/users/logout") });
|
return this.user.logout();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @deprecated use this.user.delete() */
|
||||||
public deleteAccount() {
|
public deleteAccount() {
|
||||||
return this.http.delete<void>({ url: route("/users/self") });
|
return this.user.delete();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
"test:watch": " TEST_SHUTDOWN_API_SERVER=false vitest --config ./test/vitest.config.ts"
|
"test:watch": " TEST_SHUTDOWN_API_SERVER=false vitest --config ./test/vitest.config.ts"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@faker-js/faker": "^7.5.0",
|
||||||
"@nuxtjs/eslint-config-typescript": "^11.0.0",
|
"@nuxtjs/eslint-config-typescript": "^11.0.0",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.36.2",
|
"@typescript-eslint/eslint-plugin": "^5.36.2",
|
||||||
"@typescript-eslint/parser": "^5.36.2",
|
"@typescript-eslint/parser": "^5.36.2",
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
lockfileVersion: 5.4
|
lockfileVersion: 5.4
|
||||||
|
|
||||||
specifiers:
|
specifiers:
|
||||||
|
'@faker-js/faker': ^7.5.0
|
||||||
'@iconify/vue': ^3.2.1
|
'@iconify/vue': ^3.2.1
|
||||||
'@nuxtjs/eslint-config-typescript': ^11.0.0
|
'@nuxtjs/eslint-config-typescript': ^11.0.0
|
||||||
'@nuxtjs/tailwindcss': ^5.3.2
|
'@nuxtjs/tailwindcss': ^5.3.2
|
||||||
|
@ -44,6 +45,7 @@ dependencies:
|
||||||
vue: 3.2.39
|
vue: 3.2.39
|
||||||
|
|
||||||
devDependencies:
|
devDependencies:
|
||||||
|
'@faker-js/faker': 7.5.0
|
||||||
'@nuxtjs/eslint-config-typescript': 11.0.0_7ilbxdl5iguzcjriqqcg2m5cku
|
'@nuxtjs/eslint-config-typescript': 11.0.0_7ilbxdl5iguzcjriqqcg2m5cku
|
||||||
'@typescript-eslint/eslint-plugin': 5.38.0_4gkcvl6qsi23tqqawfqgcwtp54
|
'@typescript-eslint/eslint-plugin': 5.38.0_4gkcvl6qsi23tqqawfqgcwtp54
|
||||||
'@typescript-eslint/parser': 5.38.0_7ilbxdl5iguzcjriqqcg2m5cku
|
'@typescript-eslint/parser': 5.38.0_7ilbxdl5iguzcjriqqcg2m5cku
|
||||||
|
@ -381,6 +383,11 @@ packages:
|
||||||
transitivePeerDependencies:
|
transitivePeerDependencies:
|
||||||
- supports-color
|
- supports-color
|
||||||
|
|
||||||
|
/@faker-js/faker/7.5.0:
|
||||||
|
resolution: {integrity: sha512-8wNUCCUHvfvI0gQpDUho/3gPzABffnCn5um65F8dzQ86zz6dlt4+nmAA7PQUc8L+eH+9RgR/qzy5N/8kN0Ozdw==}
|
||||||
|
engines: {node: '>=14.0.0', npm: '>=6.0.0'}
|
||||||
|
dev: true
|
||||||
|
|
||||||
/@humanwhocodes/config-array/0.10.5:
|
/@humanwhocodes/config-array/0.10.5:
|
||||||
resolution: {integrity: sha512-XVVDtp+dVvRxMoxSiSfasYaG02VEe1qH5cKgMQJWhol6HwzbcqoCMJi8dAGoYAO57jhUyhI6cWuRiTcRaDaYug==}
|
resolution: {integrity: sha512-XVVDtp+dVvRxMoxSiSfasYaG02VEe1qH5cKgMQJWhol6HwzbcqoCMJi8dAGoYAO57jhUyhI6cWuRiTcRaDaYug==}
|
||||||
engines: {node: '>=10.10.0'}
|
engines: {node: '>=10.10.0'}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { defineStore } from "pinia";
|
import { defineStore } from "pinia";
|
||||||
import { useLocalStorage } from "@vueuse/core";
|
import { useLocalStorage } from "@vueuse/core";
|
||||||
import { UserApi } from "~~/lib/api/user";
|
import { UserClient } from "~~/lib/api/user";
|
||||||
import { UserOut } from "~~/lib/api/types/data-contracts";
|
import { UserOut } from "~~/lib/api/types/data-contracts";
|
||||||
|
|
||||||
export const useAuthStore = defineStore("auth", {
|
export const useAuthStore = defineStore("auth", {
|
||||||
|
@ -23,7 +23,7 @@ export const useAuthStore = defineStore("auth", {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
actions: {
|
actions: {
|
||||||
async logout(api: UserApi) {
|
async logout(api: UserClient) {
|
||||||
const result = await api.logout();
|
const result = await api.logout();
|
||||||
|
|
||||||
if (result.error) {
|
if (result.error) {
|
||||||
|
|
Loading…
Reference in a new issue