Skip to content

Commit b2fdccb

Browse files
committed
Parse additional fields from error and notice response messages
1 parent b02af17 commit b2fdccb

File tree

6 files changed

+79
-271
lines changed

6 files changed

+79
-271
lines changed

CHANGES.md

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,11 @@
11
In next release ...
22

3-
-
3+
- The PostgreSQL error code table has been removed. Consult the documentation to
4+
map error codes to the string identifiers used in the PostgreSQL codebase.
5+
6+
- Parse additional fields into `DatabaseError` and `ClientNotice`. Note that a
7+
`detail` field has been added (previously it was added to the message) as
8+
part of this change.
49

510
## v2.0.0 (2024-02-29)
611

src/client.ts

+47-28
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,6 @@ import { EventEmitter } from 'node:events';
1515
import { Defaults } from './defaults.js';
1616
import * as logger from './logging.js';
1717

18-
import { postgresqlErrorCodes } from './errors.js';
1918
import { Queue } from './queue.js';
2019
import { Query } from './query.js';
2120

@@ -30,7 +29,6 @@ import {
3029
ClientConnectionDefaults,
3130
ClientConnectionOptions,
3231
DatabaseError,
33-
ErrorLevel,
3432
Message,
3533
Reader,
3634
RowDescription,
@@ -52,11 +50,7 @@ export interface ConnectionInfo {
5250
parameters: ReadonlyMap<string, string>;
5351
}
5452

55-
export interface ClientNotice extends DatabaseError {
56-
level: ErrorLevel;
57-
code: keyof typeof postgresqlErrorCodes;
58-
message: string;
59-
}
53+
export interface ClientNotice extends DatabaseError {}
6054

6155
export interface DataTypeError {
6256
dataType: DataType;
@@ -780,11 +774,7 @@ export class ClientImpl {
780774
}
781775

782776
private parseError(buffer: Buffer) {
783-
let level: DatabaseError['level'] | null = null;
784-
let code: DatabaseError['code'] | null = null;
785-
let message: DatabaseError['message'] | null = null;
786-
let details: string | null = null;
787-
777+
const params: Partial<ConstructorParameters<typeof DatabaseError>> = [];
788778
const length = buffer.length;
789779
let offset = 0;
790780

@@ -793,45 +783,74 @@ export class ClientImpl {
793783
if (next < 0) break;
794784

795785
const value = buffer.subarray(offset + 1, next).toString();
796-
switch (buffer[offset]) {
786+
787+
// See https://www.postgresql.org/docs/current/protocol-error-fields.html
788+
const token = buffer[offset];
789+
switch (token) {
790+
// S:
797791
case 0x53: {
798-
if (level === null) {
799-
level = value as DatabaseError['level'];
800-
}
792+
params[0] = value as DatabaseError['level'];
801793
break;
802794
}
795+
// V:
796+
// This is present only in messages generated by PostgreSQL
797+
// versions 9.6 and later, taking priority over the previous
798+
// case.
803799
case 0x56: {
804-
level = value as DatabaseError['level'];
800+
params[0] = value as DatabaseError['level'];
805801
break;
806802
}
803+
// C:
807804
case 0x43: {
808-
code = value as DatabaseError['code'];
805+
params[1] = value as DatabaseError['code'];
809806
break;
810807
}
808+
// M:
809+
case 0x4d: {
810+
params[2] = value;
811+
break;
812+
}
813+
// D:
811814
case 0x44: {
812-
details = value;
815+
params[3] = value;
813816
break;
814817
}
815-
case 0x4d: {
816-
message = value;
818+
// F:
819+
case 0x46: {
820+
params[4] = value;
817821
break;
818822
}
819-
default:
823+
// H:
824+
case 0x48: {
825+
params[5] = value;
820826
break;
827+
}
828+
// L:
829+
case 0x4c: {
830+
params[6] = parseInt(value);
831+
break;
832+
}
833+
// P:
834+
case 0x50: {
835+
params[7] = value;
836+
break;
837+
}
838+
// R:
839+
case 0x52: {
840+
params[8] = parseInt(value);
841+
break;
842+
}
821843
}
822844

823845
offset = next + 1;
824846
}
825847

848+
const [level, code, message, ...optional] = params;
826849
if (level && code && message) {
827-
return new DatabaseError(
828-
level,
829-
code,
830-
details ? `${message}: ${details}` : message,
831-
);
850+
return new DatabaseError(level, code, message, ...optional);
832851
}
833852

834-
throw new Error('Unable to parse error message.');
853+
throw new Error('Malformed error response');
835854
}
836855

837856
private handle(buffer: Buffer, offset: number, size: number): number {

src/errors.ts

-234
This file was deleted.

0 commit comments

Comments
 (0)