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