mirror of
https://github.com/Infisical/infisical.git
synced 2025-07-15 09:42:14 +00:00
Compare commits
87 Commits
users-proj
...
create-sec
Author | SHA1 | Date | |
---|---|---|---|
d659250ce8 | |||
87363eabfe | |||
d1b9c316d8 | |||
39f7354fec | |||
c46c0cb1e8 | |||
6905ffba4e | |||
64fd423c61 | |||
da1a7466d1 | |||
d3f3f34129 | |||
c8fba7ce4c | |||
82c3e943eb | |||
dc3903ff15 | |||
a9c01dcf1f | |||
586b9d9a56 | |||
6d709fba62 | |||
27beca7099 | |||
28e7e4c52d | |||
cfc0ca1f03 | |||
b96593d0ab | |||
2de5896ba4 | |||
3455ad3898 | |||
c7a32a3b05 | |||
1ebfed8c11 | |||
16d215b588 | |||
cacd9041b0 | |||
cfeffebd46 | |||
1dceedcdb4 | |||
14f03c38c3 | |||
be9f096e75 | |||
49133a044f | |||
b7fe3743db | |||
c5fded361c | |||
e676acbadf | |||
9b31a7bbb1 | |||
345be85825 | |||
f82b11851a | |||
b466b3073b | |||
46105fc315 | |||
3cf8fd2ff8 | |||
5277a50b3e | |||
dab8f0b261 | |||
4293665130 | |||
8afa65c272 | |||
4c739fd57f | |||
bcc2840020 | |||
8b3af92d23 | |||
9ca58894f0 | |||
d131314de0 | |||
9c03144f19 | |||
5495ffd78e | |||
a200469c72 | |||
85c3074216 | |||
cfc55ff283 | |||
7179b7a540 | |||
9cfb044178 | |||
105eb70fd9 | |||
9df9f4a5da | |||
afdc704423 | |||
57261cf0c8 | |||
06f6004993 | |||
f3bfb9cc5a | |||
48fb77be49 | |||
f55bcb93ba | |||
d3fb2a6a74 | |||
6a23b74481 | |||
602cf4b3c4 | |||
84ff71fef2 | |||
add5742b8c | |||
68f3964206 | |||
90374971ae | |||
3a1eadba8c | |||
5305017ce2 | |||
cf5f49d14e | |||
4f4b5be8ea | |||
3b47d7698b | |||
aa9a86df71 | |||
ca55f19926 | |||
3794521c56 | |||
2c402fbbb6 | |||
bbf52c9a48 | |||
3d6ea3251e | |||
be39e63832 | |||
464a3ccd53 | |||
63fac39fff | |||
7c62a776fb | |||
ed7fc0e5cd | |||
1ae6213387 |
86
backend/e2e-test/routes/v3/secret-recursive.spec.ts
Normal file
86
backend/e2e-test/routes/v3/secret-recursive.spec.ts
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
import { createFolder, deleteFolder } from "e2e-test/testUtils/folders";
|
||||||
|
import { createSecretV2, deleteSecretV2, getSecretsV2 } from "e2e-test/testUtils/secrets";
|
||||||
|
|
||||||
|
import { seedData1 } from "@app/db/seed-data";
|
||||||
|
|
||||||
|
describe("Secret Recursive Testing", async () => {
|
||||||
|
const projectId = seedData1.projectV3.id;
|
||||||
|
const folderAndSecretNames = [
|
||||||
|
{ name: "deep1", path: "/", expectedSecretCount: 4 },
|
||||||
|
{ name: "deep21", path: "/deep1", expectedSecretCount: 2 },
|
||||||
|
{ name: "deep3", path: "/deep1/deep2", expectedSecretCount: 1 },
|
||||||
|
{ name: "deep22", path: "/deep2", expectedSecretCount: 1 }
|
||||||
|
];
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
const rootFolderIds: string[] = [];
|
||||||
|
for (const folder of folderAndSecretNames) {
|
||||||
|
// eslint-disable-next-line no-await-in-loop
|
||||||
|
const createdFolder = await createFolder({
|
||||||
|
authToken: jwtAuthToken,
|
||||||
|
environmentSlug: "prod",
|
||||||
|
workspaceId: projectId,
|
||||||
|
secretPath: folder.path,
|
||||||
|
name: folder.name
|
||||||
|
});
|
||||||
|
|
||||||
|
if (folder.path === "/") {
|
||||||
|
rootFolderIds.push(createdFolder.id);
|
||||||
|
}
|
||||||
|
// eslint-disable-next-line no-await-in-loop
|
||||||
|
await createSecretV2({
|
||||||
|
secretPath: folder.path,
|
||||||
|
authToken: jwtAuthToken,
|
||||||
|
environmentSlug: "prod",
|
||||||
|
workspaceId: projectId,
|
||||||
|
key: folder.name,
|
||||||
|
value: folder.name
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return async () => {
|
||||||
|
await Promise.all(
|
||||||
|
rootFolderIds.map((id) =>
|
||||||
|
deleteFolder({
|
||||||
|
authToken: jwtAuthToken,
|
||||||
|
secretPath: "/",
|
||||||
|
id,
|
||||||
|
workspaceId: projectId,
|
||||||
|
environmentSlug: "prod"
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
await deleteSecretV2({
|
||||||
|
authToken: jwtAuthToken,
|
||||||
|
secretPath: "/",
|
||||||
|
workspaceId: projectId,
|
||||||
|
environmentSlug: "prod",
|
||||||
|
key: folderAndSecretNames[0].name
|
||||||
|
});
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
test.each(folderAndSecretNames)("$path recursive secret fetching", async ({ path, expectedSecretCount }) => {
|
||||||
|
const secrets = await getSecretsV2({
|
||||||
|
authToken: jwtAuthToken,
|
||||||
|
secretPath: path,
|
||||||
|
workspaceId: projectId,
|
||||||
|
environmentSlug: "prod",
|
||||||
|
recursive: true
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(secrets.secrets.length).toEqual(expectedSecretCount);
|
||||||
|
expect(secrets.secrets.sort((a, b) => a.secretKey.localeCompare(b.secretKey))).toEqual(
|
||||||
|
folderAndSecretNames
|
||||||
|
.filter((el) => el.path.startsWith(path))
|
||||||
|
.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
.map((el) =>
|
||||||
|
expect.objectContaining({
|
||||||
|
secretKey: el.name,
|
||||||
|
secretValue: el.name
|
||||||
|
})
|
||||||
|
)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
@ -97,6 +97,7 @@ export const getSecretsV2 = async (dto: {
|
|||||||
environmentSlug: string;
|
environmentSlug: string;
|
||||||
secretPath: string;
|
secretPath: string;
|
||||||
authToken: string;
|
authToken: string;
|
||||||
|
recursive?: boolean;
|
||||||
}) => {
|
}) => {
|
||||||
const getSecretsResponse = await testServer.inject({
|
const getSecretsResponse = await testServer.inject({
|
||||||
method: "GET",
|
method: "GET",
|
||||||
@ -109,7 +110,8 @@ export const getSecretsV2 = async (dto: {
|
|||||||
environment: dto.environmentSlug,
|
environment: dto.environmentSlug,
|
||||||
secretPath: dto.secretPath,
|
secretPath: dto.secretPath,
|
||||||
expandSecretReferences: "true",
|
expandSecretReferences: "true",
|
||||||
include_imports: "true"
|
include_imports: "true",
|
||||||
|
recursive: String(dto.recursive || false)
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
expect(getSecretsResponse.statusCode).toBe(200);
|
expect(getSecretsResponse.statusCode).toBe(200);
|
||||||
|
@ -112,7 +112,7 @@ export const dynamicSecretLeaseServiceFactory = ({
|
|||||||
})
|
})
|
||||||
) as object;
|
) as object;
|
||||||
|
|
||||||
const selectedTTL = ttl ?? dynamicSecretCfg.defaultTTL;
|
const selectedTTL = ttl || dynamicSecretCfg.defaultTTL;
|
||||||
const { maxTTL } = dynamicSecretCfg;
|
const { maxTTL } = dynamicSecretCfg;
|
||||||
const expireAt = new Date(new Date().getTime() + ms(selectedTTL));
|
const expireAt = new Date(new Date().getTime() + ms(selectedTTL));
|
||||||
if (maxTTL) {
|
if (maxTTL) {
|
||||||
@ -187,7 +187,7 @@ export const dynamicSecretLeaseServiceFactory = ({
|
|||||||
})
|
})
|
||||||
) as object;
|
) as object;
|
||||||
|
|
||||||
const selectedTTL = ttl ?? dynamicSecretCfg.defaultTTL;
|
const selectedTTL = ttl || dynamicSecretCfg.defaultTTL;
|
||||||
const { maxTTL } = dynamicSecretCfg;
|
const { maxTTL } = dynamicSecretCfg;
|
||||||
const expireAt = new Date(dynamicSecretLease.expireAt.getTime() + ms(selectedTTL));
|
const expireAt = new Date(dynamicSecretLease.expireAt.getTime() + ms(selectedTTL));
|
||||||
if (maxTTL) {
|
if (maxTTL) {
|
||||||
|
@ -31,7 +31,6 @@ export enum OrgPermissionSubjects {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type OrgPermissionSet =
|
export type OrgPermissionSet =
|
||||||
| [OrgPermissionActions.Read, OrgPermissionSubjects.Workspace]
|
|
||||||
| [OrgPermissionActions.Create, OrgPermissionSubjects.Workspace]
|
| [OrgPermissionActions.Create, OrgPermissionSubjects.Workspace]
|
||||||
| [OrgPermissionActions, OrgPermissionSubjects.Role]
|
| [OrgPermissionActions, OrgPermissionSubjects.Role]
|
||||||
| [OrgPermissionActions, OrgPermissionSubjects.Member]
|
| [OrgPermissionActions, OrgPermissionSubjects.Member]
|
||||||
@ -52,7 +51,6 @@ export type OrgPermissionSet =
|
|||||||
const buildAdminPermission = () => {
|
const buildAdminPermission = () => {
|
||||||
const { can, rules } = new AbilityBuilder<MongoAbility<OrgPermissionSet>>(createMongoAbility);
|
const { can, rules } = new AbilityBuilder<MongoAbility<OrgPermissionSet>>(createMongoAbility);
|
||||||
// ws permissions
|
// ws permissions
|
||||||
can(OrgPermissionActions.Read, OrgPermissionSubjects.Workspace);
|
|
||||||
can(OrgPermissionActions.Create, OrgPermissionSubjects.Workspace);
|
can(OrgPermissionActions.Create, OrgPermissionSubjects.Workspace);
|
||||||
// role permission
|
// role permission
|
||||||
can(OrgPermissionActions.Read, OrgPermissionSubjects.Role);
|
can(OrgPermissionActions.Read, OrgPermissionSubjects.Role);
|
||||||
@ -135,7 +133,6 @@ export const orgAdminPermissions = buildAdminPermission();
|
|||||||
const buildMemberPermission = () => {
|
const buildMemberPermission = () => {
|
||||||
const { can, rules } = new AbilityBuilder<MongoAbility<OrgPermissionSet>>(createMongoAbility);
|
const { can, rules } = new AbilityBuilder<MongoAbility<OrgPermissionSet>>(createMongoAbility);
|
||||||
|
|
||||||
can(OrgPermissionActions.Read, OrgPermissionSubjects.Workspace);
|
|
||||||
can(OrgPermissionActions.Create, OrgPermissionSubjects.Workspace);
|
can(OrgPermissionActions.Create, OrgPermissionSubjects.Workspace);
|
||||||
can(OrgPermissionActions.Read, OrgPermissionSubjects.Member);
|
can(OrgPermissionActions.Read, OrgPermissionSubjects.Member);
|
||||||
can(OrgPermissionActions.Read, OrgPermissionSubjects.Groups);
|
can(OrgPermissionActions.Read, OrgPermissionSubjects.Groups);
|
||||||
|
@ -18,6 +18,7 @@ import { TGroupProjectDALFactory } from "@app/services/group-project/group-proje
|
|||||||
import { TOrgDALFactory } from "@app/services/org/org-dal";
|
import { TOrgDALFactory } from "@app/services/org/org-dal";
|
||||||
import { deleteOrgMembershipFn } from "@app/services/org/org-fns";
|
import { deleteOrgMembershipFn } from "@app/services/org/org-fns";
|
||||||
import { getDefaultOrgMembershipRole } from "@app/services/org/org-role-fns";
|
import { getDefaultOrgMembershipRole } from "@app/services/org/org-role-fns";
|
||||||
|
import { OrgAuthMethod } from "@app/services/org/org-types";
|
||||||
import { TOrgMembershipDALFactory } from "@app/services/org-membership/org-membership-dal";
|
import { TOrgMembershipDALFactory } from "@app/services/org-membership/org-membership-dal";
|
||||||
import { TProjectDALFactory } from "@app/services/project/project-dal";
|
import { TProjectDALFactory } from "@app/services/project/project-dal";
|
||||||
import { TProjectBotDALFactory } from "@app/services/project-bot/project-bot-dal";
|
import { TProjectBotDALFactory } from "@app/services/project-bot/project-bot-dal";
|
||||||
@ -71,6 +72,7 @@ type TScimServiceFactoryDep = {
|
|||||||
| "deleteMembershipById"
|
| "deleteMembershipById"
|
||||||
| "transaction"
|
| "transaction"
|
||||||
| "updateMembershipById"
|
| "updateMembershipById"
|
||||||
|
| "findOrgById"
|
||||||
>;
|
>;
|
||||||
orgMembershipDAL: Pick<
|
orgMembershipDAL: Pick<
|
||||||
TOrgMembershipDALFactory,
|
TOrgMembershipDALFactory,
|
||||||
@ -288,8 +290,7 @@ export const scimServiceFactory = ({
|
|||||||
const createScimUser = async ({ externalId, email, firstName, lastName, orgId }: TCreateScimUserDTO) => {
|
const createScimUser = async ({ externalId, email, firstName, lastName, orgId }: TCreateScimUserDTO) => {
|
||||||
if (!email) throw new ScimRequestError({ detail: "Invalid request. Missing email.", status: 400 });
|
if (!email) throw new ScimRequestError({ detail: "Invalid request. Missing email.", status: 400 });
|
||||||
|
|
||||||
const org = await orgDAL.findById(orgId);
|
const org = await orgDAL.findOrgById(orgId);
|
||||||
|
|
||||||
if (!org)
|
if (!org)
|
||||||
throw new ScimRequestError({
|
throw new ScimRequestError({
|
||||||
detail: "Organization not found",
|
detail: "Organization not found",
|
||||||
@ -302,13 +303,24 @@ export const scimServiceFactory = ({
|
|||||||
status: 403
|
status: 403
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (!org.orgAuthMethod) {
|
||||||
|
throw new ScimRequestError({
|
||||||
|
detail: "Neither SAML or OIDC SSO is configured",
|
||||||
|
status: 400
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const appCfg = getConfig();
|
const appCfg = getConfig();
|
||||||
const serverCfg = await getServerCfg();
|
const serverCfg = await getServerCfg();
|
||||||
|
|
||||||
|
const aliasType = org.orgAuthMethod === OrgAuthMethod.OIDC ? UserAliasType.OIDC : UserAliasType.SAML;
|
||||||
|
const trustScimEmails =
|
||||||
|
org.orgAuthMethod === OrgAuthMethod.OIDC ? serverCfg.trustOidcEmails : serverCfg.trustSamlEmails;
|
||||||
|
|
||||||
const userAlias = await userAliasDAL.findOne({
|
const userAlias = await userAliasDAL.findOne({
|
||||||
externalId,
|
externalId,
|
||||||
orgId,
|
orgId,
|
||||||
aliasType: UserAliasType.SAML
|
aliasType
|
||||||
});
|
});
|
||||||
|
|
||||||
const { user: createdUser, orgMembership: createdOrgMembership } = await userDAL.transaction(async (tx) => {
|
const { user: createdUser, orgMembership: createdOrgMembership } = await userDAL.transaction(async (tx) => {
|
||||||
@ -349,7 +361,7 @@ export const scimServiceFactory = ({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (serverCfg.trustSamlEmails) {
|
if (trustScimEmails) {
|
||||||
user = await userDAL.findOne(
|
user = await userDAL.findOne(
|
||||||
{
|
{
|
||||||
email,
|
email,
|
||||||
@ -367,9 +379,9 @@ export const scimServiceFactory = ({
|
|||||||
);
|
);
|
||||||
user = await userDAL.create(
|
user = await userDAL.create(
|
||||||
{
|
{
|
||||||
username: serverCfg.trustSamlEmails ? email : uniqueUsername,
|
username: trustScimEmails ? email : uniqueUsername,
|
||||||
email,
|
email,
|
||||||
isEmailVerified: serverCfg.trustSamlEmails,
|
isEmailVerified: trustScimEmails,
|
||||||
firstName,
|
firstName,
|
||||||
lastName,
|
lastName,
|
||||||
authMethods: [],
|
authMethods: [],
|
||||||
@ -382,7 +394,7 @@ export const scimServiceFactory = ({
|
|||||||
await userAliasDAL.create(
|
await userAliasDAL.create(
|
||||||
{
|
{
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
aliasType: UserAliasType.SAML,
|
aliasType,
|
||||||
externalId,
|
externalId,
|
||||||
emails: email ? [email] : [],
|
emails: email ? [email] : [],
|
||||||
orgId
|
orgId
|
||||||
@ -437,7 +449,7 @@ export const scimServiceFactory = ({
|
|||||||
recipients: [email],
|
recipients: [email],
|
||||||
substitutions: {
|
substitutions: {
|
||||||
organizationName: org.name,
|
organizationName: org.name,
|
||||||
callback_url: `${appCfg.SITE_URL}/api/v1/sso/redirect/saml2/organizations/${org.slug}`
|
callback_url: `${appCfg.SITE_URL}/api/v1/sso/redirect/organizations/${org.slug}`
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -456,6 +468,14 @@ export const scimServiceFactory = ({
|
|||||||
|
|
||||||
// partial
|
// partial
|
||||||
const updateScimUser = async ({ orgMembershipId, orgId, operations }: TUpdateScimUserDTO) => {
|
const updateScimUser = async ({ orgMembershipId, orgId, operations }: TUpdateScimUserDTO) => {
|
||||||
|
const org = await orgDAL.findOrgById(orgId);
|
||||||
|
if (!org.orgAuthMethod) {
|
||||||
|
throw new ScimRequestError({
|
||||||
|
detail: "Neither SAML or OIDC SSO is configured",
|
||||||
|
status: 400
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const [membership] = await orgDAL
|
const [membership] = await orgDAL
|
||||||
.findMembership({
|
.findMembership({
|
||||||
[`${TableName.OrgMembership}.id` as "id"]: orgMembershipId,
|
[`${TableName.OrgMembership}.id` as "id"]: orgMembershipId,
|
||||||
@ -493,6 +513,9 @@ export const scimServiceFactory = ({
|
|||||||
scimPatch(scimUser, operations);
|
scimPatch(scimUser, operations);
|
||||||
|
|
||||||
const serverCfg = await getServerCfg();
|
const serverCfg = await getServerCfg();
|
||||||
|
const trustScimEmails =
|
||||||
|
org.orgAuthMethod === OrgAuthMethod.OIDC ? serverCfg.trustOidcEmails : serverCfg.trustSamlEmails;
|
||||||
|
|
||||||
await userDAL.transaction(async (tx) => {
|
await userDAL.transaction(async (tx) => {
|
||||||
await orgMembershipDAL.updateById(
|
await orgMembershipDAL.updateById(
|
||||||
membership.id,
|
membership.id,
|
||||||
@ -508,7 +531,7 @@ export const scimServiceFactory = ({
|
|||||||
firstName: scimUser.name.givenName,
|
firstName: scimUser.name.givenName,
|
||||||
email: scimUser.emails[0].value,
|
email: scimUser.emails[0].value,
|
||||||
lastName: scimUser.name.familyName,
|
lastName: scimUser.name.familyName,
|
||||||
isEmailVerified: hasEmailChanged ? serverCfg.trustSamlEmails : true
|
isEmailVerified: hasEmailChanged ? trustScimEmails : true
|
||||||
},
|
},
|
||||||
tx
|
tx
|
||||||
);
|
);
|
||||||
@ -526,6 +549,14 @@ export const scimServiceFactory = ({
|
|||||||
email,
|
email,
|
||||||
externalId
|
externalId
|
||||||
}: TReplaceScimUserDTO) => {
|
}: TReplaceScimUserDTO) => {
|
||||||
|
const org = await orgDAL.findOrgById(orgId);
|
||||||
|
if (!org.orgAuthMethod) {
|
||||||
|
throw new ScimRequestError({
|
||||||
|
detail: "Neither SAML or OIDC SSO is configured",
|
||||||
|
status: 400
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const [membership] = await orgDAL
|
const [membership] = await orgDAL
|
||||||
.findMembership({
|
.findMembership({
|
||||||
[`${TableName.OrgMembership}.id` as "id"]: orgMembershipId,
|
[`${TableName.OrgMembership}.id` as "id"]: orgMembershipId,
|
||||||
@ -555,7 +586,7 @@ export const scimServiceFactory = ({
|
|||||||
await userAliasDAL.update(
|
await userAliasDAL.update(
|
||||||
{
|
{
|
||||||
orgId,
|
orgId,
|
||||||
aliasType: UserAliasType.SAML,
|
aliasType: org.orgAuthMethod === OrgAuthMethod.OIDC ? UserAliasType.OIDC : UserAliasType.SAML,
|
||||||
userId: membership.userId
|
userId: membership.userId
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -576,7 +607,8 @@ export const scimServiceFactory = ({
|
|||||||
firstName,
|
firstName,
|
||||||
email,
|
email,
|
||||||
lastName,
|
lastName,
|
||||||
isEmailVerified: serverCfg.trustSamlEmails
|
isEmailVerified:
|
||||||
|
org.orgAuthMethod === OrgAuthMethod.OIDC ? serverCfg.trustOidcEmails : serverCfg.trustSamlEmails
|
||||||
},
|
},
|
||||||
tx
|
tx
|
||||||
);
|
);
|
||||||
|
@ -14,10 +14,12 @@ import { Strategy as GoogleStrategy } from "passport-google-oauth20";
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { getConfig } from "@app/lib/config/env";
|
import { getConfig } from "@app/lib/config/env";
|
||||||
import { NotFoundError } from "@app/lib/errors";
|
import { BadRequestError, NotFoundError } from "@app/lib/errors";
|
||||||
import { logger } from "@app/lib/logger";
|
import { logger } from "@app/lib/logger";
|
||||||
import { fetchGithubEmails } from "@app/lib/requests/github";
|
import { fetchGithubEmails } from "@app/lib/requests/github";
|
||||||
|
import { authRateLimit } from "@app/server/config/rateLimiter";
|
||||||
import { AuthMethod } from "@app/services/auth/auth-type";
|
import { AuthMethod } from "@app/services/auth/auth-type";
|
||||||
|
import { OrgAuthMethod } from "@app/services/org/org-types";
|
||||||
|
|
||||||
export const registerSsoRouter = async (server: FastifyZodProvider) => {
|
export const registerSsoRouter = async (server: FastifyZodProvider) => {
|
||||||
const appCfg = getConfig();
|
const appCfg = getConfig();
|
||||||
@ -196,6 +198,44 @@ export const registerSsoRouter = async (server: FastifyZodProvider) => {
|
|||||||
handler: () => {}
|
handler: () => {}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
server.route({
|
||||||
|
url: "/redirect/organizations/:orgSlug",
|
||||||
|
method: "GET",
|
||||||
|
config: {
|
||||||
|
rateLimit: authRateLimit
|
||||||
|
},
|
||||||
|
schema: {
|
||||||
|
params: z.object({
|
||||||
|
orgSlug: z.string().trim()
|
||||||
|
}),
|
||||||
|
querystring: z.object({
|
||||||
|
callback_port: z.string().optional()
|
||||||
|
})
|
||||||
|
},
|
||||||
|
handler: async (req, res) => {
|
||||||
|
const org = await server.services.org.findOrgBySlug(req.params.orgSlug);
|
||||||
|
if (org.orgAuthMethod === OrgAuthMethod.SAML) {
|
||||||
|
return res.redirect(
|
||||||
|
`${appCfg.SITE_URL}/api/v1/sso/redirect/saml2/organizations/${org.slug}?${
|
||||||
|
req.query.callback_port ? `callback_port=${req.query.callback_port}` : ""
|
||||||
|
}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (org.orgAuthMethod === OrgAuthMethod.OIDC) {
|
||||||
|
return res.redirect(
|
||||||
|
`${appCfg.SITE_URL}/api/v1/sso/oidc/login?orgSlug=${org.slug}${
|
||||||
|
req.query.callback_port ? `&callbackPort=${req.query.callback_port}` : ""
|
||||||
|
}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new BadRequestError({
|
||||||
|
message: "The organization does not have any SSO configured."
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
server.route({
|
server.route({
|
||||||
url: "/github",
|
url: "/github",
|
||||||
method: "GET",
|
method: "GET",
|
||||||
|
@ -120,7 +120,8 @@ export const identityKubernetesAuthServiceFactory = ({
|
|||||||
apiVersion: "authentication.k8s.io/v1",
|
apiVersion: "authentication.k8s.io/v1",
|
||||||
kind: "TokenReview",
|
kind: "TokenReview",
|
||||||
spec: {
|
spec: {
|
||||||
token: serviceAccountJwt
|
token: serviceAccountJwt,
|
||||||
|
...(identityKubernetesAuth.allowedAudience ? { audiences: [identityKubernetesAuth.allowedAudience] } : {})
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -14,6 +14,8 @@ import { DatabaseError } from "@app/lib/errors";
|
|||||||
import { buildFindFilter, ormify, selectAllTableCols, TFindFilter, TFindOpt, withTransaction } from "@app/lib/knex";
|
import { buildFindFilter, ormify, selectAllTableCols, TFindFilter, TFindOpt, withTransaction } from "@app/lib/knex";
|
||||||
import { generateKnexQueryFromScim } from "@app/lib/knex/scim";
|
import { generateKnexQueryFromScim } from "@app/lib/knex/scim";
|
||||||
|
|
||||||
|
import { OrgAuthMethod } from "./org-types";
|
||||||
|
|
||||||
export type TOrgDALFactory = ReturnType<typeof orgDALFactory>;
|
export type TOrgDALFactory = ReturnType<typeof orgDALFactory>;
|
||||||
|
|
||||||
export const orgDALFactory = (db: TDbClient) => {
|
export const orgDALFactory = (db: TDbClient) => {
|
||||||
@ -21,13 +23,78 @@ export const orgDALFactory = (db: TDbClient) => {
|
|||||||
|
|
||||||
const findOrgById = async (orgId: string) => {
|
const findOrgById = async (orgId: string) => {
|
||||||
try {
|
try {
|
||||||
const org = await db.replicaNode()(TableName.Organization).where({ id: orgId }).first();
|
const org = (await db
|
||||||
|
.replicaNode()(TableName.Organization)
|
||||||
|
.where({ [`${TableName.Organization}.id` as "id"]: orgId })
|
||||||
|
.leftJoin(TableName.SamlConfig, (qb) => {
|
||||||
|
qb.on(`${TableName.SamlConfig}.orgId`, "=", `${TableName.Organization}.id`).andOn(
|
||||||
|
`${TableName.SamlConfig}.isActive`,
|
||||||
|
"=",
|
||||||
|
db.raw("true")
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.leftJoin(TableName.OidcConfig, (qb) => {
|
||||||
|
qb.on(`${TableName.OidcConfig}.orgId`, "=", `${TableName.Organization}.id`).andOn(
|
||||||
|
`${TableName.OidcConfig}.isActive`,
|
||||||
|
"=",
|
||||||
|
db.raw("true")
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.select(selectAllTableCols(TableName.Organization))
|
||||||
|
.select(
|
||||||
|
db.raw(`
|
||||||
|
CASE
|
||||||
|
WHEN ${TableName.SamlConfig}."orgId" IS NOT NULL THEN '${OrgAuthMethod.SAML}'
|
||||||
|
WHEN ${TableName.OidcConfig}."orgId" IS NOT NULL THEN '${OrgAuthMethod.OIDC}'
|
||||||
|
ELSE ''
|
||||||
|
END as "orgAuthMethod"
|
||||||
|
`)
|
||||||
|
)
|
||||||
|
.first()) as TOrganizations & { orgAuthMethod?: string };
|
||||||
|
|
||||||
return org;
|
return org;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new DatabaseError({ error, name: "Find org by id" });
|
throw new DatabaseError({ error, name: "Find org by id" });
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const findOrgBySlug = async (orgSlug: string) => {
|
||||||
|
try {
|
||||||
|
const org = (await db
|
||||||
|
.replicaNode()(TableName.Organization)
|
||||||
|
.where({ [`${TableName.Organization}.slug` as "slug"]: orgSlug })
|
||||||
|
.leftJoin(TableName.SamlConfig, (qb) => {
|
||||||
|
qb.on(`${TableName.SamlConfig}.orgId`, "=", `${TableName.Organization}.id`).andOn(
|
||||||
|
`${TableName.SamlConfig}.isActive`,
|
||||||
|
"=",
|
||||||
|
db.raw("true")
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.leftJoin(TableName.OidcConfig, (qb) => {
|
||||||
|
qb.on(`${TableName.OidcConfig}.orgId`, "=", `${TableName.Organization}.id`).andOn(
|
||||||
|
`${TableName.OidcConfig}.isActive`,
|
||||||
|
"=",
|
||||||
|
db.raw("true")
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.select(selectAllTableCols(TableName.Organization))
|
||||||
|
.select(
|
||||||
|
db.raw(`
|
||||||
|
CASE
|
||||||
|
WHEN ${TableName.SamlConfig}."orgId" IS NOT NULL THEN '${OrgAuthMethod.SAML}'
|
||||||
|
WHEN ${TableName.OidcConfig}."orgId" IS NOT NULL THEN '${OrgAuthMethod.OIDC}'
|
||||||
|
ELSE ''
|
||||||
|
END as "orgAuthMethod"
|
||||||
|
`)
|
||||||
|
)
|
||||||
|
.first()) as TOrganizations & { orgAuthMethod?: string };
|
||||||
|
|
||||||
|
return org;
|
||||||
|
} catch (error) {
|
||||||
|
throw new DatabaseError({ error, name: "Find org by slug" });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// special query
|
// special query
|
||||||
const findAllOrgsByUserId = async (userId: string): Promise<(TOrganizations & { orgAuthMethod: string })[]> => {
|
const findAllOrgsByUserId = async (userId: string): Promise<(TOrganizations & { orgAuthMethod: string })[]> => {
|
||||||
try {
|
try {
|
||||||
@ -398,6 +465,7 @@ export const orgDALFactory = (db: TDbClient) => {
|
|||||||
findAllOrgMembers,
|
findAllOrgMembers,
|
||||||
countAllOrgMembers,
|
countAllOrgMembers,
|
||||||
findOrgById,
|
findOrgById,
|
||||||
|
findOrgBySlug,
|
||||||
findAllOrgsByUserId,
|
findAllOrgsByUserId,
|
||||||
ghostUserExists,
|
ghostUserExists,
|
||||||
findOrgMembersByUsername,
|
findOrgMembersByUsername,
|
||||||
|
@ -187,6 +187,15 @@ export const orgServiceFactory = ({
|
|||||||
return members;
|
return members;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const findOrgBySlug = async (slug: string) => {
|
||||||
|
const org = await orgDAL.findOrgBySlug(slug);
|
||||||
|
if (!org) {
|
||||||
|
throw new NotFoundError({ message: `Organization with slug '${slug}' not found` });
|
||||||
|
}
|
||||||
|
|
||||||
|
return org;
|
||||||
|
};
|
||||||
|
|
||||||
const findAllWorkspaces = async ({ actor, actorId, orgId }: TFindAllWorkspacesDTO) => {
|
const findAllWorkspaces = async ({ actor, actorId, orgId }: TFindAllWorkspacesDTO) => {
|
||||||
const organizationWorkspaceIds = new Set((await projectDAL.find({ orgId })).map((workspace) => workspace.id));
|
const organizationWorkspaceIds = new Set((await projectDAL.find({ orgId })).map((workspace) => workspace.id));
|
||||||
|
|
||||||
@ -275,6 +284,7 @@ export const orgServiceFactory = ({
|
|||||||
ForbiddenError.from(permission).throwUnlessCan(OrgPermissionActions.Edit, OrgPermissionSubjects.Settings);
|
ForbiddenError.from(permission).throwUnlessCan(OrgPermissionActions.Edit, OrgPermissionSubjects.Settings);
|
||||||
|
|
||||||
const plan = await licenseService.getPlan(orgId);
|
const plan = await licenseService.getPlan(orgId);
|
||||||
|
const currentOrg = await orgDAL.findOrgById(actorOrgId);
|
||||||
|
|
||||||
if (enforceMfa !== undefined) {
|
if (enforceMfa !== undefined) {
|
||||||
if (!plan.enforceMfa) {
|
if (!plan.enforceMfa) {
|
||||||
@ -305,6 +315,11 @@ export const orgServiceFactory = ({
|
|||||||
"Failed to enable/disable SCIM provisioning due to plan restriction. Upgrade plan to enable/disable SCIM provisioning."
|
"Failed to enable/disable SCIM provisioning due to plan restriction. Upgrade plan to enable/disable SCIM provisioning."
|
||||||
});
|
});
|
||||||
ForbiddenError.from(permission).throwUnlessCan(OrgPermissionActions.Edit, OrgPermissionSubjects.Scim);
|
ForbiddenError.from(permission).throwUnlessCan(OrgPermissionActions.Edit, OrgPermissionSubjects.Scim);
|
||||||
|
if (scimEnabled && !currentOrg.orgAuthMethod) {
|
||||||
|
throw new BadRequestError({
|
||||||
|
message: "Cannot enable SCIM when neither SAML or OIDC is configured."
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (authEnforced) {
|
if (authEnforced) {
|
||||||
@ -1132,6 +1147,7 @@ export const orgServiceFactory = ({
|
|||||||
createIncidentContact,
|
createIncidentContact,
|
||||||
deleteIncidentContact,
|
deleteIncidentContact,
|
||||||
getOrgGroups,
|
getOrgGroups,
|
||||||
listProjectMembershipsByOrgMembershipId
|
listProjectMembershipsByOrgMembershipId,
|
||||||
|
findOrgBySlug
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -74,3 +74,8 @@ export type TGetOrgGroupsDTO = TOrgPermission;
|
|||||||
export type TListProjectMembershipsByOrgMembershipIdDTO = {
|
export type TListProjectMembershipsByOrgMembershipIdDTO = {
|
||||||
orgMembershipId: string;
|
orgMembershipId: string;
|
||||||
} & TOrgPermission;
|
} & TOrgPermission;
|
||||||
|
|
||||||
|
export enum OrgAuthMethod {
|
||||||
|
OIDC = "oidc",
|
||||||
|
SAML = "saml"
|
||||||
|
}
|
||||||
|
@ -365,9 +365,8 @@ export const recursivelyGetSecretPaths = async ({
|
|||||||
folderId: p.folderId
|
folderId: p.folderId
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const pathsInCurrentDirectory = paths.filter((folder) =>
|
// path relative will start with ../ if its outside directory
|
||||||
folder.path.startsWith(currentPath === "/" ? "" : currentPath)
|
const pathsInCurrentDirectory = paths.filter((folder) => !path.relative(currentPath, folder.path).startsWith(".."));
|
||||||
);
|
|
||||||
|
|
||||||
return pathsInCurrentDirectory;
|
return pathsInCurrentDirectory;
|
||||||
};
|
};
|
||||||
|
@ -10,7 +10,7 @@ require (
|
|||||||
github.com/fatih/semgroup v1.2.0
|
github.com/fatih/semgroup v1.2.0
|
||||||
github.com/gitleaks/go-gitdiff v0.8.0
|
github.com/gitleaks/go-gitdiff v0.8.0
|
||||||
github.com/h2non/filetype v1.1.3
|
github.com/h2non/filetype v1.1.3
|
||||||
github.com/infisical/go-sdk v0.3.8
|
github.com/infisical/go-sdk v0.4.3
|
||||||
github.com/mattn/go-isatty v0.0.20
|
github.com/mattn/go-isatty v0.0.20
|
||||||
github.com/muesli/ansi v0.0.0-20221106050444-61f0cd9a192a
|
github.com/muesli/ansi v0.0.0-20221106050444-61f0cd9a192a
|
||||||
github.com/muesli/mango-cobra v1.2.0
|
github.com/muesli/mango-cobra v1.2.0
|
||||||
|
@ -265,8 +265,8 @@ github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:
|
|||||||
github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
|
github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
|
||||||
github.com/inconshreveable/mousetrap v1.0.1 h1:U3uMjPSQEBMNp1lFxmllqCPM6P5u/Xq7Pgzkat/bFNc=
|
github.com/inconshreveable/mousetrap v1.0.1 h1:U3uMjPSQEBMNp1lFxmllqCPM6P5u/Xq7Pgzkat/bFNc=
|
||||||
github.com/inconshreveable/mousetrap v1.0.1/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
github.com/inconshreveable/mousetrap v1.0.1/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
||||||
github.com/infisical/go-sdk v0.3.8 h1:0dGOhF3cwt0q5QzpnUs4lxwBiEza+DQYOyvEn7AfrM0=
|
github.com/infisical/go-sdk v0.4.3 h1:O5ZJ2eCBAZDE9PIAfBPq9Utb2CgQKrhmj9R0oFTRu4U=
|
||||||
github.com/infisical/go-sdk v0.3.8/go.mod h1:HHW7DgUqoolyQIUw/9HdpkZ3bDLwWyZ0HEtYiVaDKQw=
|
github.com/infisical/go-sdk v0.4.3/go.mod h1:6fWzAwTPIoKU49mQ2Oxu+aFnJu9n7k2JcNrZjzhHM2M=
|
||||||
github.com/jedib0t/go-pretty v4.3.0+incompatible h1:CGs8AVhEKg/n9YbUenWmNStRW2PHJzaeDodcfvRAbIo=
|
github.com/jedib0t/go-pretty v4.3.0+incompatible h1:CGs8AVhEKg/n9YbUenWmNStRW2PHJzaeDodcfvRAbIo=
|
||||||
github.com/jedib0t/go-pretty v4.3.0+incompatible/go.mod h1:XemHduiw8R651AF9Pt4FwCTKeG3oo7hrHJAoznj9nag=
|
github.com/jedib0t/go-pretty v4.3.0+incompatible/go.mod h1:XemHduiw8R651AF9Pt4FwCTKeG3oo7hrHJAoznj9nag=
|
||||||
github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
||||||
|
@ -205,6 +205,25 @@ func CallGetAllWorkSpacesUserBelongsTo(httpClient *resty.Client) (GetWorkSpacesR
|
|||||||
return workSpacesResponse, nil
|
return workSpacesResponse, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func CallGetProjectById(httpClient *resty.Client, id string) (Project, error) {
|
||||||
|
var projectResponse GetProjectByIdResponse
|
||||||
|
response, err := httpClient.
|
||||||
|
R().
|
||||||
|
SetResult(&projectResponse).
|
||||||
|
SetHeader("User-Agent", USER_AGENT).
|
||||||
|
Get(fmt.Sprintf("%v/v1/workspace/%s", config.INFISICAL_URL, id))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return Project{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if response.IsError() {
|
||||||
|
return Project{}, fmt.Errorf("CallGetProjectById: Unsuccessful response: [response=%v]", response)
|
||||||
|
}
|
||||||
|
|
||||||
|
return projectResponse.Project, nil
|
||||||
|
}
|
||||||
|
|
||||||
func CallIsAuthenticated(httpClient *resty.Client) bool {
|
func CallIsAuthenticated(httpClient *resty.Client) bool {
|
||||||
var workSpacesResponse GetWorkSpacesResponse
|
var workSpacesResponse GetWorkSpacesResponse
|
||||||
response, err := httpClient.
|
response, err := httpClient.
|
||||||
|
@ -128,6 +128,10 @@ type GetWorkSpacesResponse struct {
|
|||||||
} `json:"workspaces"`
|
} `json:"workspaces"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GetProjectByIdResponse struct {
|
||||||
|
Project Project `json:"workspace"`
|
||||||
|
}
|
||||||
|
|
||||||
type GetOrganizationsResponse struct {
|
type GetOrganizationsResponse struct {
|
||||||
Organizations []struct {
|
Organizations []struct {
|
||||||
ID string `json:"id"`
|
ID string `json:"id"`
|
||||||
@ -163,6 +167,12 @@ type Secret struct {
|
|||||||
PlainTextKey string `json:"plainTextKey"`
|
PlainTextKey string `json:"plainTextKey"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Project struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Slug string `json:"slug"`
|
||||||
|
}
|
||||||
|
|
||||||
type RawSecret struct {
|
type RawSecret struct {
|
||||||
SecretKey string `json:"secretKey,omitempty"`
|
SecretKey string `json:"secretKey,omitempty"`
|
||||||
SecretValue string `json:"secretValue,omitempty"`
|
SecretValue string `json:"secretValue,omitempty"`
|
||||||
|
571
cli/packages/cmd/dynamic_secrets.go
Normal file
571
cli/packages/cmd/dynamic_secrets.go
Normal file
@ -0,0 +1,571 @@
|
|||||||
|
/*
|
||||||
|
Copyright (c) 2023 Infisical Inc.
|
||||||
|
*/
|
||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/Infisical/infisical-merge/packages/api"
|
||||||
|
"github.com/Infisical/infisical-merge/packages/config"
|
||||||
|
"github.com/Infisical/infisical-merge/packages/visualize"
|
||||||
|
|
||||||
|
// "github.com/Infisical/infisical-merge/packages/models"
|
||||||
|
"github.com/Infisical/infisical-merge/packages/util"
|
||||||
|
// "github.com/Infisical/infisical-merge/packages/visualize"
|
||||||
|
"github.com/go-resty/resty/v2"
|
||||||
|
"github.com/posthog/posthog-go"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
|
||||||
|
infisicalSdk "github.com/infisical/go-sdk"
|
||||||
|
infisicalSdkModels "github.com/infisical/go-sdk/packages/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
var dynamicSecretCmd = &cobra.Command{
|
||||||
|
Example: `infisical dynamic-secrets`,
|
||||||
|
Short: "Used to list dynamic secrets",
|
||||||
|
Use: "dynamic-secrets",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
Args: cobra.NoArgs,
|
||||||
|
Run: getDynamicSecretList,
|
||||||
|
}
|
||||||
|
|
||||||
|
func getDynamicSecretList(cmd *cobra.Command, args []string) {
|
||||||
|
environmentName, _ := cmd.Flags().GetString("env")
|
||||||
|
if !cmd.Flags().Changed("env") {
|
||||||
|
environmentFromWorkspace := util.GetEnvFromWorkspaceFile()
|
||||||
|
if environmentFromWorkspace != "" {
|
||||||
|
environmentName = environmentFromWorkspace
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
token, err := util.GetInfisicalToken(cmd)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
projectId, err := cmd.Flags().GetString("projectId")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsPath, err := cmd.Flags().GetString("path")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse path flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var infisicalToken string
|
||||||
|
httpClient := resty.New()
|
||||||
|
|
||||||
|
if projectId == "" {
|
||||||
|
workspaceFile, err := util.GetWorkSpaceFromFile()
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to get local project details")
|
||||||
|
}
|
||||||
|
projectId = workspaceFile.WorkspaceId
|
||||||
|
}
|
||||||
|
|
||||||
|
if token != nil && (token.Type == util.SERVICE_TOKEN_IDENTIFIER || token.Type == util.UNIVERSAL_AUTH_TOKEN_IDENTIFIER) {
|
||||||
|
infisicalToken = token.Token
|
||||||
|
} else {
|
||||||
|
util.RequireLogin()
|
||||||
|
util.RequireLocalWorkspaceFile()
|
||||||
|
|
||||||
|
loggedInUserDetails, err := util.GetCurrentLoggedInUserDetails(true)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to authenticate")
|
||||||
|
}
|
||||||
|
|
||||||
|
if loggedInUserDetails.LoginExpired {
|
||||||
|
util.PrintErrorMessageAndExit("Your login session has expired, please run [infisical login] and try again")
|
||||||
|
}
|
||||||
|
infisicalToken = loggedInUserDetails.UserCredentials.JTWToken
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient.SetAuthToken(infisicalToken)
|
||||||
|
|
||||||
|
infisicalClient := infisicalSdk.NewInfisicalClient(context.Background(), infisicalSdk.Config{
|
||||||
|
SiteUrl: config.INFISICAL_URL,
|
||||||
|
UserAgent: api.USER_AGENT,
|
||||||
|
AutoTokenRefresh: false,
|
||||||
|
})
|
||||||
|
infisicalClient.Auth().SetAccessToken(infisicalToken)
|
||||||
|
|
||||||
|
projectDetails, err := api.CallGetProjectById(httpClient, projectId)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch project details")
|
||||||
|
}
|
||||||
|
|
||||||
|
dynamicSecretRootCredentials, err := infisicalClient.DynamicSecrets().List(infisicalSdk.ListDynamicSecretsRootCredentialsOptions{
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch dynamic secret root credentials details")
|
||||||
|
}
|
||||||
|
|
||||||
|
visualize.PrintAllDynamicRootCredentials(dynamicSecretRootCredentials)
|
||||||
|
Telemetry.CaptureEvent("cli-command:dynamic-secrets", posthog.NewProperties().Set("count", len(dynamicSecretRootCredentials)).Set("version", util.CLI_VERSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicSecretLeaseCmd = &cobra.Command{
|
||||||
|
Example: `lease`,
|
||||||
|
Short: "Manage leases for dynamic secrets",
|
||||||
|
Use: "lease",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicSecretLeaseCreateCmd = &cobra.Command{
|
||||||
|
Example: `lease create <dynamic secret name>"`,
|
||||||
|
Short: "Used to lease dynamic secret by name",
|
||||||
|
Use: "create [dynamic-secret]",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
Run: createDynamicSecretLeaseByName,
|
||||||
|
}
|
||||||
|
|
||||||
|
func createDynamicSecretLeaseByName(cmd *cobra.Command, args []string) {
|
||||||
|
dynamicSecretRootCredentialName := args[0]
|
||||||
|
|
||||||
|
environmentName, _ := cmd.Flags().GetString("env")
|
||||||
|
if !cmd.Flags().Changed("env") {
|
||||||
|
environmentFromWorkspace := util.GetEnvFromWorkspaceFile()
|
||||||
|
if environmentFromWorkspace != "" {
|
||||||
|
environmentName = environmentFromWorkspace
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
token, err := util.GetInfisicalToken(cmd)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
projectId, err := cmd.Flags().GetString("projectId")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
ttl, err := cmd.Flags().GetString("ttl")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsPath, err := cmd.Flags().GetString("path")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse path flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
plainOutput, err := cmd.Flags().GetBool("plain")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var infisicalToken string
|
||||||
|
httpClient := resty.New()
|
||||||
|
|
||||||
|
if projectId == "" {
|
||||||
|
workspaceFile, err := util.GetWorkSpaceFromFile()
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to get local project details")
|
||||||
|
}
|
||||||
|
projectId = workspaceFile.WorkspaceId
|
||||||
|
}
|
||||||
|
|
||||||
|
if token != nil && (token.Type == util.SERVICE_TOKEN_IDENTIFIER || token.Type == util.UNIVERSAL_AUTH_TOKEN_IDENTIFIER) {
|
||||||
|
infisicalToken = token.Token
|
||||||
|
} else {
|
||||||
|
util.RequireLogin()
|
||||||
|
util.RequireLocalWorkspaceFile()
|
||||||
|
|
||||||
|
loggedInUserDetails, err := util.GetCurrentLoggedInUserDetails(true)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to authenticate")
|
||||||
|
}
|
||||||
|
|
||||||
|
if loggedInUserDetails.LoginExpired {
|
||||||
|
util.PrintErrorMessageAndExit("Your login session has expired, please run [infisical login] and try again")
|
||||||
|
}
|
||||||
|
infisicalToken = loggedInUserDetails.UserCredentials.JTWToken
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient.SetAuthToken(infisicalToken)
|
||||||
|
|
||||||
|
infisicalClient := infisicalSdk.NewInfisicalClient(context.Background(), infisicalSdk.Config{
|
||||||
|
SiteUrl: config.INFISICAL_URL,
|
||||||
|
UserAgent: api.USER_AGENT,
|
||||||
|
AutoTokenRefresh: false,
|
||||||
|
})
|
||||||
|
infisicalClient.Auth().SetAccessToken(infisicalToken)
|
||||||
|
|
||||||
|
projectDetails, err := api.CallGetProjectById(httpClient, projectId)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch project details")
|
||||||
|
}
|
||||||
|
|
||||||
|
dynamicSecretRootCredential, err := infisicalClient.DynamicSecrets().GetByName(infisicalSdk.GetDynamicSecretRootCredentialByNameOptions{
|
||||||
|
DynamicSecretName: dynamicSecretRootCredentialName,
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch dynamic secret root credentials details")
|
||||||
|
}
|
||||||
|
|
||||||
|
leaseCredentials, _, leaseDetails, err := infisicalClient.DynamicSecrets().Leases().Create(infisicalSdk.CreateDynamicSecretLeaseOptions{
|
||||||
|
DynamicSecretName: dynamicSecretRootCredential.Name,
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
TTL: ttl,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To lease dynamic secret")
|
||||||
|
}
|
||||||
|
|
||||||
|
if plainOutput {
|
||||||
|
for key, value := range leaseCredentials {
|
||||||
|
if cred, ok := value.(string); ok {
|
||||||
|
fmt.Printf("%s=%s\n", key, cred)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fmt.Println("Dynamic Secret Leasing")
|
||||||
|
fmt.Printf("Name: %s\n", dynamicSecretRootCredential.Name)
|
||||||
|
fmt.Printf("Provider: %s\n", dynamicSecretRootCredential.Type)
|
||||||
|
fmt.Printf("Lease ID: %s\n", leaseDetails.Id)
|
||||||
|
fmt.Printf("Expire At: %s\n", leaseDetails.ExpireAt.Local().Format("02-Jan-2006 03:04:05 PM"))
|
||||||
|
visualize.PrintAllDyamicSecretLeaseCredentials(leaseCredentials)
|
||||||
|
}
|
||||||
|
|
||||||
|
Telemetry.CaptureEvent("cli-command:dynamic-secrets lease", posthog.NewProperties().Set("type", dynamicSecretRootCredential.Type).Set("version", util.CLI_VERSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicSecretLeaseRenewCmd = &cobra.Command{
|
||||||
|
Example: `lease renew <dynamic secret name>"`,
|
||||||
|
Short: "Used to renew dynamic secret lease by name",
|
||||||
|
Use: "renew [lease-id]",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
Run: renewDynamicSecretLeaseByName,
|
||||||
|
}
|
||||||
|
|
||||||
|
func renewDynamicSecretLeaseByName(cmd *cobra.Command, args []string) {
|
||||||
|
dynamicSecretLeaseId := args[0]
|
||||||
|
|
||||||
|
environmentName, _ := cmd.Flags().GetString("env")
|
||||||
|
if !cmd.Flags().Changed("env") {
|
||||||
|
environmentFromWorkspace := util.GetEnvFromWorkspaceFile()
|
||||||
|
if environmentFromWorkspace != "" {
|
||||||
|
environmentName = environmentFromWorkspace
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
token, err := util.GetInfisicalToken(cmd)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
projectId, err := cmd.Flags().GetString("projectId")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
ttl, err := cmd.Flags().GetString("ttl")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsPath, err := cmd.Flags().GetString("path")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse path flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var infisicalToken string
|
||||||
|
httpClient := resty.New()
|
||||||
|
|
||||||
|
if projectId == "" {
|
||||||
|
workspaceFile, err := util.GetWorkSpaceFromFile()
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to get local project details")
|
||||||
|
}
|
||||||
|
projectId = workspaceFile.WorkspaceId
|
||||||
|
}
|
||||||
|
|
||||||
|
if token != nil && (token.Type == util.SERVICE_TOKEN_IDENTIFIER || token.Type == util.UNIVERSAL_AUTH_TOKEN_IDENTIFIER) {
|
||||||
|
infisicalToken = token.Token
|
||||||
|
} else {
|
||||||
|
util.RequireLogin()
|
||||||
|
util.RequireLocalWorkspaceFile()
|
||||||
|
|
||||||
|
loggedInUserDetails, err := util.GetCurrentLoggedInUserDetails(true)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to authenticate")
|
||||||
|
}
|
||||||
|
|
||||||
|
if loggedInUserDetails.LoginExpired {
|
||||||
|
util.PrintErrorMessageAndExit("Your login session has expired, please run [infisical login] and try again")
|
||||||
|
}
|
||||||
|
infisicalToken = loggedInUserDetails.UserCredentials.JTWToken
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient.SetAuthToken(infisicalToken)
|
||||||
|
|
||||||
|
infisicalClient := infisicalSdk.NewInfisicalClient(context.Background(), infisicalSdk.Config{
|
||||||
|
SiteUrl: config.INFISICAL_URL,
|
||||||
|
UserAgent: api.USER_AGENT,
|
||||||
|
AutoTokenRefresh: false,
|
||||||
|
})
|
||||||
|
infisicalClient.Auth().SetAccessToken(infisicalToken)
|
||||||
|
|
||||||
|
projectDetails, err := api.CallGetProjectById(httpClient, projectId)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch project details")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch dynamic secret root credentials details")
|
||||||
|
}
|
||||||
|
|
||||||
|
leaseDetails, err := infisicalClient.DynamicSecrets().Leases().RenewById(infisicalSdk.RenewDynamicSecretLeaseOptions{
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
TTL: ttl,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
LeaseId: dynamicSecretLeaseId,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To renew dynamic secret lease")
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println("Successfully renewed dynamic secret lease")
|
||||||
|
visualize.PrintAllDynamicSecretLeases([]infisicalSdkModels.DynamicSecretLease{leaseDetails})
|
||||||
|
|
||||||
|
Telemetry.CaptureEvent("cli-command:dynamic-secrets lease renew", posthog.NewProperties().Set("version", util.CLI_VERSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicSecretLeaseRevokeCmd = &cobra.Command{
|
||||||
|
Example: `lease delete <dynamic secret name>"`,
|
||||||
|
Short: "Used to delete dynamic secret lease by name",
|
||||||
|
Use: "delete [lease-id]",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
Run: revokeDynamicSecretLeaseByName,
|
||||||
|
}
|
||||||
|
|
||||||
|
func revokeDynamicSecretLeaseByName(cmd *cobra.Command, args []string) {
|
||||||
|
dynamicSecretLeaseId := args[0]
|
||||||
|
|
||||||
|
environmentName, _ := cmd.Flags().GetString("env")
|
||||||
|
if !cmd.Flags().Changed("env") {
|
||||||
|
environmentFromWorkspace := util.GetEnvFromWorkspaceFile()
|
||||||
|
if environmentFromWorkspace != "" {
|
||||||
|
environmentName = environmentFromWorkspace
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
token, err := util.GetInfisicalToken(cmd)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
projectId, err := cmd.Flags().GetString("projectId")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsPath, err := cmd.Flags().GetString("path")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse path flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var infisicalToken string
|
||||||
|
httpClient := resty.New()
|
||||||
|
|
||||||
|
if projectId == "" {
|
||||||
|
workspaceFile, err := util.GetWorkSpaceFromFile()
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to get local project details")
|
||||||
|
}
|
||||||
|
projectId = workspaceFile.WorkspaceId
|
||||||
|
}
|
||||||
|
|
||||||
|
if token != nil && (token.Type == util.SERVICE_TOKEN_IDENTIFIER || token.Type == util.UNIVERSAL_AUTH_TOKEN_IDENTIFIER) {
|
||||||
|
infisicalToken = token.Token
|
||||||
|
} else {
|
||||||
|
util.RequireLogin()
|
||||||
|
util.RequireLocalWorkspaceFile()
|
||||||
|
|
||||||
|
loggedInUserDetails, err := util.GetCurrentLoggedInUserDetails(true)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to authenticate")
|
||||||
|
}
|
||||||
|
|
||||||
|
if loggedInUserDetails.LoginExpired {
|
||||||
|
util.PrintErrorMessageAndExit("Your login session has expired, please run [infisical login] and try again")
|
||||||
|
}
|
||||||
|
infisicalToken = loggedInUserDetails.UserCredentials.JTWToken
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient.SetAuthToken(infisicalToken)
|
||||||
|
|
||||||
|
infisicalClient := infisicalSdk.NewInfisicalClient(context.Background(), infisicalSdk.Config{
|
||||||
|
SiteUrl: config.INFISICAL_URL,
|
||||||
|
UserAgent: api.USER_AGENT,
|
||||||
|
AutoTokenRefresh: false,
|
||||||
|
})
|
||||||
|
infisicalClient.Auth().SetAccessToken(infisicalToken)
|
||||||
|
|
||||||
|
projectDetails, err := api.CallGetProjectById(httpClient, projectId)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch project details")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch dynamic secret root credentials details")
|
||||||
|
}
|
||||||
|
|
||||||
|
leaseDetails, err := infisicalClient.DynamicSecrets().Leases().DeleteById(infisicalSdk.DeleteDynamicSecretLeaseOptions{
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
LeaseId: dynamicSecretLeaseId,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To revoke dynamic secret lease")
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println("Successfully revoked dynamic secret lease")
|
||||||
|
visualize.PrintAllDynamicSecretLeases([]infisicalSdkModels.DynamicSecretLease{leaseDetails})
|
||||||
|
|
||||||
|
Telemetry.CaptureEvent("cli-command:dynamic-secrets lease revoke", posthog.NewProperties().Set("version", util.CLI_VERSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicSecretLeaseListCmd = &cobra.Command{
|
||||||
|
Example: `lease list <dynamic secret name>"`,
|
||||||
|
Short: "Used to list leases of a dynamic secret by name",
|
||||||
|
Use: "list [dynamic-secret]",
|
||||||
|
DisableFlagsInUseLine: true,
|
||||||
|
Args: cobra.ExactArgs(1),
|
||||||
|
Run: listDynamicSecretLeaseByName,
|
||||||
|
}
|
||||||
|
|
||||||
|
func listDynamicSecretLeaseByName(cmd *cobra.Command, args []string) {
|
||||||
|
dynamicSecretRootCredentialName := args[0]
|
||||||
|
|
||||||
|
environmentName, _ := cmd.Flags().GetString("env")
|
||||||
|
if !cmd.Flags().Changed("env") {
|
||||||
|
environmentFromWorkspace := util.GetEnvFromWorkspaceFile()
|
||||||
|
if environmentFromWorkspace != "" {
|
||||||
|
environmentName = environmentFromWorkspace
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
token, err := util.GetInfisicalToken(cmd)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
projectId, err := cmd.Flags().GetString("projectId")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
secretsPath, err := cmd.Flags().GetString("path")
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to parse path flag")
|
||||||
|
}
|
||||||
|
|
||||||
|
var infisicalToken string
|
||||||
|
httpClient := resty.New()
|
||||||
|
|
||||||
|
if projectId == "" {
|
||||||
|
workspaceFile, err := util.GetWorkSpaceFromFile()
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to get local project details")
|
||||||
|
}
|
||||||
|
projectId = workspaceFile.WorkspaceId
|
||||||
|
}
|
||||||
|
|
||||||
|
if token != nil && (token.Type == util.SERVICE_TOKEN_IDENTIFIER || token.Type == util.UNIVERSAL_AUTH_TOKEN_IDENTIFIER) {
|
||||||
|
infisicalToken = token.Token
|
||||||
|
} else {
|
||||||
|
util.RequireLogin()
|
||||||
|
util.RequireLocalWorkspaceFile()
|
||||||
|
|
||||||
|
loggedInUserDetails, err := util.GetCurrentLoggedInUserDetails(true)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "Unable to authenticate")
|
||||||
|
}
|
||||||
|
|
||||||
|
if loggedInUserDetails.LoginExpired {
|
||||||
|
util.PrintErrorMessageAndExit("Your login session has expired, please run [infisical login] and try again")
|
||||||
|
}
|
||||||
|
infisicalToken = loggedInUserDetails.UserCredentials.JTWToken
|
||||||
|
}
|
||||||
|
|
||||||
|
httpClient.SetAuthToken(infisicalToken)
|
||||||
|
|
||||||
|
infisicalClient := infisicalSdk.NewInfisicalClient(context.Background(), infisicalSdk.Config{
|
||||||
|
SiteUrl: config.INFISICAL_URL,
|
||||||
|
UserAgent: api.USER_AGENT,
|
||||||
|
AutoTokenRefresh: false,
|
||||||
|
})
|
||||||
|
infisicalClient.Auth().SetAccessToken(infisicalToken)
|
||||||
|
|
||||||
|
projectDetails, err := api.CallGetProjectById(httpClient, projectId)
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch project details")
|
||||||
|
}
|
||||||
|
|
||||||
|
dynamicSecretLeases, err := infisicalClient.DynamicSecrets().Leases().List(infisicalSdk.ListDynamicSecretLeasesOptions{
|
||||||
|
DynamicSecretName: dynamicSecretRootCredentialName,
|
||||||
|
ProjectSlug: projectDetails.Slug,
|
||||||
|
SecretPath: secretsPath,
|
||||||
|
EnvironmentSlug: environmentName,
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
util.HandleError(err, "To fetch dynamic secret leases list")
|
||||||
|
}
|
||||||
|
|
||||||
|
visualize.PrintAllDynamicSecretLeases(dynamicSecretLeases)
|
||||||
|
Telemetry.CaptureEvent("cli-command:dynamic-secrets lease list", posthog.NewProperties().Set("lease-count", len(dynamicSecretLeases)).Set("version", util.CLI_VERSION))
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
dynamicSecretLeaseCreateCmd.Flags().StringP("path", "p", "/", "The path from where dynamic secret should be leased from")
|
||||||
|
dynamicSecretLeaseCreateCmd.Flags().String("token", "", "Create dynamic secret leases using machine identity access token")
|
||||||
|
dynamicSecretLeaseCreateCmd.Flags().String("projectId", "", "Manually set the projectId to fetch leased from when using machine identity based auth")
|
||||||
|
dynamicSecretLeaseCreateCmd.Flags().String("ttl", "", "The lease lifetime TTL. If not provided the default TTL of dynamic secret will be used.")
|
||||||
|
dynamicSecretLeaseCreateCmd.Flags().Bool("plain", false, "Print leased credentials without formatting, one per line")
|
||||||
|
dynamicSecretLeaseCmd.AddCommand(dynamicSecretLeaseCreateCmd)
|
||||||
|
|
||||||
|
dynamicSecretLeaseListCmd.Flags().StringP("path", "p", "/", "The path from where dynamic secret should be leased from")
|
||||||
|
dynamicSecretLeaseListCmd.Flags().String("token", "", "Fetch dynamic secret leases machine identity access token")
|
||||||
|
dynamicSecretLeaseListCmd.Flags().String("projectId", "", "Manually set the projectId to fetch leased from when using machine identity based auth")
|
||||||
|
dynamicSecretLeaseCmd.AddCommand(dynamicSecretLeaseListCmd)
|
||||||
|
|
||||||
|
dynamicSecretLeaseRenewCmd.Flags().StringP("path", "p", "/", "The path from where dynamic secret should be leased from")
|
||||||
|
dynamicSecretLeaseRenewCmd.Flags().String("token", "", "Renew dynamic secrets machine identity access token")
|
||||||
|
dynamicSecretLeaseRenewCmd.Flags().String("projectId", "", "Manually set the projectId to fetch leased from when using machine identity based auth")
|
||||||
|
dynamicSecretLeaseRenewCmd.Flags().String("ttl", "", "The lease lifetime TTL. If not provided the default TTL of dynamic secret will be used.")
|
||||||
|
dynamicSecretLeaseCmd.AddCommand(dynamicSecretLeaseRenewCmd)
|
||||||
|
|
||||||
|
dynamicSecretLeaseRevokeCmd.Flags().StringP("path", "p", "/", "The path from where dynamic secret should be leased from")
|
||||||
|
dynamicSecretLeaseRevokeCmd.Flags().String("token", "", "Delete dynamic secrets using machine identity access token")
|
||||||
|
dynamicSecretLeaseRevokeCmd.Flags().String("projectId", "", "Manually set the projectId to fetch leased from when using machine identity based auth")
|
||||||
|
dynamicSecretLeaseCmd.AddCommand(dynamicSecretLeaseRevokeCmd)
|
||||||
|
|
||||||
|
dynamicSecretCmd.AddCommand(dynamicSecretLeaseCmd)
|
||||||
|
|
||||||
|
dynamicSecretCmd.Flags().String("token", "", "Fetch secrets using service token or machine identity access token")
|
||||||
|
dynamicSecretCmd.Flags().String("projectId", "", "Manually set the projectId to fetch dynamic-secret when using machine identity based auth")
|
||||||
|
dynamicSecretCmd.PersistentFlags().String("env", "dev", "Used to select the environment name on which actions should be taken on")
|
||||||
|
dynamicSecretCmd.Flags().String("path", "/", "get dynamic secret within a folder path")
|
||||||
|
rootCmd.AddCommand(dynamicSecretCmd)
|
||||||
|
}
|
39
cli/packages/visualize/dynamic_secret_leases.go
Normal file
39
cli/packages/visualize/dynamic_secret_leases.go
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
package visualize
|
||||||
|
|
||||||
|
import infisicalModels "github.com/infisical/go-sdk/packages/models"
|
||||||
|
|
||||||
|
func PrintAllDyamicSecretLeaseCredentials(leaseCredentials map[string]any) {
|
||||||
|
rows := [][]string{}
|
||||||
|
for key, value := range leaseCredentials {
|
||||||
|
if cred, ok := value.(string); ok {
|
||||||
|
rows = append(rows, []string{key, cred})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{"Key", "Value"}
|
||||||
|
|
||||||
|
GenericTable(headers, rows)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintAllDynamicRootCredentials(dynamicRootCredentials []infisicalModels.DynamicSecret) {
|
||||||
|
rows := [][]string{}
|
||||||
|
for _, el := range dynamicRootCredentials {
|
||||||
|
rows = append(rows, []string{el.Name, el.Type, el.DefaultTTL, el.MaxTTL})
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{"Name", "Provider", "Default TTL", "Max TTL"}
|
||||||
|
|
||||||
|
GenericTable(headers, rows)
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintAllDynamicSecretLeases(dynamicSecretLeases []infisicalModels.DynamicSecretLease) {
|
||||||
|
rows := [][]string{}
|
||||||
|
const timeformat = "02-Jan-2006 03:04:05 PM"
|
||||||
|
for _, el := range dynamicSecretLeases {
|
||||||
|
rows = append(rows, []string{el.Id, el.ExpireAt.Local().Format(timeformat), el.CreatedAt.Local().Format(timeformat)})
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := []string{"ID", "Expire At", "Created At"}
|
||||||
|
|
||||||
|
GenericTable(headers, rows)
|
||||||
|
}
|
@ -94,6 +94,33 @@ func getLongestValues(rows [][3]string) (longestSecretName, longestSecretType in
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func GenericTable(headers []string, rows [][]string) {
|
||||||
|
t := table.NewWriter()
|
||||||
|
t.SetOutputMirror(os.Stdout)
|
||||||
|
t.SetStyle(table.StyleLight)
|
||||||
|
|
||||||
|
// t.SetTitle(tableOptions.Title)
|
||||||
|
t.Style().Options.DrawBorder = true
|
||||||
|
t.Style().Options.SeparateHeader = true
|
||||||
|
t.Style().Options.SeparateColumns = true
|
||||||
|
|
||||||
|
tableHeaders := table.Row{}
|
||||||
|
for _, header := range headers {
|
||||||
|
tableHeaders = append(tableHeaders, header)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.AppendHeader(tableHeaders)
|
||||||
|
for _, row := range rows {
|
||||||
|
tableRow := table.Row{}
|
||||||
|
for _, val := range row {
|
||||||
|
tableRow = append(tableRow, val)
|
||||||
|
}
|
||||||
|
t.AppendRow(tableRow)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Render()
|
||||||
|
}
|
||||||
|
|
||||||
// stringWidth returns the width of a string.
|
// stringWidth returns the width of a string.
|
||||||
// ANSI escape sequences are ignored and double-width characters are handled correctly.
|
// ANSI escape sequences are ignored and double-width characters are handled correctly.
|
||||||
func stringWidth(str string) (width int) {
|
func stringWidth(str string) (width int) {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
error: CallGetRawSecretsV3: Unsuccessful response [GET https://app.infisical.com/api/v3/secrets/raw?environment=invalid-env&expandSecretReferences=true&include_imports=true&recursive=true&secretPath=%2F&workspaceId=bef697d4-849b-4a75-b284-0922f87f8ba2] [status-code=404] [response={"statusCode":404,"message":"Environment with slug 'invalid-env' in project with ID bef697d4-849b-4a75-b284-0922f87f8ba2 not found","error":"NotFound"}]
|
error: CallGetRawSecretsV3: Unsuccessful response [GET https://app.infisical.com/api/v3/secrets/raw?environment=invalid-env&expandSecretReferences=true&include_imports=true&recursive=true&secretPath=%2F&workspaceId=bef697d4-849b-4a75-b284-0922f87f8ba2] [status-code=404] [response={"error":"NotFound","message":"Environment with slug 'invalid-env' in project with ID bef697d4-849b-4a75-b284-0922f87f8ba2 not found","statusCode":404}]
|
||||||
|
|
||||||
|
|
||||||
If this issue continues, get support at https://infisical.com/slack
|
If this issue continues, get support at https://infisical.com/slack
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
Warning: Unable to fetch the latest secret(s) due to connection error, serving secrets from last successful fetch. For more info, run with --debug
|
Warning: Unable to fetch the latest secret(s) due to connection error, serving secrets from last successful fetch. For more info, run with --debug
|
||||||
┌───────────────┬──────────────┬─────────────┐
|
┌───────────────┬──────────────┬─────────────┐
|
||||||
│ SECRET NAME │ SECRET VALUE │ SECRET TYPE │
|
│ SECRET NAME │ SECRET VALUE │ SECRET TYPE │
|
||||||
├───────────────┼──────────────┼─────────────┤
|
├───────────────┼──────────────┼─────────────┤
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package tests
|
package tests
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
@ -41,11 +42,12 @@ var creds = Credentials{
|
|||||||
func ExecuteCliCommand(command string, args ...string) (string, error) {
|
func ExecuteCliCommand(command string, args ...string) (string, error) {
|
||||||
cmd := exec.Command(command, args...)
|
cmd := exec.Command(command, args...)
|
||||||
output, err := cmd.CombinedOutput()
|
output, err := cmd.CombinedOutput()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(fmt.Sprint(err) + ": " + string(output))
|
fmt.Println(fmt.Sprint(err) + ": " + FilterRequestID(strings.TrimSpace(string(output))))
|
||||||
return strings.TrimSpace(string(output)), err
|
return FilterRequestID(strings.TrimSpace(string(output))), err
|
||||||
}
|
}
|
||||||
return strings.TrimSpace(string(output)), nil
|
return FilterRequestID(strings.TrimSpace(string(output))), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func SetupCli() {
|
func SetupCli() {
|
||||||
@ -67,3 +69,34 @@ func SetupCli() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func FilterRequestID(input string) string {
|
||||||
|
// Find the JSON part of the error message
|
||||||
|
start := strings.Index(input, "{")
|
||||||
|
end := strings.LastIndex(input, "}") + 1
|
||||||
|
|
||||||
|
if start == -1 || end == -1 {
|
||||||
|
return input
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonPart := input[:start] // Pre-JSON content
|
||||||
|
|
||||||
|
// Parse the JSON object
|
||||||
|
var errorObj map[string]interface{}
|
||||||
|
if err := json.Unmarshal([]byte(input[start:end]), &errorObj); err != nil {
|
||||||
|
return input
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove requestId field
|
||||||
|
delete(errorObj, "requestId")
|
||||||
|
delete(errorObj, "reqId")
|
||||||
|
|
||||||
|
// Convert back to JSON
|
||||||
|
filtered, err := json.Marshal(errorObj)
|
||||||
|
if err != nil {
|
||||||
|
return input
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reconstruct the full string
|
||||||
|
return jsonPart + string(filtered) + input[end:]
|
||||||
|
}
|
||||||
|
@ -3,7 +3,6 @@ package tests
|
|||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/Infisical/infisical-merge/packages/util"
|
|
||||||
"github.com/bradleyjkemp/cupaloy/v2"
|
"github.com/bradleyjkemp/cupaloy/v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -96,28 +95,29 @@ func TestUserAuth_SecretsGetAll(t *testing.T) {
|
|||||||
// testUserAuth_SecretsGetAllWithoutConnection(t)
|
// testUserAuth_SecretsGetAllWithoutConnection(t)
|
||||||
}
|
}
|
||||||
|
|
||||||
func testUserAuth_SecretsGetAllWithoutConnection(t *testing.T) {
|
// disabled for the time being
|
||||||
originalConfigFile, err := util.GetConfigFile()
|
// func testUserAuth_SecretsGetAllWithoutConnection(t *testing.T) {
|
||||||
if err != nil {
|
// originalConfigFile, err := util.GetConfigFile()
|
||||||
t.Fatalf("error getting config file")
|
// if err != nil {
|
||||||
}
|
// t.Fatalf("error getting config file")
|
||||||
newConfigFile := originalConfigFile
|
// }
|
||||||
|
// newConfigFile := originalConfigFile
|
||||||
|
|
||||||
// set it to a URL that will always be unreachable
|
// // set it to a URL that will always be unreachable
|
||||||
newConfigFile.LoggedInUserDomain = "http://localhost:4999"
|
// newConfigFile.LoggedInUserDomain = "http://localhost:4999"
|
||||||
util.WriteConfigFile(&newConfigFile)
|
// util.WriteConfigFile(&newConfigFile)
|
||||||
|
|
||||||
// restore config file
|
// // restore config file
|
||||||
defer util.WriteConfigFile(&originalConfigFile)
|
// defer util.WriteConfigFile(&originalConfigFile)
|
||||||
|
|
||||||
output, err := ExecuteCliCommand(FORMATTED_CLI_NAME, "secrets", "--projectId", creds.ProjectID, "--env", creds.EnvSlug, "--include-imports=false", "--silent")
|
// output, err := ExecuteCliCommand(FORMATTED_CLI_NAME, "secrets", "--projectId", creds.ProjectID, "--env", creds.EnvSlug, "--include-imports=false", "--silent")
|
||||||
if err != nil {
|
// if err != nil {
|
||||||
t.Fatalf("error running CLI command: %v", err)
|
// t.Fatalf("error running CLI command: %v", err)
|
||||||
}
|
// }
|
||||||
|
|
||||||
// Use cupaloy to snapshot test the output
|
// // Use cupaloy to snapshot test the output
|
||||||
err = cupaloy.Snapshot(output)
|
// err = cupaloy.Snapshot(output)
|
||||||
if err != nil {
|
// if err != nil {
|
||||||
t.Fatalf("snapshot failed: %v", err)
|
// t.Fatalf("snapshot failed: %v", err)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
@ -12,6 +12,18 @@ To request time off, just submit a request in Rippling and let Maidul know at le
|
|||||||
|
|
||||||
Since Infisical's team is globally distributed, it is hard for us to keep track of all the various national holidays across many different countries. Whether you'd like to celebrate Christmas or National Brisket Day (which, by the way, is on May 28th), you are welcome to take PTO on those days – just let Maidul know at least a week ahead so that we can adjust our planning.
|
Since Infisical's team is globally distributed, it is hard for us to keep track of all the various national holidays across many different countries. Whether you'd like to celebrate Christmas or National Brisket Day (which, by the way, is on May 28th), you are welcome to take PTO on those days – just let Maidul know at least a week ahead so that we can adjust our planning.
|
||||||
|
|
||||||
## Winter Break
|
## Winter break
|
||||||
|
|
||||||
Every year, Infisical team goes on a company-wide vacation during winter holidays. This year, the winter break period starts on December 21st, 2024 and ends on January 5th, 2025. You should expect to do no scheduled work during this period, but we will have a rotation process for [high and urgent service disruptions](https://infisical.com/sla).
|
Every year, Infisical team goes on a company-wide vacation during winter holidays. This year, the winter break period starts on December 21st, 2024 and ends on January 5th, 2025. You should expect to do no scheduled work during this period, but we will have a rotation process for [high and urgent service disruptions](https://infisical.com/sla).
|
||||||
|
|
||||||
|
## Parental leave
|
||||||
|
|
||||||
|
At Infisical, we recognize that parental leave is a special and important time, significantly different from a typical vacation. We’re proud to offer parental leave to everyone, regardless of gender, and whether you’ve become a parent through childbirth or adoption.
|
||||||
|
|
||||||
|
For team members who have been with Infisical for over a year by the time of your child’s birth or adoption, you are eligible for up to 12 weeks of paid parental leave. This leave will be provided in one continuous block to allow you uninterrupted time with your family. If you have been with Infisical for less than a year, we will follow the parental leave provisions required by your local jurisdiction.
|
||||||
|
|
||||||
|
While we trust your judgment, parental leave is intended to be a distinct benefit and is not designed to be combined with our unlimited PTO policy. To ensure fairness and balance, we generally discourage combining parental leave with an extended vacation.
|
||||||
|
|
||||||
|
When you’re ready, please notify Maidul about your plans for parental leave, ideally at least four months in advance. This allows us to support you fully and arrange any necessary logistics, including salary adjustments and statutory paperwork.
|
||||||
|
|
||||||
|
We’re here to support you as you embark on this exciting new chapter in your life!
|
||||||
|
295
docs/cli/commands/dynamic-secrets.mdx
Normal file
295
docs/cli/commands/dynamic-secrets.mdx
Normal file
@ -0,0 +1,295 @@
|
|||||||
|
---
|
||||||
|
title: "infisical dynamic-secrets"
|
||||||
|
description: "Perform dynamic secret operations directly with the CLI"
|
||||||
|
---
|
||||||
|
|
||||||
|
```
|
||||||
|
infisical dynamic-secrets
|
||||||
|
```
|
||||||
|
|
||||||
|
## Description
|
||||||
|
|
||||||
|
Dynamic secrets are unique secrets generated on demand based on the provided configuration settings. For more details, refer to [dynamics secrets section](/documentation/platform/dynamic-secrets/overview).
|
||||||
|
|
||||||
|
This command enables you to perform list, lease, renew lease, and revoke lease operations on dynamic secrets within your Infisical project.
|
||||||
|
|
||||||
|
### Sub-commands
|
||||||
|
|
||||||
|
<Accordion title="infisical dynamic-secrets">
|
||||||
|
Use this command to print out all of the dynamic secrets in your project.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ infisical dynamic-secrets
|
||||||
|
```
|
||||||
|
|
||||||
|
### Environment variables
|
||||||
|
|
||||||
|
<Accordion title="INFISICAL_TOKEN">
|
||||||
|
Used to fetch dynamic secrets via a [machine identity](/documentation/platform/identities/machine-identities) instead of logged-in credentials. Simply, export this variable in the terminal before running this command.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
export INFISICAL_TOKEN=$(infisical login --method=universal-auth --client-id=<identity-client-id> --client-secret=<identity-client-secret> --silent --plain) # --plain flag will output only the token, so it can be fed to an environment variable. --silent will disable any update messages.
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="INFISICAL_DISABLE_UPDATE_CHECK">
|
||||||
|
Used to disable the check for new CLI versions. This can improve the time it takes to run this command. Recommended for production environments.
|
||||||
|
|
||||||
|
To use, simply export this variable in the terminal before running this command.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
export INFISICAL_DISABLE_UPDATE_CHECK=true
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
### Flags
|
||||||
|
|
||||||
|
<Accordion title="--projectId">
|
||||||
|
The project ID to fetch dynamic secrets from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets --projectId=<project-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--token">
|
||||||
|
The authenticated token to fetch dynamic secrets from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets --token=<token>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--env">
|
||||||
|
Used to select the environment name on which actions should be taken. Default
|
||||||
|
value: `dev`
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--path">
|
||||||
|
Use to select the project folder on which dynamic secrets will be accessed.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets --path="/" --env=dev
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
</Accordion>
|
||||||
|
<Accordion title="infisical dynamic-secrets lease create">
|
||||||
|
This command is used to create a new lease for a dynamic secret.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ infisical dynamic-secrets lease create <dynamic-secret-name>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Flags
|
||||||
|
|
||||||
|
<Accordion title="--env">
|
||||||
|
Used to select the environment name on which actions should be taken. Default
|
||||||
|
value: `dev`
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--plain">
|
||||||
|
The `--plain` flag will output dynamic secret lease credentials values without formatting, one per line.
|
||||||
|
Default value: `false`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease create dynamic-secret-postgres --plain
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--path">
|
||||||
|
The `--path` flag indicates which project folder dynamic secrets will be injected from.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease create <dynamic-secret-name> --path="/" --env=dev
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--projectId">
|
||||||
|
The project ID of the dynamic secrets to lease from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease create <dynamic-secret-name> --projectId=<project-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--token">
|
||||||
|
The authenticated token to create dynamic secret leases. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease create <dynamic-secret-name> --token=<token>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--ttl">
|
||||||
|
The lease lifetime. If not provided, the default TTL of the dynamic secret root credential will be used.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease create <dynamic-secret-name> --ttl=<ttl>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
<Accordion title="infisical dynamic-secrets lease list">
|
||||||
|
This command is used to list leases for a dynamic secret.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ infisical dynamic-secrets lease list <dynamic-secret-name>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Flags
|
||||||
|
|
||||||
|
<Accordion title="--env">
|
||||||
|
Used to select the environment name on which actions should be taken. Default
|
||||||
|
value: `dev`
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--path">
|
||||||
|
The `--path` flag indicates which project folder dynamic secrets will be injected from.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease list <dynamic-secret-name> --path="/" --env=dev
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--projectId">
|
||||||
|
The project ID of the dynamic secrets to list leases from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease list <dynamic-secret-name> --projectId=<project-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--token">
|
||||||
|
The authenticated token to list dynamic secret leases. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease list <dynamic-secret-name> --token=<token>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="infisical dynamic-secrets lease renew">
|
||||||
|
This command is used to renew a lease before it expires.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ infisical dynamic-secrets lease renew <lease-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Flags
|
||||||
|
|
||||||
|
<Accordion title="--env">
|
||||||
|
Used to select the environment name on which actions should be taken. Default
|
||||||
|
value: `dev`
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--path">
|
||||||
|
The `--path` flag indicates which project folder dynamic secrets will be renewed from.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease renew <lease-id> --path="/" --env=dev
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--projectId">
|
||||||
|
The project ID of the dynamic secret's lease from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease renew <lease-id> --projectId=<project-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--token">
|
||||||
|
The authenticated token to create dynamic secret leases. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease renew <lease-id> --token=<token>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--ttl">
|
||||||
|
The lease lifetime. If not provided, the default TTL of the dynamic secret root credential will be used.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease renew <lease-id> --ttl=<ttl>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="infisical dynamic-secrets lease delete">
|
||||||
|
This command is used to delete a lease.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ infisical dynamic-secrets lease delete <lease-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Flags
|
||||||
|
|
||||||
|
<Accordion title="--env">
|
||||||
|
Used to select the environment name on which actions should be taken. Default
|
||||||
|
value: `dev`
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--path">
|
||||||
|
The `--path` flag indicates which project folder dynamic secrets will be deleted from.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease delete <lease-id> --path="/" --env=dev
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--projectId">
|
||||||
|
The project ID of the dynamic secret's lease from. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease delete <lease-id> --projectId=<project-id>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="--token">
|
||||||
|
The authenticated token to delete dynamic secret leases. This is required when using a machine identity to authenticate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Example
|
||||||
|
infisical dynamic-secrets lease delete <lease-id> --token=<token>
|
||||||
|
```
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
</Accordion>
|
@ -3,11 +3,15 @@ title: "SCIM Overview"
|
|||||||
description: "Learn how to provision users for Infisical via SCIM."
|
description: "Learn how to provision users for Infisical via SCIM."
|
||||||
---
|
---
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
SCIM provisioning can only be enabled when either SAML or OIDC is setup for
|
||||||
|
the organization.
|
||||||
|
</Note>
|
||||||
<Info>
|
<Info>
|
||||||
SCIM provisioning is a paid feature.
|
SCIM provisioning is a paid feature. If you're using Infisical Cloud, then it
|
||||||
|
is available under the **Enterprise Tier**. If you're self-hosting Infisical,
|
||||||
If you're using Infisical Cloud, then it is available under the **Enterprise Tier**. If you're self-hosting Infisical,
|
then you should contact sales@infisical.com to purchase an enterprise license
|
||||||
then you should contact sales@infisical.com to purchase an enterprise license to use it.
|
to use it.
|
||||||
</Info>
|
</Info>
|
||||||
|
|
||||||
You can configure your organization in Infisical to have users and user groups be provisioned/deprovisioned using [SCIM](https://scim.cloud/#Implementations2) via providers like Okta, Azure, JumpCloud, etc.
|
You can configure your organization in Infisical to have users and user groups be provisioned/deprovisioned using [SCIM](https://scim.cloud/#Implementations2) via providers like Okta, Azure, JumpCloud, etc.
|
||||||
@ -20,13 +24,3 @@ SCIM providers:
|
|||||||
- [Okta SCIM](/documentation/platform/scim/okta)
|
- [Okta SCIM](/documentation/platform/scim/okta)
|
||||||
- [Azure SCIM](/documentation/platform/scim/azure)
|
- [Azure SCIM](/documentation/platform/scim/azure)
|
||||||
- [JumpCloud SCIM](/documentation/platform/scim/jumpcloud)
|
- [JumpCloud SCIM](/documentation/platform/scim/jumpcloud)
|
||||||
|
|
||||||
**FAQ**
|
|
||||||
|
|
||||||
<AccordionGroup>
|
|
||||||
<Accordion title="Why do SCIM-provisioned users have to finish setting up their account?">
|
|
||||||
Infisical's SCIM implementation accounts for retaining the end-to-end encrypted architecture of Infisical because we decouple the **authentication** and **decryption** steps in the platform.
|
|
||||||
|
|
||||||
For this reason, SCIM-provisioned users are initialized but must finish setting up their account when logging in the first time by creating a master encryption/decryption key. With this implementation, IdPs and SCIM providers cannot and will not have access to the decryption key needed to decrypt your secrets.
|
|
||||||
</Accordion>
|
|
||||||
</AccordionGroup>
|
|
281
docs/integrations/platforms/kubernetes-csi.mdx
Normal file
281
docs/integrations/platforms/kubernetes-csi.mdx
Normal file
@ -0,0 +1,281 @@
|
|||||||
|
---
|
||||||
|
title: "Kubernetes CSI"
|
||||||
|
description: "How to use Infisical to inject secrets directly into Kubernetes pods."
|
||||||
|
---
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
The Infisical CSI provider allows you to use Infisical with the [Secrets Store CSI driver](https://secrets-store-csi-driver.sigs.k8s.io) to inject secrets directly into your Kubernetes pods through a volume mount.
|
||||||
|
In contrast to the [Infisical Kubernetes Operator](https://infisical.com/docs/integrations/platforms/kubernetes), the Infisical CSI provider will allow you to sync Infisical secrets directly to pods as files, removing the need for Kubernetes secret resources.
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
flowchart LR
|
||||||
|
subgraph Secrets Management
|
||||||
|
SS(Infisical) --> CSP(Infisical CSI Provider)
|
||||||
|
CSP --> CSD(Secrets Store CSI Driver)
|
||||||
|
end
|
||||||
|
|
||||||
|
subgraph Application
|
||||||
|
CSD --> V(Volume)
|
||||||
|
V <--> P(Pod)
|
||||||
|
end
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
The following features are supported by the Infisical CSI Provider:
|
||||||
|
|
||||||
|
- Integration with Secrets Store CSI Driver for direct pod mounting
|
||||||
|
- Authentication using Kubernetes service accounts via machine identities
|
||||||
|
- Auto-syncing secrets when enabled via CSI Driver
|
||||||
|
- Configurable secret paths and file mounting locations
|
||||||
|
- Installation via Helm
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
The Infisical CSI provider is only supported for Kubernetes clusters with version >= 1.20.
|
||||||
|
|
||||||
|
## Limitations
|
||||||
|
|
||||||
|
Currently, the Infisical CSI provider only supports static secrets.
|
||||||
|
|
||||||
|
## Deploy to Kubernetes cluster
|
||||||
|
|
||||||
|
### Install Secrets Store CSI Driver
|
||||||
|
|
||||||
|
In order to use the Infisical CSI provider, you will first have to install the [Secrets Store CSI driver](https://secrets-store-csi-driver.sigs.k8s.io/getting-started/installation) to your cluster. It is important that you define
|
||||||
|
the audience value for token requests as demonstrated below. The Infisical CSI provider will **NOT WORK** if this is not set.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
helm repo add secrets-store-csi-driver https://kubernetes-sigs.github.io/secrets-store-csi-driver/charts
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
helm install csi secrets-store-csi-driver/secrets-store-csi-driver \
|
||||||
|
--namespace=kube-system \
|
||||||
|
--set "tokenRequests[0].audience=infisical" \
|
||||||
|
--set enableSecretRotation=true \
|
||||||
|
--set rotationPollInterval=2m \
|
||||||
|
--set "syncSecret.enabled=true" \
|
||||||
|
```
|
||||||
|
|
||||||
|
The flags configure the following:
|
||||||
|
|
||||||
|
- `tokenRequests[0].audience=infisical`: Sets the audience value for service account token authentication (required)
|
||||||
|
- `enableSecretRotation=true`: Enables automatic secret updates from Infisical
|
||||||
|
- `rotationPollInterval=2m`: Checks for secret updates every 2 minutes
|
||||||
|
- `syncSecret.enabled=true`: Enables syncing secrets to Kubernetes secrets
|
||||||
|
|
||||||
|
<Info>
|
||||||
|
If you do not wish to use the auto-syncing feature of the secrets store CSI
|
||||||
|
driver, you can omit the `enableSecretRotation` and the `rotationPollInterval`
|
||||||
|
flags. Do note that by default, secrets from Infisical are only fetched and
|
||||||
|
mounted during pod creation. If there are any changes made to the secrets in
|
||||||
|
Infisical, they will not propagate to the pods unless auto-syncing is enabled
|
||||||
|
for the CSI driver.
|
||||||
|
</Info>
|
||||||
|
|
||||||
|
### Install Infisical CSI Provider
|
||||||
|
|
||||||
|
You would then have to install the Infisical CSI provider to your cluster.
|
||||||
|
|
||||||
|
**Install the latest Infisical Helm repository**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
helm repo add infisical-helm-charts 'https://dl.cloudsmith.io/public/infisical/helm-charts/helm/charts/'
|
||||||
|
|
||||||
|
helm repo update
|
||||||
|
```
|
||||||
|
|
||||||
|
**Install the Helm Chart**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
helm install infisical-csi-provider infisical-helm-charts/infisical-csi-provider
|
||||||
|
```
|
||||||
|
|
||||||
|
For a list of all supported arguments for the helm installation, you can run the following:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
helm show values infisical-helm-charts/infisical-csi-provider
|
||||||
|
```
|
||||||
|
|
||||||
|
### Authentication
|
||||||
|
|
||||||
|
In order for the Infisical CSI provider to pull secrets from your Infisical project, you will have to configure
|
||||||
|
a machine identity with [Kubernetes authentication](https://infisical.com/docs/documentation/platform/identities/kubernetes-auth) configured with your cluster.
|
||||||
|
You can refer to the documentation for setting it up [here](https://infisical.com/docs/documentation/platform/identities/kubernetes-auth#guide).
|
||||||
|
|
||||||
|
<Warning>
|
||||||
|
The allowed audience field of the Kubernetes authentication settings should
|
||||||
|
match the audience specified for the Secrets Store CSI driver during
|
||||||
|
installation.
|
||||||
|
</Warning>
|
||||||
|
|
||||||
|
### Creating Secret Provider Class
|
||||||
|
|
||||||
|
With the Secrets Store CSI driver and the Infisical CSI provider installed, create a Kubernetes [SecretProviderClass](https://secrets-store-csi-driver.sigs.k8s.io/concepts.html#secretproviderclass) resource to establish
|
||||||
|
the connection between the CSI driver and the Infisical CSI provider for secret retrieval. You can create as many Secret Provider Classes as needed for your cluster.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
apiVersion: secrets-store.csi.x-k8s.io/v1
|
||||||
|
kind: SecretProviderClass
|
||||||
|
metadata:
|
||||||
|
name: my-infisical-app-csi-provider
|
||||||
|
spec:
|
||||||
|
provider: infisical
|
||||||
|
parameters:
|
||||||
|
infisicalUrl: "https://app.infisical.com"
|
||||||
|
authMethod: "kubernetes"
|
||||||
|
identityId: "ad2f8c67-cbe2-417a-b5eb-1339776ec0b3"
|
||||||
|
projectId: "09eda1f8-85a3-47a9-8a6f-e27f133b2a36"
|
||||||
|
envSlug: "prod"
|
||||||
|
secrets: |
|
||||||
|
- secretPath: "/"
|
||||||
|
fileName: "dbPassword"
|
||||||
|
secretKey: "DB_PASSWORD"
|
||||||
|
- secretPath: "/app"
|
||||||
|
fileName: "appSecret"
|
||||||
|
secretKey: "APP_SECRET"
|
||||||
|
```
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
The SecretProviderClass should be provisioned in the same namespace as the pod
|
||||||
|
you intend to mount secrets to.
|
||||||
|
</Note>
|
||||||
|
|
||||||
|
#### Supported Parameters
|
||||||
|
|
||||||
|
<Accordion title="infisicalUrl">
|
||||||
|
The base URL of your Infisical instance. If you're using Infisical Cloud US,
|
||||||
|
this should be set to `https://app.infisical.com`. If you're using Infisical
|
||||||
|
Cloud EU, then this should be set to `https://eu.infisical.com`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="caCertificate">
|
||||||
|
The CA certificate of the Infisical instance in order to establish SSL/TLS
|
||||||
|
when the instance uses a private or self-signed certificate. Unless necessary,
|
||||||
|
this should be omitted.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="authMethod">
|
||||||
|
The auth method to use for authenticating the Infisical CSI provider with
|
||||||
|
Infisical. For now, the only supported method is `kubernetes`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="identityId">
|
||||||
|
The ID of the machine identity to use for authenticating the Infisical CSI
|
||||||
|
provider with your Infisical organization. This should be the machine identity
|
||||||
|
configured with Kubernetes authentication.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="projectId">
|
||||||
|
The project ID of the Infisical project to pull secrets from.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="envSlug">
|
||||||
|
The slug of the project environment to pull secrets from.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="secrets">
|
||||||
|
An array that defines which secrets to retrieve and how to mount them. Each
|
||||||
|
entry requires three properties: `secretPath` and `secretKey` work together to
|
||||||
|
identify the source secret to fetch, while `fileName` specifies the path where
|
||||||
|
the secret's value will be mounted within the pod's filesystem.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Accordion title="audience">
|
||||||
|
The custom audience value configured for the CSI driver. This defaults to
|
||||||
|
`infisical`.
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
### Using Secret Provider Class
|
||||||
|
|
||||||
|
A pod can use the Secret Provider Class by mounting it as a CSI volume:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
apiVersion: v1
|
||||||
|
kind: Pod
|
||||||
|
metadata:
|
||||||
|
name: nginx-secrets-store
|
||||||
|
labels:
|
||||||
|
app: nginx
|
||||||
|
spec:
|
||||||
|
containers:
|
||||||
|
- name: nginx
|
||||||
|
image: nginx
|
||||||
|
volumeMounts:
|
||||||
|
- name: secrets-store-inline
|
||||||
|
mountPath: "/mnt/secrets-store"
|
||||||
|
readOnly: true
|
||||||
|
volumes:
|
||||||
|
- name: secrets-store-inline
|
||||||
|
csi:
|
||||||
|
driver: secrets-store.csi.k8s.io
|
||||||
|
readOnly: true
|
||||||
|
volumeAttributes:
|
||||||
|
secretProviderClass: "my-infisical-app-csi-provider"
|
||||||
|
```
|
||||||
|
|
||||||
|
When the pod is created, the secrets are mounted as individual files in the /mnt/secrets-store directory.
|
||||||
|
|
||||||
|
### Verifying Secret Mounts
|
||||||
|
|
||||||
|
To verify your secrets are mounted correctly:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check pod status
|
||||||
|
kubectl get pod nginx-secrets-store
|
||||||
|
|
||||||
|
# View mounted secrets
|
||||||
|
kubectl exec -it nginx-secrets-store -- ls -l /mnt/secrets-store
|
||||||
|
```
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
To troubleshoot issues with the Infisical CSI provider, refer to the logs of the Infisical CSI provider running on the same node as your pod.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
kubectl logs infisical-csi-provider-7x44t
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also refer to the logs of the secrets store CSI driver. Modify the command below with the appropriate pod and namespace of your secrets store CSI driver installation.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
kubectl logs csi-secrets-store-csi-driver-7h4jp -n=kube-system
|
||||||
|
```
|
||||||
|
|
||||||
|
**Common issues include:**
|
||||||
|
|
||||||
|
- Mismatch in the audience value of the CSI driver with the machine identity's Kubernetes auth configuration
|
||||||
|
- SecretProviderClass in the wrong namespace
|
||||||
|
- Invalid machine identity configuration
|
||||||
|
- Incorrect secret paths or keys
|
||||||
|
|
||||||
|
## Best Practices
|
||||||
|
|
||||||
|
For additional guidance on setting this up for your production cluster, you can refer to the Secrets Store CSI driver documentation [here](https://secrets-store-csi-driver.sigs.k8s.io/topics/best-practices).
|
||||||
|
|
||||||
|
## Frequently Asked Questions
|
||||||
|
|
||||||
|
<AccordionGroup>
|
||||||
|
<Accordion title="Is it possible to sync Infisical secrets as environment variables?">
|
||||||
|
Yes, but it requires an indirect approach:
|
||||||
|
|
||||||
|
1. First enable syncing to Kubernetes secrets by setting `syncSecret.enabled=true` in the CSI driver installation
|
||||||
|
2. Configure the Secret Provider Class to sync specific secrets to Kubernetes secrets
|
||||||
|
3. Use the resulting Kubernetes secrets in your pod's environment variables
|
||||||
|
|
||||||
|
This means secrets are first synced to Kubernetes secrets before they can be used as environment variables. You can find detailed examples in the [Secrets Store CSI driver documentation](https://secrets-store-csi-driver.sigs.k8s.io/topics/set-as-env-var).
|
||||||
|
|
||||||
|
</Accordion>
|
||||||
|
</AccordionGroup>
|
||||||
|
|
||||||
|
<AccordionGroup>
|
||||||
|
<Accordion title="Do I have to list out every Infisical single secret that I want to sync?">
|
||||||
|
Yes, you will need to explicitly list each secret you want to sync in the
|
||||||
|
Secret Provider Class configuration. This is a common requirement across all
|
||||||
|
CSI providers as the Secrets Store CSI Driver architecture requires specific
|
||||||
|
mapping of secrets to their mounted file locations.
|
||||||
|
</Accordion>
|
||||||
|
</AccordionGroup>
|
@ -316,6 +316,7 @@
|
|||||||
"cli/commands/init",
|
"cli/commands/init",
|
||||||
"cli/commands/run",
|
"cli/commands/run",
|
||||||
"cli/commands/secrets",
|
"cli/commands/secrets",
|
||||||
|
"cli/commands/dynamic-secrets",
|
||||||
"cli/commands/export",
|
"cli/commands/export",
|
||||||
"cli/commands/token",
|
"cli/commands/token",
|
||||||
"cli/commands/service-token",
|
"cli/commands/service-token",
|
||||||
@ -344,6 +345,7 @@
|
|||||||
"group": "Container orchestrators",
|
"group": "Container orchestrators",
|
||||||
"pages": [
|
"pages": [
|
||||||
"integrations/platforms/kubernetes",
|
"integrations/platforms/kubernetes",
|
||||||
|
"integrations/platforms/kubernetes-csi",
|
||||||
"integrations/platforms/docker-swarm-with-agent",
|
"integrations/platforms/docker-swarm-with-agent",
|
||||||
"integrations/platforms/ecs-with-agent"
|
"integrations/platforms/ecs-with-agent"
|
||||||
]
|
]
|
||||||
|
@ -1,11 +1,17 @@
|
|||||||
|
import { ParsedUrlQuery } from "querystring";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { faAngleRight, faLock } from "@fortawesome/free-solid-svg-icons";
|
import { faAngleRight, faCheck, faCopy, faLock } from "@fortawesome/free-solid-svg-icons";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
|
import { twMerge } from "tailwind-merge";
|
||||||
|
|
||||||
import { useOrganization, useWorkspace } from "@app/context";
|
import { useOrganization, useWorkspace } from "@app/context";
|
||||||
|
import { useToggle } from "@app/hooks";
|
||||||
|
|
||||||
import { Select, SelectItem, Tooltip } from "../v2";
|
import { createNotification } from "../notifications";
|
||||||
|
import { IconButton, Select, SelectItem, Tooltip } from "../v2";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
pageName: string;
|
pageName: string;
|
||||||
@ -50,6 +56,10 @@ export default function NavHeader({
|
|||||||
}: Props): JSX.Element {
|
}: Props): JSX.Element {
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace } = useWorkspace();
|
||||||
const { currentOrg } = useOrganization();
|
const { currentOrg } = useOrganization();
|
||||||
|
|
||||||
|
const [isCopied, { timedToggle: toggleIsCopied }] = useToggle(false);
|
||||||
|
const [isHoveringCopyButton, setIsHoveringCopyButton] = useState(false);
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const secretPathSegments = secretPath.split("/").filter(Boolean);
|
const secretPathSegments = secretPath.split("/").filter(Boolean);
|
||||||
@ -132,8 +142,10 @@ export default function NavHeader({
|
|||||||
)}
|
)}
|
||||||
{isFolderMode &&
|
{isFolderMode &&
|
||||||
secretPathSegments?.map((folderName, index) => {
|
secretPathSegments?.map((folderName, index) => {
|
||||||
const query = { ...router.query };
|
const query: ParsedUrlQuery & { secretPath: string } = {
|
||||||
query.secretPath = `/${secretPathSegments.slice(0, index + 1).join("/")}`;
|
...router.query,
|
||||||
|
secretPath: `/${secretPathSegments.slice(0, index + 1).join("/")}`
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -142,14 +154,59 @@ export default function NavHeader({
|
|||||||
>
|
>
|
||||||
<FontAwesomeIcon icon={faAngleRight} className="ml-3 mr-1.5 text-xs text-gray-400" />
|
<FontAwesomeIcon icon={faAngleRight} className="ml-3 mr-1.5 text-xs text-gray-400" />
|
||||||
{index + 1 === secretPathSegments?.length ? (
|
{index + 1 === secretPathSegments?.length ? (
|
||||||
<span className="text-sm font-semibold text-bunker-300">{folderName}</span>
|
<div className="flex items-center space-x-2">
|
||||||
|
<span
|
||||||
|
className={twMerge(
|
||||||
|
"text-sm font-semibold transition-all",
|
||||||
|
isHoveringCopyButton ? "text-bunker-200" : "text-bunker-300"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{folderName}
|
||||||
|
</span>
|
||||||
|
<Tooltip
|
||||||
|
className="relative right-2"
|
||||||
|
position="bottom"
|
||||||
|
content="Copy secret path"
|
||||||
|
>
|
||||||
|
<IconButton
|
||||||
|
variant="plain"
|
||||||
|
ariaLabel="copy"
|
||||||
|
onMouseEnter={() => setIsHoveringCopyButton(true)}
|
||||||
|
onMouseLeave={() => setIsHoveringCopyButton(false)}
|
||||||
|
onClick={() => {
|
||||||
|
if (isCopied) return;
|
||||||
|
|
||||||
|
navigator.clipboard.writeText(query.secretPath);
|
||||||
|
|
||||||
|
createNotification({
|
||||||
|
text: "Copied secret path to clipboard",
|
||||||
|
type: "info"
|
||||||
|
});
|
||||||
|
|
||||||
|
toggleIsCopied(2000);
|
||||||
|
}}
|
||||||
|
className="hover:bg-bunker-100/10"
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={!isCopied ? faCopy : faCheck}
|
||||||
|
size="sm"
|
||||||
|
className="cursor-pointer"
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<Link
|
<Link
|
||||||
passHref
|
passHref
|
||||||
legacyBehavior
|
legacyBehavior
|
||||||
href={{ pathname: "/project/[id]/secrets/[env]", query }}
|
href={{ pathname: "/project/[id]/secrets/[env]", query }}
|
||||||
>
|
>
|
||||||
<a className="text-sm font-semibold text-primary/80 hover:text-primary">
|
<a
|
||||||
|
className={twMerge(
|
||||||
|
"text-sm font-semibold transition-all hover:text-primary",
|
||||||
|
isHoveringCopyButton ? "text-primary" : "text-primary/80"
|
||||||
|
)}
|
||||||
|
>
|
||||||
{folderName}
|
{folderName}
|
||||||
</a>
|
</a>
|
||||||
</Link>
|
</Link>
|
||||||
|
@ -14,6 +14,7 @@ export type DatePickerProps = Omit<DayPickerProps, "selected"> & {
|
|||||||
onChange: (date?: Date) => void;
|
onChange: (date?: Date) => void;
|
||||||
popUpProps: PopoverProps;
|
popUpProps: PopoverProps;
|
||||||
popUpContentProps: PopoverContentProps;
|
popUpContentProps: PopoverContentProps;
|
||||||
|
dateFormat?: "PPP" | "PP" | "P"; // extend as needed
|
||||||
};
|
};
|
||||||
|
|
||||||
// Doc: https://react-day-picker.js.org/
|
// Doc: https://react-day-picker.js.org/
|
||||||
@ -22,6 +23,7 @@ export const DatePicker = ({
|
|||||||
onChange,
|
onChange,
|
||||||
popUpProps,
|
popUpProps,
|
||||||
popUpContentProps,
|
popUpContentProps,
|
||||||
|
dateFormat = "PPP",
|
||||||
...props
|
...props
|
||||||
}: DatePickerProps) => {
|
}: DatePickerProps) => {
|
||||||
const [timeValue, setTimeValue] = useState<string>(value ? format(value, "HH:mm") : "00:00");
|
const [timeValue, setTimeValue] = useState<string>(value ? format(value, "HH:mm") : "00:00");
|
||||||
@ -53,7 +55,7 @@ export const DatePicker = ({
|
|||||||
<Popover {...popUpProps}>
|
<Popover {...popUpProps}>
|
||||||
<PopoverTrigger asChild>
|
<PopoverTrigger asChild>
|
||||||
<Button variant="outline_bg" leftIcon={<FontAwesomeIcon icon={faCalendar} />}>
|
<Button variant="outline_bg" leftIcon={<FontAwesomeIcon icon={faCalendar} />}>
|
||||||
{value ? format(value, "PPP") : "Pick a date and time"}
|
{value ? format(value, dateFormat) : "Pick a date and time"}
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent className="w-fit p-2" {...popUpContentProps}>
|
<PopoverContent className="w-fit p-2" {...popUpContentProps}>
|
||||||
|
@ -3,7 +3,12 @@ import { twMerge } from "tailwind-merge";
|
|||||||
|
|
||||||
import { ClearIndicator, DropdownIndicator, MultiValueRemove, Option } from "../Select/components";
|
import { ClearIndicator, DropdownIndicator, MultiValueRemove, Option } from "../Select/components";
|
||||||
|
|
||||||
export const FilterableSelect = <T,>({ isMulti, closeMenuOnSelect, ...props }: Props<T>) => (
|
export const FilterableSelect = <T,>({
|
||||||
|
isMulti,
|
||||||
|
closeMenuOnSelect,
|
||||||
|
tabSelectsValue = false,
|
||||||
|
...props
|
||||||
|
}: Props<T>) => (
|
||||||
<Select
|
<Select
|
||||||
isMulti={isMulti}
|
isMulti={isMulti}
|
||||||
closeMenuOnSelect={closeMenuOnSelect ?? !isMulti}
|
closeMenuOnSelect={closeMenuOnSelect ?? !isMulti}
|
||||||
@ -26,7 +31,14 @@ export const FilterableSelect = <T,>({ isMulti, closeMenuOnSelect, ...props }: P
|
|||||||
transition: "none"
|
transition: "none"
|
||||||
})
|
})
|
||||||
}}
|
}}
|
||||||
components={{ DropdownIndicator, ClearIndicator, MultiValueRemove, Option }}
|
tabSelectsValue={tabSelectsValue}
|
||||||
|
components={{
|
||||||
|
DropdownIndicator,
|
||||||
|
ClearIndicator,
|
||||||
|
MultiValueRemove,
|
||||||
|
Option,
|
||||||
|
...props.components
|
||||||
|
}}
|
||||||
classNames={{
|
classNames={{
|
||||||
container: () => "w-full font-inter",
|
container: () => "w-full font-inter",
|
||||||
control: ({ isFocused }) =>
|
control: ({ isFocused }) =>
|
||||||
@ -46,14 +58,15 @@ export const FilterableSelect = <T,>({ isMulti, closeMenuOnSelect, ...props }: P
|
|||||||
clearIndicator: () => "p-1 hover:text-red text-bunker-400",
|
clearIndicator: () => "p-1 hover:text-red text-bunker-400",
|
||||||
indicatorSeparator: () => "bg-bunker-400",
|
indicatorSeparator: () => "bg-bunker-400",
|
||||||
dropdownIndicator: () => "text-bunker-200 p-1",
|
dropdownIndicator: () => "text-bunker-200 p-1",
|
||||||
|
menuList: () => "flex flex-col gap-1",
|
||||||
menu: () =>
|
menu: () =>
|
||||||
"mt-2 border text-sm text-mineshaft-200 bg-mineshaft-900 border-mineshaft-600 rounded-md",
|
"mt-2 p-2 border text-sm text-mineshaft-200 thin-scrollbar bg-mineshaft-900 border-mineshaft-600 rounded-md",
|
||||||
groupHeading: () => "ml-3 mt-2 mb-1 text-mineshaft-400 text-sm",
|
groupHeading: () => "ml-3 mt-2 mb-1 text-mineshaft-400 text-sm",
|
||||||
option: ({ isFocused, isSelected }) =>
|
option: ({ isFocused, isSelected }) =>
|
||||||
twMerge(
|
twMerge(
|
||||||
isFocused && "bg-mineshaft-700 active:bg-mineshaft-600",
|
isFocused && "bg-mineshaft-700 active:bg-mineshaft-600",
|
||||||
isSelected && "text-mineshaft-200",
|
isSelected && "text-mineshaft-200",
|
||||||
"hover:cursor-pointer text-xs px-3 py-2"
|
"hover:cursor-pointer rounded text-xs px-3 py-2"
|
||||||
),
|
),
|
||||||
noOptionsMessage: () => "text-mineshaft-400 p-2 rounded-md"
|
noOptionsMessage: () => "text-mineshaft-400 p-2 rounded-md"
|
||||||
}}
|
}}
|
||||||
|
@ -54,7 +54,7 @@ export const Pagination = ({
|
|||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
{startAdornment}
|
{startAdornment}
|
||||||
<div className="ml-auto mr-6 flex items-center space-x-2">
|
<div className={twMerge("mr-4 flex items-center space-x-2", startAdornment && "ml-auto")}>
|
||||||
<div className="text-xs">
|
<div className="text-xs">
|
||||||
{(page - 1) * perPage + 1} - {Math.min((page - 1) * perPage + perPage, count)} of {count}
|
{(page - 1) * perPage + 1} - {Math.min((page - 1) * perPage + perPage, count)} of {count}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { ProjectMembershipRole } from "@app/hooks/api/roles/types";
|
import { ProjectMembershipRole, TOrgRole } from "@app/hooks/api/roles/types";
|
||||||
|
|
||||||
enum OrgMembershipRole {
|
enum OrgMembershipRole {
|
||||||
Admin = "admin",
|
Admin = "admin",
|
||||||
@ -23,3 +23,8 @@ export const formatProjectRoleName = (name: string) => {
|
|||||||
|
|
||||||
export const isCustomProjectRole = (slug: string) =>
|
export const isCustomProjectRole = (slug: string) =>
|
||||||
!Object.values(ProjectMembershipRole).includes(slug as ProjectMembershipRole);
|
!Object.values(ProjectMembershipRole).includes(slug as ProjectMembershipRole);
|
||||||
|
|
||||||
|
export const findOrgMembershipRole = (roles: TOrgRole[], roleIdOrSlug: string) =>
|
||||||
|
isCustomOrgRole(roleIdOrSlug)
|
||||||
|
? roles.find((r) => r.id === roleIdOrSlug)
|
||||||
|
: roles.find((r) => r.slug === roleIdOrSlug);
|
||||||
|
@ -10,7 +10,6 @@ import { useTranslation } from "react-i18next";
|
|||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { faGithub, faSlack } from "@fortawesome/free-brands-svg-icons";
|
import { faGithub, faSlack } from "@fortawesome/free-brands-svg-icons";
|
||||||
import { faStar } from "@fortawesome/free-regular-svg-icons";
|
|
||||||
import {
|
import {
|
||||||
faAngleDown,
|
faAngleDown,
|
||||||
faArrowLeft,
|
faArrowLeft,
|
||||||
@ -22,15 +21,11 @@ import {
|
|||||||
faInfo,
|
faInfo,
|
||||||
faMobile,
|
faMobile,
|
||||||
faPlus,
|
faPlus,
|
||||||
faQuestion,
|
faQuestion
|
||||||
faStar as faSolidStar
|
|
||||||
} from "@fortawesome/free-solid-svg-icons";
|
} from "@fortawesome/free-solid-svg-icons";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { DropdownMenuTrigger } from "@radix-ui/react-dropdown-menu";
|
import { DropdownMenuTrigger } from "@radix-ui/react-dropdown-menu";
|
||||||
import { twMerge } from "tailwind-merge";
|
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
|
||||||
import { OrgPermissionCan } from "@app/components/permissions";
|
|
||||||
import { tempLocalStorage } from "@app/components/utilities/checks/tempLocalStorage";
|
import { tempLocalStorage } from "@app/components/utilities/checks/tempLocalStorage";
|
||||||
import SecurityClient from "@app/components/utilities/SecurityClient";
|
import SecurityClient from "@app/components/utilities/SecurityClient";
|
||||||
import {
|
import {
|
||||||
@ -39,20 +34,9 @@ import {
|
|||||||
DropdownMenuContent,
|
DropdownMenuContent,
|
||||||
DropdownMenuItem,
|
DropdownMenuItem,
|
||||||
Menu,
|
Menu,
|
||||||
MenuItem,
|
MenuItem
|
||||||
Select,
|
|
||||||
SelectItem,
|
|
||||||
UpgradePlanModal
|
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { NewProjectModal } from "@app/components/v2/projects/NewProjectModal";
|
import { useOrganization, useSubscription, useUser, useWorkspace } from "@app/context";
|
||||||
import {
|
|
||||||
OrgPermissionActions,
|
|
||||||
OrgPermissionSubjects,
|
|
||||||
useOrganization,
|
|
||||||
useSubscription,
|
|
||||||
useUser,
|
|
||||||
useWorkspace
|
|
||||||
} from "@app/context";
|
|
||||||
import { usePopUp, useToggle } from "@app/hooks";
|
import { usePopUp, useToggle } from "@app/hooks";
|
||||||
import {
|
import {
|
||||||
useGetAccessRequestsCount,
|
useGetAccessRequestsCount,
|
||||||
@ -62,11 +46,9 @@ import {
|
|||||||
useSelectOrganization
|
useSelectOrganization
|
||||||
} from "@app/hooks/api";
|
} from "@app/hooks/api";
|
||||||
import { MfaMethod } from "@app/hooks/api/auth/types";
|
import { MfaMethod } from "@app/hooks/api/auth/types";
|
||||||
import { Workspace } from "@app/hooks/api/types";
|
|
||||||
import { useUpdateUserProjectFavorites } from "@app/hooks/api/users/mutation";
|
|
||||||
import { useGetUserProjectFavorites } from "@app/hooks/api/users/queries";
|
|
||||||
import { AuthMethod } from "@app/hooks/api/users/types";
|
import { AuthMethod } from "@app/hooks/api/users/types";
|
||||||
import { InsecureConnectionBanner } from "@app/layouts/AppLayout/components/InsecureConnectionBanner";
|
import { InsecureConnectionBanner } from "@app/layouts/AppLayout/components/InsecureConnectionBanner";
|
||||||
|
import { ProjectSelect } from "@app/layouts/AppLayout/components/ProjectSelect";
|
||||||
import { navigateUserToOrg } from "@app/views/Login/Login.utils";
|
import { navigateUserToOrg } from "@app/views/Login/Login.utils";
|
||||||
import { Mfa } from "@app/views/Login/Mfa";
|
import { Mfa } from "@app/views/Login/Mfa";
|
||||||
import { CreateOrgModal } from "@app/views/Org/components";
|
import { CreateOrgModal } from "@app/views/Org/components";
|
||||||
@ -108,23 +90,10 @@ export const AppLayout = ({ children }: LayoutProps) => {
|
|||||||
const { workspaces, currentWorkspace } = useWorkspace();
|
const { workspaces, currentWorkspace } = useWorkspace();
|
||||||
const { orgs, currentOrg } = useOrganization();
|
const { orgs, currentOrg } = useOrganization();
|
||||||
|
|
||||||
const { data: projectFavorites } = useGetUserProjectFavorites(currentOrg?.id!);
|
|
||||||
const { mutateAsync: updateUserProjectFavorites } = useUpdateUserProjectFavorites();
|
|
||||||
const [shouldShowMfa, toggleShowMfa] = useToggle(false);
|
const [shouldShowMfa, toggleShowMfa] = useToggle(false);
|
||||||
const [requiredMfaMethod, setRequiredMfaMethod] = useState(MfaMethod.EMAIL);
|
const [requiredMfaMethod, setRequiredMfaMethod] = useState(MfaMethod.EMAIL);
|
||||||
const [mfaSuccessCallback, setMfaSuccessCallback] = useState<() => void>(() => {});
|
const [mfaSuccessCallback, setMfaSuccessCallback] = useState<() => void>(() => {});
|
||||||
|
|
||||||
const workspacesWithFaveProp = useMemo(
|
|
||||||
() =>
|
|
||||||
workspaces
|
|
||||||
.map((w): Workspace & { isFavorite: boolean } => ({
|
|
||||||
...w,
|
|
||||||
isFavorite: Boolean(projectFavorites?.includes(w.id))
|
|
||||||
}))
|
|
||||||
.sort((a, b) => Number(b.isFavorite) - Number(a.isFavorite)),
|
|
||||||
[workspaces, projectFavorites]
|
|
||||||
);
|
|
||||||
|
|
||||||
const { user } = useUser();
|
const { user } = useUser();
|
||||||
const { subscription } = useSubscription();
|
const { subscription } = useSubscription();
|
||||||
const workspaceId = currentWorkspace?.id || "";
|
const workspaceId = currentWorkspace?.id || "";
|
||||||
@ -137,17 +106,9 @@ export const AppLayout = ({ children }: LayoutProps) => {
|
|||||||
return (secretApprovalReqCount?.open || 0) + (accessApprovalRequestCount?.pendingCount || 0);
|
return (secretApprovalReqCount?.open || 0) + (accessApprovalRequestCount?.pendingCount || 0);
|
||||||
}, [secretApprovalReqCount, accessApprovalRequestCount]);
|
}, [secretApprovalReqCount, accessApprovalRequestCount]);
|
||||||
|
|
||||||
const isAddingProjectsAllowed = subscription?.workspaceLimit
|
|
||||||
? subscription.workspacesUsed < subscription.workspaceLimit
|
|
||||||
: true;
|
|
||||||
|
|
||||||
const infisicalPlatformVersion = process.env.NEXT_PUBLIC_INFISICAL_PLATFORM_VERSION;
|
const infisicalPlatformVersion = process.env.NEXT_PUBLIC_INFISICAL_PLATFORM_VERSION;
|
||||||
|
|
||||||
const { popUp, handlePopUpOpen, handlePopUpToggle } = usePopUp([
|
const { popUp, handlePopUpToggle } = usePopUp(["createOrg"] as const);
|
||||||
"addNewWs",
|
|
||||||
"upgradePlan",
|
|
||||||
"createOrg"
|
|
||||||
] as const);
|
|
||||||
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
@ -230,38 +191,6 @@ export const AppLayout = ({ children }: LayoutProps) => {
|
|||||||
putUserInOrg();
|
putUserInOrg();
|
||||||
}, [router.query.id]);
|
}, [router.query.id]);
|
||||||
|
|
||||||
const addProjectToFavorites = async (projectId: string) => {
|
|
||||||
try {
|
|
||||||
if (currentOrg?.id) {
|
|
||||||
await updateUserProjectFavorites({
|
|
||||||
orgId: currentOrg?.id,
|
|
||||||
projectFavorites: [...(projectFavorites || []), projectId]
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
createNotification({
|
|
||||||
text: "Failed to add project to favorites.",
|
|
||||||
type: "error"
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const removeProjectFromFavorites = async (projectId: string) => {
|
|
||||||
try {
|
|
||||||
if (currentOrg?.id) {
|
|
||||||
await updateUserProjectFavorites({
|
|
||||||
orgId: currentOrg?.id,
|
|
||||||
projectFavorites: [...(projectFavorites || []).filter((entry) => entry !== projectId)]
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
createNotification({
|
|
||||||
text: "Failed to remove project from favorites.",
|
|
||||||
type: "error"
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (shouldShowMfa) {
|
if (shouldShowMfa) {
|
||||||
return (
|
return (
|
||||||
<div className="flex max-h-screen min-h-screen flex-col items-center justify-center gap-2 overflow-y-auto bg-gradient-to-tr from-mineshaft-600 via-mineshaft-800 to-bunker-700">
|
<div className="flex max-h-screen min-h-screen flex-col items-center justify-center gap-2 overflow-y-auto bg-gradient-to-tr from-mineshaft-600 via-mineshaft-800 to-bunker-700">
|
||||||
@ -448,97 +377,7 @@ export const AppLayout = ({ children }: LayoutProps) => {
|
|||||||
)}
|
)}
|
||||||
{!router.asPath.includes("org") &&
|
{!router.asPath.includes("org") &&
|
||||||
(!router.asPath.includes("personal") && currentWorkspace ? (
|
(!router.asPath.includes("personal") && currentWorkspace ? (
|
||||||
<div className="mt-5 mb-4 w-full p-3">
|
<ProjectSelect />
|
||||||
<p className="ml-1.5 mb-1 text-xs font-semibold uppercase text-gray-400">
|
|
||||||
Project
|
|
||||||
</p>
|
|
||||||
<Select
|
|
||||||
defaultValue={currentWorkspace?.id}
|
|
||||||
value={currentWorkspace?.id}
|
|
||||||
className="w-full bg-mineshaft-600 py-2.5 font-medium [&>*:first-child]:truncate"
|
|
||||||
onValueChange={(value) => {
|
|
||||||
localStorage.setItem("projectData.id", value);
|
|
||||||
// this is not using react query because react query in overview is throwing error when envs are not exact same count
|
|
||||||
// to reproduce change this back to router.push and switch between two projects with different env count
|
|
||||||
// look into this on dashboard revamp
|
|
||||||
window.location.assign(`/project/${value}/secrets/overview`);
|
|
||||||
}}
|
|
||||||
position="popper"
|
|
||||||
dropdownContainerClassName="text-bunker-200 bg-mineshaft-800 border border-mineshaft-600 z-50 max-h-96 border-gray-700"
|
|
||||||
>
|
|
||||||
<div className="no-scrollbar::-webkit-scrollbar h-full no-scrollbar">
|
|
||||||
{workspacesWithFaveProp
|
|
||||||
.filter((ws) => ws.orgId === currentOrg?.id)
|
|
||||||
.map(({ id, name, isFavorite }) => (
|
|
||||||
<div
|
|
||||||
className={twMerge(
|
|
||||||
"mb-1 grid grid-cols-7 rounded-md hover:bg-mineshaft-500",
|
|
||||||
id === currentWorkspace?.id && "bg-mineshaft-500"
|
|
||||||
)}
|
|
||||||
key={id}
|
|
||||||
>
|
|
||||||
<div className="col-span-6">
|
|
||||||
<SelectItem
|
|
||||||
key={`ws-layout-list-${id}`}
|
|
||||||
value={id}
|
|
||||||
className="transition-none data-[highlighted]:bg-mineshaft-500"
|
|
||||||
>
|
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
</div>
|
|
||||||
<div className="col-span-1 flex items-center">
|
|
||||||
{isFavorite ? (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faSolidStar}
|
|
||||||
className="text-sm text-mineshaft-300 hover:text-mineshaft-400"
|
|
||||||
onClick={(e) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
removeProjectFromFavorites(id);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faStar}
|
|
||||||
className="text-sm text-mineshaft-400 hover:text-mineshaft-300"
|
|
||||||
onClick={(e) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
addProjectToFavorites(id);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
<hr className="mt-1 mb-1 h-px border-0 bg-gray-700" />
|
|
||||||
<div className="w-full">
|
|
||||||
<OrgPermissionCan
|
|
||||||
I={OrgPermissionActions.Create}
|
|
||||||
a={OrgPermissionSubjects.Workspace}
|
|
||||||
>
|
|
||||||
{(isAllowed) => (
|
|
||||||
<Button
|
|
||||||
className="w-full bg-mineshaft-700 py-2 text-bunker-200"
|
|
||||||
colorSchema="primary"
|
|
||||||
variant="outline_bg"
|
|
||||||
size="sm"
|
|
||||||
isDisabled={!isAllowed}
|
|
||||||
onClick={() => {
|
|
||||||
if (isAddingProjectsAllowed) {
|
|
||||||
handlePopUpOpen("addNewWs");
|
|
||||||
} else {
|
|
||||||
handlePopUpOpen("upgradePlan");
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
leftIcon={<FontAwesomeIcon icon={faPlus} />}
|
|
||||||
>
|
|
||||||
Add Project
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</OrgPermissionCan>
|
|
||||||
</div>
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
) : (
|
) : (
|
||||||
<Link href={`/org/${currentOrg?.id}/overview`}>
|
<Link href={`/org/${currentOrg?.id}/overview`}>
|
||||||
<div className="my-6 flex cursor-default items-center justify-center pr-2 text-sm text-mineshaft-300 hover:text-mineshaft-100">
|
<div className="my-6 flex cursor-default items-center justify-center pr-2 text-sm text-mineshaft-300 hover:text-mineshaft-100">
|
||||||
@ -816,15 +655,6 @@ export const AppLayout = ({ children }: LayoutProps) => {
|
|||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
</aside>
|
</aside>
|
||||||
<NewProjectModal
|
|
||||||
isOpen={popUp.addNewWs.isOpen}
|
|
||||||
onOpenChange={(isOpen) => handlePopUpToggle("addNewWs", isOpen)}
|
|
||||||
/>
|
|
||||||
<UpgradePlanModal
|
|
||||||
isOpen={popUp.upgradePlan.isOpen}
|
|
||||||
onOpenChange={(isOpen) => handlePopUpToggle("upgradePlan", isOpen)}
|
|
||||||
text="You have exceeded the number of projects allowed on the free plan."
|
|
||||||
/>
|
|
||||||
<CreateOrgModal
|
<CreateOrgModal
|
||||||
isOpen={popUp?.createOrg?.isOpen}
|
isOpen={popUp?.createOrg?.isOpen}
|
||||||
onClose={() => handlePopUpToggle("createOrg", false)}
|
onClose={() => handlePopUpToggle("createOrg", false)}
|
||||||
|
@ -0,0 +1,212 @@
|
|||||||
|
import { useMemo } from "react";
|
||||||
|
import { components, MenuProps, OptionProps } from "react-select";
|
||||||
|
import { faStar } from "@fortawesome/free-regular-svg-icons";
|
||||||
|
import { faChevronRight, faPlus, faStar as faSolidStar } from "@fortawesome/free-solid-svg-icons";
|
||||||
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
|
import { twMerge } from "tailwind-merge";
|
||||||
|
|
||||||
|
import { createNotification } from "@app/components/notifications";
|
||||||
|
import { OrgPermissionCan } from "@app/components/permissions";
|
||||||
|
import { Button, FilterableSelect, UpgradePlanModal } from "@app/components/v2";
|
||||||
|
import { NewProjectModal } from "@app/components/v2/projects";
|
||||||
|
import {
|
||||||
|
OrgPermissionActions,
|
||||||
|
OrgPermissionSubjects,
|
||||||
|
useOrganization,
|
||||||
|
useSubscription,
|
||||||
|
useWorkspace
|
||||||
|
} from "@app/context";
|
||||||
|
import { usePopUp } from "@app/hooks";
|
||||||
|
import { useUpdateUserProjectFavorites } from "@app/hooks/api/users/mutation";
|
||||||
|
import { useGetUserProjectFavorites } from "@app/hooks/api/users/queries";
|
||||||
|
import { Workspace } from "@app/hooks/api/workspace/types";
|
||||||
|
|
||||||
|
type TWorkspaceWithFaveProp = Workspace & { isFavorite: boolean };
|
||||||
|
|
||||||
|
const ProjectsMenu = ({ children, ...props }: MenuProps<TWorkspaceWithFaveProp>) => {
|
||||||
|
return (
|
||||||
|
<components.Menu {...props}>
|
||||||
|
{children}
|
||||||
|
<hr className="mb-2 h-px border-0 bg-mineshaft-500" />
|
||||||
|
<OrgPermissionCan I={OrgPermissionActions.Create} a={OrgPermissionSubjects.Workspace}>
|
||||||
|
{(isAllowed) => (
|
||||||
|
<Button
|
||||||
|
className="w-full bg-mineshaft-700 pt-2 text-bunker-200"
|
||||||
|
colorSchema="primary"
|
||||||
|
variant="outline_bg"
|
||||||
|
size="xs"
|
||||||
|
isDisabled={!isAllowed}
|
||||||
|
onClick={() => props.clearValue()}
|
||||||
|
leftIcon={<FontAwesomeIcon icon={faPlus} />}
|
||||||
|
>
|
||||||
|
Add Project
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</OrgPermissionCan>
|
||||||
|
</components.Menu>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const ProjectOption = ({
|
||||||
|
isSelected,
|
||||||
|
children,
|
||||||
|
data,
|
||||||
|
...props
|
||||||
|
}: OptionProps<TWorkspaceWithFaveProp>) => {
|
||||||
|
const { currentOrg } = useOrganization();
|
||||||
|
const { mutateAsync: updateUserProjectFavorites } = useUpdateUserProjectFavorites();
|
||||||
|
const { data: projectFavorites } = useGetUserProjectFavorites(currentOrg?.id!);
|
||||||
|
|
||||||
|
const removeProjectFromFavorites = async (projectId: string) => {
|
||||||
|
try {
|
||||||
|
await updateUserProjectFavorites({
|
||||||
|
orgId: currentOrg!.id,
|
||||||
|
projectFavorites: [...(projectFavorites || []).filter((entry) => entry !== projectId)]
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
createNotification({
|
||||||
|
text: "Failed to remove project from favorites.",
|
||||||
|
type: "error"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const addProjectToFavorites = async (projectId: string) => {
|
||||||
|
try {
|
||||||
|
await updateUserProjectFavorites({
|
||||||
|
orgId: currentOrg!.id,
|
||||||
|
projectFavorites: [...(projectFavorites || []), projectId]
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
createNotification({
|
||||||
|
text: "Failed to add project to favorites.",
|
||||||
|
type: "error"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<components.Option
|
||||||
|
isSelected={isSelected}
|
||||||
|
data={data}
|
||||||
|
{...props}
|
||||||
|
className={twMerge(props.className, isSelected && "bg-mineshaft-500")}
|
||||||
|
>
|
||||||
|
<div className="flex w-full items-center">
|
||||||
|
{isSelected && (
|
||||||
|
<FontAwesomeIcon className="mr-2 text-primary" icon={faChevronRight} size="xs" />
|
||||||
|
)}
|
||||||
|
<p className="truncate">{children}</p>
|
||||||
|
{data.isFavorite ? (
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={faSolidStar}
|
||||||
|
className="ml-auto text-sm text-yellow-600 hover:text-mineshaft-400"
|
||||||
|
onClick={async (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
await removeProjectFromFavorites(data.id);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={faStar}
|
||||||
|
className="ml-auto text-sm text-mineshaft-400 hover:text-mineshaft-300"
|
||||||
|
onClick={async (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
await addProjectToFavorites(data.id);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</components.Option>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ProjectSelect = () => {
|
||||||
|
const { workspaces, currentWorkspace } = useWorkspace();
|
||||||
|
const { currentOrg } = useOrganization();
|
||||||
|
const { data: projectFavorites } = useGetUserProjectFavorites(currentOrg?.id!);
|
||||||
|
|
||||||
|
const { subscription } = useSubscription();
|
||||||
|
|
||||||
|
const isAddingProjectsAllowed = subscription?.workspaceLimit
|
||||||
|
? subscription.workspacesUsed < subscription.workspaceLimit
|
||||||
|
: true;
|
||||||
|
|
||||||
|
const { popUp, handlePopUpOpen, handlePopUpToggle } = usePopUp([
|
||||||
|
"addNewWs",
|
||||||
|
"upgradePlan"
|
||||||
|
] as const);
|
||||||
|
|
||||||
|
const { options, value } = useMemo(() => {
|
||||||
|
const projectOptions = workspaces
|
||||||
|
.map((w): Workspace & { isFavorite: boolean } => ({
|
||||||
|
...w,
|
||||||
|
isFavorite: Boolean(projectFavorites?.includes(w.id))
|
||||||
|
}))
|
||||||
|
.sort((a, b) => Number(b.isFavorite) - Number(a.isFavorite));
|
||||||
|
|
||||||
|
const currentOption = projectOptions.find((option) => option.id === currentWorkspace?.id);
|
||||||
|
|
||||||
|
if (!currentOption) {
|
||||||
|
return {
|
||||||
|
options: projectOptions,
|
||||||
|
value: null
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
options: [
|
||||||
|
currentOption,
|
||||||
|
...projectOptions.filter((option) => option.id !== currentOption.id)
|
||||||
|
],
|
||||||
|
value: currentOption
|
||||||
|
};
|
||||||
|
}, [workspaces, projectFavorites, currentWorkspace]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mt-5 mb-4 w-full p-3">
|
||||||
|
<p className="ml-1.5 mb-1 text-xs font-semibold uppercase text-gray-400">Project</p>
|
||||||
|
<FilterableSelect
|
||||||
|
className="text-sm"
|
||||||
|
value={value}
|
||||||
|
filterOption={(option, inputValue) =>
|
||||||
|
option.data.name.toLowerCase().includes(inputValue.toLowerCase())
|
||||||
|
}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
getOptionValue={(option) => option.id}
|
||||||
|
onChange={(newValue) => {
|
||||||
|
// hacky use of null as indication to create project
|
||||||
|
if (!newValue) {
|
||||||
|
if (isAddingProjectsAllowed) {
|
||||||
|
handlePopUpOpen("addNewWs");
|
||||||
|
} else {
|
||||||
|
handlePopUpOpen("upgradePlan");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const project = newValue as TWorkspaceWithFaveProp;
|
||||||
|
localStorage.setItem("projectData.id", project.id);
|
||||||
|
// todo(akhi): this is not using react query because react query in overview is throwing error when envs are not exact same count
|
||||||
|
// to reproduce change this back to router.push and switch between two projects with different env count
|
||||||
|
// look into this on dashboard revamp
|
||||||
|
window.location.assign(`/project/${project.id}/secrets/overview`);
|
||||||
|
}}
|
||||||
|
options={options}
|
||||||
|
components={{
|
||||||
|
Option: ProjectOption,
|
||||||
|
Menu: ProjectsMenu
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<UpgradePlanModal
|
||||||
|
isOpen={popUp.upgradePlan.isOpen}
|
||||||
|
onOpenChange={(isOpen) => handlePopUpToggle("upgradePlan", isOpen)}
|
||||||
|
text="You have exceeded the number of projects allowed on the free plan."
|
||||||
|
/>
|
||||||
|
|
||||||
|
<NewProjectModal
|
||||||
|
isOpen={popUp.addNewWs.isOpen}
|
||||||
|
onOpenChange={(isOpen) => handlePopUpToggle("addNewWs", isOpen)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -0,0 +1 @@
|
|||||||
|
export * from "./ProjectSelect";
|
@ -1,6 +1,6 @@
|
|||||||
// REFACTOR(akhilmhdh): This file needs to be split into multiple components too complex
|
// REFACTOR(akhilmhdh): This file needs to be split into multiple components too complex
|
||||||
|
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { ReactNode, useEffect, useMemo, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
@ -9,20 +9,22 @@ import { IconProp } from "@fortawesome/fontawesome-svg-core";
|
|||||||
import { faSlack } from "@fortawesome/free-brands-svg-icons";
|
import { faSlack } from "@fortawesome/free-brands-svg-icons";
|
||||||
import { faFolderOpen, faStar } from "@fortawesome/free-regular-svg-icons";
|
import { faFolderOpen, faStar } from "@fortawesome/free-regular-svg-icons";
|
||||||
import {
|
import {
|
||||||
|
faArrowDownAZ,
|
||||||
faArrowRight,
|
faArrowRight,
|
||||||
faArrowUpRightFromSquare,
|
faArrowUpRightFromSquare,
|
||||||
|
faArrowUpZA,
|
||||||
faBorderAll,
|
faBorderAll,
|
||||||
faCheck,
|
faCheck,
|
||||||
faCheckCircle,
|
faCheckCircle,
|
||||||
faClipboard,
|
faClipboard,
|
||||||
faExclamationCircle,
|
faExclamationCircle,
|
||||||
faFileShield,
|
|
||||||
faHandPeace,
|
faHandPeace,
|
||||||
faList,
|
faList,
|
||||||
faMagnifyingGlass,
|
faMagnifyingGlass,
|
||||||
faNetworkWired,
|
faNetworkWired,
|
||||||
faPlug,
|
faPlug,
|
||||||
faPlus,
|
faPlus,
|
||||||
|
faSearch,
|
||||||
faStar as faSolidStar,
|
faStar as faSolidStar,
|
||||||
faUserPlus
|
faUserPlus
|
||||||
} from "@fortawesome/free-solid-svg-icons";
|
} from "@fortawesome/free-solid-svg-icons";
|
||||||
@ -32,7 +34,15 @@ import * as Tabs from "@radix-ui/react-tabs";
|
|||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import { OrgPermissionCan } from "@app/components/permissions";
|
import { OrgPermissionCan } from "@app/components/permissions";
|
||||||
import onboardingCheck from "@app/components/utilities/checks/OnboardingCheck";
|
import onboardingCheck from "@app/components/utilities/checks/OnboardingCheck";
|
||||||
import { Button, IconButton, Input, Skeleton, UpgradePlanModal } from "@app/components/v2";
|
import {
|
||||||
|
Button,
|
||||||
|
IconButton,
|
||||||
|
Input,
|
||||||
|
Pagination,
|
||||||
|
Skeleton,
|
||||||
|
Tooltip,
|
||||||
|
UpgradePlanModal
|
||||||
|
} from "@app/components/v2";
|
||||||
import { NewProjectModal } from "@app/components/v2/projects";
|
import { NewProjectModal } from "@app/components/v2/projects";
|
||||||
import {
|
import {
|
||||||
OrgPermissionActions,
|
OrgPermissionActions,
|
||||||
@ -42,7 +52,9 @@ import {
|
|||||||
useUser,
|
useUser,
|
||||||
useWorkspace
|
useWorkspace
|
||||||
} from "@app/context";
|
} from "@app/context";
|
||||||
|
import { usePagination, useResetPageHelper } from "@app/hooks";
|
||||||
import { useRegisterUserAction } from "@app/hooks/api";
|
import { useRegisterUserAction } from "@app/hooks/api";
|
||||||
|
import { OrderByDirection } from "@app/hooks/api/generic/types";
|
||||||
// import { fetchUserWsKey } from "@app/hooks/api/keys/queries";
|
// import { fetchUserWsKey } from "@app/hooks/api/keys/queries";
|
||||||
import { useFetchServerStatus } from "@app/hooks/api/serverDetails";
|
import { useFetchServerStatus } from "@app/hooks/api/serverDetails";
|
||||||
import { Workspace } from "@app/hooks/api/types";
|
import { Workspace } from "@app/hooks/api/types";
|
||||||
@ -81,6 +93,10 @@ enum ProjectsViewMode {
|
|||||||
LIST = "list"
|
LIST = "list"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum ProjectOrderBy {
|
||||||
|
Name = "name"
|
||||||
|
}
|
||||||
|
|
||||||
function copyToClipboard(id: string, setState: (value: boolean) => void) {
|
function copyToClipboard(id: string, setState: (value: boolean) => void) {
|
||||||
// Get the text field
|
// Get the text field
|
||||||
const copyText = document.getElementById(id) as HTMLInputElement;
|
const copyText = document.getElementById(id) as HTMLInputElement;
|
||||||
@ -496,26 +512,48 @@ const OrganizationPage = () => {
|
|||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const isWorkspaceEmpty = !isWorkspaceLoading && orgWorkspaces?.length === 0;
|
const isWorkspaceEmpty = !isProjectViewLoading && orgWorkspaces?.length === 0;
|
||||||
const filteredWorkspaces = orgWorkspaces.filter((ws) =>
|
|
||||||
ws?.name?.toLowerCase().includes(searchFilter.toLowerCase())
|
const {
|
||||||
|
setPage,
|
||||||
|
perPage,
|
||||||
|
setPerPage,
|
||||||
|
page,
|
||||||
|
offset,
|
||||||
|
limit,
|
||||||
|
toggleOrderDirection,
|
||||||
|
orderDirection
|
||||||
|
} = usePagination(ProjectOrderBy.Name, { initPerPage: 24 });
|
||||||
|
|
||||||
|
const filteredWorkspaces = useMemo(
|
||||||
|
() =>
|
||||||
|
orgWorkspaces
|
||||||
|
.filter((ws) => ws?.name?.toLowerCase().includes(searchFilter.toLowerCase()))
|
||||||
|
.sort((a, b) =>
|
||||||
|
orderDirection === OrderByDirection.ASC
|
||||||
|
? a.name.toLowerCase().localeCompare(b.name.toLowerCase())
|
||||||
|
: b.name.toLowerCase().localeCompare(a.name.toLowerCase())
|
||||||
|
),
|
||||||
|
[searchFilter, page, perPage, orderDirection, offset, limit]
|
||||||
);
|
);
|
||||||
|
|
||||||
const { workspacesWithFaveProp, favoriteWorkspaces, nonFavoriteWorkspaces } = useMemo(() => {
|
useResetPageHelper({
|
||||||
|
setPage,
|
||||||
|
offset,
|
||||||
|
totalCount: filteredWorkspaces.length
|
||||||
|
});
|
||||||
|
|
||||||
|
const { workspacesWithFaveProp } = useMemo(() => {
|
||||||
const workspacesWithFav = filteredWorkspaces
|
const workspacesWithFav = filteredWorkspaces
|
||||||
.map((w): Workspace & { isFavorite: boolean } => ({
|
.map((w): Workspace & { isFavorite: boolean } => ({
|
||||||
...w,
|
...w,
|
||||||
isFavorite: Boolean(projectFavorites?.includes(w.id))
|
isFavorite: Boolean(projectFavorites?.includes(w.id))
|
||||||
}))
|
}))
|
||||||
.sort((a, b) => Number(b.isFavorite) - Number(a.isFavorite));
|
.sort((a, b) => Number(b.isFavorite) - Number(a.isFavorite))
|
||||||
|
.slice(offset, limit * page);
|
||||||
const favWorkspaces = workspacesWithFav.filter((w) => w.isFavorite);
|
|
||||||
const nonFavWorkspaces = workspacesWithFav.filter((w) => !w.isFavorite);
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
workspacesWithFaveProp: workspacesWithFav,
|
workspacesWithFaveProp: workspacesWithFav
|
||||||
favoriteWorkspaces: favWorkspaces,
|
|
||||||
nonFavoriteWorkspaces: nonFavWorkspaces
|
|
||||||
};
|
};
|
||||||
}, [filteredWorkspaces, projectFavorites]);
|
}, [filteredWorkspaces, projectFavorites]);
|
||||||
|
|
||||||
@ -566,7 +604,7 @@ const OrganizationPage = () => {
|
|||||||
{isFavorite ? (
|
{isFavorite ? (
|
||||||
<FontAwesomeIcon
|
<FontAwesomeIcon
|
||||||
icon={faSolidStar}
|
icon={faSolidStar}
|
||||||
className="text-sm text-mineshaft-300 hover:text-mineshaft-400"
|
className="text-sm text-yellow-600 hover:text-mineshaft-400"
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
removeProjectFromFavorites(workspace.id);
|
removeProjectFromFavorites(workspace.id);
|
||||||
@ -623,11 +661,10 @@ const OrganizationPage = () => {
|
|||||||
key={workspace.id}
|
key={workspace.id}
|
||||||
className={`min-w-72 group grid h-14 cursor-pointer grid-cols-6 border-t border-l border-r border-mineshaft-600 bg-mineshaft-800 px-6 hover:bg-mineshaft-700 ${
|
className={`min-w-72 group grid h-14 cursor-pointer grid-cols-6 border-t border-l border-r border-mineshaft-600 bg-mineshaft-800 px-6 hover:bg-mineshaft-700 ${
|
||||||
index === 0 && "rounded-t-md"
|
index === 0 && "rounded-t-md"
|
||||||
} ${index === filteredWorkspaces.length - 1 && "rounded-b-md border-b"}`}
|
}`}
|
||||||
>
|
>
|
||||||
<div className="flex items-center sm:col-span-3 lg:col-span-4">
|
<div className="flex items-center sm:col-span-3 lg:col-span-4">
|
||||||
<FontAwesomeIcon icon={faFileShield} className="text-sm text-primary/70" />
|
<div className="truncate text-sm text-mineshaft-100">{workspace.name}</div>
|
||||||
<div className="ml-5 truncate text-sm text-mineshaft-100">{workspace.name}</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center justify-end sm:col-span-3 lg:col-span-2">
|
<div className="flex items-center justify-end sm:col-span-3 lg:col-span-2">
|
||||||
<div className="text-center text-sm text-mineshaft-300">
|
<div className="text-center text-sm text-mineshaft-300">
|
||||||
@ -636,7 +673,7 @@ const OrganizationPage = () => {
|
|||||||
{isFavorite ? (
|
{isFavorite ? (
|
||||||
<FontAwesomeIcon
|
<FontAwesomeIcon
|
||||||
icon={faSolidStar}
|
icon={faSolidStar}
|
||||||
className="ml-6 text-sm text-mineshaft-300 hover:text-mineshaft-400"
|
className="ml-6 text-sm text-yellow-600 hover:text-mineshaft-400"
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
removeProjectFromFavorites(workspace.id);
|
removeProjectFromFavorites(workspace.id);
|
||||||
@ -656,63 +693,75 @@ const OrganizationPage = () => {
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
const projectsGridView = (
|
let projectsComponents: ReactNode;
|
||||||
<>
|
|
||||||
{favoriteWorkspaces.length > 0 && (
|
|
||||||
<>
|
|
||||||
<p className="mt-6 text-xl font-semibold text-white">Favorites</p>
|
|
||||||
<div
|
|
||||||
className={`b grid w-full grid-cols-1 gap-4 ${
|
|
||||||
nonFavoriteWorkspaces.length > 0 && "border-b border-mineshaft-600"
|
|
||||||
} py-4 lg:grid-cols-2 xl:grid-cols-3 2xl:grid-cols-4`}
|
|
||||||
>
|
|
||||||
{favoriteWorkspaces.map((workspace) => renderProjectGridItem(workspace, true))}
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<div className="mt-4 grid w-full grid-cols-1 gap-4 lg:grid-cols-2 xl:grid-cols-3 2xl:grid-cols-4">
|
|
||||||
{isProjectViewLoading &&
|
|
||||||
Array.apply(0, Array(3)).map((_x, i) => (
|
|
||||||
<div
|
|
||||||
key={`workspace-cards-loading-${i + 1}`}
|
|
||||||
className="min-w-72 flex h-40 flex-col justify-between rounded-md border border-mineshaft-600 bg-mineshaft-800 p-4"
|
|
||||||
>
|
|
||||||
<div className="mt-0 text-lg text-mineshaft-100">
|
|
||||||
<Skeleton className="w-3/4 bg-mineshaft-600" />
|
|
||||||
</div>
|
|
||||||
<div className="mt-0 pb-6 text-sm text-mineshaft-300">
|
|
||||||
<Skeleton className="w-1/2 bg-mineshaft-600" />
|
|
||||||
</div>
|
|
||||||
<div className="flex justify-end">
|
|
||||||
<Skeleton className="w-1/2 bg-mineshaft-600" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
{!isProjectViewLoading &&
|
|
||||||
nonFavoriteWorkspaces.map((workspace) => renderProjectGridItem(workspace, false))}
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
|
|
||||||
const projectsListView = (
|
if (filteredWorkspaces.length || isProjectViewLoading) {
|
||||||
<div className="mt-4 w-full rounded-md">
|
switch (projectsViewMode) {
|
||||||
{isProjectViewLoading &&
|
case ProjectsViewMode.GRID:
|
||||||
Array.apply(0, Array(3)).map((_x, i) => (
|
projectsComponents = (
|
||||||
<div
|
<div className="mt-4 grid w-full grid-cols-1 gap-4 lg:grid-cols-2 xl:grid-cols-3 2xl:grid-cols-4">
|
||||||
key={`workspace-cards-loading-${i + 1}`}
|
{isProjectViewLoading &&
|
||||||
className={`min-w-72 group flex h-12 cursor-pointer flex-row items-center justify-between border border-mineshaft-600 bg-mineshaft-800 px-6 hover:bg-mineshaft-700 ${
|
Array.apply(0, Array(3)).map((_x, i) => (
|
||||||
i === 0 && "rounded-t-md"
|
<div
|
||||||
} ${i === 2 && "rounded-b-md border-b"}`}
|
key={`workspace-cards-loading-${i + 1}`}
|
||||||
>
|
className="min-w-72 flex h-40 flex-col justify-between rounded-md border border-mineshaft-600 bg-mineshaft-800 p-4"
|
||||||
<Skeleton className="w-full bg-mineshaft-600" />
|
>
|
||||||
|
<div className="mt-0 text-lg text-mineshaft-100">
|
||||||
|
<Skeleton className="w-3/4 bg-mineshaft-600" />
|
||||||
|
</div>
|
||||||
|
<div className="mt-0 pb-6 text-sm text-mineshaft-300">
|
||||||
|
<Skeleton className="w-1/2 bg-mineshaft-600" />
|
||||||
|
</div>
|
||||||
|
<div className="flex justify-end">
|
||||||
|
<Skeleton className="w-1/2 bg-mineshaft-600" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{!isProjectViewLoading && (
|
||||||
|
<>
|
||||||
|
{workspacesWithFaveProp.map((workspace) =>
|
||||||
|
renderProjectGridItem(workspace, workspace.isFavorite)
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
))}
|
);
|
||||||
{!isProjectViewLoading &&
|
|
||||||
workspacesWithFaveProp.map((workspace, ind) =>
|
break;
|
||||||
renderProjectListItem(workspace, workspace.isFavorite, ind)
|
case ProjectsViewMode.LIST:
|
||||||
)}
|
default:
|
||||||
</div>
|
projectsComponents = (
|
||||||
);
|
<div className="mt-4 w-full rounded-md">
|
||||||
|
{isProjectViewLoading &&
|
||||||
|
Array.apply(0, Array(3)).map((_x, i) => (
|
||||||
|
<div
|
||||||
|
key={`workspace-cards-loading-${i + 1}`}
|
||||||
|
className={`min-w-72 group flex h-12 cursor-pointer flex-row items-center justify-between border border-mineshaft-600 bg-mineshaft-800 px-6 hover:bg-mineshaft-700 ${
|
||||||
|
i === 0 && "rounded-t-md"
|
||||||
|
} ${i === 2 && "rounded-b-md border-b"}`}
|
||||||
|
>
|
||||||
|
<Skeleton className="w-full bg-mineshaft-600" />
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{!isProjectViewLoading &&
|
||||||
|
workspacesWithFaveProp.map((workspace, ind) =>
|
||||||
|
renderProjectListItem(workspace, workspace.isFavorite, ind)
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if (orgWorkspaces.length) {
|
||||||
|
projectsComponents = (
|
||||||
|
<div className="mt-4 w-full rounded-md border border-mineshaft-700 bg-mineshaft-800 px-4 py-6 text-base text-mineshaft-300">
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={faSearch}
|
||||||
|
className="mb-4 mt-2 w-full text-center text-5xl text-mineshaft-400"
|
||||||
|
/>
|
||||||
|
<div className="text-center font-light">No projects match search...</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mx-auto flex max-w-7xl flex-col justify-start bg-bunker-800 md:h-screen">
|
<div className="mx-auto flex max-w-7xl flex-col justify-start bg-bunker-800 md:h-screen">
|
||||||
@ -754,6 +803,24 @@ const OrganizationPage = () => {
|
|||||||
onChange={(e) => setSearchFilter(e.target.value)}
|
onChange={(e) => setSearchFilter(e.target.value)}
|
||||||
leftIcon={<FontAwesomeIcon icon={faMagnifyingGlass} />}
|
leftIcon={<FontAwesomeIcon icon={faMagnifyingGlass} />}
|
||||||
/>
|
/>
|
||||||
|
<div className="ml-2 flex rounded-md border border-mineshaft-600 bg-mineshaft-800 p-1">
|
||||||
|
<Tooltip content="Toggle Sort Direction">
|
||||||
|
<IconButton
|
||||||
|
className="min-w-[2.4rem] border-none hover:bg-mineshaft-600"
|
||||||
|
ariaLabel={`Sort ${
|
||||||
|
orderDirection === OrderByDirection.ASC ? "descending" : "ascending"
|
||||||
|
}`}
|
||||||
|
variant="plain"
|
||||||
|
size="xs"
|
||||||
|
colorSchema="secondary"
|
||||||
|
onClick={toggleOrderDirection}
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={orderDirection === OrderByDirection.ASC ? faArrowDownAZ : faArrowUpZA}
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
<div className="ml-2 flex rounded-md border border-mineshaft-600 bg-mineshaft-800 p-1">
|
<div className="ml-2 flex rounded-md border border-mineshaft-600 bg-mineshaft-800 p-1">
|
||||||
<IconButton
|
<IconButton
|
||||||
variant="outline_bg"
|
variant="outline_bg"
|
||||||
@ -804,9 +871,24 @@ const OrganizationPage = () => {
|
|||||||
)}
|
)}
|
||||||
</OrgPermissionCan>
|
</OrgPermissionCan>
|
||||||
</div>
|
</div>
|
||||||
{projectsViewMode === ProjectsViewMode.LIST ? projectsListView : projectsGridView}
|
{projectsComponents}
|
||||||
|
{!isProjectViewLoading && Boolean(filteredWorkspaces.length) && (
|
||||||
|
<Pagination
|
||||||
|
className={
|
||||||
|
projectsViewMode === ProjectsViewMode.GRID
|
||||||
|
? "col-span-full !justify-start border-transparent bg-transparent pl-2"
|
||||||
|
: "rounded-b-md border border-mineshaft-600"
|
||||||
|
}
|
||||||
|
perPage={perPage}
|
||||||
|
perPageList={[12, 24, 48, 96]}
|
||||||
|
count={filteredWorkspaces.length}
|
||||||
|
page={page}
|
||||||
|
onChangePage={setPage}
|
||||||
|
onChangePerPage={setPerPage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
{isWorkspaceEmpty && (
|
{isWorkspaceEmpty && (
|
||||||
<div className="w-full rounded-md border border-mineshaft-700 bg-mineshaft-800 px-4 py-6 text-base text-mineshaft-300">
|
<div className="mt-4 w-full rounded-md border border-mineshaft-700 bg-mineshaft-800 px-4 py-6 text-base text-mineshaft-300">
|
||||||
<FontAwesomeIcon
|
<FontAwesomeIcon
|
||||||
icon={faFolderOpen}
|
icon={faFolderOpen}
|
||||||
className="mb-4 mt-2 w-full text-center text-5xl text-mineshaft-400"
|
className="mb-4 mt-2 w-full text-center text-5xl text-mineshaft-400"
|
||||||
|
@ -12,6 +12,7 @@ import {
|
|||||||
DropdownMenuContent,
|
DropdownMenuContent,
|
||||||
DropdownMenuItem,
|
DropdownMenuItem,
|
||||||
DropdownMenuTrigger,
|
DropdownMenuTrigger,
|
||||||
|
FilterableSelect,
|
||||||
FormControl,
|
FormControl,
|
||||||
Select,
|
Select,
|
||||||
SelectItem
|
SelectItem
|
||||||
@ -64,7 +65,7 @@ export const LogsFilter = ({
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (workspacesInOrg.length) {
|
if (workspacesInOrg.length) {
|
||||||
setValue("projectId", workspacesInOrg[0].id);
|
setValue("project", workspacesInOrg[0]);
|
||||||
}
|
}
|
||||||
}, [workspaces]);
|
}, [workspaces]);
|
||||||
|
|
||||||
@ -111,11 +112,34 @@ export const LogsFilter = ({
|
|||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={twMerge(
|
className={twMerge(
|
||||||
"sticky top-20 z-10 flex items-center justify-between bg-bunker-800",
|
"sticky top-20 z-10 flex flex-wrap items-center justify-between bg-bunker-800",
|
||||||
className
|
className
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<div className="flex items-center space-x-2">
|
{isOrgAuditLogs && workspacesInOrg.length > 0 && (
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="project"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
label="Project"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
|
className="mr-12 w-64"
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder="Select a project..."
|
||||||
|
options={workspacesInOrg.map(({ name, id }) => ({ name, id }))}
|
||||||
|
getOptionValue={(option) => option.id}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<div className="mt-1 flex items-center space-x-2">
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="eventType"
|
name="eventType"
|
||||||
@ -123,7 +147,7 @@ export const LogsFilter = ({
|
|||||||
<FormControl label="Events">
|
<FormControl label="Events">
|
||||||
<DropdownMenu>
|
<DropdownMenu>
|
||||||
<DropdownMenuTrigger asChild>
|
<DropdownMenuTrigger asChild>
|
||||||
<div className="inline-flex w-full cursor-pointer items-center justify-between rounded-md border border-mineshaft-500 bg-mineshaft-700 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
<div className="inline-flex w-full cursor-pointer items-center justify-between whitespace-nowrap rounded-md border border-mineshaft-500 bg-mineshaft-700 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
||||||
{selectedEventTypes?.length === 1
|
{selectedEventTypes?.length === 1
|
||||||
? eventTypes.find((eventType) => eventType.value === selectedEventTypes[0])
|
? eventTypes.find((eventType) => eventType.value === selectedEventTypes[0])
|
||||||
?.label
|
?.label
|
||||||
@ -235,37 +259,6 @@ export const LogsFilter = ({
|
|||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{isOrgAuditLogs && workspacesInOrg.length > 0 && (
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="projectId"
|
|
||||||
render={({ field: { onChange, value, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
|
||||||
label="Project"
|
|
||||||
errorText={error?.message}
|
|
||||||
isError={Boolean(error)}
|
|
||||||
className="w-40"
|
|
||||||
>
|
|
||||||
<Select
|
|
||||||
value={value}
|
|
||||||
{...field}
|
|
||||||
onValueChange={(e) => onChange(e)}
|
|
||||||
className={twMerge(
|
|
||||||
"w-full border border-mineshaft-500 bg-mineshaft-700 ",
|
|
||||||
value === undefined && "text-mineshaft-400"
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{workspacesInOrg.map((project) => (
|
|
||||||
<SelectItem value={String(project.id || "")} key={project.id}>
|
|
||||||
{project.name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<Controller
|
<Controller
|
||||||
name="startDate"
|
name="startDate"
|
||||||
control={control}
|
control={control}
|
||||||
@ -275,6 +268,7 @@ export const LogsFilter = ({
|
|||||||
<DatePicker
|
<DatePicker
|
||||||
value={field.value || undefined}
|
value={field.value || undefined}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
|
dateFormat="P"
|
||||||
popUpProps={{
|
popUpProps={{
|
||||||
open: isStartDatePickerOpen,
|
open: isStartDatePickerOpen,
|
||||||
onOpenChange: setIsStartDatePickerOpen
|
onOpenChange: setIsStartDatePickerOpen
|
||||||
@ -294,6 +288,7 @@ export const LogsFilter = ({
|
|||||||
<DatePicker
|
<DatePicker
|
||||||
value={field.value || undefined}
|
value={field.value || undefined}
|
||||||
onChange={onChange}
|
onChange={onChange}
|
||||||
|
dateFormat="P"
|
||||||
popUpProps={{
|
popUpProps={{
|
||||||
open: isEndDatePickerOpen,
|
open: isEndDatePickerOpen,
|
||||||
onOpenChange: setIsEndDatePickerOpen
|
onOpenChange: setIsEndDatePickerOpen
|
||||||
@ -304,27 +299,27 @@ export const LogsFilter = ({
|
|||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
<Button
|
||||||
|
isLoading={false}
|
||||||
|
colorSchema="primary"
|
||||||
|
variant="outline_bg"
|
||||||
|
className="mt-[0.45rem]"
|
||||||
|
type="submit"
|
||||||
|
leftIcon={<FontAwesomeIcon icon={faFilterCircleXmark} />}
|
||||||
|
onClick={() =>
|
||||||
|
reset({
|
||||||
|
eventType: presets?.eventType || [],
|
||||||
|
actor: presets?.actorId,
|
||||||
|
userAgentType: undefined,
|
||||||
|
startDate: undefined,
|
||||||
|
endDate: undefined,
|
||||||
|
project: null
|
||||||
|
})
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Clear filters
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<Button
|
|
||||||
isLoading={false}
|
|
||||||
colorSchema="primary"
|
|
||||||
variant="outline_bg"
|
|
||||||
className="mt-1.5"
|
|
||||||
type="submit"
|
|
||||||
leftIcon={<FontAwesomeIcon icon={faFilterCircleXmark} />}
|
|
||||||
onClick={() =>
|
|
||||||
reset({
|
|
||||||
eventType: presets?.eventType || [],
|
|
||||||
actor: presets?.actorId,
|
|
||||||
userAgentType: undefined,
|
|
||||||
startDate: undefined,
|
|
||||||
endDate: undefined,
|
|
||||||
projectId: undefined
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Clear filters
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -47,7 +47,7 @@ export const LogsSection = ({
|
|||||||
const { control, reset, watch, setValue } = useForm<AuditLogFilterFormData>({
|
const { control, reset, watch, setValue } = useForm<AuditLogFilterFormData>({
|
||||||
resolver: yupResolver(auditLogFilterFormSchema),
|
resolver: yupResolver(auditLogFilterFormSchema),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
projectId: undefined,
|
project: null,
|
||||||
actor: presets?.actorId,
|
actor: presets?.actorId,
|
||||||
eventType: presets?.eventType || [],
|
eventType: presets?.eventType || [],
|
||||||
page: 1,
|
page: 1,
|
||||||
@ -66,7 +66,7 @@ export const LogsSection = ({
|
|||||||
const eventType = watch("eventType") as EventType[] | undefined;
|
const eventType = watch("eventType") as EventType[] | undefined;
|
||||||
const userAgentType = watch("userAgentType") as UserAgentType | undefined;
|
const userAgentType = watch("userAgentType") as UserAgentType | undefined;
|
||||||
const actor = watch("actor");
|
const actor = watch("actor");
|
||||||
const projectId = watch("projectId");
|
const projectId = watch("project")?.id;
|
||||||
|
|
||||||
const startDate = watch("startDate");
|
const startDate = watch("startDate");
|
||||||
const endDate = watch("endDate");
|
const endDate = watch("endDate");
|
||||||
|
@ -5,7 +5,7 @@ import { EventType, UserAgentType } from "@app/hooks/api/auditLogs/enums";
|
|||||||
export const auditLogFilterFormSchema = yup
|
export const auditLogFilterFormSchema = yup
|
||||||
.object({
|
.object({
|
||||||
eventMetadata: yup.object({}).optional(),
|
eventMetadata: yup.object({}).optional(),
|
||||||
projectId: yup.string().optional(),
|
project: yup.object({ id: yup.string().required(), name: yup.string().required() }).nullable(),
|
||||||
eventType: yup.array(yup.string().oneOf(Object.values(EventType), "Invalid event type")),
|
eventType: yup.array(yup.string().oneOf(Object.values(EventType), "Invalid event type")),
|
||||||
actor: yup.string(),
|
actor: yup.string(),
|
||||||
userAgentType: yup.string().oneOf(Object.values(UserAgentType), "Invalid user agent type"),
|
userAgentType: yup.string().oneOf(Object.values(UserAgentType), "Invalid user agent type"),
|
||||||
|
@ -4,7 +4,14 @@ import { zodResolver } from "@hookform/resolvers/zod";
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import { Button, FormControl, Modal, ModalContent, Select, SelectItem } from "@app/components/v2";
|
import {
|
||||||
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
|
FormControl,
|
||||||
|
Modal,
|
||||||
|
ModalClose,
|
||||||
|
ModalContent
|
||||||
|
} from "@app/components/v2";
|
||||||
import { useOrganization, useWorkspace } from "@app/context";
|
import { useOrganization, useWorkspace } from "@app/context";
|
||||||
import {
|
import {
|
||||||
useAddIdentityToWorkspace,
|
useAddIdentityToWorkspace,
|
||||||
@ -16,8 +23,8 @@ import { UsePopUpState } from "@app/hooks/usePopUp";
|
|||||||
|
|
||||||
const schema = z
|
const schema = z
|
||||||
.object({
|
.object({
|
||||||
projectId: z.string(),
|
project: z.object({ name: z.string(), id: z.string() }),
|
||||||
role: z.string()
|
role: z.object({ name: z.string(), slug: z.string() })
|
||||||
})
|
})
|
||||||
.required();
|
.required();
|
||||||
|
|
||||||
@ -32,7 +39,9 @@ type Props = {
|
|||||||
) => void;
|
) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const IdentityAddToProjectModal = ({ identityId, popUp, handlePopUpToggle }: Props) => {
|
// TODO: eventually refactor to support adding to multiple projects at once? would lose role granularity unique to project
|
||||||
|
|
||||||
|
const Content = ({ identityId, handlePopUpToggle }: Omit<Props, "popUp">) => {
|
||||||
const { currentOrg } = useOrganization();
|
const { currentOrg } = useOrganization();
|
||||||
const { workspaces } = useWorkspace();
|
const { workspaces } = useWorkspace();
|
||||||
const { mutateAsync: addIdentityToWorkspace } = useAddIdentityToWorkspace();
|
const { mutateAsync: addIdentityToWorkspace } = useAddIdentityToWorkspace();
|
||||||
@ -47,10 +56,10 @@ export const IdentityAddToProjectModal = ({ identityId, popUp, handlePopUpToggle
|
|||||||
resolver: zodResolver(schema)
|
resolver: zodResolver(schema)
|
||||||
});
|
});
|
||||||
|
|
||||||
const projectId = watch("projectId");
|
const projectId = watch("project")?.id;
|
||||||
const { data: projectMemberships } = useGetIdentityProjectMemberships(identityId);
|
const { data: projectMemberships } = useGetIdentityProjectMemberships(identityId);
|
||||||
const { data: project } = useGetWorkspaceById(projectId);
|
const { data: project, isLoading: isProjectLoading } = useGetWorkspaceById(projectId);
|
||||||
const { data: roles } = useGetProjectRoles(project?.id ?? "");
|
const { data: roles, isLoading: isRolesLoading } = useGetProjectRoles(project?.id ?? "");
|
||||||
|
|
||||||
const filteredWorkspaces = useMemo(() => {
|
const filteredWorkspaces = useMemo(() => {
|
||||||
const wsWorkspaceIds = new Map();
|
const wsWorkspaceIds = new Map();
|
||||||
@ -64,12 +73,12 @@ export const IdentityAddToProjectModal = ({ identityId, popUp, handlePopUpToggle
|
|||||||
);
|
);
|
||||||
}, [workspaces, projectMemberships]);
|
}, [workspaces, projectMemberships]);
|
||||||
|
|
||||||
const onFormSubmit = async ({ projectId: workspaceId, role }: FormData) => {
|
const onFormSubmit = async ({ project: selectedProject, role }: FormData) => {
|
||||||
try {
|
try {
|
||||||
await addIdentityToWorkspace({
|
await addIdentityToWorkspace({
|
||||||
workspaceId,
|
workspaceId: selectedProject.id,
|
||||||
identityId,
|
identityId,
|
||||||
role: role || undefined
|
role: role.slug || undefined
|
||||||
});
|
});
|
||||||
|
|
||||||
createNotification({
|
createNotification({
|
||||||
@ -91,87 +100,85 @@ export const IdentityAddToProjectModal = ({ identityId, popUp, handlePopUpToggle
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const isProjectSelected = Boolean(projectId);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<form onSubmit={handleSubmit(onFormSubmit)}>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="project"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
label="Project"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
|
className="mt-4"
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
options={filteredWorkspaces}
|
||||||
|
placeholder="Select project..."
|
||||||
|
getOptionValue={(option) => option.id}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
isLoading={isProjectSelected && isProjectLoading}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="role"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
label="Role"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
|
className="mt-4"
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
isDisabled={!isProjectSelected}
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
options={roles}
|
||||||
|
isLoading={isProjectSelected && isRolesLoading}
|
||||||
|
placeholder="Select role..."
|
||||||
|
getOptionValue={(option) => option.slug}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<Button
|
||||||
|
className="mr-4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
isLoading={isSubmitting}
|
||||||
|
isDisabled={isSubmitting}
|
||||||
|
>
|
||||||
|
Add
|
||||||
|
</Button>
|
||||||
|
<ModalClose asChild>
|
||||||
|
<Button colorSchema="secondary" variant="plain">
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</ModalClose>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const IdentityAddToProjectModal = ({ identityId, popUp, handlePopUpToggle }: Props) => {
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
isOpen={popUp?.addIdentityToProject?.isOpen}
|
isOpen={popUp?.addIdentityToProject?.isOpen}
|
||||||
onOpenChange={(isOpen) => {
|
onOpenChange={(isOpen) => {
|
||||||
handlePopUpToggle("addIdentityToProject", isOpen);
|
handlePopUpToggle("addIdentityToProject", isOpen);
|
||||||
reset();
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent title="Add Identity to Project">
|
<ModalContent bodyClassName="overflow-visible" title="Add Identity to Project">
|
||||||
<form onSubmit={handleSubmit(onFormSubmit)}>
|
<Content identityId={identityId} handlePopUpToggle={handlePopUpToggle} />
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="projectId"
|
|
||||||
defaultValue=""
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
|
||||||
label="Project"
|
|
||||||
errorText={error?.message}
|
|
||||||
isError={Boolean(error)}
|
|
||||||
className="mt-4"
|
|
||||||
>
|
|
||||||
<Select
|
|
||||||
defaultValue={field.value}
|
|
||||||
{...field}
|
|
||||||
onValueChange={(e) => onChange(e)}
|
|
||||||
className="w-full"
|
|
||||||
>
|
|
||||||
{(filteredWorkspaces || []).map(({ id, name }) => (
|
|
||||||
<SelectItem value={id} key={`project-${id}`}>
|
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="role"
|
|
||||||
defaultValue=""
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
|
||||||
label="Role"
|
|
||||||
errorText={error?.message}
|
|
||||||
isError={Boolean(error)}
|
|
||||||
className="mt-4"
|
|
||||||
>
|
|
||||||
<Select
|
|
||||||
defaultValue={field.value}
|
|
||||||
{...field}
|
|
||||||
onValueChange={(e) => onChange(e)}
|
|
||||||
className="w-full"
|
|
||||||
>
|
|
||||||
{(roles || []).map(({ name, slug }) => (
|
|
||||||
<SelectItem value={slug} key={`project-role-${slug}`}>
|
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<div className="flex items-center">
|
|
||||||
<Button
|
|
||||||
className="mr-4"
|
|
||||||
size="sm"
|
|
||||||
type="submit"
|
|
||||||
isLoading={isSubmitting}
|
|
||||||
isDisabled={isSubmitting}
|
|
||||||
>
|
|
||||||
Add
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
colorSchema="secondary"
|
|
||||||
variant="plain"
|
|
||||||
onClick={() => handlePopUpToggle("addIdentityToProject", false)}
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
|
@ -6,14 +6,14 @@ import { z } from "zod";
|
|||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
FormControl,
|
FormControl,
|
||||||
Input,
|
Input,
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent
|
||||||
Select,
|
|
||||||
SelectItem
|
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { useOrganization } from "@app/context";
|
import { useOrganization } from "@app/context";
|
||||||
|
import { findOrgMembershipRole } from "@app/helpers/roles";
|
||||||
import { useCreateGroup, useGetOrgRoles, useUpdateGroup } from "@app/hooks/api";
|
import { useCreateGroup, useGetOrgRoles, useUpdateGroup } from "@app/hooks/api";
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
@ -23,7 +23,7 @@ const GroupFormSchema = z.object({
|
|||||||
.string()
|
.string()
|
||||||
.min(5, "Slug must be at least 5 characters long")
|
.min(5, "Slug must be at least 5 characters long")
|
||||||
.max(36, "Slug must be 36 characters or fewer"),
|
.max(36, "Slug must be 36 characters or fewer"),
|
||||||
role: z.string()
|
role: z.object({ name: z.string(), slug: z.string() })
|
||||||
});
|
});
|
||||||
|
|
||||||
export type TGroupFormData = z.infer<typeof GroupFormSchema>;
|
export type TGroupFormData = z.infer<typeof GroupFormSchema>;
|
||||||
@ -62,13 +62,13 @@ export const OrgGroupModal = ({ popUp, handlePopUpClose, handlePopUpToggle }: Pr
|
|||||||
reset({
|
reset({
|
||||||
name: group.name,
|
name: group.name,
|
||||||
slug: group.slug,
|
slug: group.slug,
|
||||||
role: group?.customRole?.slug ?? group.role
|
role: group?.customRole ?? findOrgMembershipRole(roles, group.role)
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
reset({
|
reset({
|
||||||
name: "",
|
name: "",
|
||||||
slug: "",
|
slug: "",
|
||||||
role: roles[0].slug
|
role: findOrgMembershipRole(roles, currentOrg!.defaultMembershipRole)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [popUp?.group?.data, roles]);
|
}, [popUp?.group?.data, roles]);
|
||||||
@ -88,14 +88,14 @@ export const OrgGroupModal = ({ popUp, handlePopUpClose, handlePopUpToggle }: Pr
|
|||||||
id: group.groupId,
|
id: group.groupId,
|
||||||
name,
|
name,
|
||||||
slug,
|
slug,
|
||||||
role: role || undefined
|
role: role.slug || undefined
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
await createMutateAsync({
|
await createMutateAsync({
|
||||||
name,
|
name,
|
||||||
slug,
|
slug,
|
||||||
organizationId: currentOrg.id,
|
organizationId: currentOrg.id,
|
||||||
role: role || undefined
|
role: role.slug || undefined
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
handlePopUpToggle("group", false);
|
handlePopUpToggle("group", false);
|
||||||
@ -121,7 +121,10 @@ export const OrgGroupModal = ({ popUp, handlePopUpClose, handlePopUpToggle }: Pr
|
|||||||
reset();
|
reset();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent title={`${popUp?.group?.data ? "Update" : "Create"} Group`}>
|
<ModalContent
|
||||||
|
bodyClassName="overflow-visible"
|
||||||
|
title={`${popUp?.group?.data ? "Update" : "Create"} Group`}
|
||||||
|
>
|
||||||
<form onSubmit={handleSubmit(onGroupModalSubmit)}>
|
<form onSubmit={handleSubmit(onGroupModalSubmit)}>
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
@ -144,26 +147,21 @@ export const OrgGroupModal = ({ popUp, handlePopUpClose, handlePopUpToggle }: Pr
|
|||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="role"
|
name="role"
|
||||||
defaultValue=""
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
<FormControl
|
||||||
label={`${popUp?.group?.data ? "Update" : ""} Role`}
|
label={`${popUp?.group?.data ? "Update" : ""} Role`}
|
||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
isError={Boolean(error)}
|
isError={Boolean(error)}
|
||||||
className="mt-4"
|
className="mt-4"
|
||||||
>
|
>
|
||||||
<Select
|
<FilterableSelect
|
||||||
defaultValue={field.value}
|
options={roles}
|
||||||
{...field}
|
placeholder="Select role..."
|
||||||
onValueChange={(e) => onChange(e)}
|
onChange={onChange}
|
||||||
className="w-full"
|
value={value}
|
||||||
>
|
getOptionValue={(option) => option.slug}
|
||||||
{(roles || []).map(({ name, slug }) => (
|
getOptionLabel={(option) => option.name}
|
||||||
<SelectItem value={slug} key={`org-group-role-${slug}`}>
|
/>
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -9,27 +9,24 @@ import { z } from "zod";
|
|||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
FormControl,
|
FormControl,
|
||||||
FormLabel,
|
FormLabel,
|
||||||
IconButton,
|
IconButton,
|
||||||
Input,
|
Input,
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent
|
||||||
Select,
|
|
||||||
SelectItem
|
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { useOrganization } from "@app/context";
|
import { useOrganization } from "@app/context";
|
||||||
|
import { findOrgMembershipRole } from "@app/helpers/roles";
|
||||||
import { useCreateIdentity, useGetOrgRoles, useUpdateIdentity } from "@app/hooks/api";
|
import { useCreateIdentity, useGetOrgRoles, useUpdateIdentity } from "@app/hooks/api";
|
||||||
import {
|
import { useAddIdentityUniversalAuth } from "@app/hooks/api/identities";
|
||||||
// IdentityAuthMethod,
|
|
||||||
useAddIdentityUniversalAuth
|
|
||||||
} from "@app/hooks/api/identities";
|
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
const schema = z
|
const schema = z
|
||||||
.object({
|
.object({
|
||||||
name: z.string(),
|
name: z.string().min(1, "Required"),
|
||||||
role: z.string(),
|
role: z.object({ slug: z.string(), name: z.string() }),
|
||||||
metadata: z
|
metadata: z
|
||||||
.object({
|
.object({
|
||||||
key: z.string().trim().min(1),
|
key: z.string().trim().min(1),
|
||||||
@ -101,13 +98,13 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
if (identity) {
|
if (identity) {
|
||||||
reset({
|
reset({
|
||||||
name: identity.name,
|
name: identity.name,
|
||||||
role: identity?.customRole?.slug ?? identity.role,
|
role: identity.customRole ?? findOrgMembershipRole(roles, identity.role),
|
||||||
metadata: identity.metadata
|
metadata: identity.metadata
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
reset({
|
reset({
|
||||||
name: "",
|
name: "",
|
||||||
role: roles[0].slug
|
role: findOrgMembershipRole(roles, currentOrg!.defaultMembershipRole)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [popUp?.identity?.data, roles]);
|
}, [popUp?.identity?.data, roles]);
|
||||||
@ -126,7 +123,7 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
await updateMutateAsync({
|
await updateMutateAsync({
|
||||||
identityId: identity.identityId,
|
identityId: identity.identityId,
|
||||||
name,
|
name,
|
||||||
role: role || undefined,
|
role: role.slug || undefined,
|
||||||
organizationId: orgId,
|
organizationId: orgId,
|
||||||
metadata
|
metadata
|
||||||
});
|
});
|
||||||
@ -137,7 +134,7 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
|
|
||||||
const { id: createdId } = await createMutateAsync({
|
const { id: createdId } = await createMutateAsync({
|
||||||
name,
|
name,
|
||||||
role: role || undefined,
|
role: role.slug || undefined,
|
||||||
organizationId: orgId,
|
organizationId: orgId,
|
||||||
metadata
|
metadata
|
||||||
});
|
});
|
||||||
@ -184,7 +181,10 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
reset();
|
reset();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent title={`${popUp?.identity?.data ? "Update" : "Create"} Identity`}>
|
<ModalContent
|
||||||
|
bodyClassName="overflow-visible"
|
||||||
|
title={`${popUp?.identity?.data ? "Update" : "Create"} Identity`}
|
||||||
|
>
|
||||||
<form onSubmit={handleSubmit(onFormSubmit)}>
|
<form onSubmit={handleSubmit(onFormSubmit)}>
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
@ -199,26 +199,21 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="role"
|
name="role"
|
||||||
defaultValue=""
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
<FormControl
|
||||||
label={`${popUp?.identity?.data ? "Update" : ""} Role`}
|
label={`${popUp?.identity?.data ? "Update" : ""} Role`}
|
||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
isError={Boolean(error)}
|
isError={Boolean(error)}
|
||||||
className="mt-4"
|
className="mt-4"
|
||||||
>
|
>
|
||||||
<Select
|
<FilterableSelect
|
||||||
defaultValue={field.value}
|
placeholder="Select role..."
|
||||||
{...field}
|
options={roles}
|
||||||
onValueChange={(e) => onChange(e)}
|
onChange={onChange}
|
||||||
className="w-full"
|
value={value}
|
||||||
>
|
getOptionValue={(option) => option.slug}
|
||||||
{(roles || []).map(({ name, slug }) => (
|
getOptionLabel={(option) => option.name}
|
||||||
<SelectItem value={slug} key={`st-role-${slug}`}>
|
/>
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -42,7 +42,7 @@ export const IdentitySection = withPermission(
|
|||||||
? subscription.identitiesUsed < subscription.identityLimit
|
? subscription.identitiesUsed < subscription.identityLimit
|
||||||
: true;
|
: true;
|
||||||
|
|
||||||
const isEnterprise = subscription?.slug === "enterprise"
|
const isEnterprise = subscription?.slug === "enterprise";
|
||||||
|
|
||||||
const onDeleteIdentitySubmit = async (identityId: string) => {
|
const onDeleteIdentitySubmit = async (identityId: string) => {
|
||||||
try {
|
try {
|
||||||
@ -105,7 +105,7 @@ export const IdentitySection = withPermission(
|
|||||||
}}
|
}}
|
||||||
isDisabled={!isAllowed}
|
isDisabled={!isAllowed}
|
||||||
>
|
>
|
||||||
Create identity
|
Create Identity
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
</OrgPermissionCan>
|
</OrgPermissionCan>
|
||||||
|
@ -1,31 +1,21 @@
|
|||||||
import { useEffect } from "react";
|
import { useEffect } from "react";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import {
|
|
||||||
faCheckCircle,
|
|
||||||
faChevronDown,
|
|
||||||
faExclamationCircle
|
|
||||||
} from "@fortawesome/free-solid-svg-icons";
|
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
DropdownMenu,
|
FilterableSelect,
|
||||||
DropdownMenuContent,
|
|
||||||
DropdownMenuItem,
|
|
||||||
DropdownMenuTrigger,
|
|
||||||
FormControl,
|
FormControl,
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent,
|
||||||
Select,
|
Select,
|
||||||
SelectItem,
|
SelectItem,
|
||||||
TextArea,
|
TextArea
|
||||||
Tooltip
|
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { useOrganization } from "@app/context";
|
import { useOrganization } from "@app/context";
|
||||||
import { isCustomOrgRole } from "@app/helpers/roles";
|
import { findOrgMembershipRole } from "@app/helpers/roles";
|
||||||
import {
|
import {
|
||||||
useAddUsersToOrg,
|
useAddUsersToOrg,
|
||||||
useFetchServerStatus,
|
useFetchServerStatus,
|
||||||
@ -44,9 +34,18 @@ const EmailSchema = z.string().email().min(1).trim().toLowerCase();
|
|||||||
|
|
||||||
const addMemberFormSchema = z.object({
|
const addMemberFormSchema = z.object({
|
||||||
emails: z.string().min(1).trim().toLowerCase(),
|
emails: z.string().min(1).trim().toLowerCase(),
|
||||||
projectIds: z.array(z.string().min(1).trim().toLowerCase()).default([]),
|
projects: z
|
||||||
|
.array(
|
||||||
|
z.object({
|
||||||
|
name: z.string(),
|
||||||
|
id: z.string(),
|
||||||
|
slug: z.string(),
|
||||||
|
version: z.nativeEnum(ProjectVersion)
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.default([]),
|
||||||
projectRoleSlug: z.string().min(1).default(DEFAULT_ORG_AND_PROJECT_MEMBER_ROLE_SLUG),
|
projectRoleSlug: z.string().min(1).default(DEFAULT_ORG_AND_PROJECT_MEMBER_ROLE_SLUG),
|
||||||
organizationRoleSlug: z.string().min(1).default(DEFAULT_ORG_AND_PROJECT_MEMBER_ROLE_SLUG)
|
organizationRole: z.object({ name: z.string(), slug: z.string() })
|
||||||
});
|
});
|
||||||
|
|
||||||
type TAddMemberForm = z.infer<typeof addMemberFormSchema>;
|
type TAddMemberForm = z.infer<typeof addMemberFormSchema>;
|
||||||
@ -72,7 +71,7 @@ export const AddOrgMemberModal = ({
|
|||||||
const { data: organizationRoles } = useGetOrgRoles(currentOrg?.id ?? "");
|
const { data: organizationRoles } = useGetOrgRoles(currentOrg?.id ?? "");
|
||||||
const { data: serverDetails } = useFetchServerStatus();
|
const { data: serverDetails } = useFetchServerStatus();
|
||||||
const { mutateAsync: addUsersMutateAsync } = useAddUsersToOrg();
|
const { mutateAsync: addUsersMutateAsync } = useAddUsersToOrg();
|
||||||
const { data: projects } = useGetUserWorkspaces(true);
|
const { data: projects, isLoading: isProjectsLoading } = useGetUserWorkspaces(true);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
control,
|
control,
|
||||||
@ -88,25 +87,22 @@ export const AddOrgMemberModal = ({
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (organizationRoles) {
|
if (organizationRoles) {
|
||||||
reset({
|
reset({
|
||||||
organizationRoleSlug: isCustomOrgRole(currentOrg?.defaultMembershipRole!)
|
organizationRole: findOrgMembershipRole(
|
||||||
? organizationRoles?.find((role) => role.id === currentOrg?.defaultMembershipRole)?.slug!
|
organizationRoles,
|
||||||
: currentOrg?.defaultMembershipRole
|
currentOrg?.defaultMembershipRole!
|
||||||
|
)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [organizationRoles]);
|
}, [organizationRoles]);
|
||||||
|
|
||||||
const selectedProjectIds = watch("projectIds", []);
|
|
||||||
|
|
||||||
const onAddMembers = async ({
|
const onAddMembers = async ({
|
||||||
emails,
|
emails,
|
||||||
organizationRoleSlug,
|
organizationRole,
|
||||||
projectIds,
|
projects: selectedProjects,
|
||||||
projectRoleSlug
|
projectRoleSlug
|
||||||
}: TAddMemberForm) => {
|
}: TAddMemberForm) => {
|
||||||
if (!currentOrg?.id) return;
|
if (!currentOrg?.id) return;
|
||||||
|
|
||||||
const selectedProjects = projects?.filter((project) => projectIds.includes(String(project.id)));
|
|
||||||
|
|
||||||
if (selectedProjects?.length) {
|
if (selectedProjects?.length) {
|
||||||
// eslint-disable-next-line no-restricted-syntax
|
// eslint-disable-next-line no-restricted-syntax
|
||||||
for (const project of selectedProjects) {
|
for (const project of selectedProjects) {
|
||||||
@ -143,8 +139,8 @@ export const AddOrgMemberModal = ({
|
|||||||
const { data } = await addUsersMutateAsync({
|
const { data } = await addUsersMutateAsync({
|
||||||
organizationId: currentOrg?.id,
|
organizationId: currentOrg?.id,
|
||||||
inviteeEmails: emails.split(",").map((email) => email.trim()),
|
inviteeEmails: emails.split(",").map((email) => email.trim()),
|
||||||
organizationRoleSlug,
|
organizationRoleSlug: organizationRole.slug,
|
||||||
projects: projectIds.map((id) => ({ id, projectRoleSlug: [projectRoleSlug] }))
|
projects: selectedProjects.map(({ id }) => ({ id, projectRoleSlug: [projectRoleSlug] }))
|
||||||
});
|
});
|
||||||
|
|
||||||
setCompleteInviteLinks(data?.completeInviteLinks ?? null);
|
setCompleteInviteLinks(data?.completeInviteLinks ?? null);
|
||||||
@ -182,6 +178,7 @@ export const AddOrgMemberModal = ({
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent
|
<ModalContent
|
||||||
|
bodyClassName="overflow-visible"
|
||||||
title={`Invite others to ${currentOrg?.name}`}
|
title={`Invite others to ${currentOrg?.name}`}
|
||||||
subTitle={
|
subTitle={
|
||||||
<div>
|
<div>
|
||||||
@ -211,123 +208,53 @@ export const AddOrgMemberModal = ({
|
|||||||
|
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="organizationRoleSlug"
|
name="organizationRole"
|
||||||
render={({ field, fieldState: { error } }) => (
|
render={({ field: { value, onChange }, fieldState: { error } }) => (
|
||||||
<FormControl
|
<FormControl
|
||||||
tooltipText="Select which organization role you want to assign to the user."
|
tooltipText="Select which organization role you want to assign to the user."
|
||||||
label="Assign organization role"
|
label="Assign organization role"
|
||||||
isError={Boolean(error)}
|
isError={Boolean(error)}
|
||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
>
|
>
|
||||||
<div>
|
<FilterableSelect
|
||||||
<Select
|
placeholder="Select role..."
|
||||||
className="w-full"
|
options={organizationRoles}
|
||||||
{...field}
|
getOptionValue={(option) => option.slug}
|
||||||
onValueChange={(val) => field.onChange(val)}
|
getOptionLabel={(option) => option.name}
|
||||||
>
|
value={value}
|
||||||
{organizationRoles?.map((role) => (
|
onChange={onChange}
|
||||||
<SelectItem key={role.id} value={role.slug}>
|
/>
|
||||||
{role.name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex items-center justify-between gap-2">
|
<div className="flex items-start justify-between gap-2">
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="projectIds"
|
name="projects"
|
||||||
render={({ field, fieldState: { error } }) => (
|
render={({ field: { value, onChange }, fieldState: { error } }) => (
|
||||||
<FormControl
|
<FormControl
|
||||||
label="Assign users to projects (optional)"
|
label="Assign users to projects"
|
||||||
|
isOptional
|
||||||
isError={Boolean(error?.message)}
|
isError={Boolean(error?.message)}
|
||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
>
|
>
|
||||||
<DropdownMenu>
|
<FilterableSelect
|
||||||
<DropdownMenuTrigger asChild>
|
isMulti
|
||||||
{projects && projects.length > 0 ? (
|
value={value}
|
||||||
<div className="inline-flex w-full cursor-pointer items-center justify-between rounded-md border border-mineshaft-600 bg-mineshaft-900 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
onChange={onChange}
|
||||||
{/* eslint-disable-next-line no-nested-ternary */}
|
isLoading={isProjectsLoading}
|
||||||
{selectedProjectIds.length === 1
|
getOptionLabel={(project) => project.name}
|
||||||
? projects.find((project) => project.id === selectedProjectIds[0])
|
getOptionValue={(project) => project.id}
|
||||||
?.name
|
options={projects}
|
||||||
: selectedProjectIds.length === 0
|
placeholder="Select projects..."
|
||||||
? "No projects selected"
|
/>
|
||||||
: `${selectedProjectIds.length} projects selected`}
|
|
||||||
<FontAwesomeIcon icon={faChevronDown} className="text-xs" />
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="inline-flex w-full cursor-default items-center justify-between rounded-md border border-mineshaft-600 bg-mineshaft-900 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
|
||||||
No projects found
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</DropdownMenuTrigger>
|
|
||||||
<DropdownMenuContent
|
|
||||||
align="start"
|
|
||||||
className="thin-scrollbar z-[100] max-h-80"
|
|
||||||
>
|
|
||||||
{projects && projects.length > 0 ? (
|
|
||||||
projects.map((project) => {
|
|
||||||
const isSelected = selectedProjectIds.includes(String(project.id));
|
|
||||||
|
|
||||||
return (
|
|
||||||
<DropdownMenuItem
|
|
||||||
onSelect={(event) =>
|
|
||||||
projects.length > 1 && event.preventDefault()
|
|
||||||
}
|
|
||||||
onClick={() => {
|
|
||||||
if (selectedProjectIds.includes(String(project.id))) {
|
|
||||||
field.onChange(
|
|
||||||
selectedProjectIds.filter(
|
|
||||||
(projectId: string) => projectId !== String(project.id)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
field.onChange([...selectedProjectIds, String(project.id)]);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
key={`project-id-${project.id}`}
|
|
||||||
icon={
|
|
||||||
isSelected ? (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faCheckCircle}
|
|
||||||
className="pr-0.5 text-primary"
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<div className="pl-[1.01rem]" />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
iconPos="left"
|
|
||||||
className="w-[28.4rem] text-sm"
|
|
||||||
>
|
|
||||||
<div className="flex items-center gap-2 capitalize">
|
|
||||||
{project.name}
|
|
||||||
{project.version !== ProjectVersion.V3 && (
|
|
||||||
<Tooltip content="Project is not compatible with this action, please upgrade this project.">
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faExclamationCircle}
|
|
||||||
className="text-xs opacity-50"
|
|
||||||
/>
|
|
||||||
</Tooltip>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</DropdownMenuItem>
|
|
||||||
);
|
|
||||||
})
|
|
||||||
) : (
|
|
||||||
<div />
|
|
||||||
)}
|
|
||||||
</DropdownMenuContent>
|
|
||||||
</DropdownMenu>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex min-w-fit justify-end">
|
<div className="mt-[0.15rem] flex min-w-fit justify-end">
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="projectRoleSlug"
|
name="projectRoleSlug"
|
||||||
@ -340,7 +267,7 @@ export const AddOrgMemberModal = ({
|
|||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
<Select
|
<Select
|
||||||
isDisabled={selectedProjectIds.length === 0}
|
isDisabled={watch("projects", []).length === 0}
|
||||||
defaultValue={DEFAULT_ORG_AND_PROJECT_MEMBER_ROLE_SLUG}
|
defaultValue={DEFAULT_ORG_AND_PROJECT_MEMBER_ROLE_SLUG}
|
||||||
{...field}
|
{...field}
|
||||||
onValueChange={(val) => field.onChange(val)}
|
onValueChange={(val) => field.onChange(val)}
|
||||||
|
@ -148,7 +148,8 @@ export const UserPage = withPermission(
|
|||||||
onClick={() =>
|
onClick={() =>
|
||||||
handlePopUpOpen("orgMembership", {
|
handlePopUpOpen("orgMembership", {
|
||||||
membershipId: membership.id,
|
membershipId: membership.id,
|
||||||
role: membership.role
|
role: membership.role,
|
||||||
|
roleId: membership.roleId
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
disabled={!isAllowed}
|
disabled={!isAllowed}
|
||||||
|
@ -100,6 +100,7 @@ export const UserDetailsSection = ({ membershipId, handlePopUpOpen }: Props) =>
|
|||||||
handlePopUpOpen("orgMembership", {
|
handlePopUpOpen("orgMembership", {
|
||||||
membershipId: membership.id,
|
membershipId: membership.id,
|
||||||
role: membership.role,
|
role: membership.role,
|
||||||
|
roleId: membership.roleId,
|
||||||
metadata: membership.metadata
|
metadata: membership.metadata
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import { useEffect } from "react";
|
import { useEffect } from "react";
|
||||||
import { Controller, useFieldArray, useForm } from "react-hook-form";
|
import { Controller, useFieldArray, useForm } from "react-hook-form";
|
||||||
|
import { SingleValue } from "react-select";
|
||||||
import { faPlus, faTrash } from "@fortawesome/free-solid-svg-icons";
|
import { faPlus, faTrash } from "@fortawesome/free-solid-svg-icons";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
@ -8,21 +9,21 @@ import { z } from "zod";
|
|||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
FormControl,
|
FormControl,
|
||||||
FormLabel,
|
FormLabel,
|
||||||
IconButton,
|
IconButton,
|
||||||
Input,
|
Input,
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent
|
||||||
Select,
|
|
||||||
SelectItem
|
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { useOrganization, useSubscription } from "@app/context";
|
import { useOrganization, useSubscription } from "@app/context";
|
||||||
|
import { findOrgMembershipRole, isCustomOrgRole } from "@app/helpers/roles";
|
||||||
import { useGetOrgRoles, useUpdateOrgMembership } from "@app/hooks/api";
|
import { useGetOrgRoles, useUpdateOrgMembership } from "@app/hooks/api";
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
const schema = z.object({
|
const schema = z.object({
|
||||||
role: z.string(),
|
role: z.object({ name: z.string(), slug: z.string() }),
|
||||||
metadata: z
|
metadata: z
|
||||||
.object({
|
.object({
|
||||||
key: z.string().trim().min(1),
|
key: z.string().trim().min(1),
|
||||||
@ -45,7 +46,7 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
const { currentOrg } = useOrganization();
|
const { currentOrg } = useOrganization();
|
||||||
const orgId = currentOrg?.id || "";
|
const orgId = currentOrg?.id || "";
|
||||||
|
|
||||||
const { data: roles } = useGetOrgRoles(orgId);
|
const { data: roles = [] } = useGetOrgRoles(orgId);
|
||||||
|
|
||||||
const { mutateAsync: updateOrgMembership } = useUpdateOrgMembership();
|
const { mutateAsync: updateOrgMembership } = useUpdateOrgMembership();
|
||||||
|
|
||||||
@ -66,6 +67,7 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
const popUpData = popUp?.orgMembership?.data as {
|
const popUpData = popUp?.orgMembership?.data as {
|
||||||
membershipId: string;
|
membershipId: string;
|
||||||
role: string;
|
role: string;
|
||||||
|
roleId?: string;
|
||||||
metadata: { key: string; value: string }[];
|
metadata: { key: string; value: string }[];
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -74,12 +76,12 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
|
|
||||||
if (popUpData) {
|
if (popUpData) {
|
||||||
reset({
|
reset({
|
||||||
role: popUpData.role,
|
role: findOrgMembershipRole(roles, popUpData.roleId ?? popUpData.role),
|
||||||
metadata: popUpData.metadata
|
metadata: popUpData.metadata
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
reset({
|
reset({
|
||||||
role: roles[0].slug
|
role: findOrgMembershipRole(roles, currentOrg!.defaultMembershipRole!)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [popUp?.orgMembership?.data, roles]);
|
}, [popUp?.orgMembership?.data, roles]);
|
||||||
@ -91,7 +93,7 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
await updateOrgMembership({
|
await updateOrgMembership({
|
||||||
organizationId: orgId,
|
organizationId: orgId,
|
||||||
membershipId: popUpData.membershipId,
|
membershipId: popUpData.membershipId,
|
||||||
role,
|
role: role.slug,
|
||||||
metadata
|
metadata
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -123,23 +125,26 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
reset();
|
reset();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent title="Update Membership">
|
<ModalContent bodyClassName="overflow-visible" title="Update Membership">
|
||||||
<form onSubmit={handleSubmit(onFormSubmit)}>
|
<form onSubmit={handleSubmit(onFormSubmit)}>
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="role"
|
name="role"
|
||||||
defaultValue=""
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
<FormControl
|
||||||
label="Update Organization Role"
|
label="Update Organization Role"
|
||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
isError={Boolean(error)}
|
isError={Boolean(error)}
|
||||||
>
|
>
|
||||||
<Select
|
<FilterableSelect
|
||||||
defaultValue={field.value}
|
placeholder="Select role..."
|
||||||
{...field}
|
options={roles}
|
||||||
onValueChange={(e) => {
|
onChange={(newValue) => {
|
||||||
const isCustomRole = !["admin", "member", "no-access"].includes(e);
|
const role = newValue as SingleValue<(typeof roles)[number]>;
|
||||||
|
|
||||||
|
if (!role) return;
|
||||||
|
|
||||||
|
const isCustomRole = isCustomOrgRole(role.slug);
|
||||||
|
|
||||||
if (isCustomRole && subscription && !subscription?.rbac) {
|
if (isCustomRole && subscription && !subscription?.rbac) {
|
||||||
handlePopUpOpen("upgradePlan", {
|
handlePopUpOpen("upgradePlan", {
|
||||||
@ -149,16 +154,12 @@ export const UserOrgMembershipModal = ({ popUp, handlePopUpOpen, handlePopUpTogg
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
onChange(e);
|
onChange(role);
|
||||||
}}
|
}}
|
||||||
className="w-full"
|
value={value}
|
||||||
>
|
getOptionValue={(option) => option.slug}
|
||||||
{(roles || []).map(({ name, slug }) => (
|
getOptionLabel={(option) => option.name}
|
||||||
<SelectItem value={slug} key={`st-role-${slug}`}>
|
/>
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -1,6 +1,27 @@
|
|||||||
import { faFolder } from "@fortawesome/free-solid-svg-icons";
|
import { useMemo } from "react";
|
||||||
|
import {
|
||||||
|
faArrowDown,
|
||||||
|
faArrowUp,
|
||||||
|
faMagnifyingGlass,
|
||||||
|
faSearch,
|
||||||
|
faUser
|
||||||
|
} from "@fortawesome/free-solid-svg-icons";
|
||||||
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
|
|
||||||
import { EmptyState, Table, TableContainer, TBody, Th, THead, Tr } from "@app/components/v2";
|
import {
|
||||||
|
EmptyState,
|
||||||
|
IconButton,
|
||||||
|
Input,
|
||||||
|
Pagination,
|
||||||
|
Table,
|
||||||
|
TableContainer,
|
||||||
|
TBody,
|
||||||
|
Th,
|
||||||
|
THead,
|
||||||
|
Tr
|
||||||
|
} from "@app/components/v2";
|
||||||
|
import { usePagination, useResetPageHelper } from "@app/hooks";
|
||||||
|
import { OrderByDirection } from "@app/hooks/api/generic/types";
|
||||||
import { OrgUser } from "@app/hooks/api/types";
|
import { OrgUser } from "@app/hooks/api/types";
|
||||||
import { useListUserGroupMemberships } from "@app/hooks/api/users/queries";
|
import { useListUserGroupMemberships } from "@app/hooks/api/users/queries";
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
@ -12,31 +33,106 @@ type Props = {
|
|||||||
handlePopUpOpen: (popUpName: keyof UsePopUpState<["removeUserFromGroup"]>, data?: {}) => void;
|
handlePopUpOpen: (popUpName: keyof UsePopUpState<["removeUserFromGroup"]>, data?: {}) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum UserGroupsOrderBy {
|
||||||
|
Name = "name"
|
||||||
|
}
|
||||||
|
|
||||||
export const UserGroupsTable = ({ handlePopUpOpen, orgMembership }: Props) => {
|
export const UserGroupsTable = ({ handlePopUpOpen, orgMembership }: Props) => {
|
||||||
const { data: groups, isLoading } = useListUserGroupMemberships(orgMembership.user.username);
|
const { data: groupMemberships = [], isLoading } = useListUserGroupMemberships(
|
||||||
|
orgMembership.user.username
|
||||||
|
);
|
||||||
|
|
||||||
|
const {
|
||||||
|
search,
|
||||||
|
setSearch,
|
||||||
|
setPage,
|
||||||
|
page,
|
||||||
|
perPage,
|
||||||
|
setPerPage,
|
||||||
|
offset,
|
||||||
|
orderDirection,
|
||||||
|
toggleOrderDirection
|
||||||
|
} = usePagination(UserGroupsOrderBy.Name, { initPerPage: 10 });
|
||||||
|
|
||||||
|
const filteredGroupMemberships = useMemo(
|
||||||
|
() =>
|
||||||
|
groupMemberships
|
||||||
|
.filter((group) => group.name.toLowerCase().includes(search.trim().toLowerCase()))
|
||||||
|
.sort((a, b) => {
|
||||||
|
const [membershipOne, membershipTwo] =
|
||||||
|
orderDirection === OrderByDirection.ASC ? [a, b] : [b, a];
|
||||||
|
|
||||||
|
return membershipOne.name.toLowerCase().localeCompare(membershipTwo.name.toLowerCase());
|
||||||
|
}),
|
||||||
|
[groupMemberships, orderDirection, search]
|
||||||
|
);
|
||||||
|
|
||||||
|
useResetPageHelper({
|
||||||
|
totalCount: filteredGroupMemberships.length,
|
||||||
|
offset,
|
||||||
|
setPage
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TableContainer>
|
<div>
|
||||||
<Table>
|
<Input
|
||||||
<THead>
|
value={search}
|
||||||
<Tr>
|
onChange={(e) => setSearch(e.target.value)}
|
||||||
<Th>Name</Th>
|
leftIcon={<FontAwesomeIcon icon={faMagnifyingGlass} />}
|
||||||
<Th className="w-5" />
|
placeholder="Search groups..."
|
||||||
</Tr>
|
/>
|
||||||
</THead>
|
<TableContainer className="mt-4">
|
||||||
<TBody>
|
<Table>
|
||||||
{groups?.map((group) => (
|
<THead>
|
||||||
<UserGroupsRow
|
<Tr>
|
||||||
key={`user-group-${group.id}`}
|
<Th className="w-full">
|
||||||
group={group}
|
<div className="flex items-center">
|
||||||
handlePopUpOpen={handlePopUpOpen}
|
Name
|
||||||
/>
|
<IconButton
|
||||||
))}
|
variant="plain"
|
||||||
</TBody>
|
className="ml-2"
|
||||||
</Table>
|
ariaLabel="sort"
|
||||||
{!isLoading && !groups?.length && (
|
onClick={toggleOrderDirection}
|
||||||
<EmptyState title="This user has not been assigned to any groups" icon={faFolder} />
|
>
|
||||||
)}
|
<FontAwesomeIcon
|
||||||
</TableContainer>
|
icon={orderDirection === OrderByDirection.DESC ? faArrowUp : faArrowDown}
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
</div>
|
||||||
|
</Th>
|
||||||
|
<Th className="w-5" />
|
||||||
|
</Tr>
|
||||||
|
</THead>
|
||||||
|
<TBody>
|
||||||
|
{filteredGroupMemberships.slice(offset, perPage * page).map((group) => (
|
||||||
|
<UserGroupsRow
|
||||||
|
key={`user-group-${group.id}`}
|
||||||
|
group={group}
|
||||||
|
handlePopUpOpen={handlePopUpOpen}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</TBody>
|
||||||
|
</Table>
|
||||||
|
{Boolean(filteredGroupMemberships.length) && (
|
||||||
|
<Pagination
|
||||||
|
count={filteredGroupMemberships.length}
|
||||||
|
page={page}
|
||||||
|
perPage={perPage}
|
||||||
|
onChangePage={setPage}
|
||||||
|
onChangePerPage={setPerPage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{!isLoading && !filteredGroupMemberships?.length && (
|
||||||
|
<EmptyState
|
||||||
|
title={
|
||||||
|
groupMemberships.length
|
||||||
|
? "No groups match search..."
|
||||||
|
: "This user has not been assigned to any groups"
|
||||||
|
}
|
||||||
|
icon={groupMemberships.length ? faSearch : faUser}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</TableContainer>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -5,7 +5,7 @@ import { zodResolver } from "@hookform/resolvers/zod";
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import { Button, FormControl, Modal, ModalContent, Select, SelectItem } from "@app/components/v2";
|
import { Button, FilterableSelect, FormControl, Modal, ModalContent } from "@app/components/v2";
|
||||||
import { useOrganization, useWorkspace } from "@app/context";
|
import { useOrganization, useWorkspace } from "@app/context";
|
||||||
import {
|
import {
|
||||||
useAddGroupToWorkspace,
|
useAddGroupToWorkspace,
|
||||||
@ -16,8 +16,8 @@ import {
|
|||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
const schema = z.object({
|
const schema = z.object({
|
||||||
id: z.string(),
|
group: z.object({ id: z.string(), name: z.string() }),
|
||||||
role: z.string()
|
role: z.object({ slug: z.string(), name: z.string() })
|
||||||
});
|
});
|
||||||
|
|
||||||
export type FormData = z.infer<typeof schema>;
|
export type FormData = z.infer<typeof schema>;
|
||||||
@ -27,7 +27,9 @@ type Props = {
|
|||||||
handlePopUpToggle: (popUpName: keyof UsePopUpState<["group"]>, state?: boolean) => void;
|
handlePopUpToggle: (popUpName: keyof UsePopUpState<["group"]>, state?: boolean) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const GroupModal = ({ popUp, handlePopUpToggle }: Props) => {
|
// TODO: update backend to support adding multiple roles at once
|
||||||
|
|
||||||
|
const Content = ({ popUp, handlePopUpToggle }: Props) => {
|
||||||
const { currentOrg } = useOrganization();
|
const { currentOrg } = useOrganization();
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace } = useWorkspace();
|
||||||
|
|
||||||
@ -59,12 +61,12 @@ export const GroupModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
resolver: zodResolver(schema)
|
resolver: zodResolver(schema)
|
||||||
});
|
});
|
||||||
|
|
||||||
const onFormSubmit = async ({ id, role }: FormData) => {
|
const onFormSubmit = async ({ group, role }: FormData) => {
|
||||||
try {
|
try {
|
||||||
await addGroupToWorkspaceMutateAsync({
|
await addGroupToWorkspaceMutateAsync({
|
||||||
projectId: currentWorkspace?.id || "",
|
projectId: currentWorkspace?.id || "",
|
||||||
groupId: id,
|
groupId: group.id,
|
||||||
role: role || undefined
|
role: role.slug || undefined
|
||||||
});
|
});
|
||||||
|
|
||||||
reset();
|
reset();
|
||||||
@ -82,95 +84,84 @@ export const GroupModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
return filteredGroupMembershipOrgs.length ? (
|
||||||
|
<form onSubmit={handleSubmit(onFormSubmit)}>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="group"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl label="Group" errorText={error?.message} isError={Boolean(error)}>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
getOptionValue={(option) => option.id}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
options={filteredGroupMembershipOrgs}
|
||||||
|
placeholder="Select group..."
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="role"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
label="Role"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
|
className="mt-4"
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
getOptionValue={(option) => option.slug}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
options={roles}
|
||||||
|
placeholder="Select role..."
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<div className="mt-6 flex items-center">
|
||||||
|
<Button
|
||||||
|
className="mr-4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
isLoading={isSubmitting}
|
||||||
|
isDisabled={isSubmitting}
|
||||||
|
>
|
||||||
|
{popUp?.group?.data ? "Update" : "Add"}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorSchema="secondary"
|
||||||
|
variant="plain"
|
||||||
|
onClick={() => handlePopUpToggle("group", false)}
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
) : (
|
||||||
|
<div className="flex flex-col space-y-4">
|
||||||
|
<div className="text-sm">
|
||||||
|
All groups in your organization have already been added to this project.
|
||||||
|
</div>
|
||||||
|
<Link href={`/org/${currentWorkspace?.orgId}/members`}>
|
||||||
|
<Button variant="outline_bg">Create a new group</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const GroupModal = ({ popUp, handlePopUpToggle }: Props) => {
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
isOpen={popUp?.group?.isOpen}
|
isOpen={popUp?.group?.isOpen}
|
||||||
onOpenChange={(isOpen) => {
|
onOpenChange={(isOpen) => handlePopUpToggle("group", isOpen)}
|
||||||
handlePopUpToggle("group", isOpen);
|
|
||||||
reset();
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<ModalContent title="Add Group to Project">
|
<ModalContent bodyClassName="overflow-visible" title="Add Group to Project">
|
||||||
{filteredGroupMembershipOrgs.length ? (
|
<Content popUp={popUp} handlePopUpToggle={handlePopUpToggle} />
|
||||||
<form onSubmit={handleSubmit(onFormSubmit)}>
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="id"
|
|
||||||
defaultValue={filteredGroupMembershipOrgs?.[0]?.id}
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl label="Group" errorText={error?.message} isError={Boolean(error)}>
|
|
||||||
<Select
|
|
||||||
defaultValue={field.value}
|
|
||||||
{...field}
|
|
||||||
onValueChange={(e) => onChange(e)}
|
|
||||||
className="w-full border border-mineshaft-600"
|
|
||||||
placeholder="Select group..."
|
|
||||||
>
|
|
||||||
{filteredGroupMembershipOrgs.map(({ name, id }) => (
|
|
||||||
<SelectItem value={id} key={`org-group-${id}`}>
|
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="role"
|
|
||||||
defaultValue=""
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
|
||||||
label="Role"
|
|
||||||
errorText={error?.message}
|
|
||||||
isError={Boolean(error)}
|
|
||||||
className="mt-4"
|
|
||||||
>
|
|
||||||
<Select
|
|
||||||
defaultValue={field.value}
|
|
||||||
{...field}
|
|
||||||
onValueChange={(e) => onChange(e)}
|
|
||||||
className="w-full"
|
|
||||||
placeholder="Select role..."
|
|
||||||
>
|
|
||||||
{(roles || []).map(({ name, slug }) => (
|
|
||||||
<SelectItem value={slug} key={`st-role-${slug}`}>
|
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<div className="mt-6 flex items-center">
|
|
||||||
<Button
|
|
||||||
className="mr-4"
|
|
||||||
size="sm"
|
|
||||||
type="submit"
|
|
||||||
isLoading={isSubmitting}
|
|
||||||
isDisabled={isSubmitting}
|
|
||||||
>
|
|
||||||
{popUp?.group?.data ? "Update" : "Create"}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
colorSchema="secondary"
|
|
||||||
variant="plain"
|
|
||||||
onClick={() => handlePopUpToggle("group", false)}
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
) : (
|
|
||||||
<div className="flex flex-col space-y-4">
|
|
||||||
<div className="text-sm">
|
|
||||||
All groups in your organization have already been added to this project.
|
|
||||||
</div>
|
|
||||||
<Link href={`/org/${currentWorkspace?.orgId}/members`}>
|
|
||||||
<Button variant="outline_bg">Create a new group</Button>
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
|
@ -181,7 +181,7 @@ export const IdentityTab = withProjectPermission(
|
|||||||
onClick={() => handlePopUpOpen("identity")}
|
onClick={() => handlePopUpOpen("identity")}
|
||||||
isDisabled={!isAllowed}
|
isDisabled={!isAllowed}
|
||||||
>
|
>
|
||||||
Add identity
|
Add Identity
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
</ProjectPermissionCan>
|
</ProjectPermissionCan>
|
||||||
|
@ -1,12 +1,19 @@
|
|||||||
import { useEffect, useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { yupResolver } from "@hookform/resolvers/yup";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import * as yup from "yup";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import { Button, FormControl, Modal, ModalClose, ModalContent } from "@app/components/v2";
|
import {
|
||||||
import { ComboBox } from "@app/components/v2/ComboBox";
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
|
FormControl,
|
||||||
|
Modal,
|
||||||
|
ModalClose,
|
||||||
|
ModalContent,
|
||||||
|
Spinner
|
||||||
|
} from "@app/components/v2";
|
||||||
import { useOrganization, useWorkspace } from "@app/context";
|
import { useOrganization, useWorkspace } from "@app/context";
|
||||||
import {
|
import {
|
||||||
useAddIdentityToWorkspace,
|
useAddIdentityToWorkspace,
|
||||||
@ -16,37 +23,30 @@ import {
|
|||||||
} from "@app/hooks/api";
|
} from "@app/hooks/api";
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
const schema = yup
|
const schema = z.object({
|
||||||
.object({
|
identity: z.object({ name: z.string(), id: z.string() }),
|
||||||
identity: yup.object({
|
role: z.object({ name: z.string(), slug: z.string() })
|
||||||
id: yup.string().required("Identity id is required"),
|
});
|
||||||
name: yup.string().required("Identity name is required")
|
|
||||||
}),
|
|
||||||
role: yup.object({
|
|
||||||
slug: yup.string().required("role slug is required"),
|
|
||||||
name: yup.string().required("role name is required")
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.required();
|
|
||||||
|
|
||||||
export type FormData = yup.InferType<typeof schema>;
|
export type FormData = z.infer<typeof schema>;
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
popUp: UsePopUpState<["identity"]>;
|
popUp: UsePopUpState<["identity"]>;
|
||||||
handlePopUpToggle: (popUpName: keyof UsePopUpState<["identity"]>, state?: boolean) => void;
|
handlePopUpToggle: (popUpName: keyof UsePopUpState<["identity"]>, state?: boolean) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
const Content = ({ popUp, handlePopUpToggle }: Props) => {
|
||||||
const { currentOrg } = useOrganization();
|
const { currentOrg } = useOrganization();
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace } = useWorkspace();
|
||||||
|
|
||||||
const organizationId = currentOrg?.id || "";
|
const organizationId = currentOrg?.id || "";
|
||||||
const workspaceId = currentWorkspace?.id || "";
|
const workspaceId = currentWorkspace?.id || "";
|
||||||
|
|
||||||
const { data: identityMembershipOrgsData } = useGetIdentityMembershipOrgs({
|
const { data: identityMembershipOrgsData, isLoading: isMembershipsLoading } =
|
||||||
organizationId,
|
useGetIdentityMembershipOrgs({
|
||||||
limit: 20000 // TODO: this is temp to preserve functionality for larger projects, will replace with combobox in separate PR
|
organizationId,
|
||||||
});
|
limit: 20000 // TODO: this is temp to preserve functionality for larger projects, will replace with combobox in separate PR
|
||||||
|
});
|
||||||
const identityMembershipOrgs = identityMembershipOrgsData?.identityMemberships;
|
const identityMembershipOrgs = identityMembershipOrgsData?.identityMemberships;
|
||||||
const { data: identityMembershipsData } = useGetWorkspaceIdentityMemberships({
|
const { data: identityMembershipsData } = useGetWorkspaceIdentityMemberships({
|
||||||
workspaceId,
|
workspaceId,
|
||||||
@ -54,11 +54,7 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
});
|
});
|
||||||
const identityMemberships = identityMembershipsData?.identityMemberships;
|
const identityMemberships = identityMembershipsData?.identityMemberships;
|
||||||
|
|
||||||
const {
|
const { data: roles, isLoading: isRolesLoading } = useGetProjectRoles(workspaceId);
|
||||||
data: roles,
|
|
||||||
isLoading: isRolesLoading,
|
|
||||||
isFetched: isRolesFetched
|
|
||||||
} = useGetProjectRoles(workspaceId);
|
|
||||||
|
|
||||||
const { mutateAsync: addIdentityToWorkspaceMutateAsync } = useAddIdentityToWorkspace();
|
const { mutateAsync: addIdentityToWorkspaceMutateAsync } = useAddIdentityToWorkspace();
|
||||||
|
|
||||||
@ -76,18 +72,11 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
control,
|
control,
|
||||||
handleSubmit,
|
handleSubmit,
|
||||||
reset,
|
reset,
|
||||||
setValue,
|
|
||||||
formState: { isSubmitting }
|
formState: { isSubmitting }
|
||||||
} = useForm<FormData>({
|
} = useForm<FormData>({
|
||||||
resolver: yupResolver(schema)
|
resolver: zodResolver(schema)
|
||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isRolesFetched || !roles) return;
|
|
||||||
|
|
||||||
setValue("role", { name: roles[0]?.name, slug: roles[0]?.slug });
|
|
||||||
}, [isRolesFetched, roles]);
|
|
||||||
|
|
||||||
const onFormSubmit = async ({ identity, role }: FormData) => {
|
const onFormSubmit = async ({ identity, role }: FormData) => {
|
||||||
try {
|
try {
|
||||||
await addIdentityToWorkspaceMutateAsync({
|
await addIdentityToWorkspaceMutateAsync({
|
||||||
@ -125,104 +114,93 @@ export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (isMembershipsLoading || isRolesLoading)
|
||||||
|
return (
|
||||||
|
<div className="flex w-full items-center justify-center py-10">
|
||||||
|
<Spinner className="text-mineshaft-400" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
return filteredIdentityMembershipOrgs.length ? (
|
||||||
|
<form onSubmit={handleSubmit(onFormSubmit)}>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="identity"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl label="Identity" errorText={error?.message} isError={Boolean(error)}>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder="Select identity..."
|
||||||
|
options={filteredIdentityMembershipOrgs.map((membership) => membership.identity)}
|
||||||
|
getOptionValue={(option) => option.id}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Controller
|
||||||
|
control={control}
|
||||||
|
name="role"
|
||||||
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
label="Role"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
|
className="mt-4"
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
options={roles}
|
||||||
|
placeholder="Select role..."
|
||||||
|
getOptionValue={(option) => option.slug}
|
||||||
|
getOptionLabel={(option) => option.name}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<Button
|
||||||
|
className="mr-4"
|
||||||
|
size="sm"
|
||||||
|
type="submit"
|
||||||
|
isLoading={isSubmitting}
|
||||||
|
isDisabled={isSubmitting}
|
||||||
|
>
|
||||||
|
{popUp?.identity?.data ? "Update" : "Add"}
|
||||||
|
</Button>
|
||||||
|
<ModalClose asChild>
|
||||||
|
<Button colorSchema="secondary" variant="plain">
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</ModalClose>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
) : (
|
||||||
|
<div className="flex flex-col space-y-4">
|
||||||
|
<div className="text-sm">
|
||||||
|
All identities in your organization have already been added to this project.
|
||||||
|
</div>
|
||||||
|
<Link href={`/org/${currentWorkspace?.orgId}/members`}>
|
||||||
|
<Button isDisabled={isRolesLoading} isLoading={isRolesLoading} variant="outline_bg">
|
||||||
|
Create a new identity
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const IdentityModal = ({ popUp, handlePopUpToggle }: Props) => {
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
isOpen={popUp?.identity?.isOpen}
|
isOpen={popUp?.identity?.isOpen}
|
||||||
onOpenChange={(isOpen) => {
|
onOpenChange={(isOpen) => {
|
||||||
handlePopUpToggle("identity", isOpen);
|
handlePopUpToggle("identity", isOpen);
|
||||||
reset();
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalContent title="Add Identity to Project" bodyClassName="overflow-visible">
|
<ModalContent title="Add Identity to Project" bodyClassName="overflow-visible">
|
||||||
{filteredIdentityMembershipOrgs.length ? (
|
<Content popUp={popUp} handlePopUpToggle={handlePopUpToggle} />
|
||||||
<form onSubmit={handleSubmit(onFormSubmit)}>
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="identity"
|
|
||||||
defaultValue={{
|
|
||||||
id: filteredIdentityMembershipOrgs?.[0]?.identity?.id,
|
|
||||||
name: filteredIdentityMembershipOrgs?.[0]?.identity?.name
|
|
||||||
}}
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl label="Identity" errorText={error?.message} isError={Boolean(error)}>
|
|
||||||
<ComboBox
|
|
||||||
className="w-full"
|
|
||||||
by="id"
|
|
||||||
value={{ id: field.value.id, name: field.value.name }}
|
|
||||||
defaultValue={{ id: field.value.id, name: field.value.name }}
|
|
||||||
onSelectChange={(value) => onChange({ id: value.id, name: value.name })}
|
|
||||||
displayValue={(el) => el.name}
|
|
||||||
onFilter={({ value }, filterQuery) =>
|
|
||||||
value.name.toLowerCase().includes(filterQuery.toLowerCase())
|
|
||||||
}
|
|
||||||
items={filteredIdentityMembershipOrgs.map(({ identity }) => ({
|
|
||||||
key: identity.id,
|
|
||||||
value: { id: identity.id, name: identity.name },
|
|
||||||
label: identity.name
|
|
||||||
}))}
|
|
||||||
/>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Controller
|
|
||||||
control={control}
|
|
||||||
name="role"
|
|
||||||
defaultValue={{ name: "", slug: "" }}
|
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl
|
|
||||||
label="Role"
|
|
||||||
errorText={error?.message}
|
|
||||||
isError={Boolean(error)}
|
|
||||||
className="mt-4"
|
|
||||||
>
|
|
||||||
<ComboBox
|
|
||||||
className="w-full"
|
|
||||||
by="slug"
|
|
||||||
value={{ slug: field.value.slug, name: field.value.name }}
|
|
||||||
defaultValue={{ slug: field.value.slug, name: field.value.name }}
|
|
||||||
onSelectChange={(value) => onChange({ slug: value.slug, name: value.name })}
|
|
||||||
displayValue={(el) => el.name}
|
|
||||||
onFilter={({ value }, filterQuery) =>
|
|
||||||
value.name.toLowerCase().includes(filterQuery.toLowerCase())
|
|
||||||
}
|
|
||||||
items={(roles || []).map(({ slug, name }) => ({
|
|
||||||
key: slug,
|
|
||||||
value: { slug, name },
|
|
||||||
label: name
|
|
||||||
}))}
|
|
||||||
/>
|
|
||||||
</FormControl>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<div className="flex items-center">
|
|
||||||
<Button
|
|
||||||
className="mr-4"
|
|
||||||
size="sm"
|
|
||||||
type="submit"
|
|
||||||
isLoading={isSubmitting}
|
|
||||||
isDisabled={isSubmitting}
|
|
||||||
>
|
|
||||||
{popUp?.identity?.data ? "Update" : "Create"}
|
|
||||||
</Button>
|
|
||||||
<ModalClose asChild>
|
|
||||||
<Button colorSchema="secondary" variant="plain">
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
</ModalClose>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
) : (
|
|
||||||
<div className="flex flex-col space-y-4">
|
|
||||||
<div className="text-sm">
|
|
||||||
All identities in your organization have already been added to this project.
|
|
||||||
</div>
|
|
||||||
<Link href={`/org/${currentWorkspace?.orgId}/members`}>
|
|
||||||
<Button isDisabled={isRolesLoading} isLoading={isRolesLoading} variant="outline_bg">
|
|
||||||
Create a new identity
|
|
||||||
</Button>
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
|
@ -2,24 +2,11 @@ import { useMemo } from "react";
|
|||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { faCheckCircle, faChevronDown } from "@fortawesome/free-solid-svg-icons";
|
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { twMerge } from "tailwind-merge";
|
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import { Button, FilterableSelect, FormControl, Modal, ModalContent } from "@app/components/v2";
|
||||||
Button,
|
|
||||||
DropdownMenu,
|
|
||||||
DropdownMenuContent,
|
|
||||||
DropdownMenuItem,
|
|
||||||
DropdownMenuTrigger,
|
|
||||||
FilterableSelect,
|
|
||||||
FormControl,
|
|
||||||
Modal,
|
|
||||||
ModalContent
|
|
||||||
} from "@app/components/v2";
|
|
||||||
import { useOrganization, useWorkspace } from "@app/context";
|
import { useOrganization, useWorkspace } from "@app/context";
|
||||||
import {
|
import {
|
||||||
useAddUsersToOrg,
|
useAddUsersToOrg,
|
||||||
@ -33,7 +20,7 @@ import { UsePopUpState } from "@app/hooks/usePopUp";
|
|||||||
|
|
||||||
const addMemberFormSchema = z.object({
|
const addMemberFormSchema = z.object({
|
||||||
orgMemberships: z.array(z.object({ label: z.string().trim(), value: z.string().trim() })).min(1),
|
orgMemberships: z.array(z.object({ label: z.string().trim(), value: z.string().trim() })).min(1),
|
||||||
projectRoleSlugs: z.array(z.string().trim().min(1)).min(1)
|
projectRoleSlugs: z.array(z.object({ slug: z.string().trim(), name: z.string().trim() })).min(1)
|
||||||
});
|
});
|
||||||
|
|
||||||
type TAddMemberForm = z.infer<typeof addMemberFormSchema>;
|
type TAddMemberForm = z.infer<typeof addMemberFormSchema>;
|
||||||
@ -64,7 +51,7 @@ export const AddMemberModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
formState: { isSubmitting, errors }
|
formState: { isSubmitting, errors }
|
||||||
} = useForm<TAddMemberForm>({
|
} = useForm<TAddMemberForm>({
|
||||||
resolver: zodResolver(addMemberFormSchema),
|
resolver: zodResolver(addMemberFormSchema),
|
||||||
defaultValues: { orgMemberships: [], projectRoleSlugs: [ProjectMembershipRole.Member] }
|
defaultValues: { orgMemberships: [], projectRoleSlugs: [] }
|
||||||
});
|
});
|
||||||
|
|
||||||
const { mutateAsync: addMembersToProject } = useAddUsersToOrg();
|
const { mutateAsync: addMembersToProject } = useAddUsersToOrg();
|
||||||
@ -94,7 +81,7 @@ export const AddMemberModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
{
|
{
|
||||||
slug: currentWorkspace.slug,
|
slug: currentWorkspace.slug,
|
||||||
id: currentWorkspace.id,
|
id: currentWorkspace.id,
|
||||||
projectRoleSlug: projectRoleSlugs
|
projectRoleSlug: projectRoleSlugs.map((role) => role.slug)
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
@ -172,78 +159,23 @@ export const AddMemberModal = ({ popUp, handlePopUpToggle }: Props) => {
|
|||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="projectRoleSlugs"
|
name="projectRoleSlugs"
|
||||||
render={({ field }) => (
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
<FormControl
|
<FormControl
|
||||||
className="w-full"
|
className="w-full"
|
||||||
label="Select roles"
|
label="Select roles"
|
||||||
tooltipText="Select the roles that you wish to assign to the users"
|
tooltipText="Select the roles that you wish to assign to the users"
|
||||||
|
errorText={error?.message}
|
||||||
|
isError={Boolean(error)}
|
||||||
>
|
>
|
||||||
<DropdownMenu>
|
<FilterableSelect
|
||||||
<DropdownMenuTrigger asChild>
|
options={roles}
|
||||||
{roles && roles.length > 0 ? (
|
placeholder="Select roles..."
|
||||||
<div className="inline-flex w-full cursor-pointer items-center justify-between rounded-md border border-mineshaft-600 bg-mineshaft-900 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
value={value}
|
||||||
{/* eslint-disable-next-line no-nested-ternary */}
|
onChange={onChange}
|
||||||
{selectedRoleSlugs.length === 1
|
isMulti
|
||||||
? roles.find((role) => role.slug === selectedRoleSlugs[0])?.name
|
getOptionValue={(option) => option.slug}
|
||||||
: selectedRoleSlugs.length === 0
|
getOptionLabel={(option) => option.name}
|
||||||
? "Select at least one role"
|
/>
|
||||||
: `${selectedRoleSlugs.length} roles selected`}
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faChevronDown}
|
|
||||||
className={twMerge("ml-2 text-xs")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="inline-flex w-full cursor-default items-center justify-between rounded-md border border-mineshaft-600 bg-mineshaft-900 px-3 py-2 font-inter text-sm font-normal text-bunker-200 outline-none data-[placeholder]:text-mineshaft-200">
|
|
||||||
No roles found
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</DropdownMenuTrigger>
|
|
||||||
<DropdownMenuContent
|
|
||||||
align="start"
|
|
||||||
className="thin-scrollbar z-[100] max-h-80"
|
|
||||||
>
|
|
||||||
{roles && roles.length > 0 ? (
|
|
||||||
roles.map((role) => {
|
|
||||||
const isSelected = selectedRoleSlugs.includes(role.slug);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<DropdownMenuItem
|
|
||||||
onSelect={(event) => roles.length > 1 && event.preventDefault()}
|
|
||||||
onClick={() => {
|
|
||||||
if (selectedRoleSlugs.includes(String(role.slug))) {
|
|
||||||
field.onChange(
|
|
||||||
selectedRoleSlugs.filter(
|
|
||||||
(roleSlug: string) => roleSlug !== String(role.slug)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
field.onChange([...selectedRoleSlugs, role.slug]);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
key={`role-slug-${role.slug}`}
|
|
||||||
icon={
|
|
||||||
isSelected ? (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={faCheckCircle}
|
|
||||||
className="pr-0.5 text-primary"
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<div className="pl-[1.01rem]" />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
iconPos="left"
|
|
||||||
className="w-[28.4rem] text-sm"
|
|
||||||
>
|
|
||||||
{role.name}
|
|
||||||
</DropdownMenuItem>
|
|
||||||
);
|
|
||||||
})
|
|
||||||
) : (
|
|
||||||
<div />
|
|
||||||
)}
|
|
||||||
</DropdownMenuContent>
|
|
||||||
</DropdownMenu>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -22,7 +22,11 @@ import {
|
|||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { useWorkspace } from "@app/context";
|
import { useWorkspace } from "@app/context";
|
||||||
import { policyDetails } from "@app/helpers/policies";
|
import { policyDetails } from "@app/helpers/policies";
|
||||||
import { useCreateSecretApprovalPolicy, useListWorkspaceGroups, useUpdateSecretApprovalPolicy } from "@app/hooks/api";
|
import {
|
||||||
|
useCreateSecretApprovalPolicy,
|
||||||
|
useListWorkspaceGroups,
|
||||||
|
useUpdateSecretApprovalPolicy
|
||||||
|
} from "@app/hooks/api";
|
||||||
import {
|
import {
|
||||||
useCreateAccessApprovalPolicy,
|
useCreateAccessApprovalPolicy,
|
||||||
useUpdateAccessApprovalPolicy
|
useUpdateAccessApprovalPolicy
|
||||||
@ -46,7 +50,11 @@ const formSchema = z
|
|||||||
name: z.string().optional(),
|
name: z.string().optional(),
|
||||||
secretPath: z.string().optional(),
|
secretPath: z.string().optional(),
|
||||||
approvals: z.number().min(1),
|
approvals: z.number().min(1),
|
||||||
approvers: z.object({type: z.nativeEnum(ApproverType), id: z.string()}).array().min(1).default([]),
|
approvers: z
|
||||||
|
.object({ type: z.nativeEnum(ApproverType), id: z.string() })
|
||||||
|
.array()
|
||||||
|
.min(1)
|
||||||
|
.default([]),
|
||||||
policyType: z.nativeEnum(PolicyType),
|
policyType: z.nativeEnum(PolicyType),
|
||||||
enforcementLevel: z.nativeEnum(EnforcementLevel)
|
enforcementLevel: z.nativeEnum(EnforcementLevel)
|
||||||
})
|
})
|
||||||
@ -100,6 +108,8 @@ export const AccessPolicyForm = ({
|
|||||||
|
|
||||||
const policyName = policyDetails[watch("policyType")]?.name || "Policy";
|
const policyName = policyDetails[watch("policyType")]?.name || "Policy";
|
||||||
|
|
||||||
|
const approversRequired = watch("approvals") || 1;
|
||||||
|
|
||||||
const handleCreatePolicy = async (data: TFormSchema) => {
|
const handleCreatePolicy = async (data: TFormSchema) => {
|
||||||
if (!projectId) return;
|
if (!projectId) return;
|
||||||
|
|
||||||
@ -169,12 +179,6 @@ export const AccessPolicyForm = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const formatEnforcementLevel = (level: EnforcementLevel) => {
|
|
||||||
if (level === EnforcementLevel.Hard) return "Hard";
|
|
||||||
if (level === EnforcementLevel.Soft) return "Soft";
|
|
||||||
return level;
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal isOpen={isOpen} onOpenChange={onToggle}>
|
<Modal isOpen={isOpen} onOpenChange={onToggle}>
|
||||||
<ModalContent title={isEditMode ? `Edit ${policyName}` : "Create Policy"}>
|
<ModalContent title={isEditMode ? `Edit ${policyName}` : "Create Policy"}>
|
||||||
@ -257,14 +261,15 @@ export const AccessPolicyForm = ({
|
|||||||
name="secretPath"
|
name="secretPath"
|
||||||
render={({ field, fieldState: { error } }) => (
|
render={({ field, fieldState: { error } }) => (
|
||||||
<FormControl
|
<FormControl
|
||||||
label="Secret Path"
|
tooltipText="Secret paths support glob patterns. For example, '/**' will match all paths."
|
||||||
isError={Boolean(error)}
|
label="Secret Path"
|
||||||
errorText={error?.message}
|
isError={Boolean(error)}
|
||||||
|
errorText={error?.message}
|
||||||
>
|
>
|
||||||
<Input {...field} value={field.value || ""} />
|
<Input {...field} value={field.value || ""} />
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="approvals"
|
name="approvals"
|
||||||
@ -295,9 +300,11 @@ export const AccessPolicyForm = ({
|
|||||||
errorText={error?.message}
|
errorText={error?.message}
|
||||||
tooltipText="Determines the level of enforcement for required approvers of a request"
|
tooltipText="Determines the level of enforcement for required approvers of a request"
|
||||||
helperText={
|
helperText={
|
||||||
field.value === EnforcementLevel.Hard
|
<div className="ml-1">
|
||||||
? "All approvers must approve the request."
|
{field.value === EnforcementLevel.Hard
|
||||||
: "All approvers must approve the request; however, the requester can bypass approval requirements in emergencies."
|
? `Hard enforcement requires at least ${approversRequired} approver(s) to approve the request.`
|
||||||
|
: `At least ${approversRequired} approver(s) must approve the request; however, the requester can bypass approval requirements in emergencies.`}
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<Select
|
<Select
|
||||||
@ -307,12 +314,8 @@ export const AccessPolicyForm = ({
|
|||||||
>
|
>
|
||||||
{Object.values(EnforcementLevel).map((level) => {
|
{Object.values(EnforcementLevel).map((level) => {
|
||||||
return (
|
return (
|
||||||
<SelectItem
|
<SelectItem value={level} key={`enforcement-level-${level}`}>
|
||||||
value={level}
|
<span className="capitalize">{level}</span>
|
||||||
key={`enforcement-level-${level}`}
|
|
||||||
className="text-xs"
|
|
||||||
>
|
|
||||||
{formatEnforcementLevel(level)}
|
|
||||||
</SelectItem>
|
</SelectItem>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
@ -320,7 +323,12 @@ export const AccessPolicyForm = ({
|
|||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<p>Approvers</p>
|
<div className="mb-2">
|
||||||
|
<p>Approvers</p>
|
||||||
|
<p className="font-inter text-xs text-mineshaft-300 opacity-90">
|
||||||
|
Select members or groups that are allowed to approve requests from this policy.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="approvers"
|
name="approvers"
|
||||||
@ -334,7 +342,11 @@ export const AccessPolicyForm = ({
|
|||||||
<DropdownMenuTrigger asChild>
|
<DropdownMenuTrigger asChild>
|
||||||
<Input
|
<Input
|
||||||
isReadOnly
|
isReadOnly
|
||||||
value={value?.filter((e) => e.type=== ApproverType.User).length ? `${value.filter((e) => e.type=== ApproverType.User).length} selected` : "None"}
|
value={
|
||||||
|
value?.filter((e) => e.type === ApproverType.User).length
|
||||||
|
? `${value.filter((e) => e.type === ApproverType.User).length} selected`
|
||||||
|
: "None"
|
||||||
|
}
|
||||||
className="text-left"
|
className="text-left"
|
||||||
/>
|
/>
|
||||||
</DropdownMenuTrigger>
|
</DropdownMenuTrigger>
|
||||||
@ -347,15 +359,22 @@ export const AccessPolicyForm = ({
|
|||||||
</DropdownMenuLabel>
|
</DropdownMenuLabel>
|
||||||
{members.map(({ user }) => {
|
{members.map(({ user }) => {
|
||||||
const { id: userId } = user;
|
const { id: userId } = user;
|
||||||
const isChecked = value?.filter((el: {id: string, type: ApproverType}) => el.id === userId && el.type === ApproverType.User).length > 0;
|
const isChecked =
|
||||||
|
value?.filter(
|
||||||
|
(el: { id: string; type: ApproverType }) =>
|
||||||
|
el.id === userId && el.type === ApproverType.User
|
||||||
|
).length > 0;
|
||||||
return (
|
return (
|
||||||
<DropdownMenuItem
|
<DropdownMenuItem
|
||||||
onClick={(evt) => {
|
onClick={(evt) => {
|
||||||
evt.preventDefault();
|
evt.preventDefault();
|
||||||
onChange(
|
onChange(
|
||||||
isChecked
|
isChecked
|
||||||
? value?.filter((el: {id: string, type: ApproverType}) => el.id !== userId && el.type !== ApproverType.User)
|
? value?.filter(
|
||||||
: [...(value || []), {id:userId, type: ApproverType.User}]
|
(el: { id: string; type: ApproverType }) =>
|
||||||
|
el.id !== userId && el.type !== ApproverType.User
|
||||||
|
)
|
||||||
|
: [...(value || []), { id: userId, type: ApproverType.User }]
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
key={`create-policy-members-${userId}`}
|
key={`create-policy-members-${userId}`}
|
||||||
@ -384,7 +403,13 @@ export const AccessPolicyForm = ({
|
|||||||
<DropdownMenuTrigger asChild>
|
<DropdownMenuTrigger asChild>
|
||||||
<Input
|
<Input
|
||||||
isReadOnly
|
isReadOnly
|
||||||
value={value?.filter((e) => e.type=== ApproverType.Group).length ? `${value?.filter((e) => e.type=== ApproverType.Group).length} selected` : "None"}
|
value={
|
||||||
|
value?.filter((e) => e.type === ApproverType.Group).length
|
||||||
|
? `${
|
||||||
|
value?.filter((e) => e.type === ApproverType.Group).length
|
||||||
|
} selected`
|
||||||
|
: "None"
|
||||||
|
}
|
||||||
className="text-left"
|
className="text-left"
|
||||||
/>
|
/>
|
||||||
</DropdownMenuTrigger>
|
</DropdownMenuTrigger>
|
||||||
@ -395,28 +420,36 @@ export const AccessPolicyForm = ({
|
|||||||
<DropdownMenuLabel>
|
<DropdownMenuLabel>
|
||||||
Select groups that are allowed to approve requests
|
Select groups that are allowed to approve requests
|
||||||
</DropdownMenuLabel>
|
</DropdownMenuLabel>
|
||||||
{groups && groups.map(({ group }) => {
|
{groups &&
|
||||||
const { id } = group;
|
groups.map(({ group }) => {
|
||||||
const isChecked = value?.filter((el: {id: string, type: ApproverType}) => el.id === id && el.type === ApproverType.Group).length > 0;
|
const { id } = group;
|
||||||
|
const isChecked =
|
||||||
|
value?.filter(
|
||||||
|
(el: { id: string; type: ApproverType }) =>
|
||||||
|
el.id === id && el.type === ApproverType.Group
|
||||||
|
).length > 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DropdownMenuItem
|
<DropdownMenuItem
|
||||||
onClick={(evt) => {
|
onClick={(evt) => {
|
||||||
evt.preventDefault();
|
evt.preventDefault();
|
||||||
onChange(
|
onChange(
|
||||||
isChecked
|
isChecked
|
||||||
? value?.filter((el: {id: string, type: ApproverType}) => el.id !== id && el.type !== ApproverType.Group)
|
? value?.filter(
|
||||||
: [...(value || []), {id, type: ApproverType.Group}]
|
(el: { id: string; type: ApproverType }) =>
|
||||||
);
|
el.id !== id && el.type !== ApproverType.Group
|
||||||
}}
|
)
|
||||||
key={`create-policy-members-${id}`}
|
: [...(value || []), { id, type: ApproverType.Group }]
|
||||||
iconPos="right"
|
);
|
||||||
icon={isChecked && <FontAwesomeIcon icon={faCheckCircle} />}
|
}}
|
||||||
>
|
key={`create-policy-members-${id}`}
|
||||||
{group.name}
|
iconPos="right"
|
||||||
</DropdownMenuItem>
|
icon={isChecked && <FontAwesomeIcon icon={faCheckCircle} />}
|
||||||
);
|
>
|
||||||
})}
|
{group.name}
|
||||||
|
</DropdownMenuItem>
|
||||||
|
);
|
||||||
|
})}
|
||||||
</DropdownMenuContent>
|
</DropdownMenuContent>
|
||||||
</DropdownMenu>
|
</DropdownMenu>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
@ -6,6 +6,7 @@ import { z } from "zod";
|
|||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
|
FilterableSelect,
|
||||||
FormControl,
|
FormControl,
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent,
|
||||||
@ -17,7 +18,7 @@ import { useSubscription, useWorkspace } from "@app/context";
|
|||||||
import { useCreateSecretImport } from "@app/hooks/api";
|
import { useCreateSecretImport } from "@app/hooks/api";
|
||||||
|
|
||||||
const typeSchema = z.object({
|
const typeSchema = z.object({
|
||||||
environment: z.string().trim(),
|
environment: z.object({ name: z.string(), slug: z.string() }),
|
||||||
secretPath: z
|
secretPath: z
|
||||||
.string()
|
.string()
|
||||||
.trim()
|
.trim()
|
||||||
@ -80,7 +81,7 @@ export const CreateSecretImportForm = ({
|
|||||||
path: secretPath,
|
path: secretPath,
|
||||||
isReplication,
|
isReplication,
|
||||||
import: {
|
import: {
|
||||||
environment: importedEnv,
|
environment: importedEnv.slug,
|
||||||
path: importedSecPath
|
path: importedSecPath
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -88,8 +89,9 @@ export const CreateSecretImportForm = ({
|
|||||||
reset();
|
reset();
|
||||||
createNotification({
|
createNotification({
|
||||||
type: "success",
|
type: "success",
|
||||||
text: `Successfully linked. ${isReplication ? "Please refresh the dashboard to view changes" : ""
|
text: `Successfully linked. ${
|
||||||
}`
|
isReplication ? "Please refresh the dashboard to view changes" : ""
|
||||||
|
}`
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
@ -111,6 +113,7 @@ export const CreateSecretImportForm = ({
|
|||||||
return (
|
return (
|
||||||
<Modal isOpen={isOpen} onOpenChange={onTogglePopUp}>
|
<Modal isOpen={isOpen} onOpenChange={onTogglePopUp}>
|
||||||
<ModalContent
|
<ModalContent
|
||||||
|
bodyClassName="overflow-visible"
|
||||||
title="Add Secret Link"
|
title="Add Secret Link"
|
||||||
subTitle="To inherit secrets from another environment or folder"
|
subTitle="To inherit secrets from another environment or folder"
|
||||||
>
|
>
|
||||||
@ -118,21 +121,16 @@ export const CreateSecretImportForm = ({
|
|||||||
<Controller
|
<Controller
|
||||||
control={control}
|
control={control}
|
||||||
name="environment"
|
name="environment"
|
||||||
defaultValue={environments?.[0]?.slug}
|
render={({ field: { onChange, value }, fieldState: { error } }) => (
|
||||||
render={({ field: { onChange, ...field }, fieldState: { error } }) => (
|
|
||||||
<FormControl label="Environment" errorText={error?.message} isError={Boolean(error)}>
|
<FormControl label="Environment" errorText={error?.message} isError={Boolean(error)}>
|
||||||
<Select
|
<FilterableSelect
|
||||||
defaultValue={field.value}
|
options={environments}
|
||||||
{...field}
|
getOptionLabel={(option) => option.name}
|
||||||
onValueChange={(e) => onChange(e)}
|
getOptionValue={(option) => option.slug}
|
||||||
className="w-full"
|
placeholder="Select environment..."
|
||||||
>
|
value={value}
|
||||||
{environments.map(({ name, slug }) => (
|
onChange={onChange}
|
||||||
<SelectItem value={slug} key={slug}>
|
/>
|
||||||
{name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
@ -142,7 +140,7 @@ export const CreateSecretImportForm = ({
|
|||||||
defaultValue="/"
|
defaultValue="/"
|
||||||
render={({ field, fieldState: { error } }) => (
|
render={({ field, fieldState: { error } }) => (
|
||||||
<FormControl label="Secret Path" isError={Boolean(error)} errorText={error?.message}>
|
<FormControl label="Secret Path" isError={Boolean(error)} errorText={error?.message}>
|
||||||
<SecretPathInput {...field} environment={selectedEnvironment} />
|
<SecretPathInput {...field} environment={selectedEnvironment?.slug} />
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
@ -15,7 +15,10 @@ const formSchema = z.object({
|
|||||||
name: z
|
name: z
|
||||||
.string()
|
.string()
|
||||||
.trim()
|
.trim()
|
||||||
.regex(/^[a-zA-Z0-9-_]+$/, "Folder name cannot contain spaces. Only underscore and dashes")
|
.regex(
|
||||||
|
/^[a-zA-Z0-9-_]+$/,
|
||||||
|
"Folder name can only contain letters, numbers, dashes, and underscores"
|
||||||
|
)
|
||||||
});
|
});
|
||||||
type TFormData = z.infer<typeof formSchema>;
|
type TFormData = z.infer<typeof formSchema>;
|
||||||
|
|
||||||
|
@ -1,14 +1,7 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import { subject } from "@casl/ability";
|
import { subject } from "@casl/ability";
|
||||||
import {
|
import { faClone, faFileImport, faSquareCheck } from "@fortawesome/free-solid-svg-icons";
|
||||||
faClone,
|
|
||||||
faFileImport,
|
|
||||||
faKey,
|
|
||||||
faSearch,
|
|
||||||
faSquareCheck,
|
|
||||||
faSquareXmark
|
|
||||||
} from "@fortawesome/free-solid-svg-icons";
|
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
@ -16,17 +9,13 @@ import { z } from "zod";
|
|||||||
import { ProjectPermissionCan } from "@app/components/permissions";
|
import { ProjectPermissionCan } from "@app/components/permissions";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
Checkbox,
|
FilterableSelect,
|
||||||
EmptyState,
|
|
||||||
FormControl,
|
FormControl,
|
||||||
IconButton,
|
IconButton,
|
||||||
Input,
|
|
||||||
Modal,
|
Modal,
|
||||||
ModalContent,
|
ModalContent,
|
||||||
ModalTrigger,
|
ModalTrigger,
|
||||||
Select,
|
Switch,
|
||||||
SelectItem,
|
|
||||||
Skeleton,
|
|
||||||
Tooltip
|
Tooltip
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { SecretPathInput } from "@app/components/v2/SecretPathInput";
|
import { SecretPathInput } from "@app/components/v2/SecretPathInput";
|
||||||
@ -35,14 +24,17 @@ import { useDebounce } from "@app/hooks";
|
|||||||
import { useGetProjectSecrets } from "@app/hooks/api";
|
import { useGetProjectSecrets } from "@app/hooks/api";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
environment: z.string().trim(),
|
environment: z.object({ name: z.string(), slug: z.string() }),
|
||||||
secretPath: z
|
secretPath: z
|
||||||
.string()
|
.string()
|
||||||
.trim()
|
.trim()
|
||||||
.transform((val) =>
|
.transform((val) =>
|
||||||
typeof val === "string" && val.at(-1) === "/" && val.length > 1 ? val.slice(0, -1) : val
|
typeof val === "string" && val.at(-1) === "/" && val.length > 1 ? val.slice(0, -1) : val
|
||||||
),
|
),
|
||||||
secrets: z.record(z.string().optional().nullable())
|
secrets: z
|
||||||
|
.object({ key: z.string(), value: z.string().optional() })
|
||||||
|
.array()
|
||||||
|
.min(1, "Select one or more secrets to copy")
|
||||||
});
|
});
|
||||||
|
|
||||||
type TFormSchema = z.infer<typeof formSchema>;
|
type TFormSchema = z.infer<typeof formSchema>;
|
||||||
@ -68,7 +60,6 @@ export const CopySecretsFromBoard = ({
|
|||||||
onToggle,
|
onToggle,
|
||||||
onParsedEnv
|
onParsedEnv
|
||||||
}: Props) => {
|
}: Props) => {
|
||||||
const [searchFilter, setSearchFilter] = useState("");
|
|
||||||
const [shouldIncludeValues, setShouldIncludeValues] = useState(true);
|
const [shouldIncludeValues, setShouldIncludeValues] = useState(true);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
@ -80,7 +71,7 @@ export const CopySecretsFromBoard = ({
|
|||||||
formState: { isDirty }
|
formState: { isDirty }
|
||||||
} = useForm<TFormSchema>({
|
} = useForm<TFormSchema>({
|
||||||
resolver: zodResolver(formSchema),
|
resolver: zodResolver(formSchema),
|
||||||
defaultValues: { secretPath: "/", environment: environments?.[0]?.slug }
|
defaultValues: { secretPath: "/", environment: environments?.[0] }
|
||||||
});
|
});
|
||||||
|
|
||||||
const envCopySecPath = watch("secretPath");
|
const envCopySecPath = watch("secretPath");
|
||||||
@ -89,7 +80,7 @@ export const CopySecretsFromBoard = ({
|
|||||||
|
|
||||||
const { data: secrets, isLoading: isSecretsLoading } = useGetProjectSecrets({
|
const { data: secrets, isLoading: isSecretsLoading } = useGetProjectSecrets({
|
||||||
workspaceId,
|
workspaceId,
|
||||||
environment: selectedEnvSlug,
|
environment: selectedEnvSlug.slug,
|
||||||
secretPath: debouncedEnvCopySecretPath,
|
secretPath: debouncedEnvCopySecretPath,
|
||||||
options: {
|
options: {
|
||||||
enabled:
|
enabled:
|
||||||
@ -101,29 +92,22 @@ export const CopySecretsFromBoard = ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setValue("secrets", {});
|
setValue("secrets", []);
|
||||||
setSearchFilter("");
|
}, [debouncedEnvCopySecretPath, selectedEnvSlug]);
|
||||||
}, [debouncedEnvCopySecretPath]);
|
|
||||||
|
|
||||||
const handleSecSelectAll = () => {
|
const handleSecSelectAll = () => {
|
||||||
if (secrets) {
|
if (secrets) {
|
||||||
setValue(
|
setValue("secrets", secrets, { shouldDirty: true });
|
||||||
"secrets",
|
|
||||||
secrets?.reduce((prev, curr) => ({ ...prev, [curr.key]: curr.value }), {}),
|
|
||||||
{ shouldDirty: true }
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleFormSubmit = async (data: TFormSchema) => {
|
const handleFormSubmit = async (data: TFormSchema) => {
|
||||||
const secretsToBePulled: Record<string, { value: string; comments: string[] }> = {};
|
const secretsToBePulled: Record<string, { value: string; comments: string[] }> = {};
|
||||||
Object.keys(data.secrets || {}).forEach((key) => {
|
data.secrets.forEach(({ key, value }) => {
|
||||||
if (data.secrets[key]) {
|
secretsToBePulled[key] = {
|
||||||
secretsToBePulled[key] = {
|
value: (shouldIncludeValues && value) || "",
|
||||||
value: (shouldIncludeValues && data.secrets[key]) || "",
|
comments: [""]
|
||||||
comments: [""]
|
};
|
||||||
};
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
onParsedEnv(secretsToBePulled);
|
onParsedEnv(secretsToBePulled);
|
||||||
onToggle(false);
|
onToggle(false);
|
||||||
@ -136,7 +120,6 @@ export const CopySecretsFromBoard = ({
|
|||||||
onOpenChange={(state) => {
|
onOpenChange={(state) => {
|
||||||
onToggle(state);
|
onToggle(state);
|
||||||
reset();
|
reset();
|
||||||
setSearchFilter("");
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ModalTrigger asChild>
|
<ModalTrigger asChild>
|
||||||
@ -165,6 +148,7 @@ export const CopySecretsFromBoard = ({
|
|||||||
</div>
|
</div>
|
||||||
</ModalTrigger>
|
</ModalTrigger>
|
||||||
<ModalContent
|
<ModalContent
|
||||||
|
bodyClassName="overflow-visible"
|
||||||
className="max-w-2xl"
|
className="max-w-2xl"
|
||||||
title="Copy Secret From An Environment"
|
title="Copy Secret From An Environment"
|
||||||
subTitle="Copy/paste secrets from other environments into this context"
|
subTitle="Copy/paste secrets from other environments into this context"
|
||||||
@ -176,22 +160,14 @@ export const CopySecretsFromBoard = ({
|
|||||||
name="environment"
|
name="environment"
|
||||||
render={({ field: { value, onChange } }) => (
|
render={({ field: { value, onChange } }) => (
|
||||||
<FormControl label="Environment" isRequired className="w-1/3">
|
<FormControl label="Environment" isRequired className="w-1/3">
|
||||||
<Select
|
<FilterableSelect
|
||||||
value={value}
|
value={value}
|
||||||
onValueChange={(val) => onChange(val)}
|
onChange={onChange}
|
||||||
className="w-full border border-mineshaft-500"
|
options={environments}
|
||||||
defaultValue={environments?.[0]?.slug}
|
placeholder="Select environment..."
|
||||||
position="popper"
|
getOptionLabel={(option) => option.name}
|
||||||
>
|
getOptionValue={(option) => option.slug}
|
||||||
{environments.map((sourceEnvironment) => (
|
/>
|
||||||
<SelectItem
|
|
||||||
value={sourceEnvironment.slug}
|
|
||||||
key={`source-environment-${sourceEnvironment.slug}`}
|
|
||||||
>
|
|
||||||
{sourceEnvironment.name}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
@ -203,7 +179,7 @@ export const CopySecretsFromBoard = ({
|
|||||||
<SecretPathInput
|
<SecretPathInput
|
||||||
{...field}
|
{...field}
|
||||||
placeholder="Provide a path, default is /"
|
placeholder="Provide a path, default is /"
|
||||||
environment={selectedEnvSlug}
|
environment={selectedEnvSlug?.slug}
|
||||||
/>
|
/>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
@ -212,72 +188,57 @@ export const CopySecretsFromBoard = ({
|
|||||||
<div className="border-t border-mineshaft-600 pt-4">
|
<div className="border-t border-mineshaft-600 pt-4">
|
||||||
<div className="mb-4 flex items-center justify-between">
|
<div className="mb-4 flex items-center justify-between">
|
||||||
<div>Secrets</div>
|
<div>Secrets</div>
|
||||||
<div className="flex w-1/2 items-center space-x-2">
|
</div>
|
||||||
<Input
|
<div className="flex w-full items-start gap-3">
|
||||||
placeholder="Search for secret"
|
<Controller
|
||||||
value={searchFilter}
|
control={control}
|
||||||
|
name="secrets"
|
||||||
|
render={({ field: { value, onChange }, fieldState: { error } }) => (
|
||||||
|
<FormControl
|
||||||
|
className="flex-1"
|
||||||
|
isError={Boolean(error)}
|
||||||
|
errorText={error?.message}
|
||||||
|
>
|
||||||
|
<FilterableSelect
|
||||||
|
placeholder={
|
||||||
|
// eslint-disable-next-line no-nested-ternary
|
||||||
|
isSecretsLoading
|
||||||
|
? "Loading secrets..."
|
||||||
|
: secrets?.length
|
||||||
|
? "Select secrets..."
|
||||||
|
: "No secrets found..."
|
||||||
|
}
|
||||||
|
isLoading={isSecretsLoading}
|
||||||
|
options={secrets}
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
isMulti
|
||||||
|
getOptionValue={(option) => option.key}
|
||||||
|
getOptionLabel={(option) => option.key}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Tooltip content="Select All">
|
||||||
|
<IconButton
|
||||||
|
className="mt-1 h-9 w-9"
|
||||||
|
ariaLabel="Select all"
|
||||||
|
variant="outline_bg"
|
||||||
size="xs"
|
size="xs"
|
||||||
leftIcon={<FontAwesomeIcon icon={faSearch} />}
|
onClick={handleSecSelectAll}
|
||||||
onChange={(evt) => setSearchFilter(evt.target.value)}
|
>
|
||||||
/>
|
<FontAwesomeIcon icon={faSquareCheck} size="lg" />
|
||||||
<Tooltip content="Select All">
|
</IconButton>
|
||||||
<IconButton
|
</Tooltip>
|
||||||
ariaLabel="Select all"
|
|
||||||
variant="outline_bg"
|
|
||||||
size="xs"
|
|
||||||
onClick={handleSecSelectAll}
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={faSquareCheck} size="lg" />
|
|
||||||
</IconButton>
|
|
||||||
</Tooltip>
|
|
||||||
<Tooltip content="Unselect All">
|
|
||||||
<IconButton
|
|
||||||
ariaLabel="UnSelect all"
|
|
||||||
variant="outline_bg"
|
|
||||||
size="xs"
|
|
||||||
onClick={() => reset()}
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={faSquareXmark} size="lg" />
|
|
||||||
</IconButton>
|
|
||||||
</Tooltip>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
{!isSecretsLoading && !secrets?.length && (
|
<div className="my-6 ml-2">
|
||||||
<EmptyState title="No secrets found" icon={faKey} />
|
<Switch
|
||||||
)}
|
|
||||||
<div className="thin-scrollbar grid max-h-64 grid-cols-2 gap-4 overflow-auto ">
|
|
||||||
{isSecretsLoading &&
|
|
||||||
Array.apply(0, Array(2)).map((_x, i) => (
|
|
||||||
<Skeleton key={`secret-pull-loading-${i + 1}`} className="bg-mineshaft-700" />
|
|
||||||
))}
|
|
||||||
|
|
||||||
{secrets
|
|
||||||
?.filter(({ key }) => key.toLowerCase().includes(searchFilter.toLowerCase()))
|
|
||||||
?.map(({ id, key, value: secVal }) => (
|
|
||||||
<Controller
|
|
||||||
key={`pull-secret--${id}`}
|
|
||||||
control={control}
|
|
||||||
name={`secrets.${key}`}
|
|
||||||
render={({ field: { value, onChange } }) => (
|
|
||||||
<Checkbox
|
|
||||||
id={`pull-secret-${id}`}
|
|
||||||
isChecked={Boolean(value)}
|
|
||||||
onCheckedChange={(isChecked) => onChange(isChecked ? secVal : "")}
|
|
||||||
>
|
|
||||||
{key}
|
|
||||||
</Checkbox>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
<div className="mt-6 mb-4">
|
|
||||||
<Checkbox
|
|
||||||
id="populate-include-value"
|
id="populate-include-value"
|
||||||
isChecked={shouldIncludeValues}
|
isChecked={shouldIncludeValues}
|
||||||
onCheckedChange={(isChecked) => setShouldIncludeValues(isChecked as boolean)}
|
onCheckedChange={(isChecked) => setShouldIncludeValues(isChecked as boolean)}
|
||||||
>
|
>
|
||||||
Include secret values
|
Include secret values
|
||||||
</Checkbox>
|
</Switch>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center space-x-4">
|
<div className="flex items-center space-x-4">
|
||||||
<Button
|
<Button
|
||||||
@ -285,7 +246,7 @@ export const CopySecretsFromBoard = ({
|
|||||||
type="submit"
|
type="submit"
|
||||||
isDisabled={!isDirty}
|
isDisabled={!isDirty}
|
||||||
>
|
>
|
||||||
Paste Secrets
|
Copy Secrets
|
||||||
</Button>
|
</Button>
|
||||||
<Button variant="plain" colorSchema="secondary" onClick={() => onToggle(false)}>
|
<Button variant="plain" colorSchema="secondary" onClick={() => onToggle(false)}>
|
||||||
Cancel
|
Cancel
|
||||||
|
@ -1128,7 +1128,6 @@ export const SecretOverviewPage = () => {
|
|||||||
>
|
>
|
||||||
<CreateSecretForm
|
<CreateSecretForm
|
||||||
secretPath={secretPath}
|
secretPath={secretPath}
|
||||||
getSecretByKey={getSecretByKey}
|
|
||||||
onClose={() => handlePopUpClose("addSecretsInAllEnvs")}
|
onClose={() => handlePopUpClose("addSecretsInAllEnvs")}
|
||||||
/>
|
/>
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
import { ClipboardEvent } from "react";
|
import { ClipboardEvent } from "react";
|
||||||
import { Controller, useForm } from "react-hook-form";
|
import { Controller, useForm } from "react-hook-form";
|
||||||
import { subject } from "@casl/ability";
|
import { subject } from "@casl/ability";
|
||||||
import { faTriangleExclamation, faWarning } from "@fortawesome/free-solid-svg-icons";
|
import { faTriangleExclamation } from "@fortawesome/free-solid-svg-icons";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
import { createNotification } from "@app/components/notifications";
|
import { createNotification } from "@app/components/notifications";
|
||||||
import { Button, Checkbox, FormControl, FormLabel, Input, Tooltip } from "@app/components/v2";
|
import { Button, FilterableSelect, FormControl, Input } from "@app/components/v2";
|
||||||
import { CreatableSelect } from "@app/components/v2/CreatableSelect";
|
import { CreatableSelect } from "@app/components/v2/CreatableSelect";
|
||||||
import { InfisicalSecretInput } from "@app/components/v2/InfisicalSecretInput";
|
import { InfisicalSecretInput } from "@app/components/v2/InfisicalSecretInput";
|
||||||
import {
|
import {
|
||||||
@ -17,20 +17,14 @@ import {
|
|||||||
useWorkspace
|
useWorkspace
|
||||||
} from "@app/context";
|
} from "@app/context";
|
||||||
import { getKeyValue } from "@app/helpers/parseEnvVar";
|
import { getKeyValue } from "@app/helpers/parseEnvVar";
|
||||||
import {
|
import { useCreateFolder, useCreateSecretV3, useCreateWsTag, useGetWsTags } from "@app/hooks/api";
|
||||||
useCreateFolder,
|
import { SecretType } from "@app/hooks/api/types";
|
||||||
useCreateSecretV3,
|
|
||||||
useCreateWsTag,
|
|
||||||
useGetWsTags,
|
|
||||||
useUpdateSecretV3
|
|
||||||
} from "@app/hooks/api";
|
|
||||||
import { SecretType, SecretV3RawSanitized } from "@app/hooks/api/types";
|
|
||||||
|
|
||||||
const typeSchema = z
|
const typeSchema = z
|
||||||
.object({
|
.object({
|
||||||
key: z.string().trim().min(1, "Key is required"),
|
key: z.string().trim().min(1, "Key is required"),
|
||||||
value: z.string().optional(),
|
value: z.string().optional(),
|
||||||
environments: z.record(z.boolean().optional()),
|
environments: z.object({ name: z.string(), slug: z.string() }).array(),
|
||||||
tags: z.array(z.object({ label: z.string().trim(), value: z.string().trim() })).optional()
|
tags: z.array(z.object({ label: z.string().trim(), value: z.string().trim() })).optional()
|
||||||
})
|
})
|
||||||
.refine((data) => data.key !== undefined, {
|
.refine((data) => data.key !== undefined, {
|
||||||
@ -41,22 +35,19 @@ type TFormSchema = z.infer<typeof typeSchema>;
|
|||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
secretPath?: string;
|
secretPath?: string;
|
||||||
getSecretByKey: (slug: string, key: string) => SecretV3RawSanitized | undefined;
|
|
||||||
// modal props
|
// modal props
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const CreateSecretForm = ({ secretPath = "/", getSecretByKey, onClose }: Props) => {
|
export const CreateSecretForm = ({ secretPath = "/", onClose }: Props) => {
|
||||||
const {
|
const {
|
||||||
register,
|
register,
|
||||||
handleSubmit,
|
handleSubmit,
|
||||||
control,
|
control,
|
||||||
reset,
|
reset,
|
||||||
watch,
|
|
||||||
setValue,
|
setValue,
|
||||||
formState: { isSubmitting, errors }
|
formState: { isSubmitting, errors }
|
||||||
} = useForm<TFormSchema>({ resolver: zodResolver(typeSchema) });
|
} = useForm<TFormSchema>({ resolver: zodResolver(typeSchema) });
|
||||||
const newSecretKey = watch("key");
|
|
||||||
|
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace } = useWorkspace();
|
||||||
const { permission } = useProjectPermission();
|
const { permission } = useProjectPermission();
|
||||||
@ -65,22 +56,13 @@ export const CreateSecretForm = ({ secretPath = "/", getSecretByKey, onClose }:
|
|||||||
const environments = currentWorkspace?.environments || [];
|
const environments = currentWorkspace?.environments || [];
|
||||||
|
|
||||||
const { mutateAsync: createSecretV3 } = useCreateSecretV3();
|
const { mutateAsync: createSecretV3 } = useCreateSecretV3();
|
||||||
const { mutateAsync: updateSecretV3 } = useUpdateSecretV3();
|
|
||||||
const { mutateAsync: createFolder } = useCreateFolder();
|
const { mutateAsync: createFolder } = useCreateFolder();
|
||||||
const { data: projectTags, isLoading: isTagsLoading } = useGetWsTags(
|
const { data: projectTags, isLoading: isTagsLoading } = useGetWsTags(
|
||||||
canReadTags ? workspaceId : ""
|
canReadTags ? workspaceId : ""
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleFormSubmit = async ({ key, value, environments: selectedEnv, tags }: TFormSchema) => {
|
const handleFormSubmit = async ({ key, value, environments: selectedEnv, tags }: TFormSchema) => {
|
||||||
const environmentsSelected = environments.filter(({ slug }) => selectedEnv[slug]);
|
const promises = selectedEnv.map(async (env) => {
|
||||||
const isEnvironmentsSelected = environmentsSelected.length;
|
|
||||||
|
|
||||||
if (!isEnvironmentsSelected) {
|
|
||||||
createNotification({ type: "error", text: "Select at least one environment" });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const promises = environmentsSelected.map(async (env) => {
|
|
||||||
const environment = env.slug;
|
const environment = env.slug;
|
||||||
// create folder if not existing
|
// create folder if not existing
|
||||||
if (secretPath !== "/") {
|
if (secretPath !== "/") {
|
||||||
@ -106,21 +88,7 @@ export const CreateSecretForm = ({ secretPath = "/", getSecretByKey, onClose }:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const isEdit = getSecretByKey(environment, key) !== undefined;
|
// TODO: add back ability to overwrite - need to fetch secrets by key to check for conflicts as previous method broke with pagination
|
||||||
if (isEdit) {
|
|
||||||
return {
|
|
||||||
...(await updateSecretV3({
|
|
||||||
environment,
|
|
||||||
workspaceId,
|
|
||||||
secretPath,
|
|
||||||
secretKey: key,
|
|
||||||
secretValue: value || "",
|
|
||||||
type: SecretType.Shared,
|
|
||||||
tagIds: tags?.map((el) => el.value)
|
|
||||||
})),
|
|
||||||
environment
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...(await createSecretV3({
|
...(await createSecretV3({
|
||||||
@ -278,54 +246,33 @@ export const CreateSecretForm = ({ secretPath = "/", getSecretByKey, onClose }:
|
|||||||
</FormControl>
|
</FormControl>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
<FormLabel label="Environments" className="mb-2" />
|
<Controller
|
||||||
<div className="thin-scrollbar grid max-h-64 grid-cols-3 gap-4 overflow-auto py-2">
|
control={control}
|
||||||
{environments
|
render={({ field: { value, onChange }, fieldState: { error } }) => (
|
||||||
.filter((environmentSlug) =>
|
<FormControl label="Environments" isError={Boolean(error)} errorText={error?.message}>
|
||||||
permission.can(
|
<FilterableSelect
|
||||||
ProjectPermissionActions.Create,
|
isMulti
|
||||||
subject(ProjectPermissionSub.Secrets, {
|
options={environments.filter((environment) =>
|
||||||
environment: environmentSlug.slug,
|
permission.can(
|
||||||
secretPath,
|
ProjectPermissionActions.Create,
|
||||||
secretName: "*",
|
subject(ProjectPermissionSub.Secrets, {
|
||||||
secretTags: ["*"]
|
environment: environment.slug,
|
||||||
})
|
secretPath,
|
||||||
)
|
secretName: "*",
|
||||||
)
|
secretTags: ["*"]
|
||||||
.map((env) => {
|
})
|
||||||
return (
|
)
|
||||||
<Controller
|
)}
|
||||||
name={`environments.${env.slug}`}
|
value={value}
|
||||||
key={`secret-input-${env.slug}`}
|
onChange={onChange}
|
||||||
control={control}
|
placeholder="Select environments to create secret in..."
|
||||||
render={({ field }) => (
|
getOptionLabel={(option) => option.name}
|
||||||
<Checkbox
|
getOptionValue={(option) => option.slug}
|
||||||
isChecked={field.value}
|
/>
|
||||||
onCheckedChange={field.onChange}
|
</FormControl>
|
||||||
id={`secret-input-${env.slug}`}
|
)}
|
||||||
className="!justify-start"
|
name="environments"
|
||||||
>
|
/>
|
||||||
<span className="flex w-full flex-row items-center justify-start whitespace-pre-wrap">
|
|
||||||
<span title={env.name} className="truncate">
|
|
||||||
{env.name}
|
|
||||||
</span>
|
|
||||||
<span>
|
|
||||||
{getSecretByKey(env.slug, newSecretKey) && (
|
|
||||||
<Tooltip
|
|
||||||
className="max-w-[150px]"
|
|
||||||
content="Secret already exists, and it will be overwritten"
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={faWarning} className="ml-1 text-yellow-400" />
|
|
||||||
</Tooltip>
|
|
||||||
)}
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</Checkbox>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
<div className="mt-7 flex items-center">
|
<div className="mt-7 flex items-center">
|
||||||
<Button
|
<Button
|
||||||
isDisabled={isSubmitting}
|
isDisabled={isSubmitting}
|
||||||
|
@ -19,7 +19,6 @@ import { SecretTagsTable } from "./SecretTagsTable";
|
|||||||
type DeleteModalData = { name: string; id: string };
|
type DeleteModalData = { name: string; id: string };
|
||||||
|
|
||||||
export const SecretTagsSection = (): JSX.Element => {
|
export const SecretTagsSection = (): JSX.Element => {
|
||||||
|
|
||||||
const { popUp, handlePopUpToggle, handlePopUpClose, handlePopUpOpen } = usePopUp([
|
const { popUp, handlePopUpToggle, handlePopUpClose, handlePopUpOpen } = usePopUp([
|
||||||
"CreateSecretTag",
|
"CreateSecretTag",
|
||||||
"deleteTagConfirmation"
|
"deleteTagConfirmation"
|
||||||
@ -65,7 +64,7 @@ export const SecretTagsSection = (): JSX.Element => {
|
|||||||
}}
|
}}
|
||||||
isDisabled={!isAllowed}
|
isDisabled={!isAllowed}
|
||||||
>
|
>
|
||||||
Create tag
|
Create Tag
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
</ProjectPermissionCan>
|
</ProjectPermissionCan>
|
||||||
|
@ -1,10 +1,20 @@
|
|||||||
import { faTags, faTrashCan } from "@fortawesome/free-solid-svg-icons";
|
import { useMemo } from "react";
|
||||||
|
import {
|
||||||
|
faArrowDown,
|
||||||
|
faArrowUp,
|
||||||
|
faMagnifyingGlass,
|
||||||
|
faSearch,
|
||||||
|
faTag,
|
||||||
|
faTrashCan
|
||||||
|
} from "@fortawesome/free-solid-svg-icons";
|
||||||
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
|
||||||
|
|
||||||
import { ProjectPermissionCan } from "@app/components/permissions";
|
import { ProjectPermissionCan } from "@app/components/permissions";
|
||||||
import {
|
import {
|
||||||
EmptyState,
|
EmptyState,
|
||||||
IconButton,
|
IconButton,
|
||||||
|
Input,
|
||||||
|
Pagination,
|
||||||
Table,
|
Table,
|
||||||
TableContainer,
|
TableContainer,
|
||||||
TableSkeleton,
|
TableSkeleton,
|
||||||
@ -15,7 +25,9 @@ import {
|
|||||||
Tr
|
Tr
|
||||||
} from "@app/components/v2";
|
} from "@app/components/v2";
|
||||||
import { ProjectPermissionActions, ProjectPermissionSub, useWorkspace } from "@app/context";
|
import { ProjectPermissionActions, ProjectPermissionSub, useWorkspace } from "@app/context";
|
||||||
|
import { usePagination, useResetPageHelper } from "@app/hooks";
|
||||||
import { useGetWsTags } from "@app/hooks/api";
|
import { useGetWsTags } from "@app/hooks/api";
|
||||||
|
import { OrderByDirection } from "@app/hooks/api/generic/types";
|
||||||
import { UsePopUpState } from "@app/hooks/usePopUp";
|
import { UsePopUpState } from "@app/hooks/usePopUp";
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
@ -31,59 +43,124 @@ type Props = {
|
|||||||
) => void;
|
) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum TagsOrderBy {
|
||||||
|
Slug = "slug"
|
||||||
|
}
|
||||||
|
|
||||||
export const SecretTagsTable = ({ handlePopUpOpen }: Props) => {
|
export const SecretTagsTable = ({ handlePopUpOpen }: Props) => {
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace } = useWorkspace();
|
||||||
const { data, isLoading } = useGetWsTags(currentWorkspace?.id ?? "");
|
const { data: tags = [], isLoading } = useGetWsTags(currentWorkspace?.id ?? "");
|
||||||
|
|
||||||
|
const {
|
||||||
|
search,
|
||||||
|
setSearch,
|
||||||
|
setPage,
|
||||||
|
page,
|
||||||
|
perPage,
|
||||||
|
setPerPage,
|
||||||
|
offset,
|
||||||
|
orderDirection,
|
||||||
|
toggleOrderDirection
|
||||||
|
} = usePagination(TagsOrderBy.Slug, { initPerPage: 10 });
|
||||||
|
|
||||||
|
const filteredTags = useMemo(
|
||||||
|
() =>
|
||||||
|
tags
|
||||||
|
.filter((tag) => tag.slug.toLowerCase().includes(search.trim().toLowerCase()))
|
||||||
|
.sort((a, b) => {
|
||||||
|
const [tagOne, tagTwo] = orderDirection === OrderByDirection.ASC ? [a, b] : [b, a];
|
||||||
|
|
||||||
|
return tagOne.slug.toLowerCase().localeCompare(tagTwo.slug.toLowerCase());
|
||||||
|
}),
|
||||||
|
[tags, orderDirection, search]
|
||||||
|
);
|
||||||
|
|
||||||
|
useResetPageHelper({
|
||||||
|
totalCount: filteredTags.length,
|
||||||
|
offset,
|
||||||
|
setPage
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TableContainer className="mt-4">
|
<div>
|
||||||
<Table>
|
<Input
|
||||||
<THead>
|
value={search}
|
||||||
<Tr>
|
onChange={(e) => setSearch(e.target.value)}
|
||||||
<Th>Slug</Th>
|
leftIcon={<FontAwesomeIcon icon={faMagnifyingGlass} />}
|
||||||
<Th aria-label="button" />
|
placeholder="Search tags..."
|
||||||
</Tr>
|
/>
|
||||||
</THead>
|
<TableContainer className="mt-4">
|
||||||
<TBody>
|
<Table>
|
||||||
{isLoading && <TableSkeleton columns={3} innerKey="secret-tags" />}
|
<THead>
|
||||||
{!isLoading &&
|
|
||||||
data &&
|
|
||||||
data.map(({ id, slug }) => (
|
|
||||||
<Tr key={id}>
|
|
||||||
<Td>{slug}</Td>
|
|
||||||
<Td className="flex items-center justify-end">
|
|
||||||
<ProjectPermissionCan
|
|
||||||
I={ProjectPermissionActions.Delete}
|
|
||||||
a={ProjectPermissionSub.Tags}
|
|
||||||
>
|
|
||||||
{(isAllowed) => (
|
|
||||||
<IconButton
|
|
||||||
onClick={() =>
|
|
||||||
handlePopUpOpen("deleteTagConfirmation", {
|
|
||||||
name: slug,
|
|
||||||
id
|
|
||||||
})
|
|
||||||
}
|
|
||||||
colorSchema="danger"
|
|
||||||
ariaLabel="update"
|
|
||||||
isDisabled={!isAllowed}
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={faTrashCan} />
|
|
||||||
</IconButton>
|
|
||||||
)}
|
|
||||||
</ProjectPermissionCan>
|
|
||||||
</Td>
|
|
||||||
</Tr>
|
|
||||||
))}
|
|
||||||
{!isLoading && data && data?.length === 0 && (
|
|
||||||
<Tr>
|
<Tr>
|
||||||
<Td colSpan={3}>
|
<Th className="w-full">
|
||||||
<EmptyState title="No secret tags found" icon={faTags} />
|
<div className="flex items-center">
|
||||||
</Td>
|
Slug
|
||||||
|
<IconButton
|
||||||
|
variant="plain"
|
||||||
|
className="ml-2"
|
||||||
|
ariaLabel="sort"
|
||||||
|
onClick={toggleOrderDirection}
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon={orderDirection === OrderByDirection.DESC ? faArrowUp : faArrowDown}
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
</div>
|
||||||
|
</Th>
|
||||||
|
<Th aria-label="button" />
|
||||||
</Tr>
|
</Tr>
|
||||||
)}
|
</THead>
|
||||||
</TBody>
|
<TBody>
|
||||||
</Table>
|
{isLoading && <TableSkeleton columns={3} innerKey="secret-tags" />}
|
||||||
</TableContainer>
|
{!isLoading &&
|
||||||
|
filteredTags.slice(offset, perPage * page).map(({ id, slug }) => (
|
||||||
|
<Tr key={id}>
|
||||||
|
<Td>{slug}</Td>
|
||||||
|
<Td className="flex items-center justify-end">
|
||||||
|
<ProjectPermissionCan
|
||||||
|
I={ProjectPermissionActions.Delete}
|
||||||
|
a={ProjectPermissionSub.Tags}
|
||||||
|
>
|
||||||
|
{(isAllowed) => (
|
||||||
|
<IconButton
|
||||||
|
onClick={() =>
|
||||||
|
handlePopUpOpen("deleteTagConfirmation", {
|
||||||
|
name: slug,
|
||||||
|
id
|
||||||
|
})
|
||||||
|
}
|
||||||
|
size="xs"
|
||||||
|
colorSchema="danger"
|
||||||
|
ariaLabel="update"
|
||||||
|
variant="plain"
|
||||||
|
isDisabled={!isAllowed}
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon icon={faTrashCan} />
|
||||||
|
</IconButton>
|
||||||
|
)}
|
||||||
|
</ProjectPermissionCan>
|
||||||
|
</Td>
|
||||||
|
</Tr>
|
||||||
|
))}
|
||||||
|
</TBody>
|
||||||
|
</Table>
|
||||||
|
{Boolean(filteredTags.length) && (
|
||||||
|
<Pagination
|
||||||
|
count={filteredTags.length}
|
||||||
|
page={page}
|
||||||
|
perPage={perPage}
|
||||||
|
onChangePage={setPage}
|
||||||
|
onChangePerPage={setPerPage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{!isLoading && !filteredTags?.length && (
|
||||||
|
<EmptyState
|
||||||
|
title={tags.length ? "No tags match search..." : "No tags found for project"}
|
||||||
|
icon={tags.length ? faSearch : faTag}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</TableContainer>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
Reference in New Issue
Block a user