• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# Postgres Related regression tests
2#
3# Type the following command to launch start the tests:
4# $ test/run_tests -P "load_contrib('postgres')" -t test/contrib/postgres.uts
5
6+ postgres
7
8= postgres initialization
9
10from scapy.contrib.postgres import *
11
12ssl_request = "\x00\x00\x00\x08\x04\xd2\x16\x2f"
13
14startup = Startup(
15    b"\x00\x00\x00\x57\x00\x03\x00\x00\x75\x73\x65\x72\x00\x70\x6f\x73"
16    b"\x74\x67\x72\x65\x73\x00\x64\x61\x74\x61\x62\x61\x73\x65\x00\x70"
17    b"\x6f\x73\x74\x67\x72\x65\x73\x00\x61\x70\x70\x6c\x69\x63\x61\x74"
18    b"\x69\x6f\x6e\x5f\x6e\x61\x6d\x65\x00\x70\x73\x71\x6c\x00\x63\x6c"
19    b"\x69\x65\x6e\x74\x5f\x65\x6e\x63\x6f\x64\x69\x6e\x67\x00\x57\x49"
20    b"\x4e\x31\x32\x35\x32\x00\x00"
21)
22
23assert startup.len == 87
24assert startup.protocol_version_major == 3
25assert startup.protocol_version_minor == 0
26assert (
27    startup.options
28    == b"user\x00postgres\x00database\x00postgres\x00application_name\x00psql\x00client_encoding\x00WIN1252\x00\x00"
29)
30
31init_packet = (
32    b"\x52\x00\x00\x00\x08\x00\x00\x00\x00\x53\x00\x00\x00\x1a\x61\x70"
33    b"\x70\x6c\x69\x63\x61\x74\x69\x6f\x6e\x5f\x6e\x61\x6d\x65\x00\x70"
34    b"\x73\x71\x6c\x00\x53\x00\x00\x00\x1c\x63\x6c\x69\x65\x6e\x74\x5f"
35    b"\x65\x6e\x63\x6f\x64\x69\x6e\x67\x00\x57\x49\x4e\x31\x32\x35\x32"
36    b"\x00\x53\x00\x00\x00\x17\x44\x61\x74\x65\x53\x74\x79\x6c\x65\x00"
37    b"\x49\x53\x4f\x2c\x20\x4d\x44\x59\x00\x53\x00\x00\x00\x26\x64\x65"
38    b"\x66\x61\x75\x6c\x74\x5f\x74\x72\x61\x6e\x73\x61\x63\x74\x69\x6f"
39    b"\x6e\x5f\x72\x65\x61\x64\x5f\x6f\x6e\x6c\x79\x00\x6f\x66\x66\x00"
40    b"\x53\x00\x00\x00\x17\x69\x6e\x5f\x68\x6f\x74\x5f\x73\x74\x61\x6e"
41    b"\x64\x62\x79\x00\x6f\x66\x66\x00\x53\x00\x00\x00\x19\x69\x6e\x74"
42    b"\x65\x67\x65\x72\x5f\x64\x61\x74\x65\x74\x69\x6d\x65\x73\x00\x6f"
43    b"\x6e\x00\x53\x00\x00\x00\x1b\x49\x6e\x74\x65\x72\x76\x61\x6c\x53"
44    b"\x74\x79\x6c\x65\x00\x70\x6f\x73\x74\x67\x72\x65\x73\x00\x53\x00"
45    b"\x00\x00\x14\x69\x73\x5f\x73\x75\x70\x65\x72\x75\x73\x65\x72\x00"
46    b"\x6f\x6e\x00\x53\x00\x00\x00\x19\x73\x65\x72\x76\x65\x72\x5f\x65"
47    b"\x6e\x63\x6f\x64\x69\x6e\x67\x00\x55\x54\x46\x38\x00\x53\x00\x00"
48    b"\x00\x32\x73\x65\x72\x76\x65\x72\x5f\x76\x65\x72\x73\x69\x6f\x6e"
49    b"\x00\x31\x34\x2e\x32\x20\x28\x44\x65\x62\x69\x61\x6e\x20\x31\x34"
50    b"\x2e\x32\x2d\x31\x2e\x70\x67\x64\x67\x31\x31\x30\x2b\x31\x29\x00"
51    b"\x53\x00\x00\x00\x23\x73\x65\x73\x73\x69\x6f\x6e\x5f\x61\x75\x74"
52    b"\x68\x6f\x72\x69\x7a\x61\x74\x69\x6f\x6e\x00\x70\x6f\x73\x74\x67"
53    b"\x72\x65\x73\x00\x53\x00\x00\x00\x23\x73\x74\x61\x6e\x64\x61\x72"
54    b"\x64\x5f\x63\x6f\x6e\x66\x6f\x72\x6d\x69\x6e\x67\x5f\x73\x74\x72"
55    b"\x69\x6e\x67\x73\x00\x6f\x6e\x00\x53\x00\x00\x00\x15\x54\x69\x6d"
56    b"\x65\x5a\x6f\x6e\x65\x00\x45\x74\x63\x2f\x55\x54\x43\x00\x4b\x00"
57    b"\x00\x00\x0c\x00\x00\x01\x7f\x43\x4c\x36\xa5\x5a\x00\x00\x00\x05\x49"
58)
59
60= postgres backend sequence
61
62init = PostgresBackend(init_packet)
63
64assert isinstance(init.contents[0], Authentication)
65assert init.contents[0].len == 8
66assert init.contents[0].method == 0
67assert len(init.contents) == 16
68assert isinstance(init.contents[1], ParameterStatus)
69assert init.contents[1].len == 26
70assert init.contents[1].parameter == b"application_name"
71assert init.contents[1].value == b"psql"
72
73= simple queries
74
75simple_query_packet = (
76    b"\x51\x00\x00\x00\x15\x53\x45\x4c\x45\x43\x54\x20\x56\x45\x52\x53"
77    b"\x49\x4f\x4e\x28\x29\x00"
78)
79simple_query = PostgresFrontend(simple_query_packet)
80
81assert isinstance(simple_query.contents[0], Query)
82assert simple_query.contents[0].len == 21
83assert simple_query.contents[0].query == b"SELECT VERSION()"
84
85pair = SignedIntStrPair(b"\x00\x00\x00\x04\x01\x02\x03\x04")
86
87assert pair.len == 4
88assert pair.data == b"\x01\x02\x03\x04"
89
90command_response_packet = (
91    b"\x54\x00\x00\x00\x20\x00\x01\x76\x65\x72\x73\x69\x6f\x6e\x00\x00"
92    b"\x00\x00\x00\x00\x00\x00\x00\x00\x19\xff\xff\xff\xff\xff\xff\x00"
93    b"\x00\x44\x00\x00\x00\x85\x00\x01\x00\x00\x00\x7b\x50\x6f\x73\x74"
94    b"\x67\x72\x65\x53\x51\x4c\x20\x31\x34\x2e\x32\x20\x28\x44\x65\x62"
95    b"\x69\x61\x6e\x20\x31\x34\x2e\x32\x2d\x31\x2e\x70\x67\x64\x67\x31"
96    b"\x31\x30\x2b\x31\x29\x20\x6f\x6e\x20\x78\x38\x36\x5f\x36\x34\x2d"
97    b"\x70\x63\x2d\x6c\x69\x6e\x75\x78\x2d\x67\x6e\x75\x2c\x20\x63\x6f"
98    b"\x6d\x70\x69\x6c\x65\x64\x20\x62\x79\x20\x67\x63\x63\x20\x28\x44"
99    b"\x65\x62\x69\x61\x6e\x20\x31\x30\x2e\x32\x2e\x31\x2d\x36\x29\x20"
100    b"\x31\x30\x2e\x32\x2e\x31\x20\x32\x30\x32\x31\x30\x31\x31\x30\x2c"
101    b"\x20\x36\x34\x2d\x62\x69\x74\x43\x00\x00\x00\x0d\x53\x45\x4c\x45"
102    b"\x43\x54\x20\x31\x00\x5a\x00\x00\x00\x05\x49"
103)
104
105= row data response
106
107command_response = PostgresBackend(command_response_packet)
108
109assert len(command_response.contents) == 4
110assert isinstance(command_response.contents[0], RowDescription)
111rd = command_response.contents[0]
112assert rd.len == 32
113assert rd.numfields == 1
114assert rd.cols[0].col == b"version"
115assert rd.cols[0].tableoid == 0
116assert rd.cols[0].colno == 0
117assert rd.cols[0].typeoid == 25
118assert rd.cols[0].typelen == -1
119assert rd.cols[0].format == 0
120assert rd.cols[0].typemod == -1
121
122assert isinstance(command_response.contents[1], DataRow)
123assert command_response.contents[1].len == 133
124assert command_response.contents[1].numfields == 1
125assert len(command_response.contents[1].data) == 1
126assert isinstance(command_response.contents[1].data[0], SignedIntStrPair)
127assert command_response.contents[1].data[0].len == 123
128assert (
129    command_response.contents[1].data[0].data
130    == b"PostgreSQL 14.2 (Debian 14.2-1.pgdg110+1) on x86_64-pc-linux-gnu, compiled by gcc (Debian 10.2.1-6) 10.2.1 20210110, 64-bit"
131)
132
133assert isinstance(command_response.contents[2], CommandComplete)
134assert isinstance(command_response.contents[3], ReadyForQuery)
135
136three_col_rd = RowDescription(
137    b"\x54\x00\x00\x00\x55\x00\x03\x6e\x61\x6d\x65\x00\x00\x00\x00\x00"
138    b"\x00\x00\x00\x00\x00\x19\xff\xff\xff\xff\xff\xff\x00\x00\x73\x65"
139    b"\x74\x74\x69\x6e\x67\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x19"
140    b"\xff\xff\xff\xff\xff\xff\x00\x00\x64\x65\x73\x63\x72\x69\x70\x74"
141    b"\x69\x6f\x6e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x19\xff\xff"
142    b"\xff\xff\xff\xff\x00\x00"
143)
144assert three_col_rd.len == 85
145assert three_col_rd.numfields == 3
146assert len(three_col_rd.cols) == 3
147
148three_col_dr = DataRow(
149    b"\x44\x00\x00\x00\x63\x00\x03\x00\x00\x00\x17\x61\x6c\x6c\x6f\x77"
150    b"\x5f\x73\x79\x73\x74\x65\x6d\x5f\x74\x61\x62\x6c\x65\x5f\x6d\x6f"
151    b"\x64\x73\x00\x00\x00\x03\x6f\x66\x66\x00\x00\x00\x37\x41\x6c\x6c"
152    b"\x6f\x77\x73\x20\x6d\x6f\x64\x69\x66\x69\x63\x61\x74\x69\x6f\x6e"
153    b"\x73\x20\x6f\x66\x20\x74\x68\x65\x20\x73\x74\x72\x75\x63\x74\x75"
154    b"\x72\x65\x20\x6f\x66\x20\x73\x79\x73\x74\x65\x6d\x20\x74\x61\x62"
155    b"\x6c\x65\x73\x2e"
156)
157
158assert three_col_dr.numfields == 3
159assert len(three_col_dr.data) == 3
160assert three_col_dr.data[0].len == 23
161assert three_col_dr.data[0].data == b"allow_system_table_mods"
162assert three_col_dr.data[1].len == 3
163assert three_col_dr.data[1].data == b"off"
164assert three_col_dr.data[2].len == 55
165assert (
166    three_col_dr.data[2].data
167    == b"Allows modifications of the structure of system tables."
168)
169
170= errors
171
172error_response = ErrorResponse(
173    b"\x45\x00\x00\x00\x69\x53\x45\x52\x52\x4f\x52\x00\x56\x45\x52\x52"
174    b"\x4f\x52\x00\x43\x34\x32\x50\x30\x31\x00\x4d\x72\x65\x6c\x61\x74"
175    b"\x69\x6f\x6e\x20\x22\x66\x6f\x6f\x62\x61\x72\x22\x20\x64\x6f\x65"
176    b"\x73\x20\x6e\x6f\x74\x20\x65\x78\x69\x73\x74\x00\x50\x31\x35\x00"
177    b"\x46\x70\x61\x72\x73\x65\x5f\x72\x65\x6c\x61\x74\x69\x6f\x6e\x2e"
178    b"\x63\x00\x4c\x31\x33\x38\x31\x00\x52\x70\x61\x72\x73\x65\x72\x4f"
179    b"\x70\x65\x6e\x54\x61\x62\x6c\x65\x00\x00"
180)
181
182assert len(error_response.error_fields) == 8
183assert error_response.error_fields[0] == ("Severity", b"ERROR")
184assert error_response.error_fields[7] == ("Routine", b"parserOpenTable")
185
186= copy data response and request
187
188copyin_response = CopyInResponse(b"\x47\x00\x00\x00\x0f\x00\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00")
189
190assert copyin_response.len == 15
191assert copyin_response.format == 0  # Text
192assert len(copyin_response.cols) == 4
193assert copyin_response.ncols == 4
194assert copyin_response.cols[0] == 0  # Text
195assert copyin_response.cols[1] == 0  # Text
196assert copyin_response.cols[2] == 0  # Text
197assert copyin_response.cols[3] == 0  # Text
198
199copydata_in = PostgresFrontend(b"\x64\x00\x00\x00\x10\x31\x2c\x42\x6f\x62\x2c\x32\x33\x2c\x31\x0d" \
200b"\x0a\x64\x00\x00\x00\x12\x32\x2c\x53\x61\x6c\x6c\x79\x2c\x34\x33" \
201b"\x2c\x32\x0d\x0a\x64\x00\x00\x00\x14\x33\x2c\x50\x61\x72\x64\x65" \
202b"\x65\x70\x2c\x35\x34\x2c\x33\x0d\x0a\x64\x00\x00\x00\x0f\x34\x2c" \
203b"\x53\x75\x2c\x33\x32\x2c\x34\x0d\x0a\x64\x00\x00\x00\x0f\x35\x2c" \
204b"\x58\x69\x2c\x34\x33\x2c\x35\x0d\x0a\x64\x00\x00\x00\x0e\x36\x2c" \
205b"\x50\x69\x70\x2c\x36\x36\x2c\x36\x63\x00\x00\x00\x04"
206)
207
208copyout_response = CopyOutResponse(b"\x48\x00\x00\x00\x0f\x00\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00")
209assert copyout_response.len == 15
210
211# Combined message
212copydata_out = PostgresBackend(b"\x48\x00\x00\x00\x0f\x00\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00" \
213b"\x64\x00\x00\x00\x0f\x31\x09\x42\x6f\x62\x09\x32\x33\x09\x31\x0a" \
214b"\x64\x00\x00\x00\x11\x32\x09\x53\x61\x6c\x6c\x79\x09\x34\x33\x09" \
215b"\x32\x0a\x64\x00\x00\x00\x13\x33\x09\x50\x61\x72\x64\x65\x65\x70" \
216b"\x09\x35\x34\x09\x33\x0a\x64\x00\x00\x00\x0e\x34\x09\x53\x75\x09" \
217b"\x33\x32\x09\x34\x0a\x64\x00\x00\x00\x0e\x35\x09\x58\x69\x09\x34" \
218b"\x33\x09\x35\x0a\x64\x00\x00\x00\x0f\x36\x09\x50\x69\x70\x09\x36" \
219b"\x36\x09\x36\x0a\x63\x00\x00\x00\x04\x43\x00\x00\x00\x0b\x43\x4f" \
220b"\x50\x59\x20\x36\x00\x5a\x00\x00\x00\x05\x49")
221
222assert len(copydata_out.contents) == 10
223assert copydata_out.contents[0].len == 15
224assert isinstance(copydata_out.contents[0], CopyOutResponse)
225assert isinstance(copydata_out.contents[1], CopyData)
226assert copydata_out.contents[1].len == 15
227assert copydata_out.contents[1].data == b'1\tBob\t23\t1\n'
228assert isinstance(copydata_out.contents[2], CopyData)
229assert copydata_out.contents[2].data == b'2\tSally\t43\t2\n'
230assert isinstance(copydata_out.contents[3], CopyData)
231assert copydata_out.contents[3].data == b'3\tPardeep\t54\t3\n'
232assert isinstance(copydata_out.contents[4], CopyData)
233assert copydata_out.contents[4].data == b'4\tSu\t32\t4\n'
234assert isinstance(copydata_out.contents[5], CopyData)
235assert copydata_out.contents[5].data == b'5\tXi\t43\t5\n'
236assert isinstance(copydata_out.contents[6], CopyData)
237assert copydata_out.contents[6].data == b'6\tPip\t66\t6\n'
238assert isinstance(copydata_out.contents[7], CopyDone)
239assert isinstance(copydata_out.contents[8], CommandComplete)
240assert isinstance(copydata_out.contents[9], ReadyForQuery)
241
242= Check example request packet
243
244request = PostgresFrontend(
245    b"\x50\x00\x00\x00\x64\x00\x53\x45\x4c\x45\x43\x54\x20\x44\x5f\x4e"
246    b"\x45\x58\x54\x5f\x4f\x5f\x49\x44\x2c\x20\x44\x5f\x54\x41\x58\x20"
247    b"\x20\x20\x46\x52\x4f\x4d\x20\x64\x69\x73\x74\x72\x69\x63\x74\x20"
248    b"\x57\x48\x45\x52\x45\x20\x44\x5f\x57\x5f\x49\x44\x20\x3d\x20\x24"
249    b"\x31\x20\x41\x4e\x44\x20\x44\x5f\x49\x44\x20\x3d\x20\x24\x32\x20"
250    b"\x46\x4f\x52\x20\x55\x50\x44\x41\x54\x45\x00\x00\x02\x00\x00\x00"
251    b"\x17\x00\x00\x00\x17\x42\x00\x00\x00\x20\x00\x00\x00\x02\x00\x01"
252    b"\x00\x01\x00\x02\x00\x00\x00\x04\x00\x00\x00\x14\x00\x00\x00\x04"
253    b"\x00\x00\x00\x0a\x00\x00\x44\x00\x00\x00\x06\x50\x00\x45\x00\x00"
254    b"\x00\x09\x00\x00\x00\x00\x00\x53\x00\x00\x00\x04"
255)
256
257assert len(request.contents) == 5
258assert isinstance(request.contents[0], Parse)
259assert isinstance(request.contents[1], Bind)
260assert isinstance(request.contents[2], Describe)
261assert isinstance(request.contents[3], Execute)
262assert isinstance(request.contents[4], Sync)
263
264= Check parse decoding
265
266parse_msg = request.contents[0]
267assert parse_msg.len == 100
268assert parse_msg.destination == b""
269assert parse_msg.query == b"SELECT D_NEXT_O_ID, D_TAX   FROM district WHERE D_W_ID = $1 AND D_ID = $2 FOR UPDATE"
270assert parse_msg.num_param_dtypes == 2
271assert parse_msg.params[0] == 23
272assert parse_msg.params[1] == 23
273
274= Check bind decoding
275
276bind_msg = request.contents[1]
277assert bind_msg.len == 32
278assert bind_msg.destination == b""
279assert bind_msg.statement == b""
280assert bind_msg.codes_count == 2
281assert bind_msg.codes[0] == 1
282assert bind_msg.codes[1] == 1
283assert bind_msg.values_count == 2
284assert bind_msg.values[0].len == 4
285assert bind_msg.values[0].data == b"\x00\x00\x00\x14"
286assert bind_msg.values[1].len == 4
287assert bind_msg.values[1].data == b"\x00\x00\x00\x0a"
288assert bind_msg.results_count == 0
289
290= Check describe decoding
291
292describe_msg = request.contents[2]
293assert describe_msg.len == 6
294assert describe_msg.close_type == b"P"
295assert describe_msg.statement == b""
296
297= Check execute decoding
298
299exec_msg = request.contents[3]
300assert exec_msg.len == 9
301assert exec_msg.portal == b""
302assert exec_msg.rows == 0
303
304= Check sync decoding
305
306sync_msg = request.contents[4]
307assert sync_msg.len == 4
308