Merge branch 'master' into pc/bug-fixing

This commit is contained in:
Peter Clement 2024-03-07 09:34:41 +00:00 committed by GitHub
commit a7551c7c6e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
25 changed files with 305 additions and 92 deletions

View File

@ -44,7 +44,8 @@
"no-undef": "off", "no-undef": "off",
"no-prototype-builtins": "off", "no-prototype-builtins": "off",
"local-rules/no-budibase-imports": "error", "local-rules/no-budibase-imports": "error",
"local-rules/no-test-com": "error" "local-rules/no-test-com": "error",
"local-rules/email-domain-example-com": "error"
} }
}, },
{ {

View File

@ -51,4 +51,41 @@ module.exports = {
} }
}, },
}, },
"email-domain-example-com": {
meta: {
type: "problem",
docs: {
description:
"enforce using the example.com domain for generator.email calls",
category: "Possible Errors",
recommended: false,
},
fixable: "code",
schema: [],
},
create: function (context) {
return {
CallExpression(node) {
if (
node.callee.type === "MemberExpression" &&
node.callee.object.name === "generator" &&
node.callee.property.name === "email" &&
node.arguments.length === 0
) {
context.report({
node,
message:
"Prefer using generator.email with the domain \"{ domain: 'example.com' }\".",
fix: function (fixer) {
return fixer.replaceText(
node,
'generator.email({ domain: "example.com" })'
)
},
})
}
},
}
},
},
} }

View File

@ -6,7 +6,7 @@ import env from "../environment"
import * as accounts from "../accounts" import * as accounts from "../accounts"
import { UserDB } from "../users" import { UserDB } from "../users"
import { sdk } from "@budibase/shared-core" import { sdk } from "@budibase/shared-core"
import { User } from "@budibase/types" import { User, UserMetadata } from "@budibase/types"
const EXPIRY_SECONDS = 3600 const EXPIRY_SECONDS = 3600
@ -15,7 +15,7 @@ const EXPIRY_SECONDS = 3600
*/ */
async function populateFromDB(userId: string, tenantId: string) { async function populateFromDB(userId: string, tenantId: string) {
const db = tenancy.getTenantDB(tenantId) const db = tenancy.getTenantDB(tenantId)
const user = await db.get<any>(userId) const user = await db.get<UserMetadata>(userId)
user.budibaseAccess = true user.budibaseAccess = true
if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) { if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) {
const account = await accounts.getAccount(user.email) const account = await accounts.getAccount(user.email)

View File

@ -1,66 +1,57 @@
import PouchDB from "pouchdb"
import { getPouchDB, closePouchDB } from "./couch" import { getPouchDB, closePouchDB } from "./couch"
import { DocumentType } from "../constants" import { DocumentType } from "../constants"
class Replication { class Replication {
source: any source: PouchDB.Database
target: any target: PouchDB.Database
replication: any
/** constructor({ source, target }: { source: string; target: string }) {
*
* @param source - the DB you want to replicate or rollback to
* @param target - the DB you want to replicate to, or rollback from
*/
constructor({ source, target }: any) {
this.source = getPouchDB(source) this.source = getPouchDB(source)
this.target = getPouchDB(target) this.target = getPouchDB(target)
} }
close() { async close() {
return Promise.all([closePouchDB(this.source), closePouchDB(this.target)]) await Promise.all([closePouchDB(this.source), closePouchDB(this.target)])
} }
promisify(operation: any, opts = {}) { replicate(opts: PouchDB.Replication.ReplicateOptions = {}) {
return new Promise(resolve => { return new Promise<PouchDB.Replication.ReplicationResult<{}>>(resolve => {
operation(this.target, opts) this.source.replicate
.on("denied", function (err: any) { .to(this.target, opts)
.on("denied", function (err) {
// a document failed to replicate (e.g. due to permissions) // a document failed to replicate (e.g. due to permissions)
throw new Error(`Denied: Document failed to replicate ${err}`) throw new Error(`Denied: Document failed to replicate ${err}`)
}) })
.on("complete", function (info: any) { .on("complete", function (info) {
return resolve(info) return resolve(info)
}) })
.on("error", function (err: any) { .on("error", function (err) {
throw new Error(`Replication Error: ${err}`) throw new Error(`Replication Error: ${err}`)
}) })
}) })
} }
/** appReplicateOpts(
* Two way replication operation, intended to be promise based. opts: PouchDB.Replication.ReplicateOptions = {}
* @param opts - PouchDB replication options ): PouchDB.Replication.ReplicateOptions {
*/ if (typeof opts.filter === "string") {
sync(opts = {}) { return opts
this.replication = this.promisify(this.source.sync, opts) }
return this.replication
}
/** const filter = opts.filter
* One way replication operation, intended to be promise based. delete opts.filter
* @param opts - PouchDB replication options
*/
replicate(opts = {}) {
this.replication = this.promisify(this.source.replicate.to, opts)
return this.replication
}
appReplicateOpts() {
return { return {
filter: (doc: any) => { ...opts,
filter: (doc: any, params: any) => {
if (doc._id && doc._id.startsWith(DocumentType.AUTOMATION_LOG)) { if (doc._id && doc._id.startsWith(DocumentType.AUTOMATION_LOG)) {
return false return false
} }
return doc._id !== DocumentType.APP_METADATA if (doc._id === DocumentType.APP_METADATA) {
return false
}
return filter ? filter(doc, params) : true
}, },
} }
} }
@ -75,10 +66,6 @@ class Replication {
// take the opportunity to remove deleted tombstones // take the opportunity to remove deleted tombstones
await this.replicate() await this.replicate()
} }
cancel() {
this.replication.cancel()
}
} }
export default Replication export default Replication

