From e83930d22bbdd4da94d3a5e6dc09485c697f9df6 Mon Sep 17 00:00:00 2001 From: Kevin Hahn Date: Thu, 31 Oct 2024 19:29:46 +0700 Subject: [PATCH] upgrade hotchocolate to 14 (#1167) * upgrade hot chocolate to v14 * workaround projection issue * Fix admin dashboard * Fix: dotnet run generate-gql-schema --------- Co-authored-by: Tim Haasdyk --- .../IsLanguageForgeProjectDataLoader.cs | 2 +- .../LexBoxApi/GraphQL/GraphQlSetupKernel.cs | 26 +- backend/LexBoxApi/GraphQL/LexQueries.cs | 6 +- backend/LexBoxApi/GraphQL/OrgMutations.cs | 8 +- backend/LexBoxApi/GraphQL/ProjectMutations.cs | 22 +- backend/LexBoxApi/LexBoxApi.csproj | 14 +- backend/LexBoxApi/Program.cs | 3 +- .../Services/DevGqlSchemaWriterService.cs | 19 +- frontend/schema.graphql | 432 +++++++++--------- 9 files changed, 280 insertions(+), 252 deletions(-) diff --git a/backend/LexBoxApi/GraphQL/CustomTypes/IsLanguageForgeProjectDataLoader.cs b/backend/LexBoxApi/GraphQL/CustomTypes/IsLanguageForgeProjectDataLoader.cs index a86c8440b..0489e7243 100644 --- a/backend/LexBoxApi/GraphQL/CustomTypes/IsLanguageForgeProjectDataLoader.cs +++ b/backend/LexBoxApi/GraphQL/CustomTypes/IsLanguageForgeProjectDataLoader.cs @@ -20,7 +20,7 @@ public IsLanguageForgeProjectDataLoader( IBatchScheduler batchScheduler, [FromKeyedServices(ResiliencePolicyName)] ResiliencePipeline> resiliencePipeline, - DataLoaderOptions? options = null) + DataLoaderOptions options) : base(batchScheduler, options) { _resiliencePipeline = resiliencePipeline; diff --git a/backend/LexBoxApi/GraphQL/GraphQlSetupKernel.cs b/backend/LexBoxApi/GraphQL/GraphQlSetupKernel.cs index 33b75e1f5..abd27c078 100644 --- a/backend/LexBoxApi/GraphQL/GraphQlSetupKernel.cs +++ b/backend/LexBoxApi/GraphQL/GraphQlSetupKernel.cs @@ -1,10 +1,7 @@ using DataAnnotatedModelValidations; using HotChocolate.Diagnostics; -using LexBoxApi.Auth; using LexBoxApi.GraphQL.CustomFilters; using LexBoxApi.Services; -using LexBoxApi.Services.Email; -using LexCore.ServiceInterfaces; using LexData; namespace LexBoxApi.GraphQL; @@ -18,15 +15,14 @@ public static void AddLexGraphQL(this IServiceCollection services, IHostEnvironm if (forceGenerateSchema || env.IsDevelopment()) services.AddHostedService(); - services.AddGraphQLServer() + services + .AddGraphQLServer() + .ModifyCostOptions(options => + { + // See: https://github.com/sillsdev/languageforge-lexbox/issues/1179 + options.EnforceCostLimits = false; + }) .InitializeOnStartup() - .RegisterDbContext() - .RegisterService() - .RegisterService() - .RegisterService() - .RegisterService() - .RegisterService() - .RegisterService() .AddDataAnnotationsValidator() .AddSorting(descriptor => { @@ -39,11 +35,11 @@ public static void AddLexGraphQL(this IServiceCollection services, IHostEnvironm descriptor.AddDeterministicInvariantContainsFilter(); }) .AddProjections() - .SetPagingOptions(new() + .ModifyPagingOptions(options => { - DefaultPageSize = 100, - MaxPageSize = 1000, - IncludeTotalCount = true + options.DefaultPageSize = 100; + options.MaxPageSize = 1000; + options.IncludeTotalCount = true; }) .AddAuthorization() .AddLexBoxApiTypes() diff --git a/backend/LexBoxApi/GraphQL/LexQueries.cs b/backend/LexBoxApi/GraphQL/LexQueries.cs index bd70ef7af..442fcd8f3 100644 --- a/backend/LexBoxApi/GraphQL/LexQueries.cs +++ b/backend/LexBoxApi/GraphQL/LexQueries.cs @@ -162,7 +162,11 @@ public IQueryable UsersInMyOrg(LexBoxDbContext context, LoggedInContext lo IPermissionService permissionService, IResolverContext context) { - var org = await dbContext.Orgs.Where(o => o.Id == orgId).AsNoTracking().Project(context).SingleOrDefaultAsync(); + //todo remove this workaround once the issue is fixed + var projectContext = + context.GetLocalStateOrDefault("HotChocolate.Data.Projections.ProxyContext") ?? + context; + var org = await dbContext.Orgs.Where(o => o.Id == orgId).AsNoTracking().Project(projectContext).SingleOrDefaultAsync(); if (org is null) return org; // Site admins and org admins can see everything if (permissionService.CanEditOrg(orgId)) return org; diff --git a/backend/LexBoxApi/GraphQL/OrgMutations.cs b/backend/LexBoxApi/GraphQL/OrgMutations.cs index 1a66bb492..54c71f6ce 100644 --- a/backend/LexBoxApi/GraphQL/OrgMutations.cs +++ b/backend/LexBoxApi/GraphQL/OrgMutations.cs @@ -66,7 +66,7 @@ public async Task DeleteOrg(Guid orgId, public async Task> AddProjectToOrg( LexBoxDbContext dbContext, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, Guid orgId, Guid projectId) { @@ -100,7 +100,7 @@ public async Task> AddProjectToOrg( public async Task AddProjectsToOrg( LexBoxDbContext dbContext, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, IResolverContext resolverContext, Guid orgId, Guid[] projectIds) @@ -139,7 +139,7 @@ public async Task> AddProjectToOrg( public async Task> RemoveProjectFromOrg( LexBoxDbContext dbContext, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, Guid orgId, Guid projectId) { @@ -185,7 +185,7 @@ public async Task> SetOrgMemberRole( OrgRole role, string emailOrUsername, bool canInvite, - [Service] IEmailService emailService) + IEmailService emailService) { var org = await dbContext.Orgs.FindAsync(orgId); NotFoundException.ThrowIfNull(org); diff --git a/backend/LexBoxApi/GraphQL/ProjectMutations.cs b/backend/LexBoxApi/GraphQL/ProjectMutations.cs index 3ceeeb163..ccdb9046e 100644 --- a/backend/LexBoxApi/GraphQL/ProjectMutations.cs +++ b/backend/LexBoxApi/GraphQL/ProjectMutations.cs @@ -37,8 +37,8 @@ public record CreateProjectResponse(Guid? Id, CreateProjectResult Result); LoggedInContext loggedInContext, IPermissionService permissionService, CreateProjectInput input, - [Service] ProjectService projectService, - [Service] IEmailService emailService) + ProjectService projectService, + IEmailService emailService) { if (!loggedInContext.User.IsAdmin) { @@ -73,7 +73,7 @@ public async Task> AddProjectMember( LoggedInContext loggedInContext, AddProjectMemberInput input, LexBoxDbContext dbContext, - [Service] IEmailService emailService) + IEmailService emailService) { await permissionService.AssertCanManageProject(input.ProjectId); var project = await dbContext.Projects.FindAsync(input.ProjectId); @@ -248,7 +248,7 @@ public async Task> AskToJoinProject( LoggedInContext loggedInContext, Guid projectId, LexBoxDbContext dbContext, - [Service] IEmailService emailService) + IEmailService emailService) { await permissionService.AssertCanAskToJoinProject(projectId); @@ -320,7 +320,7 @@ public async Task> ChangeProjectDescription(ChangeProjectDes [UseProjection] public async Task> SetProjectConfidentiality(SetProjectConfidentialityInput input, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { await permissionService.AssertCanManageProject(input.ProjectId); @@ -342,7 +342,7 @@ public async Task> SetProjectConfidentiality(SetProjectConfi public async Task> SetRetentionPolicy( SetRetentionPolicyInput input, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { await permissionService.AssertCanManageProject(input.ProjectId); @@ -363,7 +363,7 @@ public async Task> SetRetentionPolicy( [UseProjection] public async Task> UpdateProjectLexEntryCount(string code, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { var projectId = await projectService.LookupProjectId(code); @@ -382,7 +382,7 @@ public async Task> UpdateProjectLexEntryCount(string code, [UseProjection] public async Task> UpdateProjectLanguageList(string code, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { var projectId = await projectService.LookupProjectId(code); @@ -401,7 +401,7 @@ public async Task> UpdateProjectLanguageList(string code, [UseProjection] public async Task> UpdateLangProjectId(string code, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { var projectId = await projectService.LookupProjectId(code); @@ -420,7 +420,7 @@ public async Task> UpdateLangProjectId(string code, [UseProjection] public async Task> UpdateFLExModelVersion(string code, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext) { var projectId = await projectService.LookupProjectId(code); @@ -499,7 +499,7 @@ public async Task DeleteDraftProject( public async Task> SoftDeleteProject( Guid projectId, IPermissionService permissionService, - [Service] ProjectService projectService, + ProjectService projectService, LexBoxDbContext dbContext, IHgService hgService) { diff --git a/backend/LexBoxApi/LexBoxApi.csproj b/backend/LexBoxApi/LexBoxApi.csproj index e1d319dc9..baffc9fb9 100644 --- a/backend/LexBoxApi/LexBoxApi.csproj +++ b/backend/LexBoxApi/LexBoxApi.csproj @@ -14,16 +14,16 @@ - - - + + + runtime; build; native; contentfiles; analyzers; buildtransitive all - - - - + + + + diff --git a/backend/LexBoxApi/Program.cs b/backend/LexBoxApi/Program.cs index 904dee824..a25d36415 100644 --- a/backend/LexBoxApi/Program.cs +++ b/backend/LexBoxApi/Program.cs @@ -3,6 +3,7 @@ using System.Text.Json.Serialization; using AppAny.Quartz.EntityFrameworkCore.Migrations; using AppAny.Quartz.EntityFrameworkCore.Migrations.PostgreSQL; +using HotChocolate.AspNetCore; using LexBoxApi; using LexBoxApi.Auth; using LexBoxApi.Auth.Attributes; @@ -167,7 +168,7 @@ app.UseAuthentication(); app.UseAuthorization(); app.MapSecurityTxt(); -app.MapBananaCakePop("/api/graphql/ui").AllowAnonymous(); +app.MapNitroApp("/api/graphql/ui").WithOptions(new (){ServeMode = GraphQLToolServeMode.Embedded}).AllowAnonymous(); if (app.Environment.IsDevelopment()) //required for vite to generate types app.MapGraphQLSchema("/api/graphql/schema.graphql").AllowAnonymous(); diff --git a/backend/LexBoxApi/Services/DevGqlSchemaWriterService.cs b/backend/LexBoxApi/Services/DevGqlSchemaWriterService.cs index 712286921..ec291a45a 100644 --- a/backend/LexBoxApi/Services/DevGqlSchemaWriterService.cs +++ b/backend/LexBoxApi/Services/DevGqlSchemaWriterService.cs @@ -1,5 +1,10 @@ using HotChocolate.Execution; +using LexBoxApi.Auth; using LexBoxApi.GraphQL; +using LexBoxApi.GraphQL.CustomTypes; +using LexBoxApi.Services.Email; +using LexCore.ServiceInterfaces; +using LexData; using Microsoft.Extensions.Hosting.Internal; namespace LexBoxApi.Services; @@ -18,9 +23,17 @@ public static bool IsSchemaGenerationRequest(string[] args) public static async Task GenerateGqlSchema(string[] args) { var builder = Host.CreateApplicationBuilder(args); - builder.Services.AddLogging(); - builder.Services.AddSingleton(); - builder.Services.AddLexGraphQL(builder.Environment, true); + builder.Services + .AddLogging() + .AddSingleton() + .AddScoped() + .AddScoped() + .AddScoped() + .AddScoped((services) => new LoggedInContext(null!, null!)) + .AddScoped((services) => new LexBoxDbContext(null!, null!)) + .AddScoped() + .AddScoped() + .AddLexGraphQL(builder.Environment, true); var host = builder.Build(); await host.StartAsync(); await host.StopAsync(); diff --git a/frontend/schema.graphql b/frontend/schema.graphql index 5ec6a27d9..7624aaa32 100644 --- a/frontend/schema.graphql +++ b/frontend/schema.graphql @@ -237,38 +237,38 @@ type MeDto { } type Mutation { - createOrganization(input: CreateOrganizationInput!): CreateOrganizationPayload! - deleteOrg(input: DeleteOrgInput!): DeleteOrgPayload! @authorize(policy: "AdminRequiredPolicy") - addProjectToOrg(input: AddProjectToOrgInput!): AddProjectToOrgPayload! - addProjectsToOrg(input: AddProjectsToOrgInput!): AddProjectsToOrgPayload! - removeProjectFromOrg(input: RemoveProjectFromOrgInput!): RemoveProjectFromOrgPayload! - setOrgMemberRole(input: SetOrgMemberRoleInput!): SetOrgMemberRolePayload! - changeOrgMemberRole(input: ChangeOrgMemberRoleInput!): ChangeOrgMemberRolePayload! - leaveOrg(input: LeaveOrgInput!): LeaveOrgPayload! - bulkAddOrgMembers(input: BulkAddOrgMembersInput!): BulkAddOrgMembersPayload! - changeOrgName(input: ChangeOrgNameInput!): ChangeOrgNamePayload! - createProject(input: CreateProjectInput!): CreateProjectPayload! @authorize(policy: "VerifiedEmailRequiredPolicy") - addProjectMember(input: AddProjectMemberInput!): AddProjectMemberPayload! - bulkAddProjectMembers(input: BulkAddProjectMembersInput!): BulkAddProjectMembersPayload! @authorize(policy: "AdminRequiredPolicy") - changeProjectMemberRole(input: ChangeProjectMemberRoleInput!): ChangeProjectMemberRolePayload! - askToJoinProject(input: AskToJoinProjectInput!): AskToJoinProjectPayload! - changeProjectName(input: ChangeProjectNameInput!): ChangeProjectNamePayload! - changeProjectDescription(input: ChangeProjectDescriptionInput!): ChangeProjectDescriptionPayload! - setProjectConfidentiality(input: SetProjectConfidentialityInput!): SetProjectConfidentialityPayload! - setRetentionPolicy(input: SetRetentionPolicyInput!): SetRetentionPolicyPayload! - updateProjectLexEntryCount(input: UpdateProjectLexEntryCountInput!): UpdateProjectLexEntryCountPayload! - updateProjectLanguageList(input: UpdateProjectLanguageListInput!): UpdateProjectLanguageListPayload! - updateLangProjectId(input: UpdateLangProjectIdInput!): UpdateLangProjectIdPayload! - updateFLExModelVersion(input: UpdateFLExModelVersionInput!): UpdateFLExModelVersionPayload! - leaveProject(input: LeaveProjectInput!): LeaveProjectPayload! - removeProjectMember(input: RemoveProjectMemberInput!): RemoveProjectMemberPayload! - deleteDraftProject(input: DeleteDraftProjectInput!): DeleteDraftProjectPayload! @authorize(policy: "AdminRequiredPolicy") - softDeleteProject(input: SoftDeleteProjectInput!): SoftDeleteProjectPayload! - changeUserAccountBySelf(input: ChangeUserAccountBySelfInput!): ChangeUserAccountBySelfPayload! - changeUserAccountByAdmin(input: ChangeUserAccountByAdminInput!): ChangeUserAccountByAdminPayload! @authorize(policy: "AdminRequiredPolicy") - createGuestUserByAdmin(input: CreateGuestUserByAdminInput!): CreateGuestUserByAdminPayload! @authorize(policy: "AdminRequiredPolicy") - deleteUserByAdminOrSelf(input: DeleteUserByAdminOrSelfInput!): DeleteUserByAdminOrSelfPayload! - setUserLocked(input: SetUserLockedInput!): SetUserLockedPayload! @authorize(policy: "AdminRequiredPolicy") + createOrganization(input: CreateOrganizationInput!): CreateOrganizationPayload! @cost(weight: "10") + deleteOrg(input: DeleteOrgInput!): DeleteOrgPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + addProjectToOrg(input: AddProjectToOrgInput!): AddProjectToOrgPayload! @cost(weight: "10") + addProjectsToOrg(input: AddProjectsToOrgInput!): AddProjectsToOrgPayload! @cost(weight: "10") + removeProjectFromOrg(input: RemoveProjectFromOrgInput!): RemoveProjectFromOrgPayload! @cost(weight: "10") + setOrgMemberRole(input: SetOrgMemberRoleInput!): SetOrgMemberRolePayload! @cost(weight: "10") + changeOrgMemberRole(input: ChangeOrgMemberRoleInput!): ChangeOrgMemberRolePayload! @cost(weight: "10") + leaveOrg(input: LeaveOrgInput!): LeaveOrgPayload! @cost(weight: "10") + bulkAddOrgMembers(input: BulkAddOrgMembersInput!): BulkAddOrgMembersPayload! @cost(weight: "10") + changeOrgName(input: ChangeOrgNameInput!): ChangeOrgNamePayload! @cost(weight: "10") + createProject(input: CreateProjectInput!): CreateProjectPayload! @authorize(policy: "VerifiedEmailRequiredPolicy") @cost(weight: "10") + addProjectMember(input: AddProjectMemberInput!): AddProjectMemberPayload! @cost(weight: "10") + bulkAddProjectMembers(input: BulkAddProjectMembersInput!): BulkAddProjectMembersPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + changeProjectMemberRole(input: ChangeProjectMemberRoleInput!): ChangeProjectMemberRolePayload! @cost(weight: "10") + askToJoinProject(input: AskToJoinProjectInput!): AskToJoinProjectPayload! @cost(weight: "10") + changeProjectName(input: ChangeProjectNameInput!): ChangeProjectNamePayload! @cost(weight: "10") + changeProjectDescription(input: ChangeProjectDescriptionInput!): ChangeProjectDescriptionPayload! @cost(weight: "10") + setProjectConfidentiality(input: SetProjectConfidentialityInput!): SetProjectConfidentialityPayload! @cost(weight: "10") + setRetentionPolicy(input: SetRetentionPolicyInput!): SetRetentionPolicyPayload! @cost(weight: "10") + updateProjectLexEntryCount(input: UpdateProjectLexEntryCountInput!): UpdateProjectLexEntryCountPayload! @cost(weight: "10") + updateProjectLanguageList(input: UpdateProjectLanguageListInput!): UpdateProjectLanguageListPayload! @cost(weight: "10") + updateLangProjectId(input: UpdateLangProjectIdInput!): UpdateLangProjectIdPayload! @cost(weight: "10") + updateFLExModelVersion(input: UpdateFLExModelVersionInput!): UpdateFLExModelVersionPayload! @cost(weight: "10") + leaveProject(input: LeaveProjectInput!): LeaveProjectPayload! @cost(weight: "10") + removeProjectMember(input: RemoveProjectMemberInput!): RemoveProjectMemberPayload! @cost(weight: "10") + deleteDraftProject(input: DeleteDraftProjectInput!): DeleteDraftProjectPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + softDeleteProject(input: SoftDeleteProjectInput!): SoftDeleteProjectPayload! @cost(weight: "10") + changeUserAccountBySelf(input: ChangeUserAccountBySelfInput!): ChangeUserAccountBySelfPayload! @cost(weight: "10") + changeUserAccountByAdmin(input: ChangeUserAccountByAdminInput!): ChangeUserAccountByAdminPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + createGuestUserByAdmin(input: CreateGuestUserByAdminInput!): CreateGuestUserByAdminPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + deleteUserByAdminOrSelf(input: DeleteUserByAdminOrSelfInput!): DeleteUserByAdminOrSelfPayload! @cost(weight: "10") + setUserLocked(input: SetUserLockedInput!): SetUserLockedPayload! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") } type NotFoundError implements Error { @@ -372,9 +372,9 @@ type Project { createdDate: DateTime! id: UUID! users: [ProjectUsers!]! - changesets: [Changeset!]! + changesets: [Changeset!]! @cost(weight: "10") hasAbandonedTransactions: Boolean! - isLanguageForgeProject: Boolean! + isLanguageForgeProject: Boolean! @cost(weight: "10") parentId: UUID name: String! description: String @@ -425,22 +425,22 @@ type ProjectWritingSystems { } type Query { - myProjects(orderBy: [ProjectSortInput!]): [Project!]! - projects(withDeleted: Boolean! = false where: ProjectFilterInput orderBy: [ProjectSortInput!]): [Project!]! @authorize(policy: "AdminRequiredPolicy") - myDraftProjects(orderBy: [DraftProjectSortInput!]): [DraftProject!]! - draftProjects(where: DraftProjectFilterInput orderBy: [DraftProjectSortInput!]): [DraftProject!]! @authorize(policy: "AdminRequiredPolicy") - projectsByLangCodeAndOrg(input: ProjectsByLangCodeAndOrgInput! orderBy: [ProjectSortInput!]): [Project!]! - projectsInMyOrg(input: ProjectsInMyOrgInput! where: ProjectFilterInput orderBy: [ProjectSortInput!]): [Project!]! - projectById(projectId: UUID!): Project - projectByCode(code: String!): Project - draftProjectByCode(code: String!): DraftProject @authorize(policy: "AdminRequiredPolicy") - orgs(where: OrganizationFilterInput orderBy: [OrganizationSortInput!]): [Organization!]! - myOrgs(where: OrganizationFilterInput orderBy: [OrganizationSortInput!]): [Organization!]! - usersInMyOrg(skip: Int take: Int where: UserFilterInput orderBy: [UserSortInput!]): UsersInMyOrgCollectionSegment - orgById(orgId: UUID!): OrgById - users(skip: Int take: Int where: UserFilterInput orderBy: [UserSortInput!]): UsersCollectionSegment @authorize(policy: "AdminRequiredPolicy") - me: MeDto - orgMemberById(orgId: UUID! userId: UUID!): OrgMemberDto + myProjects(orderBy: [ProjectSortInput!] @cost(weight: "10")): [Project!]! @cost(weight: "10") + projects(withDeleted: Boolean! = false where: ProjectFilterInput @cost(weight: "10") orderBy: [ProjectSortInput!] @cost(weight: "10")): [Project!]! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + myDraftProjects(orderBy: [DraftProjectSortInput!] @cost(weight: "10")): [DraftProject!]! @cost(weight: "10") + draftProjects(where: DraftProjectFilterInput @cost(weight: "10") orderBy: [DraftProjectSortInput!] @cost(weight: "10")): [DraftProject!]! @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + projectsByLangCodeAndOrg(input: ProjectsByLangCodeAndOrgInput! orderBy: [ProjectSortInput!] @cost(weight: "10")): [Project!]! @cost(weight: "10") + projectsInMyOrg(input: ProjectsInMyOrgInput! where: ProjectFilterInput @cost(weight: "10") orderBy: [ProjectSortInput!] @cost(weight: "10")): [Project!]! @cost(weight: "10") + projectById(projectId: UUID!): Project @cost(weight: "10") + projectByCode(code: String!): Project @cost(weight: "10") + draftProjectByCode(code: String!): DraftProject @authorize(policy: "AdminRequiredPolicy") @cost(weight: "10") + orgs(where: OrganizationFilterInput @cost(weight: "10") orderBy: [OrganizationSortInput!] @cost(weight: "10")): [Organization!]! @cost(weight: "10") + myOrgs(where: OrganizationFilterInput @cost(weight: "10") orderBy: [OrganizationSortInput!] @cost(weight: "10")): [Organization!]! @cost(weight: "10") + usersInMyOrg(skip: Int take: Int where: UserFilterInput @cost(weight: "10") orderBy: [UserSortInput!] @cost(weight: "10")): UsersInMyOrgCollectionSegment @listSize(assumedSize: 1000, slicingArguments: [ "take" ], sizedFields: [ "items" ]) @cost(weight: "10") + orgById(orgId: UUID!): OrgById @cost(weight: "10") + users(skip: Int take: Int where: UserFilterInput @cost(weight: "10") orderBy: [UserSortInput!] @cost(weight: "10")): UsersCollectionSegment @authorize(policy: "AdminRequiredPolicy") @listSize(assumedSize: 1000, slicingArguments: [ "take" ], sizedFields: [ "items" ]) @cost(weight: "10") + me: MeDto @cost(weight: "10") + orgMemberById(orgId: UUID! userId: UUID!): OrgMemberDto @cost(weight: "10") meAuth: LexAuthUser! testingThrowsError: LexAuthUser! isAdmin: IsAdminResponse! @authorize(policy: "AdminRequiredPolicy") @@ -545,7 +545,7 @@ type UsersCollectionSegment { pageInfo: CollectionSegmentInfo! "A flattened list of the items." items: [User!] - totalCount: Int! + totalCount: Int! @cost(weight: "10") } "A segment of a collection." @@ -554,7 +554,7 @@ type UsersInMyOrgCollectionSegment { pageInfo: CollectionSegmentInfo! "A flattened list of the items." items: [User!] - totalCount: Int! + totalCount: Int! @cost(weight: "10") } union AddProjectMemberError = NotFoundError | DbError | ProjectMembersMustBeVerified | ProjectMembersMustBeVerifiedForRole | ProjectMemberInvitedByEmail | InvalidEmailError | AlreadyExistsError @@ -642,8 +642,8 @@ input AskToJoinProjectInput { } input BooleanOperationFilterInput { - eq: Boolean - neq: Boolean + eq: Boolean @cost(weight: "10") + neq: Boolean @cost(weight: "10") } input BulkAddOrgMembersInput { @@ -726,18 +726,18 @@ input CreateProjectInput { } input DateTimeOperationFilterInput { - eq: DateTime - neq: DateTime - in: [DateTime] - nin: [DateTime] - gt: DateTime - ngt: DateTime - gte: DateTime - ngte: DateTime - lt: DateTime - nlt: DateTime - lte: DateTime - nlte: DateTime + eq: DateTime @cost(weight: "10") + neq: DateTime @cost(weight: "10") + in: [DateTime] @cost(weight: "10") + nin: [DateTime] @cost(weight: "10") + gt: DateTime @cost(weight: "10") + ngt: DateTime @cost(weight: "10") + gte: DateTime @cost(weight: "10") + ngte: DateTime @cost(weight: "10") + lt: DateTime @cost(weight: "10") + nlt: DateTime @cost(weight: "10") + lte: DateTime @cost(weight: "10") + nlte: DateTime @cost(weight: "10") } input DeleteDraftProjectInput { @@ -770,18 +770,18 @@ input DraftProjectFilterInput { } input DraftProjectSortInput { - name: SortEnumType - description: SortEnumType - code: SortEnumType - type: SortEnumType - retentionPolicy: SortEnumType - projectManager: UserSortInput - projectManagerId: SortEnumType - orgId: SortEnumType - isConfidential: SortEnumType - id: SortEnumType - createdDate: SortEnumType - updatedDate: SortEnumType + name: SortEnumType @cost(weight: "10") + description: SortEnumType @cost(weight: "10") + code: SortEnumType @cost(weight: "10") + type: SortEnumType @cost(weight: "10") + retentionPolicy: SortEnumType @cost(weight: "10") + projectManager: UserSortInput @cost(weight: "10") + projectManagerId: SortEnumType @cost(weight: "10") + orgId: SortEnumType @cost(weight: "10") + isConfidential: SortEnumType @cost(weight: "10") + id: SortEnumType @cost(weight: "10") + createdDate: SortEnumType @cost(weight: "10") + updatedDate: SortEnumType @cost(weight: "10") } input FLExWsIdFilterInput { @@ -803,25 +803,25 @@ input FlexProjectMetadataFilterInput { } input FlexProjectMetadataSortInput { - projectId: SortEnumType - lexEntryCount: SortEnumType - langProjectId: SortEnumType - flexModelVersion: SortEnumType + projectId: SortEnumType @cost(weight: "10") + lexEntryCount: SortEnumType @cost(weight: "10") + langProjectId: SortEnumType @cost(weight: "10") + flexModelVersion: SortEnumType @cost(weight: "10") } input IntOperationFilterInput { - eq: Int - neq: Int - in: [Int] - nin: [Int] - gt: Int - ngt: Int - gte: Int - ngte: Int - lt: Int - nlt: Int - lte: Int - nlte: Int + eq: Int @cost(weight: "10") + neq: Int @cost(weight: "10") + in: [Int] @cost(weight: "10") + nin: [Int] @cost(weight: "10") + gt: Int @cost(weight: "10") + ngt: Int @cost(weight: "10") + gte: Int @cost(weight: "10") + ngte: Int @cost(weight: "10") + lt: Int @cost(weight: "10") + nlt: Int @cost(weight: "10") + lte: Int @cost(weight: "10") + nlte: Int @cost(weight: "10") } input LeaveOrgInput { @@ -833,45 +833,45 @@ input LeaveProjectInput { } input ListFilterInputTypeOfFLExWsIdFilterInput { - all: FLExWsIdFilterInput - none: FLExWsIdFilterInput - some: FLExWsIdFilterInput - any: Boolean + all: FLExWsIdFilterInput @cost(weight: "10") + none: FLExWsIdFilterInput @cost(weight: "10") + some: FLExWsIdFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input ListFilterInputTypeOfOrgMemberFilterInput { - all: OrgMemberFilterInput - none: OrgMemberFilterInput - some: OrgMemberFilterInput - any: Boolean + all: OrgMemberFilterInput @cost(weight: "10") + none: OrgMemberFilterInput @cost(weight: "10") + some: OrgMemberFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input ListFilterInputTypeOfOrganizationFilterInput { - all: OrganizationFilterInput - none: OrganizationFilterInput - some: OrganizationFilterInput - any: Boolean + all: OrganizationFilterInput @cost(weight: "10") + none: OrganizationFilterInput @cost(weight: "10") + some: OrganizationFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input ListFilterInputTypeOfProjectFilterInput { - all: ProjectFilterInput - none: ProjectFilterInput - some: ProjectFilterInput - any: Boolean + all: ProjectFilterInput @cost(weight: "10") + none: ProjectFilterInput @cost(weight: "10") + some: ProjectFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input ListFilterInputTypeOfProjectUsersFilterInput { - all: ProjectUsersFilterInput - none: ProjectUsersFilterInput - some: ProjectUsersFilterInput - any: Boolean + all: ProjectUsersFilterInput @cost(weight: "10") + none: ProjectUsersFilterInput @cost(weight: "10") + some: ProjectUsersFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input ListFilterInputTypeOfUserFilterInput { - all: UserFilterInput - none: UserFilterInput - some: UserFilterInput - any: Boolean + all: UserFilterInput @cost(weight: "10") + none: UserFilterInput @cost(weight: "10") + some: UserFilterInput @cost(weight: "10") + any: Boolean @cost(weight: "10") } input OrgMemberFilterInput { @@ -888,10 +888,10 @@ input OrgMemberFilterInput { } input OrgRoleOperationFilterInput { - eq: OrgRole - neq: OrgRole - in: [OrgRole!] - nin: [OrgRole!] + eq: OrgRole @cost(weight: "10") + neq: OrgRole @cost(weight: "10") + in: [OrgRole!] @cost(weight: "10") + nin: [OrgRole!] @cost(weight: "10") } input OrganizationFilterInput { @@ -908,12 +908,12 @@ input OrganizationFilterInput { } input OrganizationSortInput { - name: SortEnumType - memberCount: SortEnumType - projectCount: SortEnumType - id: SortEnumType - createdDate: SortEnumType - updatedDate: SortEnumType + name: SortEnumType @cost(weight: "10") + memberCount: SortEnumType @cost(weight: "10") + projectCount: SortEnumType @cost(weight: "10") + id: SortEnumType @cost(weight: "10") + createdDate: SortEnumType @cost(weight: "10") + updatedDate: SortEnumType @cost(weight: "10") } input ProjectFilterInput { @@ -941,44 +941,44 @@ input ProjectFilterInput { } input ProjectMigrationStatusOperationFilterInput { - eq: ProjectMigrationStatus - neq: ProjectMigrationStatus - in: [ProjectMigrationStatus!] - nin: [ProjectMigrationStatus!] + eq: ProjectMigrationStatus @cost(weight: "10") + neq: ProjectMigrationStatus @cost(weight: "10") + in: [ProjectMigrationStatus!] @cost(weight: "10") + nin: [ProjectMigrationStatus!] @cost(weight: "10") } input ProjectRoleOperationFilterInput { - eq: ProjectRole - neq: ProjectRole - in: [ProjectRole!] - nin: [ProjectRole!] + eq: ProjectRole @cost(weight: "10") + neq: ProjectRole @cost(weight: "10") + in: [ProjectRole!] @cost(weight: "10") + nin: [ProjectRole!] @cost(weight: "10") } input ProjectSortInput { - parentId: SortEnumType - code: SortEnumType - name: SortEnumType - description: SortEnumType - retentionPolicy: SortEnumType - type: SortEnumType - isConfidential: SortEnumType - flexProjectMetadata: FlexProjectMetadataSortInput - lastCommit: SortEnumType - deletedDate: SortEnumType - resetStatus: SortEnumType - projectOrigin: SortEnumType - migratedDate: SortEnumType - userCount: SortEnumType - id: SortEnumType - createdDate: SortEnumType - updatedDate: SortEnumType + parentId: SortEnumType @cost(weight: "10") + code: SortEnumType @cost(weight: "10") + name: SortEnumType @cost(weight: "10") + description: SortEnumType @cost(weight: "10") + retentionPolicy: SortEnumType @cost(weight: "10") + type: SortEnumType @cost(weight: "10") + isConfidential: SortEnumType @cost(weight: "10") + flexProjectMetadata: FlexProjectMetadataSortInput @cost(weight: "10") + lastCommit: SortEnumType @cost(weight: "10") + deletedDate: SortEnumType @cost(weight: "10") + resetStatus: SortEnumType @cost(weight: "10") + projectOrigin: SortEnumType @cost(weight: "10") + migratedDate: SortEnumType @cost(weight: "10") + userCount: SortEnumType @cost(weight: "10") + id: SortEnumType @cost(weight: "10") + createdDate: SortEnumType @cost(weight: "10") + updatedDate: SortEnumType @cost(weight: "10") } input ProjectTypeOperationFilterInput { - eq: ProjectType - neq: ProjectType - in: [ProjectType!] - nin: [ProjectType!] + eq: ProjectType @cost(weight: "10") + neq: ProjectType @cost(weight: "10") + in: [ProjectType!] @cost(weight: "10") + nin: [ProjectType!] @cost(weight: "10") } input ProjectUsersFilterInput { @@ -1021,17 +1021,17 @@ input RemoveProjectMemberInput { } input ResetStatusOperationFilterInput { - eq: ResetStatus - neq: ResetStatus - in: [ResetStatus!] - nin: [ResetStatus!] + eq: ResetStatus @cost(weight: "10") + neq: ResetStatus @cost(weight: "10") + in: [ResetStatus!] @cost(weight: "10") + nin: [ResetStatus!] @cost(weight: "10") } input RetentionPolicyOperationFilterInput { - eq: RetentionPolicy - neq: RetentionPolicy - in: [RetentionPolicy!] - nin: [RetentionPolicy!] + eq: RetentionPolicy @cost(weight: "10") + neq: RetentionPolicy @cost(weight: "10") + in: [RetentionPolicy!] @cost(weight: "10") + nin: [RetentionPolicy!] @cost(weight: "10") } input SetOrgMemberRoleInput { @@ -1063,18 +1063,18 @@ input SoftDeleteProjectInput { input StringOperationFilterInput { and: [StringOperationFilterInput!] or: [StringOperationFilterInput!] - eq: String - neq: String - contains: String - ncontains: String - in: [String] - nin: [String] - startsWith: String - nstartsWith: String - endsWith: String - nendsWith: String - icontains: String - ieq: String + eq: String @cost(weight: "10") + neq: String @cost(weight: "10") + contains: String @cost(weight: "20") + ncontains: String @cost(weight: "20") + in: [String] @cost(weight: "10") + nin: [String] @cost(weight: "10") + startsWith: String @cost(weight: "20") + nstartsWith: String @cost(weight: "20") + endsWith: String @cost(weight: "20") + nendsWith: String @cost(weight: "20") + icontains: String @cost(weight: "10") + ieq: String @cost(weight: "10") } input UpdateFLExModelVersionInput { @@ -1120,44 +1120,48 @@ input UserFilterInput { } input UserSortInput { - name: SortEnumType - email: SortEnumType - localizationCode: SortEnumType - isAdmin: SortEnumType - passwordHash: SortEnumType - salt: SortEnumType - passwordStrength: SortEnumType - lastActive: SortEnumType - emailVerified: SortEnumType - canCreateProjects: SortEnumType - createdById: SortEnumType - createdBy: UserSortInput - locked: SortEnumType - username: SortEnumType - googleId: SortEnumType - id: SortEnumType - createdDate: SortEnumType - updatedDate: SortEnumType + name: SortEnumType @cost(weight: "10") + email: SortEnumType @cost(weight: "10") + localizationCode: SortEnumType @cost(weight: "10") + isAdmin: SortEnumType @cost(weight: "10") + passwordHash: SortEnumType @cost(weight: "10") + salt: SortEnumType @cost(weight: "10") + passwordStrength: SortEnumType @cost(weight: "10") + lastActive: SortEnumType @cost(weight: "10") + emailVerified: SortEnumType @cost(weight: "10") + canCreateProjects: SortEnumType @cost(weight: "10") + createdById: SortEnumType @cost(weight: "10") + createdBy: UserSortInput @cost(weight: "10") + locked: SortEnumType @cost(weight: "10") + username: SortEnumType @cost(weight: "10") + googleId: SortEnumType @cost(weight: "10") + id: SortEnumType @cost(weight: "10") + createdDate: SortEnumType @cost(weight: "10") + updatedDate: SortEnumType @cost(weight: "10") } input UuidOperationFilterInput { - eq: UUID - neq: UUID - in: [UUID] - nin: [UUID] - gt: UUID - ngt: UUID - gte: UUID - ngte: UUID - lt: UUID - nlt: UUID - lte: UUID - nlte: UUID -} - + eq: UUID @cost(weight: "10") + neq: UUID @cost(weight: "10") + in: [UUID] @cost(weight: "10") + nin: [UUID] @cost(weight: "10") + gt: UUID @cost(weight: "10") + ngt: UUID @cost(weight: "10") + gte: UUID @cost(weight: "10") + ngte: UUID @cost(weight: "10") + lt: UUID @cost(weight: "10") + nlt: UUID @cost(weight: "10") + lte: UUID @cost(weight: "10") + nlte: UUID @cost(weight: "10") +} + +"Defines when a policy shall be executed." enum ApplyPolicy { + "Before the resolver was executed." BEFORE_RESOLVER + "After the resolver was executed." AFTER_RESOLVER + "The policy is applied in the validation step before the execution." VALIDATION } @@ -1232,8 +1236,18 @@ enum UserRole { USER } +"The authorize directive." directive @authorize("The name of the authorization policy that determines access to the annotated resource." policy: String "Roles that are allowed to access the annotated resource." roles: [String!] "Defines when when the authorize directive shall be applied.By default the authorize directives are applied during the validation phase." apply: ApplyPolicy! = BEFORE_RESOLVER) repeatable on OBJECT | FIELD_DEFINITION +"The purpose of the `cost` directive is to define a `weight` for GraphQL types, fields, and arguments. Static analysis can use these weights when calculating the overall cost of a query or response." +directive @cost("The `weight` argument defines what value to add to the overall cost for every appearance, or possible appearance, of a type, field, argument, etc." weight: String!) on SCALAR | OBJECT | FIELD_DEFINITION | ARGUMENT_DEFINITION | ENUM | INPUT_FIELD_DEFINITION + +"The purpose of the `@listSize` directive is to either inform the static analysis about the size of returned lists (if that information is statically available), or to point the analysis to where to find that information." +directive @listSize("The `assumedSize` argument can be used to statically define the maximum length of a list returned by a field." assumedSize: Int "The `slicingArguments` argument can be used to define which of the field's arguments with numeric type are slicing arguments, so that their value determines the size of the list returned by that field. It may specify a list of multiple slicing arguments." slicingArguments: [String!] "The `sizedFields` argument can be used to define that the value of the `assumedSize` argument or of a slicing argument does not affect the size of a list returned by a field itself, but that of a list returned by one of its sub-fields." sizedFields: [String!] "The `requireOneSlicingArgument` argument can be used to inform the static analysis that it should expect that exactly one of the defined slicing arguments is present in a query. If that is not the case (i.e., if none or multiple slicing arguments are present), the static analysis may throw an error." requireOneSlicingArgument: Boolean! = true) on FIELD_DEFINITION + +"The `@specifiedBy` directive is used within the type system definition language to provide a URL for specifying the behavior of custom scalar definitions." +directive @specifiedBy("The specifiedBy URL points to a human-readable specification. This field will only read a result for scalar types." url: String!) on SCALAR + scalar DateTime @specifiedBy(url: "https:\/\/www.graphql-scalars.com\/date-time") "The `Long` scalar type represents non-fractional signed whole 64-bit numeric values. Long can represent values between -(2^63) and 2^63 - 1."