mirror of
https://github.com/twentyhq/twenty.git
synced 2024-12-28 06:46:24 +03:00
Catch query timeout exceptions (#5680)
Query read timeouts happen when a remote server is not available. It breaks: - the remote server show page - the record table page of imported remote tables This PR will catch the exception so it does not go to Sentry in both cases. Also did 2 renaming.
This commit is contained in:
parent
5e1dfde3e4
commit
c60a3e49cd
@ -1,6 +1,6 @@
|
|||||||
import { CommandFactory } from 'nest-commander';
|
import { CommandFactory } from 'nest-commander';
|
||||||
|
|
||||||
import { filterException } from 'src/engine/utils/global-exception-handler.util';
|
import { shouldFilterException } from 'src/engine/utils/global-exception-handler.util';
|
||||||
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
||||||
import { LoggerService } from 'src/engine/integrations/logger/logger.service';
|
import { LoggerService } from 'src/engine/integrations/logger/logger.service';
|
||||||
|
|
||||||
@ -10,7 +10,7 @@ async function bootstrap() {
|
|||||||
const errorHandler = (err: Error) => {
|
const errorHandler = (err: Error) => {
|
||||||
loggerService.error(err?.message, err?.name);
|
loggerService.error(err?.message, err?.name);
|
||||||
|
|
||||||
if (filterException(err)) {
|
if (shouldFilterException(err)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -20,6 +20,9 @@ export const typeORMMetadataModuleOptions: TypeOrmModuleOptions = {
|
|||||||
rejectUnauthorized: false,
|
rejectUnauthorized: false,
|
||||||
}
|
}
|
||||||
: undefined,
|
: undefined,
|
||||||
|
extra: {
|
||||||
|
query_timeout: 10000,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
export const connectionSource = new DataSource(
|
export const connectionSource = new DataSource(
|
||||||
typeORMMetadataModuleOptions as DataSourceOptions,
|
typeORMMetadataModuleOptions as DataSourceOptions,
|
||||||
|
@ -38,6 +38,9 @@ export class TypeORMService implements OnModuleInit, OnModuleDestroy {
|
|||||||
rejectUnauthorized: false,
|
rejectUnauthorized: false,
|
||||||
}
|
}
|
||||||
: undefined,
|
: undefined,
|
||||||
|
extra: {
|
||||||
|
query_timeout: 10000,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,6 +3,7 @@ import {
|
|||||||
Inject,
|
Inject,
|
||||||
Injectable,
|
Injectable,
|
||||||
Logger,
|
Logger,
|
||||||
|
RequestTimeoutException,
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { EventEmitter2 } from '@nestjs/event-emitter';
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
||||||
|
|
||||||
@ -49,7 +50,8 @@ import { QueryRunnerArgsFactory } from 'src/engine/api/graphql/workspace-query-r
|
|||||||
import { QueryResultGettersFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-result-getters.factory';
|
import { QueryResultGettersFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-result-getters.factory';
|
||||||
import { assertMutationNotOnRemoteObject } from 'src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util';
|
import { assertMutationNotOnRemoteObject } from 'src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util';
|
||||||
import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids';
|
import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids';
|
||||||
import { assertIsValidUuid } from 'src/engine/api/graphql/workspace-query-runner/utils/assertIsValidUuid.util';
|
import { assertIsValidUuid } from 'src/engine/api/graphql/workspace-query-runner/utils/assert-is-valid-uuid.util';
|
||||||
|
import { isQueryTimeoutError } from 'src/engine/utils/query-timeout.util';
|
||||||
|
|
||||||
import { WorkspaceQueryRunnerOptions } from './interfaces/query-runner-option.interface';
|
import { WorkspaceQueryRunnerOptions } from './interfaces/query-runner-option.interface';
|
||||||
import {
|
import {
|
||||||
@ -577,28 +579,30 @@ export class WorkspaceQueryRunnerService {
|
|||||||
workspaceId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
await workspaceDataSource?.query(`
|
try {
|
||||||
SET search_path TO ${this.workspaceDataSourceService.getSchemaName(
|
return await workspaceDataSource?.transaction(
|
||||||
workspaceId,
|
async (transactionManager) => {
|
||||||
)};
|
await transactionManager.query(`
|
||||||
`);
|
SET search_path TO ${this.workspaceDataSourceService.getSchemaName(
|
||||||
|
workspaceId,
|
||||||
|
)};
|
||||||
|
`);
|
||||||
|
|
||||||
return await workspaceDataSource?.transaction(
|
const results = transactionManager.query<PGGraphQLResult>(
|
||||||
async (transactionManager) => {
|
`SELECT graphql.resolve($1);`,
|
||||||
await transactionManager.query(`
|
[query],
|
||||||
SET search_path TO ${this.workspaceDataSourceService.getSchemaName(
|
);
|
||||||
workspaceId,
|
|
||||||
)};
|
|
||||||
`);
|
|
||||||
|
|
||||||
const results = transactionManager.query<PGGraphQLResult>(
|
return results;
|
||||||
`SELECT graphql.resolve($1);`,
|
},
|
||||||
[query],
|
);
|
||||||
);
|
} catch (error) {
|
||||||
|
if (isQueryTimeoutError(error)) {
|
||||||
|
throw new RequestTimeoutException(error.message);
|
||||||
|
}
|
||||||
|
|
||||||
return results;
|
throw error;
|
||||||
},
|
}
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async parseResult<Result>(
|
private async parseResult<Result>(
|
||||||
|
@ -11,7 +11,7 @@ import { GraphQLContext } from 'src/engine/api/graphql/graphql-config/interfaces
|
|||||||
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
||||||
import {
|
import {
|
||||||
convertExceptionToGraphQLError,
|
convertExceptionToGraphQLError,
|
||||||
filterException,
|
shouldFilterException,
|
||||||
} from 'src/engine/utils/global-exception-handler.util';
|
} from 'src/engine/utils/global-exception-handler.util';
|
||||||
|
|
||||||
export type ExceptionHandlerPluginOptions = {
|
export type ExceptionHandlerPluginOptions = {
|
||||||
@ -69,7 +69,7 @@ export const useExceptionHandler = <PluginContext extends GraphQLContext>(
|
|||||||
}>(
|
}>(
|
||||||
(acc, err) => {
|
(acc, err) => {
|
||||||
// Filter out exceptions that we don't want to be captured by exception handler
|
// Filter out exceptions that we don't want to be captured by exception handler
|
||||||
if (filterException(err?.originalError ?? err)) {
|
if (shouldFilterException(err?.originalError ?? err)) {
|
||||||
acc.filtered.push(err);
|
acc.filtered.push(err);
|
||||||
} else {
|
} else {
|
||||||
acc.unfiltered.push(err);
|
acc.unfiltered.push(err);
|
||||||
|
@ -1,4 +1,8 @@
|
|||||||
import { BadRequestException, Injectable } from '@nestjs/common';
|
import {
|
||||||
|
BadRequestException,
|
||||||
|
Injectable,
|
||||||
|
RequestTimeoutException,
|
||||||
|
} from '@nestjs/common';
|
||||||
import { InjectRepository } from '@nestjs/typeorm';
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
import { EntityManager, Repository } from 'typeorm';
|
import { EntityManager, Repository } from 'typeorm';
|
||||||
@ -12,6 +16,7 @@ import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/work
|
|||||||
import { DistantTables } from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/types/distant-table';
|
import { DistantTables } from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/types/distant-table';
|
||||||
import { STRIPE_DISTANT_TABLES } from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/utils/stripe-distant-tables.util';
|
import { STRIPE_DISTANT_TABLES } from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/utils/stripe-distant-tables.util';
|
||||||
import { PostgresTableSchemaColumn } from 'src/engine/metadata-modules/remote-server/types/postgres-table-schema-column';
|
import { PostgresTableSchemaColumn } from 'src/engine/metadata-modules/remote-server/types/postgres-table-schema-column';
|
||||||
|
import { isQueryTimeoutError } from 'src/engine/utils/query-timeout.util';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class DistantTableService {
|
export class DistantTableService {
|
||||||
@ -70,44 +75,54 @@ export class DistantTableService {
|
|||||||
workspaceId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
const distantTables = await workspaceDataSource.transaction(
|
try {
|
||||||
async (entityManager: EntityManager) => {
|
const distantTables = await workspaceDataSource.transaction(
|
||||||
await entityManager.query(`CREATE SCHEMA "${tmpSchemaName}"`);
|
async (entityManager: EntityManager) => {
|
||||||
|
await entityManager.query(`CREATE SCHEMA "${tmpSchemaName}"`);
|
||||||
|
|
||||||
const tableLimitationsOptions = tableName
|
const tableLimitationsOptions = tableName
|
||||||
? ` LIMIT TO ("${tableName}")`
|
? ` LIMIT TO ("${tableName}")`
|
||||||
: '';
|
: '';
|
||||||
|
|
||||||
await entityManager.query(
|
await entityManager.query(
|
||||||
`IMPORT FOREIGN SCHEMA "${remoteServer.schema}"${tableLimitationsOptions} FROM SERVER "${remoteServer.foreignDataWrapperId}" INTO "${tmpSchemaName}"`,
|
`IMPORT FOREIGN SCHEMA "${remoteServer.schema}"${tableLimitationsOptions} FROM SERVER "${remoteServer.foreignDataWrapperId}" INTO "${tmpSchemaName}"`,
|
||||||
|
);
|
||||||
|
|
||||||
|
const createdForeignTableNames = await entityManager.query(
|
||||||
|
`SELECT table_name, column_name, data_type, udt_name FROM information_schema.columns WHERE table_schema = '${tmpSchemaName}'`,
|
||||||
|
);
|
||||||
|
|
||||||
|
await entityManager.query(`DROP SCHEMA "${tmpSchemaName}" CASCADE`);
|
||||||
|
|
||||||
|
return createdForeignTableNames.reduce(
|
||||||
|
(acc, { table_name, column_name, data_type, udt_name }) => {
|
||||||
|
if (!acc[table_name]) {
|
||||||
|
acc[table_name] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
acc[table_name].push({
|
||||||
|
columnName: column_name,
|
||||||
|
dataType: data_type,
|
||||||
|
udtName: udt_name,
|
||||||
|
});
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
return distantTables;
|
||||||
|
} catch (error) {
|
||||||
|
if (isQueryTimeoutError(error)) {
|
||||||
|
throw new RequestTimeoutException(
|
||||||
|
`Could not find distant tables: ${error.message}`,
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const createdForeignTableNames = await entityManager.query(
|
throw error;
|
||||||
`SELECT table_name, column_name, data_type, udt_name FROM information_schema.columns WHERE table_schema = '${tmpSchemaName}'`,
|
}
|
||||||
);
|
|
||||||
|
|
||||||
await entityManager.query(`DROP SCHEMA "${tmpSchemaName}" CASCADE`);
|
|
||||||
|
|
||||||
return createdForeignTableNames.reduce(
|
|
||||||
(acc, { table_name, column_name, data_type, udt_name }) => {
|
|
||||||
if (!acc[table_name]) {
|
|
||||||
acc[table_name] = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
acc[table_name].push({
|
|
||||||
columnName: column_name,
|
|
||||||
dataType: data_type,
|
|
||||||
udtName: udt_name,
|
|
||||||
});
|
|
||||||
|
|
||||||
return acc;
|
|
||||||
},
|
|
||||||
{},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
return distantTables;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private getDistantTablesFromStaticSchema(
|
private getDistantTablesFromStaticSchema(
|
||||||
|
@ -10,6 +10,7 @@ import {
|
|||||||
NotFoundError,
|
NotFoundError,
|
||||||
ConflictError,
|
ConflictError,
|
||||||
MethodNotAllowedError,
|
MethodNotAllowedError,
|
||||||
|
TimeoutError,
|
||||||
} from 'src/engine/utils/graphql-errors.util';
|
} from 'src/engine/utils/graphql-errors.util';
|
||||||
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
||||||
|
|
||||||
@ -19,6 +20,7 @@ const graphQLPredefinedExceptions = {
|
|||||||
403: ForbiddenError,
|
403: ForbiddenError,
|
||||||
404: NotFoundError,
|
404: NotFoundError,
|
||||||
405: MethodNotAllowedError,
|
405: MethodNotAllowedError,
|
||||||
|
408: TimeoutError,
|
||||||
409: ConflictError,
|
409: ConflictError,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -32,7 +34,7 @@ export const handleExceptionAndConvertToGraphQLError = (
|
|||||||
return convertExceptionToGraphQLError(exception);
|
return convertExceptionToGraphQLError(exception);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const filterException = (exception: Error): boolean => {
|
export const shouldFilterException = (exception: Error): boolean => {
|
||||||
if (exception instanceof HttpException && exception.getStatus() < 500) {
|
if (exception instanceof HttpException && exception.getStatus() < 500) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -45,7 +47,7 @@ export const handleException = (
|
|||||||
exceptionHandlerService: ExceptionHandlerService,
|
exceptionHandlerService: ExceptionHandlerService,
|
||||||
user?: ExceptionHandlerUser,
|
user?: ExceptionHandlerUser,
|
||||||
): void => {
|
): void => {
|
||||||
if (filterException(exception)) {
|
if (shouldFilterException(exception)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,3 +157,11 @@ export class ConflictError extends BaseGraphQLError {
|
|||||||
Object.defineProperty(this, 'name', { value: 'ConflictError' });
|
Object.defineProperty(this, 'name', { value: 'ConflictError' });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class TimeoutError extends BaseGraphQLError {
|
||||||
|
constructor(message: string, extensions?: Record<string, any>) {
|
||||||
|
super(message, 'TIMEOUT', extensions);
|
||||||
|
|
||||||
|
Object.defineProperty(this, 'name', { value: 'TimeoutError' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
export const isQueryTimeoutError = (error: Error) => {
|
||||||
|
return error.message.includes('Query read timeout');
|
||||||
|
};
|
@ -5,7 +5,7 @@ import {
|
|||||||
MessageQueueJobData,
|
MessageQueueJobData,
|
||||||
} from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface';
|
} from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface';
|
||||||
|
|
||||||
import { filterException } from 'src/engine/utils/global-exception-handler.util';
|
import { shouldFilterException } from 'src/engine/utils/global-exception-handler.util';
|
||||||
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service';
|
||||||
import { LoggerService } from 'src/engine/integrations/logger/logger.service';
|
import { LoggerService } from 'src/engine/integrations/logger/logger.service';
|
||||||
import { JobsModule } from 'src/engine/integrations/message-queue/jobs.module';
|
import { JobsModule } from 'src/engine/integrations/message-queue/jobs.module';
|
||||||
@ -54,7 +54,7 @@ async function bootstrap() {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
loggerService?.error(err?.message, err?.name);
|
loggerService?.error(err?.message, err?.name);
|
||||||
|
|
||||||
if (!filterException(err)) {
|
if (!shouldFilterException(err)) {
|
||||||
exceptionHandlerService?.captureExceptions([err]);
|
exceptionHandlerService?.captureExceptions([err]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user