View File

@ -101,10 +101,7 @@ export function getBuiltinRole(roleId: string): Role | undefined {
/** /**
* Works through the inheritance ranks to see how far up the builtin stack this ID is. * Works through the inheritance ranks to see how far up the builtin stack this ID is.
*/ */
export function builtinRoleToNumber(id?: string) { export function builtinRoleToNumber(id: string) {
if (!id) {
return 0
}
const builtins = getBuiltinRoles() const builtins = getBuiltinRoles()
const MAX = Object.values(builtins).length + 1 const MAX = Object.values(builtins).length + 1
if (id === BUILTIN_IDS.ADMIN || id === BUILTIN_IDS.BUILDER) { if (id === BUILTIN_IDS.ADMIN || id === BUILTIN_IDS.BUILDER) {

View File

@ -18,7 +18,7 @@ export const account = (partial: Partial<Account> = {}): Account => {
return { return {
accountId: uuid(), accountId: uuid(),
tenantId: generator.word(), tenantId: generator.word(),
email: generator.email(), email: generator.email({ domain: "example.com" }),
tenantName: generator.word(), tenantName: generator.word(),
hosting: Hosting.SELF, hosting: Hosting.SELF,
createdAt: Date.now(), createdAt: Date.now(),

View File

@ -13,7 +13,7 @@ interface CreateUserRequestFields {
export function createUserRequest(userData?: Partial<CreateUserRequestFields>) { export function createUserRequest(userData?: Partial<CreateUserRequestFields>) {
const defaultValues = { const defaultValues = {
externalId: uuid(), externalId: uuid(),
email: generator.email(), email: `${uuid()}@example.com`,
firstName: generator.first(), firstName: generator.first(),
lastName: generator.last(), lastName: generator.last(),
username: generator.name(), username: generator.name(),

View File

@ -139,10 +139,22 @@
{/each} {/each}
</div> </div>
<div class="search-input"> <div class="search-input">
<div class="input-wrapper"> <div class="input-wrapper" style={`width: ${value ? "425" : "510"}px`}>
<Input bind:value={searchTerm} thin placeholder="Search Icon" /> <Input
bind:value={searchTerm}
on:keyup={event => {
if (event.key === "Enter") {
searchForIcon()
}
}}
thin
placeholder="Search Icon"
/>
</div> </div>
<Button secondary on:click={searchForIcon}>Search</Button> <Button secondary on:click={searchForIcon}>Search</Button>
{#if value}
<Button primary on:click={() => (value = null)}>Clear</Button>
{/if}
</div> </div>
<div class="page-area"> <div class="page-area">
<div class="pager"> <div class="pager">
@ -239,6 +251,7 @@
flex-flow: row nowrap; flex-flow: row nowrap;
width: 100%; width: 100%;
padding-right: 15px; padding-right: 15px;
gap: 10px;
} }
.input-wrapper { .input-wrapper {
width: 510px; width: 510px;

View File

@ -525,6 +525,38 @@
"barTitle": "Disable button", "barTitle": "Disable button",
"key": "disabled" "key": "disabled"
}, },
{
"type": "icon",
"label": "Icon",
"key": "icon"
},
{
"type": "select",
"label": "Gap",
"key": "gap",
"showInBar": true,
"barStyle": "picker",
"dependsOn": "icon",
"options": [
{
"label": "None",
"value": "N"
},
{
"label": "Small",
"value": "S"
},
{
"label": "Medium",
"value": "M"
},
{
"label": "Large",
"value": "L"
}
],
"defaultValue": "M"
},
{ {
"type": "event", "type": "event",
"label": "On click", "label": "On click",

View File

@ -13,9 +13,10 @@
export let size = "M" export let size = "M"
export let type = "cta" export let type = "cta"
export let quiet = false export let quiet = false
export let icon = null
export let gap = "M"
// For internal use only for now - not defined in the manifest // For internal use only for now - not defined in the manifest
export let icon = null
export let active = false export let active = false
const handleOnClick = async () => { const handleOnClick = async () => {
@ -47,7 +48,7 @@
{#key $component.editing} {#key $component.editing}
<button <button
class={`spectrum-Button spectrum-Button--size${size} spectrum-Button--${type}`} class={`spectrum-Button spectrum-Button--size${size} spectrum-Button--${type} gap-${gap}`}
class:spectrum-Button--quiet={quiet} class:spectrum-Button--quiet={quiet}
disabled={disabled || handlingOnClick} disabled={disabled || handlingOnClick}
use:styleable={$component.styles} use:styleable={$component.styles}
@ -58,15 +59,7 @@
class:active class:active
> >
{#if icon} {#if icon}
<svg <i class="{icon} {size}" />
class:hasText={componentText?.length > 0}
class="spectrum-Icon spectrum-Icon--size{size.toUpperCase()}"
focusable="false"
aria-hidden="true"
aria-label={icon}
>
<use xlink:href="#spectrum-icon-18-{icon}" />
</svg>
{/if} {/if}
{componentText} {componentText}
</button> </button>
@ -92,4 +85,13 @@
.active { .active {
color: var(--spectrum-global-color-blue-600); color: var(--spectrum-global-color-blue-600);
} }
.gap-S {
gap: 8px;
}
.gap-M {
gap: 16px;
}
.gap-L {
gap: 32px;
}
</style> </style>

View File

@ -20,7 +20,7 @@
wrap: true, wrap: true,
}} }}
> >
{#each buttons as { text, type, quiet, disabled, onClick, size }} {#each buttons as { text, type, quiet, disabled, onClick, size, icon, gap }}
<BlockComponent <BlockComponent
type="button" type="button"
props={{ props={{
@ -29,6 +29,8 @@
type, type,
quiet, quiet,
disabled, disabled,
icon,
gap,
size: size || "M", size: size || "M",
}} }}
/> />

View File

@ -92,9 +92,9 @@
{#if schemaLoaded} {#if schemaLoaded}
<Button <Button
onClick={openEditor} onClick={openEditor}
icon="Properties" icon="ri-filter-3-line"
text="Filter" text="Filter"
{size} size="XL"
type="secondary" type="secondary"
quiet quiet
active={filters?.length > 0} active={filters?.length > 0}

@ -1 +1 @@
Subproject commit 22a278da720d92991dabdcd4cb6c96e7abe29781 Subproject commit 2b322d0f4b71ba96664d383b94c30445ead7ac5b

View File

@ -8,6 +8,7 @@ module FetchMock {
let mockSearch = false let mockSearch = false
const func = async (url: any, opts: any) => { const func = async (url: any, opts: any) => {
const { host, pathname } = new URL(url)
function json(body: any, status = 200) { function json(body: any, status = 200) {
return { return {
status, status,
@ -34,7 +35,7 @@ module FetchMock {
} }
} }
if (url.includes("/api/global")) { if (pathname.includes("/api/global")) {
const user = { const user = {
email: "test@example.com", email: "test@example.com",
_id: "us_test@example.com", _id: "us_test@example.com",
@ -47,31 +48,31 @@ module FetchMock {
global: false, global: false,
}, },
} }
return url.endsWith("/users") && opts.method === "GET" return pathname.endsWith("/users") && opts.method === "GET"
? json([user]) ? json([user])
: json(user) : json(user)
} }
// mocked data based on url // mocked data based on url
else if (url.includes("api/apps")) { else if (pathname.includes("api/apps")) {
return json({ return json({
app1: { app1: {
url: "/app1", url: "/app1",
}, },
}) })
} else if (url.includes("example.com")) { } else if (host.includes("example.com")) {
return json({ return json({
body: opts.body, body: opts.body,
url, url,
method: opts.method, method: opts.method,
}) })
} else if (url.includes("invalid.com")) { } else if (host.includes("invalid.com")) {
return json( return json(
{ {
invalid: true, invalid: true,
}, },
404 404
) )
} else if (mockSearch && url.includes("_search")) { } else if (mockSearch && pathname.includes("_search")) {
const body = opts.body const body = opts.body
const parts = body.split("tableId:") const parts = body.split("tableId:")
let tableId let tableId
@ -90,7 +91,7 @@ module FetchMock {
], ],
bookmark: "test", bookmark: "test",
}) })
} else if (url.includes("google.com")) { } else if (host.includes("google.com")) {
return json({ return json({
url, url,
opts, opts,
@ -177,7 +178,7 @@ module FetchMock {
} else if (url === "https://www.googleapis.com/oauth2/v4/token") { } else if (url === "https://www.googleapis.com/oauth2/v4/token") {
// any valid response // any valid response
return json({}) return json({})
} else if (url.includes("failonce.com")) { } else if (host.includes("failonce.com")) {
failCount++ failCount++
if (failCount === 1) { if (failCount === 1) {
return json({ message: "error" }, 500) return json({ message: "error" }, 500)

View File

@ -106,6 +106,21 @@ export async function save(ctx: UserCtx<SaveRoleRequest, SaveRoleResponse>) {
) )
role._rev = result.rev role._rev = result.rev
ctx.body = role ctx.body = role
const devDb = context.getDevAppDB()
const prodDb = context.getProdAppDB()
if (await prodDb.exists()) {
const replication = new dbCore.Replication({
source: devDb.name,
target: prodDb.name,
})
await replication.replicate({
filter: (doc: any, params: any) => {
return doc._id && doc._id.startsWith("role_")
},
})
}
} }
export async function destroy(ctx: UserCtx<void, DestroyRoleResponse>) { export async function destroy(ctx: UserCtx<void, DestroyRoleResponse>) {

View File

@ -1,6 +1,6 @@
import { generateUserFlagID, InternalTables } from "../../db/utils" import { generateUserFlagID, InternalTables } from "../../db/utils"
import { getFullUser } from "../../utilities/users" import { getFullUser } from "../../utilities/users"
import { context } from "@budibase/backend-core" import { cache, context } from "@budibase/backend-core"
import { import {
ContextUserMetadata, ContextUserMetadata,
Ctx, Ctx,

View File

@ -16,8 +16,9 @@ import * as setup from "./utilities"
import { AppStatus } from "../../../db/utils" import { AppStatus } from "../../../db/utils"
import { events, utils, context } from "@budibase/backend-core" import { events, utils, context } from "@budibase/backend-core"
import env from "../../../environment" import env from "../../../environment"
import type { App } from "@budibase/types" import { type App } from "@budibase/types"
import tk from "timekeeper" import tk from "timekeeper"
import * as uuid from "uuid"
describe("/applications", () => { describe("/applications", () => {
let config = setup.getConfig() let config = setup.getConfig()
@ -251,7 +252,7 @@ describe("/applications", () => {
describe("permissions", () => { describe("permissions", () => {
it("should only return apps a user has access to", async () => { it("should only return apps a user has access to", async () => {
const user = await config.createUser({ let user = await config.createUser({
builder: { global: false }, builder: { global: false },
admin: { global: false }, admin: { global: false },
}) })
@ -260,6 +261,81 @@ describe("/applications", () => {
const apps = await config.api.application.fetch() const apps = await config.api.application.fetch()
expect(apps).toHaveLength(0) expect(apps).toHaveLength(0)
}) })
user = await config.globalUser({
...user,
builder: {
apps: [config.getProdAppId()],
},
})
await config.withUser(user, async () => {
const apps = await config.api.application.fetch()
expect(apps).toHaveLength(1)
})
})
it("should only return apps a user has access to through a custom role", async () => {
let user = await config.createUser({
builder: { global: false },
admin: { global: false },
})
await config.withUser(user, async () => {
const apps = await config.api.application.fetch()
expect(apps).toHaveLength(0)
})
const role = await config.api.roles.save({
name: "Test",
inherits: "PUBLIC",
permissionId: "read_only",
version: "name",
})
user = await config.globalUser({
...user,
roles: {
[config.getProdAppId()]: role.name,
},
})
await config.withUser(user, async () => {
const apps = await config.api.application.fetch()
expect(apps).toHaveLength(1)
})
})
it.only("should only return apps a user has access to through a custom role on a group", async () => {
let user = await config.createUser({
builder: { global: false },
admin: { global: false },
})
await config.withUser(user, async () => {
const apps = await config.api.application.fetch()
expect(apps).toHaveLength(0)
})
const roleName = uuid.v4().replace(/-/g, "")
const role = await config.api.roles.save({
name: roleName,
inherits: "PUBLIC",
permissionId: "read_only",
version: "name",
})
const group = await config.createGroup(role._id!)
user = await config.globalUser({
...user,
userGroups: [group._id!],
})
await config.withUser(user, async () => {
const apps = await config.api.application.fetch()
expect(apps).toHaveLength(1)
})
}) })
}) })
}) })

View File

@ -10,3 +10,4 @@ process.env.MOCK_REDIS = "1"
process.env.PLATFORM_URL = "http://localhost:10000" process.env.PLATFORM_URL = "http://localhost:10000"
process.env.REDIS_PASSWORD = "budibase" process.env.REDIS_PASSWORD = "budibase"
process.env.BUDIBASE_VERSION = "0.0.0+jest" process.env.BUDIBASE_VERSION = "0.0.0+jest"
process.env.WORKER_URL = "http://localhost:10000"

View File

@ -299,11 +299,11 @@ export default class TestConfiguration {
} }
} }
withUser(user: User, f: () => Promise<void>) { async withUser(user: User, f: () => Promise<void>) {
const oldUser = this.user const oldUser = this.user
this.user = user this.user = user
try { try {
return f() return await f()
} finally { } finally {
this.user = oldUser this.user = oldUser
} }
@ -347,7 +347,7 @@ export default class TestConfiguration {
lastName = generator.last(), lastName = generator.last(),
builder = { global: true }, builder = { global: true },
admin = { global: false }, admin = { global: false },
email = generator.email(), email = generator.email({ domain: "example.com" }),
tenantId = this.getTenantId(), tenantId = this.getTenantId(),
roles = {}, roles = {},
} = config } = config
@ -363,6 +363,7 @@ export default class TestConfiguration {
_id, _id,
...existing, ...existing,
...config, ...config,
_rev: existing._rev,
email, email,
roles, roles,
tenantId, tenantId,
@ -372,11 +373,12 @@ export default class TestConfiguration {
admin, admin,
} }
await sessions.createASession(_id, { await sessions.createASession(_id, {
sessionId: "sessionid", sessionId: this.sessionIdForUser(_id),
tenantId: this.getTenantId(), tenantId: this.getTenantId(),
csrfToken: this.csrfToken, csrfToken: this.csrfToken,
}) })
const resp = await db.put(user) const resp = await db.put(user)
await cache.user.invalidateUser(_id)
return { return {
_rev: resp.rev, _rev: resp.rev,
...user, ...user,
@ -384,9 +386,7 @@ export default class TestConfiguration {
} }
async createUser(user: Partial<User> = {}): Promise<User> { async createUser(user: Partial<User> = {}): Promise<User> {
const resp = await this.globalUser(user) return await this.globalUser(user)
await cache.user.invalidateUser(resp._id!)
return resp
} }
async createGroup(roleId: string = roles.BUILTIN_ROLE_IDS.BASIC) { async createGroup(roleId: string = roles.BUILTIN_ROLE_IDS.BASIC) {
@ -416,6 +416,10 @@ export default class TestConfiguration {
}) })
} }
sessionIdForUser(userId: string): string {
return `sessionid-${userId}`
}
async login({ async login({
roleId, roleId,
userId, userId,
@ -442,13 +446,13 @@ export default class TestConfiguration {
}) })
} }
await sessions.createASession(userId, { await sessions.createASession(userId, {
sessionId: "sessionid", sessionId: this.sessionIdForUser(userId),
tenantId: this.getTenantId(), tenantId: this.getTenantId(),
}) })
// have to fake this // have to fake this
const authObj = { const authObj = {
userId, userId,
sessionId: "sessionid", sessionId: this.sessionIdForUser(userId),
tenantId: this.getTenantId(), tenantId: this.getTenantId(),
} }
const authToken = jwt.sign(authObj, coreEnv.JWT_SECRET as Secret) const authToken = jwt.sign(authObj, coreEnv.JWT_SECRET as Secret)
@ -470,7 +474,7 @@ export default class TestConfiguration {
const user = this.getUser() const user = this.getUser()
const authObj: AuthToken = { const authObj: AuthToken = {
userId: user._id!, userId: user._id!,
sessionId: "sessionid", sessionId: this.sessionIdForUser(user._id!),
tenantId, tenantId,
} }
const authToken = jwt.sign(authObj, coreEnv.JWT_SECRET as Secret) const authToken = jwt.sign(authObj, coreEnv.JWT_SECRET as Secret)
@ -508,7 +512,7 @@ export default class TestConfiguration {
async basicRoleHeaders() { async basicRoleHeaders() {
return await this.roleHeaders({ return await this.roleHeaders({
email: generator.email(), email: generator.email({ domain: "example.com" }),
builder: false, builder: false,
prodApp: true, prodApp: true,
roleId: roles.BUILTIN_ROLE_IDS.BASIC, roleId: roles.BUILTIN_ROLE_IDS.BASIC,
@ -516,7 +520,7 @@ export default class TestConfiguration {
} }
async roleHeaders({ async roleHeaders({
email = generator.email(), email = generator.email({ domain: "example.com" }),
roleId = roles.BUILTIN_ROLE_IDS.ADMIN, roleId = roles.BUILTIN_ROLE_IDS.ADMIN,
builder = false, builder = false,
prodApp = true, prodApp = true,

View File

@ -11,6 +11,7 @@ import { BackupAPI } from "./backup"
import { AttachmentAPI } from "./attachment" import { AttachmentAPI } from "./attachment"
import { UserAPI } from "./user" import { UserAPI } from "./user"
import { QueryAPI } from "./query" import { QueryAPI } from "./query"
import { RoleAPI } from "./role"
export default class API { export default class API {
table: TableAPI table: TableAPI
@ -25,6 +26,7 @@ export default class API {
attachment: AttachmentAPI attachment: AttachmentAPI
user: UserAPI user: UserAPI
query: QueryAPI query: QueryAPI
roles: RoleAPI
constructor(config: TestConfiguration) { constructor(config: TestConfiguration) {
this.table = new TableAPI(config) this.table = new TableAPI(config)
@ -39,5 +41,6 @@ export default class API {
this.attachment = new AttachmentAPI(config) this.attachment = new AttachmentAPI(config)
this.user = new UserAPI(config) this.user = new UserAPI(config)
this.query = new QueryAPI(config) this.query = new QueryAPI(config)
this.roles = new RoleAPI(config)
} }
} }

View File

@ -0,0 +1,41 @@
import {
AccessibleRolesResponse,
FetchRolesResponse,
FindRoleResponse,
SaveRoleRequest,
SaveRoleResponse,
} from "@budibase/types"
import { Expectations, TestAPI } from "./base"
export class RoleAPI extends TestAPI {
fetch = async (expectations?: Expectations) => {
return await this._get<FetchRolesResponse>(`/api/roles`, {
expectations,
})
}
find = async (roleId: string, expectations?: Expectations) => {
return await this._get<FindRoleResponse>(`/api/roles/${roleId}`, {
expectations,
})
}
save = async (body: SaveRoleRequest, expectations?: Expectations) => {
return await this._post<SaveRoleResponse>(`/api/roles`, {
body,
expectations,
})
}
destroy = async (roleId: string, expectations?: Expectations) => {
return await this._delete(`/api/roles/${roleId}`, {
expectations,
})
}
accesssible = async (expectations?: Expectations) => {
return await this._get<AccessibleRolesResponse>(`/api/roles/accessible`, {
expectations,
})
}
}

View File

@ -5,4 +5,5 @@ export interface Role extends Document {
inherits?: string inherits?: string
permissions: { [key: string]: string[] } permissions: { [key: string]: string[] }
version?: string version?: string
name: string
} }

View File

@ -147,7 +147,7 @@ describe("/api/global/groups", () => {
await Promise.all( await Promise.all(
Array.from({ length: 30 }).map(async (_, i) => { Array.from({ length: 30 }).map(async (_, i) => {
const email = `user${i}@${generator.domain()}` const email = `user${i}@example.com`
const user = await config.api.users.saveUser({ const user = await config.api.users.saveUser({
...structures.users.user(), ...structures.users.user(),
email, email,

View File

@ -84,7 +84,7 @@ describe("Accounts", () => {
}) })
it("searches by email", async () => { it("searches by email", async () => {
const email = generator.email() const email = generator.email({ domain: "example.com" })
// Empty result // Empty result
const [_, emptyBody] = await config.api.accounts.search(email, "email") const [_, emptyBody] = await config.api.accounts.search(email, "email")

View File

@ -4,7 +4,7 @@ import { generator } from "../../shared"
export const generateUser = ( export const generateUser = (
overrides: Partial<User> = {} overrides: Partial<User> = {}
): CreateUserParams => ({ ): CreateUserParams => ({
email: generator.email(), email: generator.email({ domain: "example.com" }),
roles: { roles: {
[generator.string({ length: 32, alpha: true, numeric: true })]: [generator.string({ length: 32, alpha: true, numeric: true })]:
generator.word(), generator.word(),