1import os 2import socket 3 4import pytest 5 6from unit.applications.proto import TestApplicationProto 7from unit.option import option 8from unit.utils import waitforfiles 9 10 11class TestStatic(TestApplicationProto): 12 prerequisites = {} 13 14 def setup_method(self): 15 os.makedirs(option.temp_dir + '/assets/dir') 16 with open(option.temp_dir + '/assets/index.html', 'w') as index, open( 17 option.temp_dir + '/assets/README', 'w' 18 ) as readme, open( 19 option.temp_dir + '/assets/log.log', 'w' 20 ) as log, open( 21 option.temp_dir + '/assets/dir/file', 'w' 22 ) as file: 23 index.write('0123456789') 24 readme.write('readme') 25 log.write('[debug]') 26 file.write('blah') 27 28 self._load_conf( 29 { 30 "listeners": {"*:7080": {"pass": "routes"}}, 31 "routes": [{"action": {"share": option.temp_dir + "/assets"}}], 32 "settings": { 33 "http": { 34 "static": { 35 "mime_types": {"text/plain": [".log", "README"]} 36 } 37 } 38 }, 39 } 40 ) 41 42 def test_static_index(self): 43 assert self.get(url='/index.html')['body'] == '0123456789', 'index' 44 assert self.get(url='/')['body'] == '0123456789', 'index 2' 45 assert self.get(url='//')['body'] == '0123456789', 'index 3' 46 assert self.get(url='/.')['body'] == '0123456789', 'index 4' 47 assert self.get(url='/./')['body'] == '0123456789', 'index 5' 48 assert self.get(url='/?blah')['body'] == '0123456789', 'index vars' 49 assert self.get(url='/#blah')['body'] == '0123456789', 'index anchor' 50 assert self.get(url='/dir/')['status'] == 404, 'index not found' 51 52 resp = self.get(url='/index.html/') 53 assert resp['status'] == 404, 'index not found 2 status' 54 assert ( 55 resp['headers']['Content-Type'] == 'text/html' 56 ), 'index not found 2 Content-Type' 57 58 def test_static_large_file(self, temp_dir): 59 file_size = 32 * 1024 * 1024 60 with open(temp_dir + '/assets/large', 'wb') as f: 61 f.seek(file_size - 1) 62 f.write(b'\0') 63 64 assert ( 65 len(self.get(url='/large', read_buffer_size=1024 * 1024)['body']) 66 == file_size 67 ), 'large file' 68 69 def test_static_etag(self, temp_dir): 70 etag = self.get(url='/')['headers']['ETag'] 71 etag_2 = self.get(url='/README')['headers']['ETag'] 72 73 assert etag != etag_2, 'different ETag' 74 assert etag == self.get(url='/')['headers']['ETag'], 'same ETag' 75 76 with open(temp_dir + '/assets/index.html', 'w') as f: 77 f.write('blah') 78 79 assert etag != self.get(url='/')['headers']['ETag'], 'new ETag' 80 81 def test_static_redirect(self): 82 resp = self.get(url='/dir') 83 assert resp['status'] == 301, 'redirect status' 84 assert resp['headers']['Location'] == '/dir/', 'redirect Location' 85 assert 'Content-Type' not in resp['headers'], 'redirect Content-Type' 86 87 def test_static_space_in_name(self, temp_dir): 88 os.rename( 89 temp_dir + '/assets/dir/file', temp_dir + '/assets/dir/fi le', 90 ) 91 assert waitforfiles(temp_dir + '/assets/dir/fi le') 92 assert self.get(url='/dir/fi le')['body'] == 'blah', 'file name' 93 94 os.rename(temp_dir + '/assets/dir', temp_dir + '/assets/di r') 95 assert waitforfiles(temp_dir + '/assets/di r/fi le') 96 assert self.get(url='/di r/fi le')['body'] == 'blah', 'dir name' 97 98 os.rename(temp_dir + '/assets/di r', temp_dir + '/assets/ di r ') 99 assert waitforfiles(temp_dir + '/assets/ di r /fi le') 100 assert ( 101 self.get(url='/ di r /fi le')['body'] == 'blah' 102 ), 'dir name enclosing' 103 104 assert ( 105 self.get(url='/%20di%20r%20/fi le')['body'] == 'blah' 106 ), 'dir encoded' 107 assert ( 108 self.get(url='/ di r %2Ffi le')['body'] == 'blah' 109 ), 'slash encoded' 110 assert ( 111 self.get(url='/ di r /fi%20le')['body'] == 'blah' 112 ), 'file encoded' 113 assert ( 114 self.get(url='/%20di%20r%20%2Ffi%20le')['body'] == 'blah' 115 ), 'encoded' 116 assert ( 117 self.get(url='/%20%64%69%20%72%20%2F%66%69%20%6C%65')['body'] 118 == 'blah' 119 ), 'encoded 2' 120 121 os.rename( 122 temp_dir + '/assets/ di r /fi le', 123 temp_dir + '/assets/ di r / fi le ', 124 ) 125 assert waitforfiles(temp_dir + '/assets/ di r / fi le ') 126 assert ( 127 self.get(url='/%20di%20r%20/%20fi%20le%20')['body'] == 'blah' 128 ), 'file name enclosing' 129 130 try: 131 open(temp_dir + '/ф а', 'a').close() 132 utf8 = True 133 134 except KeyboardInterrupt: 135 raise 136 137 except: 138 utf8 = False 139 140 if utf8: 141 os.rename( 142 temp_dir + '/assets/ di r / fi le ', 143 temp_dir + '/assets/ di r /фа йл', 144 ) 145 assert waitforfiles(temp_dir + '/assets/ di r /фа йл') 146 assert ( 147 self.get(url='/ di r /фа йл')['body'] == 'blah' 148 ), 'file name 2' 149 150 os.rename( 151 temp_dir + '/assets/ di r ', temp_dir + '/assets/ди ректория', 152 ) 153 assert waitforfiles(temp_dir + '/assets/ди ректория/фа йл') 154 assert ( 155 self.get(url='/ди ректория/фа йл')['body'] == 'blah' 156 ), 'dir name 2' 157 158 def test_static_unix_socket(self, temp_dir): 159 sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) 160 sock.bind(temp_dir + '/assets/unix_socket') 161 162 assert self.get(url='/unix_socket')['status'] == 404, 'socket' 163 164 sock.close() 165 166 def test_static_unix_fifo(self, temp_dir): 167 os.mkfifo(temp_dir + '/assets/fifo') 168 169 assert self.get(url='/fifo')['status'] == 404, 'fifo' 170 171 def test_static_symlink(self, temp_dir): 172 os.symlink(temp_dir + '/assets/dir', temp_dir + '/assets/link') 173 174 assert self.get(url='/dir')['status'] == 301, 'dir' 175 assert self.get(url='/dir/file')['status'] == 200, 'file' 176 assert self.get(url='/link')['status'] == 301, 'symlink dir' 177 assert self.get(url='/link/file')['status'] == 200, 'symlink file' 178 179 def test_static_method(self): 180 resp = self.head() 181 assert resp['status'] == 200, 'HEAD status' 182 assert resp['body'] == '', 'HEAD empty body' 183 184 assert self.delete()['status'] == 405, 'DELETE' 185 assert self.post()['status'] == 405, 'POST' 186 assert self.put()['status'] == 405, 'PUT' 187 188 def test_static_path(self): 189 assert self.get(url='/dir/../dir/file')['status'] == 200, 'relative' 190 191 assert self.get(url='./')['status'] == 400, 'path invalid' 192 assert self.get(url='../')['status'] == 400, 'path invalid 2' 193 assert self.get(url='/..')['status'] == 400, 'path invalid 3' 194 assert self.get(url='../assets/')['status'] == 400, 'path invalid 4' 195 assert self.get(url='/../assets/')['status'] == 400, 'path invalid 5' 196 197 def test_static_two_clients(self): 198 _, sock = self.get(url='/', start=True, no_recv=True) 199 _, sock2 = self.get(url='/', start=True, no_recv=True) 200 201 assert sock.recv(1) == b'H', 'client 1' 202 assert sock2.recv(1) == b'H', 'client 2' 203 assert sock.recv(1) == b'T', 'client 1 again' 204 assert sock2.recv(1) == b'T', 'client 2 again' 205 206 sock.close() 207 sock2.close() 208 209 def test_static_mime_types(self): 210 assert 'success' in self.conf( 211 { 212 "text/x-code/x-blah/x-blah": "readme", 213 "text/plain": [".html", ".log", "file"], 214 }, 215 'settings/http/static/mime_types', 216 ), 'configure mime_types' 217 218 assert ( 219 self.get(url='/README')['headers']['Content-Type'] 220 == 'text/x-code/x-blah/x-blah' 221 ), 'mime_types string case insensitive' 222 assert ( 223 self.get(url='/index.html')['headers']['Content-Type'] 224 == 'text/plain' 225 ), 'mime_types html' 226 assert ( 227 self.get(url='/')['headers']['Content-Type'] == 'text/plain' 228 ), 'mime_types index default' 229 assert ( 230 self.get(url='/dir/file')['headers']['Content-Type'] 231 == 'text/plain' 232 ), 'mime_types file in dir' 233 234 def test_static_mime_types_partial_match(self): 235 assert 'success' in self.conf( 236 {"text/x-blah": ["ile", "fil", "f", "e", ".file"],}, 237 'settings/http/static/mime_types', 238 ), 'configure mime_types' 239 assert 'Content-Type' not in self.get(url='/dir/file'), 'partial match' 240 241 def test_static_mime_types_reconfigure(self): 242 assert 'success' in self.conf( 243 { 244 "text/x-code": "readme", 245 "text/plain": [".html", ".log", "file"], 246 }, 247 'settings/http/static/mime_types', 248 ), 'configure mime_types' 249 250 assert self.conf_get('settings/http/static/mime_types') == { 251 'text/x-code': 'readme', 252 'text/plain': ['.html', '.log', 'file'], 253 }, 'mime_types get' 254 assert ( 255 self.conf_get('settings/http/static/mime_types/text%2Fx-code') 256 == 'readme' 257 ), 'mime_types get string' 258 assert self.conf_get( 259 'settings/http/static/mime_types/text%2Fplain' 260 ) == ['.html', '.log', 'file'], 'mime_types get array' 261 assert ( 262 self.conf_get('settings/http/static/mime_types/text%2Fplain/1') 263 == '.log' 264 ), 'mime_types get array element' 265 266 assert 'success' in self.conf_delete( 267 'settings/http/static/mime_types/text%2Fplain/2' 268 ), 'mime_types remove array element' 269 assert ( 270 'Content-Type' not in self.get(url='/dir/file')['headers'] 271 ), 'mime_types removed' 272 273 assert 'success' in self.conf_post( 274 '"file"', 'settings/http/static/mime_types/text%2Fplain' 275 ), 'mime_types add array element' 276 assert ( 277 self.get(url='/dir/file')['headers']['Content-Type'] 278 == 'text/plain' 279 ), 'mime_types reverted' 280 281 assert 'success' in self.conf( 282 '"file"', 'settings/http/static/mime_types/text%2Fplain' 283 ), 'configure mime_types update' 284 assert ( 285 self.get(url='/dir/file')['headers']['Content-Type'] 286 == 'text/plain' 287 ), 'mime_types updated' 288 assert ( 289 'Content-Type' not in self.get(url='/log.log')['headers'] 290 ), 'mime_types updated 2' 291 292 assert 'success' in self.conf( 293 '".log"', 'settings/http/static/mime_types/text%2Fblahblahblah' 294 ), 'configure mime_types create' 295 assert ( 296 self.get(url='/log.log')['headers']['Content-Type'] 297 == 'text/blahblahblah' 298 ), 'mime_types create' 299 300 def test_static_mime_types_correct(self): 301 assert 'error' in self.conf( 302 {"text/x-code": "readme", "text/plain": "readme"}, 303 'settings/http/static/mime_types', 304 ), 'mime_types same extensions' 305 assert 'error' in self.conf( 306 {"text/x-code": [".h", ".c"], "text/plain": ".c"}, 307 'settings/http/static/mime_types', 308 ), 'mime_types same extensions array' 309 assert 'error' in self.conf( 310 {"text/x-code": [".h", ".c", "readme"], "text/plain": "README",}, 311 'settings/http/static/mime_types', 312 ), 'mime_types same extensions case insensitive' 313 314 @pytest.mark.skip('not yet') 315 def test_static_mime_types_invalid(self, temp_dir): 316 assert 'error' in self.http( 317 b"""PUT /config/settings/http/static/mime_types/%0%00% HTTP/1.1\r 318Host: localhost\r 319Connection: close\r 320Content-Length: 6\r 321\r 322\"blah\"""", 323 raw_resp=True, 324 raw=True, 325 sock_type='unix', 326 addr=temp_dir + '/control.unit.sock', 327 ), 'mime_types invalid' 328