Add scaffolding

This commit is contained in:
adrinr 2023-03-09 16:15:50 +01:00
parent d083553373
commit 4068faf9f3
7 changed files with 122 additions and 0 deletions

View File

@ -2,3 +2,4 @@ export * from "./environmentVariables"
export * from "./auditLogs" export * from "./auditLogs"
export * from "./events" export * from "./events"
export * from "./configs" export * from "./configs"
export * from "./scim"

View File

@ -0,0 +1 @@
export * from "./users"

View File

@ -0,0 +1,32 @@
export interface ScimUser {
schemas: ["urn:ietf:params:scim:schemas:core:2.0:User"]
id: string
externalId: string
meta: {
resourceType: "User"
created: string
lastModified: string
}
userName: string
name: {
formatted: string
familyName: string
givenName: string
}
active: boolean
emails: [
{
value: string
type: "work"
primary: true
}
]
}
export interface ScimListResponse {
schemas: ["urn:ietf:params:scim:api:messages:2.0:ListResponse"]
totalResults: number
Resources: ScimUser[]
startIndex: number
itemsPerPage: number
}

View File

@ -0,0 +1,41 @@
import { InviteUsersResponse, User } from "@budibase/types"
jest.mock("nodemailer")
import { TestConfiguration, mocks, structures } from "../../../../../tests"
const sendMailMock = mocks.email.mock()
import { events, tenancy, accounts as _accounts } from "@budibase/backend-core"
import * as userSdk from "../../../../../sdk/users"
const accounts = jest.mocked(_accounts)
describe("/api/global/scim/v2/users", () => {
const config = new TestConfiguration()
beforeAll(async () => {
await config.beforeAll()
})
afterAll(async () => {
await config.afterAll()
})
beforeEach(() => {
jest.clearAllMocks()
})
describe("GET /api/global/scim/v2/users", () => {
describe("no users exist", () => {
it("should retrieve empty list", async () => {
const response = await config.api.scimUsersAPI.get()
expect(response).toEqual({
Resources: [],
itemsPerPage: 20,
schemas: ["urn:ietf:params:scim:api:messages:2.0:ListResponse"],
startIndex: 1,
totalResults: 0,
})
})
})
})
})

View File

@ -20,6 +20,9 @@ import {
auth, auth,
constants, constants,
env as coreEnv, env as coreEnv,
db as dbCore,
encryption,
utils,
} from "@budibase/backend-core" } from "@budibase/backend-core"
import structures, { CSRF_TOKEN } from "./structures" import structures, { CSRF_TOKEN } from "./structures"
import { SaveUserResponse, User, AuthToken } from "@budibase/types" import { SaveUserResponse, User, AuthToken } from "@budibase/types"
@ -31,6 +34,7 @@ class TestConfiguration {
api: API api: API
tenantId: string tenantId: string
user?: User user?: User
apiKey?: string
userPassword = "test" userPassword = "test"
constructor(opts: { openServer: boolean } = { openServer: true }) { constructor(opts: { openServer: boolean } = { openServer: true }) {
@ -201,6 +205,12 @@ class TestConfiguration {
return { [constants.Header.API_KEY]: coreEnv.INTERNAL_API_KEY } return { [constants.Header.API_KEY]: coreEnv.INTERNAL_API_KEY }
} }
bearerAPIHeaders() {
return {
[constants.Header.AUTHORIZATION]: `Bearer ${this.apiKey}`,
}
}
adminOnlyResponse = () => { adminOnlyResponse = () => {
return { message: "Admin user only endpoint.", status: 403 } return { message: "Admin user only endpoint.", status: 403 }
} }
@ -213,6 +223,20 @@ class TestConfiguration {
}) })
await context.doInTenant(this.tenantId!, async () => { await context.doInTenant(this.tenantId!, async () => {
this.user = await this.createUser(user) this.user = await this.createUser(user)
const db = context.getGlobalDB()
const id = dbCore.generateDevInfoID(this.user._id)
// TODO: dry
this.apiKey = encryption.encrypt(
`${this.tenantId}${dbCore.SEPARATOR}${utils.newid()}`
)
const devInfo = {
_id: id,
userId: this.user._id,
apiKey: this.apiKey,
}
await db.put(devInfo)
}) })
} }

View File

@ -15,6 +15,8 @@ import { RolesAPI } from "./roles"
import { TemplatesAPI } from "./templates" import { TemplatesAPI } from "./templates"
import { LicenseAPI } from "./license" import { LicenseAPI } from "./license"
import { AuditLogAPI } from "./auditLogs" import { AuditLogAPI } from "./auditLogs"
import { ScimUsersAPI } from "./scim/users"
export default class API { export default class API {
accounts: AccountAPI accounts: AccountAPI
auth: AuthAPI auth: AuthAPI
@ -32,6 +34,7 @@ export default class API {
templates: TemplatesAPI templates: TemplatesAPI
license: LicenseAPI license: LicenseAPI
auditLogs: AuditLogAPI auditLogs: AuditLogAPI
scimUsersAPI: ScimUsersAPI
constructor(config: TestConfiguration) { constructor(config: TestConfiguration) {
this.accounts = new AccountAPI(config) this.accounts = new AccountAPI(config)
@ -50,5 +53,6 @@ export default class API {
this.templates = new TemplatesAPI(config) this.templates = new TemplatesAPI(config)
this.license = new LicenseAPI(config) this.license = new LicenseAPI(config)
this.auditLogs = new AuditLogAPI(config) this.auditLogs = new AuditLogAPI(config)
this.scimUsersAPI = new ScimUsersAPI(config)
} }
} }

View File

@ -0,0 +1,19 @@
import { AccountMetadata, ScimListResponse } from "@budibase/types"
import TestConfiguration from "../../TestConfiguration"
import { TestAPI } from "../base"
export class ScimUsersAPI extends TestAPI {
constructor(config: TestConfiguration) {
super(config)
}
get = async ({ expect }: { expect: number } = { expect: 200 }) => {
const res = await this.request
.get(`/api/global/scim/v2/users`)
.set(this.config.bearerAPIHeaders())
.expect("Content-Type", /json/)
.expect(expect)
return res.body as ScimListResponse
}
}