Linux server1.dn-server.com 4.18.0-553.89.1.lve.el8.x86_64 #1 SMP Wed Dec 10 13:58:50 UTC 2025 x86_64
LiteSpeed
Server IP : 195.201.204.189 & Your IP : 216.73.216.198
Domains :
Cant Read [ /etc/named.conf ]
User : beriska1
Terminal
Auto Root
Create File
Create Folder
Localroot Suggester
Backdoor Destroyer
Readme
/
opt /
alt /
python33 /
lib64 /
python3.3 /
test /
Delete
Unzip
Name
Size
Permission
Date
Action
__pycache__
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
audiodata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
capath
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
cjkencodings
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
data
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
decimaltestdata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
encoded_modules
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
imghdrdata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
namespace_pkgs
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
sndhdrdata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
subprocessdata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
support
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
test_email
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
test_importlib
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
test_json
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
tracedmodules
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
xmltestdata
[ DIR ]
drwxr-xr-x
2026-05-01 04:20
185test.db
16
KB
-rw-r--r--
2017-09-19 12:02
Sine-1000Hz-300ms.aif
60.25
KB
-rw-r--r--
2017-09-19 12:02
__init__.py
47
B
-rw-r--r--
2024-04-17 20:28
__main__.py
474
B
-rw-r--r--
2024-04-17 20:28
audiotests.py
9.34
KB
-rw-r--r--
2024-04-17 20:28
autotest.py
211
B
-rw-r--r--
2024-04-17 20:28
bad_coding.py
24
B
-rw-r--r--
2024-04-17 20:28
bad_coding2.py
30
B
-rw-r--r--
2024-04-17 20:28
badcert.pem
1.88
KB
-rw-r--r--
2017-09-19 12:02
badkey.pem
2.11
KB
-rw-r--r--
2017-09-19 12:02
badsyntax_3131.py
32
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future3.py
172
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future4.py
153
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future5.py
184
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future6.py
161
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future7.py
196
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future8.py
122
B
-rw-r--r--
2024-04-17 20:28
badsyntax_future9.py
142
B
-rw-r--r--
2024-04-17 20:28
badsyntax_pep3120.py
14
B
-rw-r--r--
2024-04-17 20:28
buffer_tests.py
10.79
KB
-rw-r--r--
2024-04-17 20:28
cfgparser.1
21
B
-rw-r--r--
2017-09-19 12:02
cfgparser.2
19.02
KB
-rw-r--r--
2017-09-19 12:02
cfgparser.3
1.55
KB
-rw-r--r--
2017-09-19 12:02
check_soundcard.vbs
411
B
-rw-r--r--
2017-09-19 12:02
cmath_testcases.txt
133.86
KB
-rw-r--r--
2017-09-19 12:02
coding20731.py
18
B
-rw-r--r--
2024-04-17 20:28
curses_tests.py
1.22
KB
-rwxr-xr-x
2024-04-17 20:28
datetimetester.py
146.82
KB
-rw-r--r--
2024-04-17 20:28
dh1024.pem
300
B
-rw-r--r--
2017-09-19 12:02
dis_module.py
76
B
-rw-r--r--
2024-04-17 20:28
doctest_aliases.py
240
B
-rw-r--r--
2024-04-17 20:28
double_const.py
1.18
KB
-rw-r--r--
2024-04-17 20:28
empty.vbs
70
B
-rw-r--r--
2017-09-19 12:02
exception_hierarchy.txt
1.69
KB
-rw-r--r--
2017-09-19 12:02
floating_points.txt
15.92
KB
-rw-r--r--
2017-09-19 12:02
fork_wait.py
2.1
KB
-rw-r--r--
2024-04-17 20:28
formatfloat_testcases.txt
7.45
KB
-rw-r--r--
2017-09-19 12:02
future_test1.py
229
B
-rw-r--r--
2024-04-17 20:28
future_test2.py
149
B
-rw-r--r--
2024-04-17 20:28
gdb_sample.py
153
B
-rw-r--r--
2024-04-17 20:28
ieee754.txt
3.21
KB
-rw-r--r--
2017-09-19 12:02
inspect_fodder.py
854
B
-rw-r--r--
2024-04-17 20:28
inspect_fodder2.py
1.39
KB
-rw-r--r--
2024-04-17 20:28
keycert.passwd.pem
1.79
KB
-rw-r--r--
2017-09-19 12:02
keycert.pem
1.74
KB
-rw-r--r--
2017-09-19 12:02
keycert2.pem
1.75
KB
-rw-r--r--
2017-09-19 12:02
list_tests.py
17.26
KB
-rw-r--r--
2024-04-17 20:28
lock_tests.py
25.47
KB
-rw-r--r--
2024-04-17 20:28
mailcap.txt
1.24
KB
-rw-r--r--
2017-09-19 12:02
make_ssl_certs.py
1.84
KB
-rw-r--r--
2024-04-17 20:28
mapping_tests.py
21.5
KB
-rw-r--r--
2024-04-17 20:28
math_testcases.txt
23.19
KB
-rw-r--r--
2017-09-19 12:02
memory_watchdog.py
859
B
-rw-r--r--
2024-04-17 20:28
mime.types
47.37
KB
-rw-r--r--
2017-09-19 12:02
mock_socket.py
3.32
KB
-rw-r--r--
2024-04-17 20:28
mp_fork_bomb.py
323
B
-rw-r--r--
2024-04-17 20:28
multibytecodec_support.py
14.25
KB
-rw-r--r--
2024-04-17 20:28
nokia.pem
1.88
KB
-rw-r--r--
2017-09-19 12:02
nullbytecert.pem
5.31
KB
-rw-r--r--
2017-09-19 12:02
nullcert.pem
0
B
-rw-r--r--
2017-09-19 12:02
outstanding_bugs.py
370
B
-rw-r--r--
2024-04-17 20:28
pickletester.py
60.31
KB
-rw-r--r--
2024-04-17 20:28
profilee.py
2.97
KB
-rw-r--r--
2024-04-17 20:28
pstats.pck
65.05
KB
-rw-r--r--
2017-09-19 12:02
pyclbr_input.py
648
B
-rw-r--r--
2024-04-17 20:28
pydoc_mod.py
461
B
-rw-r--r--
2024-04-17 20:28
pydocfodder.py
6.18
KB
-rw-r--r--
2024-04-17 20:28
pystone.py
7.22
KB
-rwxr-xr-x
2024-04-17 20:28
randv2_32.pck
7.34
KB
-rw-r--r--
2017-09-19 12:02
randv2_64.pck
7.19
KB
-rw-r--r--
2017-09-19 12:02
randv3.pck
7.82
KB
-rw-r--r--
2017-09-19 12:02
re_tests.py
31.06
KB
-rwxr-xr-x
2024-04-17 20:28
regrtest.py
63.54
KB
-rwxr-xr-x
2024-04-17 20:28
relimport.py
27
B
-rw-r--r--
2024-04-17 20:28
reperf.py
538
B
-rw-r--r--
2024-04-17 20:28
sample_doctest.py
1.02
KB
-rw-r--r--
2024-04-17 20:28
sample_doctest_no_docstrings.py
227
B
-rw-r--r--
2024-04-17 20:28
sample_doctest_no_doctests.py
269
B
-rw-r--r--
2024-04-17 20:28
script_helper.py
5.22
KB
-rw-r--r--
2024-04-17 20:28
selfsigned_pythontestdotnet.pem
956
B
-rw-r--r--
2017-09-19 12:02
seq_tests.py
13.49
KB
-rw-r--r--
2024-04-17 20:28
sgml_input.html
8.1
KB
-rw-r--r--
2017-09-19 12:02
sha256.pem
8.15
KB
-rw-r--r--
2017-09-19 12:02
sortperf.py
4.69
KB
-rw-r--r--
2024-04-17 20:28
ssl_cert.pem
867
B
-rw-r--r--
2017-09-19 12:02
ssl_key.passwd.pem
963
B
-rw-r--r--
2017-09-19 12:02
ssl_key.pem
916
B
-rw-r--r--
2017-09-19 12:02
ssl_servers.py
6.82
KB
-rw-r--r--
2024-04-17 20:28
string_tests.py
62.74
KB
-rw-r--r--
2024-04-17 20:28
test___all__.py
3.86
KB
-rw-r--r--
2024-04-17 20:28
test___future__.py
2.44
KB
-rw-r--r--
2024-04-17 20:28
test__locale.py
6.89
KB
-rw-r--r--
2024-04-17 20:28
test__osx_support.py
11.41
KB
-rw-r--r--
2024-04-17 20:28
test_abc.py
12.76
KB
-rw-r--r--
2024-04-17 20:28
test_abstract_numbers.py
1.49
KB
-rw-r--r--
2024-04-17 20:28
test_aifc.py
14.48
KB
-rw-r--r--
2024-04-17 20:28
test_argparse.py
158.6
KB
-rw-r--r--
2024-04-17 20:28
test_array.py
43.88
KB
-rw-r--r--
2024-04-17 20:28
test_ast.py
43.49
KB
-rw-r--r--
2024-04-17 20:28
test_asynchat.py
9.08
KB
-rw-r--r--
2024-04-17 20:28
test_asyncore.py
26.05
KB
-rw-r--r--
2024-04-17 20:28
test_atexit.py
3.53
KB
-rw-r--r--
2024-04-17 20:28
test_audioop.py
17.39
KB
-rw-r--r--
2024-04-17 20:28
test_augassign.py
7.44
KB
-rw-r--r--
2024-04-17 20:28
test_base64.py
14.34
KB
-rw-r--r--
2024-04-17 20:28
test_bigaddrspace.py
2.92
KB
-rw-r--r--
2024-04-17 20:28
test_bigmem.py
44.18
KB
-rw-r--r--
2024-04-17 20:28
test_binascii.py
9.94
KB
-rw-r--r--
2024-04-17 20:28
test_binhex.py
1.47
KB
-rw-r--r--
2024-04-17 20:28
test_binop.py
12.36
KB
-rw-r--r--
2024-04-17 20:28
test_bisect.py
13.65
KB
-rw-r--r--
2024-04-17 20:28
test_bool.py
11.81
KB
-rw-r--r--
2024-04-17 20:28
test_buffer.py
155.09
KB
-rw-r--r--
2024-04-17 20:28
test_bufio.py
2.54
KB
-rw-r--r--
2024-04-17 20:28
test_builtin.py
54.84
KB
-rw-r--r--
2024-04-17 20:28
test_bytes.py
52.86
KB
-rw-r--r--
2024-04-17 20:28
test_bz2.py
30.8
KB
-rw-r--r--
2024-04-17 20:28
test_calendar.py
38.7
KB
-rw-r--r--
2024-04-17 20:28
test_call.py
3.08
KB
-rw-r--r--
2024-04-17 20:28
test_capi.py
13.81
KB
-rw-r--r--
2024-04-17 20:28
test_cgi.py
17.34
KB
-rw-r--r--
2024-04-17 20:28
test_cgitb.py
2.49
KB
-rw-r--r--
2024-04-17 20:28
test_charmapcodec.py
1.75
KB
-rw-r--r--
2024-04-17 20:28
test_class.py
13.79
KB
-rw-r--r--
2024-04-17 20:28
test_cmath.py
20.38
KB
-rw-r--r--
2024-04-17 20:28
test_cmd.py
6.08
KB
-rw-r--r--
2024-04-17 20:28
test_cmd_line.py
15.35
KB
-rw-r--r--
2024-04-17 20:28
test_cmd_line_script.py
16.95
KB
-rw-r--r--
2024-04-17 20:28
test_code.py
3.54
KB
-rw-r--r--
2024-04-17 20:28
test_code_module.py
2.42
KB
-rw-r--r--
2024-04-17 20:28
test_codeccallbacks.py
33.17
KB
-rw-r--r--
2024-04-17 20:28
test_codecencodings_cn.py
3.38
KB
-rw-r--r--
2024-04-17 20:28
test_codecencodings_hk.py
775
B
-rw-r--r--
2024-04-17 20:28
test_codecencodings_iso2022.py
1.43
KB
-rw-r--r--
2024-04-17 20:28
test_codecencodings_jp.py
4.86
KB
-rw-r--r--
2024-04-17 20:28
test_codecencodings_kr.py
3.03
KB
-rw-r--r--
2024-04-17 20:28
test_codecencodings_tw.py
755
B
-rw-r--r--
2024-04-17 20:28
test_codecmaps_cn.py
930
B
-rw-r--r--
2024-04-17 20:28
test_codecmaps_hk.py
505
B
-rw-r--r--
2024-04-17 20:28
test_codecmaps_jp.py
1.88
KB
-rw-r--r--
2024-04-17 20:28
test_codecmaps_kr.py
1.34
KB
-rw-r--r--
2024-04-17 20:28
test_codecmaps_tw.py
880
B
-rw-r--r--
2024-04-17 20:28
test_codecs.py
96.02
KB
-rw-r--r--
2024-04-17 20:28
test_codeop.py
7.45
KB
-rw-r--r--
2024-04-17 20:28
test_coding.py
2.51
KB
-rw-r--r--
2024-04-17 20:28
test_collections.py
54.31
KB
-rw-r--r--
2024-04-17 20:28
test_colorsys.py
2.77
KB
-rw-r--r--
2024-04-17 20:28
test_compare.py
1.63
KB
-rw-r--r--
2024-04-17 20:28
test_compile.py
18.62
KB
-rw-r--r--
2024-04-17 20:28
test_compileall.py
15.73
KB
-rw-r--r--
2024-04-17 20:28
test_complex.py
27.09
KB
-rw-r--r--
2024-04-17 20:28
test_concurrent_futures.py
22.57
KB
-rw-r--r--
2024-04-17 20:28
test_configparser.py
69.26
KB
-rw-r--r--
2024-04-17 20:28
test_contains.py
2.58
KB
-rw-r--r--
2024-04-17 20:28
test_contextlib.py
21.39
KB
-rw-r--r--
2024-04-17 20:28
test_copy.py
21.9
KB
-rw-r--r--
2024-04-17 20:28
test_copyreg.py
4.12
KB
-rw-r--r--
2024-04-17 20:28
test_cprofile.py
5.32
KB
-rw-r--r--
2024-04-17 20:28
test_crashers.py
1.18
KB
-rw-r--r--
2024-04-17 20:28
test_crypt.py
1.24
KB
-rw-r--r--
2024-04-17 20:28
test_csv.py
40.9
KB
-rw-r--r--
2024-04-17 20:28
test_ctypes.py
400
B
-rw-r--r--
2024-04-17 20:28
test_curses.py
10.7
KB
-rw-r--r--
2024-04-17 20:28
test_datetime.py
1.81
KB
-rw-r--r--
2024-04-17 20:28
test_dbm.py
5.52
KB
-rw-r--r--
2024-04-17 20:28
test_dbm_dumb.py
5.49
KB
-rw-r--r--
2024-04-17 20:28
test_dbm_gnu.py
2.85
KB
-rw-r--r--
2024-04-17 20:28
test_dbm_ndbm.py
1.13
KB
-rw-r--r--
2024-04-17 20:28
test_decimal.py
197.66
KB
-rw-r--r--
2024-04-17 20:28
test_decorators.py
9.6
KB
-rw-r--r--
2024-04-17 20:28
test_defaultdict.py
5.94
KB
-rw-r--r--
2024-04-17 20:28
test_deque.py
25
KB
-rw-r--r--
2024-04-17 20:28
test_descr.py
156.64
KB
-rw-r--r--
2024-04-17 20:28
test_descrtut.py
11.5
KB
-rw-r--r--
2024-04-17 20:28
test_devpoll.py
3.49
KB
-rw-r--r--
2024-04-17 20:28
test_dict.py
31.11
KB
-rw-r--r--
2024-04-17 20:28
test_dictcomps.py
3.69
KB
-rw-r--r--
2024-04-17 20:28
test_dictviews.py
8.51
KB
-rw-r--r--
2024-04-17 20:28
test_difflib.py
10.78
KB
-rw-r--r--
2024-04-17 20:28
test_difflib_expect.html
100.86
KB
-rw-r--r--
2017-09-19 12:02
test_dis.py
12.33
KB
-rw-r--r--
2024-04-17 20:28
test_distutils.py
375
B
-rw-r--r--
2024-04-17 20:28
test_doctest.py
81.78
KB
-rw-r--r--
2024-04-17 20:28
test_doctest.txt
300
B
-rw-r--r--
2017-09-19 12:02
test_doctest2.py
2.3
KB
-rw-r--r--
2024-04-17 20:28
test_doctest2.txt
392
B
-rw-r--r--
2017-09-19 12:02
test_doctest3.txt
82
B
-rw-r--r--
2017-09-19 12:02
test_doctest4.txt
244
B
-rw-r--r--
2017-09-19 12:02
test_docxmlrpc.py
8.33
KB
-rw-r--r--
2024-04-17 20:28
test_dummy_thread.py
6.99
KB
-rw-r--r--
2024-04-17 20:28
test_dummy_threading.py
1.76
KB
-rw-r--r--
2024-04-17 20:28
test_dynamic.py
4.37
KB
-rw-r--r--
2024-04-17 20:28
test_enumerate.py
7.96
KB
-rw-r--r--
2024-04-17 20:28
test_eof.py
855
B
-rw-r--r--
2024-04-17 20:28
test_epoll.py
7.48
KB
-rw-r--r--
2024-04-17 20:28
test_errno.py
1.14
KB
-rw-r--r--
2024-04-17 20:28
test_exception_variations.py
3.94
KB
-rw-r--r--
2024-04-17 20:28
test_exceptions.py
33.47
KB
-rw-r--r--
2024-04-17 20:28
test_extcall.py
8.64
KB
-rw-r--r--
2024-04-17 20:28
test_faulthandler.py
17.69
KB
-rw-r--r--
2024-04-17 20:28
test_fcntl.py
4.4
KB
-rw-r--r--
2024-04-17 20:28
test_file.py
11.1
KB
-rw-r--r--
2024-04-17 20:28
test_file_eintr.py
10.09
KB
-rw-r--r--
2024-04-17 20:28
test_filecmp.py
5.5
KB
-rw-r--r--
2024-04-17 20:28
test_fileinput.py
33.37
KB
-rw-r--r--
2024-04-17 20:28
test_fileio.py
15.11
KB
-rw-r--r--
2024-04-17 20:28
test_float.py
57.46
KB
-rw-r--r--
2024-04-17 20:28
test_flufl.py
836
B
-rw-r--r--
2024-04-17 20:28
test_fnmatch.py
2.84
KB
-rw-r--r--
2024-04-17 20:28
test_fork1.py
3.69
KB
-rw-r--r--
2024-04-17 20:28
test_format.py
16.04
KB
-rw-r--r--
2024-04-17 20:28
test_fractions.py
24.45
KB
-rw-r--r--
2024-04-17 20:28
test_frozen.py
2.81
KB
-rw-r--r--
2024-04-17 20:28
test_ftplib.py
35.85
KB
-rw-r--r--
2024-04-17 20:28
test_funcattrs.py
12.7
KB
-rw-r--r--
2024-04-17 20:28
test_functools.py
29.11
KB
-rw-r--r--
2024-04-17 20:28
test_future.py
3.77
KB
-rw-r--r--
2024-04-17 20:28
test_future3.py
490
B
-rw-r--r--
2024-04-17 20:28
test_future4.py
105
B
-rw-r--r--
2024-04-17 20:28
test_future5.py
510
B
-rw-r--r--
2024-04-17 20:28
test_gc.py
29
KB
-rw-r--r--
2024-04-17 20:28
test_gdb.py
35.86
KB
-rw-r--r--
2024-04-17 20:28
test_generators.py
48.5
KB
-rw-r--r--
2024-04-17 20:28
test_genericpath.py
13.3
KB
-rw-r--r--
2024-04-17 20:28
test_genexps.py
7.12
KB
-rw-r--r--
2024-04-17 20:28
test_getargs2.py
23.74
KB
-rw-r--r--
2024-04-17 20:28
test_getopt.py
6.8
KB
-rw-r--r--
2024-04-17 20:28
test_getpass.py
5.95
KB
-rw-r--r--
2024-04-17 20:28
test_gettext.py
22.11
KB
-rw-r--r--
2024-04-17 20:28
test_glob.py
6.69
KB
-rw-r--r--
2024-04-17 20:28
test_global.py
1.25
KB
-rw-r--r--
2024-04-17 20:28
test_grammar.py
31.96
KB
-rw-r--r--
2024-04-17 20:28
test_grp.py
3.18
KB
-rw-r--r--
2024-04-17 20:28
test_gzip.py
17.79
KB
-rw-r--r--
2024-04-17 20:28
test_hash.py
7.25
KB
-rw-r--r--
2024-04-17 20:28
test_hashlib.py
18.79
KB
-rw-r--r--
2024-04-17 20:28
test_heapq.py
14.14
KB
-rw-r--r--
2024-04-17 20:28
test_hmac.py
17.23
KB
-rw-r--r--
2024-04-17 20:28
test_html.py
579
B
-rw-r--r--
2024-04-17 20:28
test_htmlparser.py
30.74
KB
-rw-r--r--
2024-04-17 20:28
test_http_cookiejar.py
70
KB
-rw-r--r--
2024-04-17 20:28
test_http_cookies.py
9.98
KB
-rw-r--r--
2024-04-17 20:28
test_httplib.py
37.59
KB
-rw-r--r--
2024-04-17 20:28
test_httpservers.py
26.35
KB
-rw-r--r--
2024-04-17 20:28
test_idle.py
1.3
KB
-rw-r--r--
2024-04-17 20:28
test_imaplib.py
15.67
KB
-rw-r--r--
2024-04-17 20:28
test_imghdr.py
4.31
KB
-rw-r--r--
2024-04-17 20:28
test_imp.py
18.89
KB
-rw-r--r--
2024-04-17 20:28
test_import.py
38.55
KB
-rw-r--r--
2024-04-17 20:28
test_importhooks.py
7.83
KB
-rw-r--r--
2024-04-17 20:28
test_index.py
8.31
KB
-rw-r--r--
2024-04-17 20:28
test_inspect.py
87.19
KB
-rw-r--r--
2024-04-17 20:28
test_int.py
15.07
KB
-rw-r--r--
2024-04-17 20:28
test_int_literal.py
6.96
KB
-rw-r--r--
2024-04-17 20:28
test_io.py
119.84
KB
-rw-r--r--
2024-04-17 20:28
test_ioctl.py
3.25
KB
-rw-r--r--
2024-04-17 20:28
test_ipaddress.py
72.5
KB
-rw-r--r--
2024-04-17 20:28
test_isinstance.py
9.93
KB
-rw-r--r--
2024-04-17 20:28
test_iter.py
28.5
KB
-rw-r--r--
2024-04-17 20:28
test_iterlen.py
7.4
KB
-rw-r--r--
2024-04-17 20:28
test_itertools.py
82.48
KB
-rw-r--r--
2024-04-17 20:28
test_keywordonlyarg.py
6.7
KB
-rw-r--r--
2024-04-17 20:28
test_kqueue.py
6.61
KB
-rw-r--r--
2024-04-17 20:28
test_largefile.py
6.4
KB
-rw-r--r--
2024-04-17 20:28
test_lib2to3.py
637
B
-rw-r--r--
2024-04-17 20:28
test_linecache.py
4.04
KB
-rw-r--r--
2024-04-17 20:28
test_list.py
4.12
KB
-rw-r--r--
2024-04-17 20:28
test_listcomps.py
3.76
KB
-rw-r--r--
2024-04-17 20:28
test_locale.py
18.35
KB
-rw-r--r--
2024-04-17 20:28
test_logging.py
133.5
KB
-rw-r--r--
2024-04-17 20:28
test_long.py
49.64
KB
-rw-r--r--
2024-04-17 20:28
test_longexp.py
308
B
-rw-r--r--
2024-04-17 20:28
test_lzma.py
69.44
KB
-rw-r--r--
2024-04-17 20:28
test_macpath.py
4.82
KB
-rw-r--r--
2024-04-17 20:28
test_macurl2path.py
1.8
KB
-rw-r--r--
2024-04-17 20:28
test_mailbox.py
90.48
KB
-rw-r--r--
2024-04-17 20:28
test_mailcap.py
9
KB
-rw-r--r--
2024-04-17 20:28
test_marshal.py
11.9
KB
-rw-r--r--
2024-04-17 20:28
test_math.py
45.89
KB
-rw-r--r--
2024-04-17 20:28
test_memoryio.py
28.07
KB
-rw-r--r--
2024-04-17 20:28
test_memoryview.py
15.41
KB
-rw-r--r--
2024-04-17 20:28
test_metaclass.py
6.2
KB
-rw-r--r--
2024-04-17 20:28
test_mimetypes.py
4.18
KB
-rw-r--r--
2024-04-17 20:28
test_minidom.py
61.08
KB
-rw-r--r--
2024-04-17 20:28
test_mmap.py
25.91
KB
-rw-r--r--
2024-04-17 20:28
test_module.py
5.82
KB
-rw-r--r--
2024-04-17 20:28
test_modulefinder.py
8.24
KB
-rw-r--r--
2024-04-17 20:28
test_msilib.py
1.43
KB
-rw-r--r--
2024-04-17 20:28
test_multibytecodec.py
11.27
KB
-rw-r--r--
2024-04-17 20:28
test_multiprocessing.py
108.06
KB
-rw-r--r--
2024-04-17 20:28
test_namespace_pkgs.py
9.19
KB
-rw-r--r--
2024-04-17 20:28
test_netrc.py
4.5
KB
-rw-r--r--
2024-04-17 20:28
test_nis.py
1.19
KB
-rw-r--r--
2024-04-17 20:28
test_nntplib.py
54.81
KB
-rw-r--r--
2024-04-17 20:28
test_normalization.py
3.15
KB
-rw-r--r--
2024-04-17 20:28
test_ntpath.py
15.13
KB
-rw-r--r--
2024-04-17 20:28
test_numeric_tower.py
7.27
KB
-rw-r--r--
2024-04-17 20:28
test_opcodes.py
2.61
KB
-rw-r--r--
2024-04-17 20:28
test_openpty.py
779
B
-rw-r--r--
2024-04-17 20:28
test_operator.py
16.59
KB
-rw-r--r--
2024-04-17 20:28
test_optparse.py
60.74
KB
-rw-r--r--
2024-04-17 20:28
test_os.py
80.83
KB
-rw-r--r--
2024-04-17 20:28
test_ossaudiodev.py
7.05
KB
-rw-r--r--
2024-04-17 20:28
test_osx_env.py
1.31
KB
-rw-r--r--
2024-04-17 20:28
test_parser.py
25.5
KB
-rw-r--r--
2024-04-17 20:28
test_pdb.py
21.79
KB
-rw-r--r--
2024-04-17 20:28
test_peepholer.py
12.47
KB
-rw-r--r--
2024-04-17 20:28
test_pep247.py
2.03
KB
-rw-r--r--
2024-04-17 20:28
test_pep263.py
2.74
KB
-rw-r--r--
2024-04-17 20:28
test_pep277.py
7.7
KB
-rw-r--r--
2024-04-17 20:28
test_pep292.py
9.32
KB
-rw-r--r--
2024-04-17 20:28
test_pep3120.py
1.24
KB
-rw-r--r--
2024-04-17 20:28
test_pep3131.py
977
B
-rw-r--r--
2024-04-17 20:28
test_pep3151.py
7.34
KB
-rw-r--r--
2024-04-17 20:28
test_pep352.py
6.98
KB
-rw-r--r--
2024-04-17 20:28
test_pep380.py
28.42
KB
-rw-r--r--
2024-04-17 20:28
test_pickle.py
4.41
KB
-rw-r--r--
2024-04-17 20:28
test_pickletools.py
643
B
-rw-r--r--
2024-04-17 20:28
test_pipes.py
6.32
KB
-rw-r--r--
2024-04-17 20:28
test_pkg.py
9.39
KB
-rw-r--r--
2024-04-17 20:28
test_pkgimport.py
2.71
KB
-rw-r--r--
2024-04-17 20:28
test_pkgutil.py
12.25
KB
-rw-r--r--
2024-04-17 20:28
test_platform.py
12.12
KB
-rw-r--r--
2024-04-17 20:28
test_plistlib.py
8.42
KB
-rw-r--r--
2024-04-17 20:28
test_poll.py
6.26
KB
-rw-r--r--
2024-04-17 20:28
test_popen.py
1.92
KB
-rw-r--r--
2024-04-17 20:28
test_poplib.py
11.67
KB
-rw-r--r--
2024-04-17 20:28
test_posix.py
47.3
KB
-rw-r--r--
2024-04-17 20:28
test_posixpath.py
25.56
KB
-rw-r--r--
2024-04-17 20:28
test_pow.py
4.49
KB
-rw-r--r--
2024-04-17 20:28
test_pprint.py
27.85
KB
-rw-r--r--
2024-04-17 20:28
test_print.py
4.57
KB
-rw-r--r--
2024-04-17 20:28
test_profile.py
6.84
KB
-rw-r--r--
2024-04-17 20:28
test_property.py
7.63
KB
-rw-r--r--
2024-04-17 20:28
test_pstats.py
1.26
KB
-rw-r--r--
2024-04-17 20:28
test_pty.py
11.12
KB
-rw-r--r--
2024-04-17 20:28
test_pulldom.py
12.17
KB
-rw-r--r--
2024-04-17 20:28
test_pwd.py
4.13
KB
-rw-r--r--
2024-04-17 20:28
test_py_compile.py
2.25
KB
-rw-r--r--
2024-04-17 20:28
test_pyclbr.py
6.8
KB
-rw-r--r--
2024-04-17 20:28
test_pydoc.py
23.26
KB
-rw-r--r--
2024-04-17 20:28
test_pyexpat.py
24.45
KB
-rw-r--r--
2024-04-17 20:28
test_queue.py
12.84
KB
-rw-r--r--
2024-04-17 20:28
test_quopri.py
7.53
KB
-rw-r--r--
2024-04-17 20:28
test_raise.py
11.08
KB
-rw-r--r--
2024-04-17 20:28
test_random.py
21.67
KB
-rw-r--r--
2024-04-17 20:28
test_range.py
22.62
KB
-rw-r--r--
2024-04-17 20:28
test_re.py
56.47
KB
-rw-r--r--
2024-04-17 20:28
test_readline.py
1.66
KB
-rw-r--r--
2024-04-17 20:28
test_reprlib.py
14.21
KB
-rw-r--r--
2024-04-17 20:28
test_resource.py
5.17
KB
-rw-r--r--
2024-04-17 20:28
test_richcmp.py
10.83
KB
-rw-r--r--
2024-04-17 20:28
test_rlcompleter.py
2.74
KB
-rw-r--r--
2024-04-17 20:28
test_robotparser.py
6.9
KB
-rw-r--r--
2024-04-17 20:28
test_runpy.py
23.24
KB
-rw-r--r--
2024-04-17 20:28
test_sax.py
33.48
KB
-rw-r--r--
2024-04-17 20:28
test_sched.py
6.48
KB
-rw-r--r--
2024-04-17 20:28
test_scope.py
19.59
KB
-rw-r--r--
2024-04-17 20:28
test_select.py
2.7
KB
-rw-r--r--
2024-04-17 20:28
test_set.py
62.31
KB
-rw-r--r--
2024-04-17 20:28
test_setcomps.py
3.7
KB
-rw-r--r--
2024-04-17 20:28
test_shelve.py
5.7
KB
-rw-r--r--
2024-04-17 20:28
test_shlex.py
5.77
KB
-rw-r--r--
2024-04-17 20:28
test_shutil.py
66.87
KB
-rw-r--r--
2024-04-17 20:28
test_signal.py
31.59
KB
-rw-r--r--
2024-04-17 20:28
test_site.py
17.32
KB
-rw-r--r--
2024-04-17 20:28
test_slice.py
4.33
KB
-rw-r--r--
2024-04-17 20:28
test_smtpd.py
22.06
KB
-rw-r--r--
2024-04-17 20:28
test_smtplib.py
33.58
KB
-rw-r--r--
2024-04-17 20:28
test_smtpnet.py
1.85
KB
-rw-r--r--
2024-04-17 20:28
test_sndhdr.py
869
B
-rw-r--r--
2024-04-17 20:28
test_socket.py
181.8
KB
-rw-r--r--
2024-04-17 20:28
test_socketserver.py
10.72
KB
-rw-r--r--
2024-04-17 20:28
test_sort.py
8.95
KB
-rw-r--r--
2024-04-17 20:28
test_sqlite.py
893
B
-rw-r--r--
2024-04-17 20:28
test_ssl.py
94.63
KB
-rw-r--r--
2024-04-17 20:28
test_startfile.py
1.21
KB
-rw-r--r--
2024-04-17 20:28
test_stat.py
6.29
KB
-rw-r--r--
2024-04-17 20:28
test_strftime.py
7.55
KB
-rw-r--r--
2024-04-17 20:28
test_string.py
6.11
KB
-rw-r--r--
2024-04-17 20:28
test_stringprep.py
3.12
KB
-rw-r--r--
2024-04-17 20:28
test_strlit.py
8.03
KB
-rw-r--r--
2024-04-17 20:28
test_strptime.py
25.84
KB
-rw-r--r--
2024-04-17 20:28
test_strtod.py
20.11
KB
-rw-r--r--
2024-04-17 20:28
test_struct.py
24.67
KB
-rw-r--r--
2024-04-17 20:28
test_structmembers.py
4.77
KB
-rw-r--r--
2024-04-17 20:28
test_structseq.py
3.96
KB
-rw-r--r--
2024-04-17 20:28
test_subprocess.py
96.46
KB
-rw-r--r--
2024-04-17 20:28
test_sunau.py
3.55
KB
-rw-r--r--
2024-04-17 20:28
test_sundry.py
2.12
KB
-rw-r--r--
2024-04-17 20:28
test_super.py
3.5
KB
-rw-r--r--
2024-04-17 20:28
test_support.py
10.03
KB
-rw-r--r--
2024-04-17 20:28
test_symtable.py
5.82
KB
-rw-r--r--
2024-04-17 20:28
test_syntax.py
17.64
KB
-rw-r--r--
2024-04-17 20:28
test_sys.py
33.13
KB
-rw-r--r--
2024-04-17 20:28
test_sys_setprofile.py
11.09
KB
-rw-r--r--
2024-04-17 20:28
test_sys_settrace.py
24.37
KB
-rw-r--r--
2024-04-17 20:28
test_sysconfig.py
15.75
KB
-rw-r--r--
2024-04-17 20:28
test_syslog.py
1.08
KB
-rw-r--r--
2024-04-17 20:28
test_tarfile.py
70.04
KB
-rw-r--r--
2024-04-17 20:28
test_tcl.py
20.19
KB
-rw-r--r--
2024-04-17 20:28
test_telnetlib.py
14.84
KB
-rw-r--r--
2024-04-17 20:28
test_tempfile.py
39.85
KB
-rw-r--r--
2024-04-17 20:28
test_textwrap.py
29.83
KB
-rw-r--r--
2024-04-17 20:28
test_thread.py
8.1
KB
-rw-r--r--
2024-04-17 20:28
test_threaded_import.py
8.12
KB
-rw-r--r--
2024-04-17 20:28
test_threadedtempfile.py
2.14
KB
-rw-r--r--
2024-04-17 20:28
test_threading.py
33.87
KB
-rw-r--r--
2024-04-17 20:28
test_threading_local.py
6.19
KB
-rw-r--r--
2024-04-17 20:28
test_threadsignals.py
8.88
KB
-rw-r--r--
2024-04-17 20:28
test_time.py
25.86
KB
-rw-r--r--
2024-04-17 20:28
test_timeit.py
10.83
KB
-rw-r--r--
2024-04-17 20:28
test_timeout.py
11.11
KB
-rw-r--r--
2024-04-17 20:28
test_tk.py
747
B
-rw-r--r--
2024-04-17 20:28
test_tokenize.py
47.22
KB
-rw-r--r--
2024-04-17 20:28
test_tools.py
12.48
KB
-rw-r--r--
2024-04-17 20:28
test_trace.py
14.48
KB
-rw-r--r--
2024-04-17 20:28
test_traceback.py
14.02
KB
-rw-r--r--
2024-04-17 20:28
test_ttk_guionly.py
1.05
KB
-rw-r--r--
2024-04-17 20:28
test_ttk_textonly.py
405
B
-rw-r--r--
2024-04-17 20:28
test_tuple.py
6.78
KB
-rw-r--r--
2024-04-17 20:28
test_typechecks.py
2.64
KB
-rw-r--r--
2024-04-17 20:28
test_types.py
41.34
KB
-rw-r--r--
2024-04-17 20:28
test_ucn.py
9.41
KB
-rw-r--r--
2024-04-17 20:28
test_unary.py
1.71
KB
-rw-r--r--
2024-04-17 20:28
test_unicode.py
104.9
KB
-rw-r--r--
2024-04-17 20:28
test_unicode_file.py
5.8
KB
-rw-r--r--
2024-04-17 20:28
test_unicodedata.py
12.06
KB
-rw-r--r--
2024-04-17 20:28
test_unittest.py
286
B
-rw-r--r--
2024-04-17 20:28
test_univnewlines.py
3.83
KB
-rw-r--r--
2024-04-17 20:28
test_unpack.py
2.56
KB
-rw-r--r--
2024-04-17 20:28
test_unpack_ex.py
4.04
KB
-rw-r--r--
2024-04-17 20:28
test_urllib.py
55.37
KB
-rw-r--r--
2024-04-17 20:28
test_urllib2.py
60.39
KB
-rw-r--r--
2024-04-17 20:28
test_urllib2_localnet.py
21.78
KB
-rw-r--r--
2024-04-17 20:28
test_urllib2net.py
12.94
KB
-rw-r--r--
2024-04-17 20:28
test_urllib_response.py
871
B
-rw-r--r--
2024-04-17 20:28
test_urllibnet.py
8.89
KB
-rw-r--r--
2024-04-17 20:28
test_urlparse.py
50.39
KB
-rw-r--r--
2024-04-17 20:28
test_userdict.py
6.32
KB
-rw-r--r--
2024-04-17 20:28
test_userlist.py
1.85
KB
-rw-r--r--
2024-04-17 20:28
test_userstring.py
1.44
KB
-rw-r--r--
2024-04-17 20:28
test_uu.py
6.71
KB
-rw-r--r--
2024-04-17 20:28
test_uuid.py
21.6
KB
-rw-r--r--
2024-04-17 20:28
test_venv.py
7.32
KB
-rw-r--r--
2024-04-17 20:28
test_wait3.py
1.2
KB
-rw-r--r--
2024-04-17 20:28
test_wait4.py
1.11
KB
-rw-r--r--
2024-04-17 20:28
test_warnings.py
32.76
KB
-rw-r--r--
2024-04-17 20:28
test_wave.py
4.39
KB
-rw-r--r--
2024-04-17 20:28
test_weakref.py
49.49
KB
-rw-r--r--
2024-04-17 20:28
test_weakset.py
15.18
KB
-rw-r--r--
2024-04-17 20:28
test_webbrowser.py
5.65
KB
-rw-r--r--
2024-04-17 20:28
test_winreg.py
19.99
KB
-rw-r--r--
2024-04-17 20:28
test_winsound.py
8.86
KB
-rw-r--r--
2024-04-17 20:28
test_with.py
25.88
KB
-rw-r--r--
2024-04-17 20:28
test_wsgiref.py
21.84
KB
-rw-r--r--
2024-04-17 20:28
test_xdrlib.py
1.56
KB
-rw-r--r--
2024-04-17 20:28
test_xml_dom_minicompat.py
2.91
KB
-rw-r--r--
2024-04-17 20:28
test_xml_etree.py
88.5
KB
-rw-r--r--
2024-04-17 20:28
test_xml_etree_c.py
2.53
KB
-rw-r--r--
2024-04-17 20:28
test_xmlrpc.py
41.85
KB
-rw-r--r--
2024-04-17 20:28
test_xmlrpc_net.py
1.89
KB
-rw-r--r--
2024-04-17 20:28
test_zipfile.py
65.66
KB
-rw-r--r--
2024-04-17 20:28
test_zipfile64.py
4.17
KB
-rw-r--r--
2024-04-17 20:28
test_zipimport.py
17
KB
-rw-r--r--
2024-04-17 20:28
test_zipimport_support.py
10.28
KB
-rw-r--r--
2024-04-17 20:28
test_zlib.py
26.14
KB
-rw-r--r--
2024-04-17 20:28
testcodec.py
1.02
KB
-rw-r--r--
2024-04-17 20:28
testtar.tar
425
KB
-rw-r--r--
2017-09-19 12:02
tf_inherit_check.py
578
B
-rw-r--r--
2024-04-17 20:28
threaded_import_hangers.py
1.45
KB
-rw-r--r--
2024-04-17 20:28
time_hashlib.py
2.82
KB
-rw-r--r--
2024-04-17 20:28
tokenize_tests-latin1-coding-cookie-and-utf8-bom-sig.txt
444
B
-rw-r--r--
2017-09-19 12:02
tokenize_tests-no-coding-cookie-and-utf8-bom-sig-only.txt
303
B
-rw-r--r--
2017-09-19 12:02
tokenize_tests-utf8-coding-cookie-and-no-utf8-bom-sig.txt
421
B
-rw-r--r--
2017-09-19 12:02
tokenize_tests-utf8-coding-cookie-and-utf8-bom-sig.txt
327
B
-rw-r--r--
2017-09-19 12:02
tokenize_tests.txt
2.65
KB
-rw-r--r--
2017-09-19 12:02
warning_tests.py
240
B
-rw-r--r--
2024-04-17 20:28
win_console_handler.py
1.38
KB
-rw-r--r--
2024-04-17 20:28
xmltests.py
499
B
-rw-r--r--
2024-04-17 20:28
zip_cp437_header.zip
270
B
-rw-r--r--
2017-09-19 12:02
zipdir.zip
374
B
-rw-r--r--
2017-09-19 12:02
Save
Rename
import io import socket import datetime import textwrap import unittest import functools import contextlib from test import support from nntplib import NNTP, GroupInfo import nntplib try: import ssl except ImportError: ssl = None TIMEOUT = 30 # TODO: # - test the `file` arg to more commands # - test error conditions # - test auth and `usenetrc` class NetworkedNNTPTestsMixin: def test_welcome(self): welcome = self.server.getwelcome() self.assertEqual(str, type(welcome)) def test_help(self): resp, lines = self.server.help() self.assertTrue(resp.startswith("100 "), resp) for line in lines: self.assertEqual(str, type(line)) def test_list(self): resp, groups = self.server.list() if len(groups) > 0: self.assertEqual(GroupInfo, type(groups[0])) self.assertEqual(str, type(groups[0].group)) def test_list_active(self): resp, groups = self.server.list(self.GROUP_PAT) if len(groups) > 0: self.assertEqual(GroupInfo, type(groups[0])) self.assertEqual(str, type(groups[0].group)) def test_unknown_command(self): with self.assertRaises(nntplib.NNTPPermanentError) as cm: self.server._shortcmd("XYZZY") resp = cm.exception.response self.assertTrue(resp.startswith("500 "), resp) def test_newgroups(self): # gmane gets a constant influx of new groups. In order not to stress # the server too much, we choose a recent date in the past. dt = datetime.date.today() - datetime.timedelta(days=7) resp, groups = self.server.newgroups(dt) if len(groups) > 0: self.assertIsInstance(groups[0], GroupInfo) self.assertIsInstance(groups[0].group, str) def test_description(self): def _check_desc(desc): # Sanity checks self.assertIsInstance(desc, str) self.assertNotIn(self.GROUP_NAME, desc) desc = self.server.description(self.GROUP_NAME) _check_desc(desc) # Another sanity check self.assertIn("Python", desc) # With a pattern desc = self.server.description(self.GROUP_PAT) _check_desc(desc) # Shouldn't exist desc = self.server.description("zk.brrtt.baz") self.assertEqual(desc, '') def test_descriptions(self): resp, descs = self.server.descriptions(self.GROUP_PAT) # 215 for LIST NEWSGROUPS, 282 for XGTITLE self.assertTrue( resp.startswith("215 ") or resp.startswith("282 "), resp) self.assertIsInstance(descs, dict) desc = descs[self.GROUP_NAME] self.assertEqual(desc, self.server.description(self.GROUP_NAME)) def test_group(self): result = self.server.group(self.GROUP_NAME) self.assertEqual(5, len(result)) resp, count, first, last, group = result self.assertEqual(group, self.GROUP_NAME) self.assertIsInstance(count, int) self.assertIsInstance(first, int) self.assertIsInstance(last, int) self.assertLessEqual(first, last) self.assertTrue(resp.startswith("211 "), resp) def test_date(self): resp, date = self.server.date() self.assertIsInstance(date, datetime.datetime) # Sanity check self.assertGreaterEqual(date.year, 1995) self.assertLessEqual(date.year, 2030) def _check_art_dict(self, art_dict): # Some sanity checks for a field dictionary returned by OVER / XOVER self.assertIsInstance(art_dict, dict) # NNTP has 7 mandatory fields self.assertGreaterEqual(art_dict.keys(), {"subject", "from", "date", "message-id", "references", ":bytes", ":lines"} ) for v in art_dict.values(): self.assertIsInstance(v, (str, type(None))) def test_xover(self): resp, count, first, last, name = self.server.group(self.GROUP_NAME) resp, lines = self.server.xover(last - 5, last) if len(lines) == 0: self.skipTest("no articles retrieved") # The 'last' article is not necessarily part of the output (cancelled?) art_num, art_dict = lines[0] self.assertGreaterEqual(art_num, last - 5) self.assertLessEqual(art_num, last) self._check_art_dict(art_dict) def test_over(self): resp, count, first, last, name = self.server.group(self.GROUP_NAME) start = last - 10 # The "start-" article range form resp, lines = self.server.over((start, None)) art_num, art_dict = lines[0] self._check_art_dict(art_dict) # The "start-end" article range form resp, lines = self.server.over((start, last)) art_num, art_dict = lines[-1] # The 'last' article is not necessarily part of the output (cancelled?) self.assertGreaterEqual(art_num, start) self.assertLessEqual(art_num, last) self._check_art_dict(art_dict) # XXX The "message_id" form is unsupported by gmane # 503 Overview by message-ID unsupported def test_xhdr(self): resp, count, first, last, name = self.server.group(self.GROUP_NAME) resp, lines = self.server.xhdr('subject', last) for line in lines: self.assertEqual(str, type(line[1])) def check_article_resp(self, resp, article, art_num=None): self.assertIsInstance(article, nntplib.ArticleInfo) if art_num is not None: self.assertEqual(article.number, art_num) for line in article.lines: self.assertIsInstance(line, bytes) # XXX this could exceptionally happen... self.assertNotIn(article.lines[-1], (b".", b".\n", b".\r\n")) def test_article_head_body(self): resp, count, first, last, name = self.server.group(self.GROUP_NAME) # Try to find an available article for art_num in (last, first, last - 1): try: resp, head = self.server.head(art_num) except nntplib.NNTPTemporaryError as e: if not e.response.startswith("423 "): raise # "423 No such article" => choose another one continue break else: self.skipTest("could not find a suitable article number") self.assertTrue(resp.startswith("221 "), resp) self.check_article_resp(resp, head, art_num) resp, body = self.server.body(art_num) self.assertTrue(resp.startswith("222 "), resp) self.check_article_resp(resp, body, art_num) resp, article = self.server.article(art_num) self.assertTrue(resp.startswith("220 "), resp) self.check_article_resp(resp, article, art_num) # Tolerate running the tests from behind a NNTP virus checker blacklist = lambda line: line.startswith(b'X-Antivirus') filtered_head_lines = [line for line in head.lines if not blacklist(line)] filtered_lines = [line for line in article.lines if not blacklist(line)] self.assertEqual(filtered_lines, filtered_head_lines + [b''] + body.lines) def test_capabilities(self): # The server under test implements NNTP version 2 and has a # couple of well-known capabilities. Just sanity check that we # got them. def _check_caps(caps): caps_list = caps['LIST'] self.assertIsInstance(caps_list, (list, tuple)) self.assertIn('OVERVIEW.FMT', caps_list) self.assertGreaterEqual(self.server.nntp_version, 2) _check_caps(self.server.getcapabilities()) # This re-emits the command resp, caps = self.server.capabilities() _check_caps(caps) @unittest.skipUnless(ssl, 'requires SSL support') def test_starttls(self): file = self.server.file sock = self.server.sock try: self.server.starttls() except nntplib.NNTPPermanentError: self.skipTest("STARTTLS not supported by server.") else: # Check that the socket and internal pseudo-file really were # changed. self.assertNotEqual(file, self.server.file) self.assertNotEqual(sock, self.server.sock) # Check that the new socket really is an SSL one self.assertIsInstance(self.server.sock, ssl.SSLSocket) # Check that trying starttls when it's already active fails. self.assertRaises(ValueError, self.server.starttls) def test_zlogin(self): # This test must be the penultimate because further commands will be # refused. baduser = "notarealuser" badpw = "notarealpassword" # Check that bogus credentials cause failure self.assertRaises(nntplib.NNTPError, self.server.login, user=baduser, password=badpw, usenetrc=False) # FIXME: We should check that correct credentials succeed, but that # would require valid details for some server somewhere to be in the # test suite, I think. Gmane is anonymous, at least as used for the # other tests. def test_zzquit(self): # This test must be called last, hence the name cls = type(self) try: self.server.quit() finally: cls.server = None @classmethod def wrap_methods(cls): # Wrap all methods in a transient_internet() exception catcher # XXX put a generic version in test.support? def wrap_meth(meth): @functools.wraps(meth) def wrapped(self): with support.transient_internet(self.NNTP_HOST): meth(self) return wrapped for name in dir(cls): if not name.startswith('test_'): continue meth = getattr(cls, name) if not callable(meth): continue # Need to use a closure so that meth remains bound to its current # value setattr(cls, name, wrap_meth(meth)) def test_with_statement(self): def is_connected(): if not hasattr(server, 'file'): return False try: server.help() except (socket.error, EOFError): return False return True with self.NNTP_CLASS(self.NNTP_HOST, timeout=TIMEOUT, usenetrc=False) as server: self.assertTrue(is_connected()) self.assertTrue(server.help()) self.assertFalse(is_connected()) with self.NNTP_CLASS(self.NNTP_HOST, timeout=TIMEOUT, usenetrc=False) as server: server.quit() self.assertFalse(is_connected()) NetworkedNNTPTestsMixin.wrap_methods() class NetworkedNNTPTests(NetworkedNNTPTestsMixin, unittest.TestCase): # This server supports STARTTLS (gmane doesn't) NNTP_HOST = 'news.trigofacile.com' GROUP_NAME = 'fr.comp.lang.python' GROUP_PAT = 'fr.comp.lang.*' NNTP_CLASS = NNTP @classmethod def setUpClass(cls): support.requires("network") with support.transient_internet(cls.NNTP_HOST): cls.server = cls.NNTP_CLASS(cls.NNTP_HOST, timeout=TIMEOUT, usenetrc=False) @classmethod def tearDownClass(cls): if cls.server is not None: cls.server.quit() @unittest.skipUnless(ssl, 'requires SSL support') class NetworkedNNTP_SSLTests(NetworkedNNTPTests): # Technical limits for this public NNTP server (see http://www.aioe.org): # "Only two concurrent connections per IP address are allowed and # 400 connections per day are accepted from each IP address." NNTP_HOST = 'nntp.aioe.org' GROUP_NAME = 'comp.lang.python' GROUP_PAT = 'comp.lang.*' NNTP_CLASS = getattr(nntplib, 'NNTP_SSL', None) # Disabled as it produces too much data test_list = None # Disabled as the connection will already be encrypted. test_starttls = None # # Non-networked tests using a local server (or something mocking it). # class _NNTPServerIO(io.RawIOBase): """A raw IO object allowing NNTP commands to be received and processed by a handler. The handler can push responses which can then be read from the IO object.""" def __init__(self, handler): io.RawIOBase.__init__(self) # The channel from the client self.c2s = io.BytesIO() # The channel to the client self.s2c = io.BytesIO() self.handler = handler self.handler.start(self.c2s.readline, self.push_data) def readable(self): return True def writable(self): return True def push_data(self, data): """Push (buffer) some data to send to the client.""" pos = self.s2c.tell() self.s2c.seek(0, 2) self.s2c.write(data) self.s2c.seek(pos) def write(self, b): """The client sends us some data""" pos = self.c2s.tell() self.c2s.write(b) self.c2s.seek(pos) self.handler.process_pending() return len(b) def readinto(self, buf): """The client wants to read a response""" self.handler.process_pending() b = self.s2c.read(len(buf)) n = len(b) buf[:n] = b return n class MockedNNTPTestsMixin: # Override in derived classes handler_class = None def setUp(self): super().setUp() self.make_server() def tearDown(self): super().tearDown() del self.server def make_server(self, *args, **kwargs): self.handler = self.handler_class() self.sio = _NNTPServerIO(self.handler) # Using BufferedRWPair instead of BufferedRandom ensures the file # isn't seekable. file = io.BufferedRWPair(self.sio, self.sio) self.server = nntplib._NNTPBase(file, 'test.server', *args, **kwargs) return self.server class MockedNNTPWithReaderModeMixin(MockedNNTPTestsMixin): def setUp(self): super().setUp() self.make_server(readermode=True) class NNTPv1Handler: """A handler for RFC 977""" welcome = "200 NNTP mock server" def start(self, readline, push_data): self.in_body = False self.allow_posting = True self._readline = readline self._push_data = push_data self._logged_in = False self._user_sent = False # Our welcome self.handle_welcome() def _decode(self, data): return str(data, "utf-8", "surrogateescape") def process_pending(self): if self.in_body: while True: line = self._readline() if not line: return self.body.append(line) if line == b".\r\n": break try: meth, tokens = self.body_callback meth(*tokens, body=self.body) finally: self.body_callback = None self.body = None self.in_body = False while True: line = self._decode(self._readline()) if not line: return if not line.endswith("\r\n"): raise ValueError("line doesn't end with \\r\\n: {!r}".format(line)) line = line[:-2] cmd, *tokens = line.split() #meth = getattr(self.handler, "handle_" + cmd.upper(), None) meth = getattr(self, "handle_" + cmd.upper(), None) if meth is None: self.handle_unknown() else: try: meth(*tokens) except Exception as e: raise ValueError("command failed: {!r}".format(line)) from e else: if self.in_body: self.body_callback = meth, tokens self.body = [] def expect_body(self): """Flag that the client is expected to post a request body""" self.in_body = True def push_data(self, data): """Push some binary data""" self._push_data(data) def push_lit(self, lit): """Push a string literal""" lit = textwrap.dedent(lit) lit = "\r\n".join(lit.splitlines()) + "\r\n" lit = lit.encode('utf-8') self.push_data(lit) def handle_unknown(self): self.push_lit("500 What?") def handle_welcome(self): self.push_lit(self.welcome) def handle_QUIT(self): self.push_lit("205 Bye!") def handle_DATE(self): self.push_lit("111 20100914001155") def handle_GROUP(self, group): if group == "fr.comp.lang.python": self.push_lit("211 486 761 1265 fr.comp.lang.python") else: self.push_lit("411 No such group {}".format(group)) def handle_HELP(self): self.push_lit("""\ 100 Legal commands authinfo user Name|pass Password|generic <prog> <args> date help Report problems to <root@example.org> .""") def handle_STAT(self, message_spec=None): if message_spec is None: self.push_lit("412 No newsgroup selected") elif message_spec == "3000234": self.push_lit("223 3000234 <45223423@example.com>") elif message_spec == "<45223423@example.com>": self.push_lit("223 0 <45223423@example.com>") else: self.push_lit("430 No Such Article Found") def handle_NEXT(self): self.push_lit("223 3000237 <668929@example.org> retrieved") def handle_LAST(self): self.push_lit("223 3000234 <45223423@example.com> retrieved") def handle_LIST(self, action=None, param=None): if action is None: self.push_lit("""\ 215 Newsgroups in form "group high low flags". comp.lang.python 0000052340 0000002828 y comp.lang.python.announce 0000001153 0000000993 m free.it.comp.lang.python 0000000002 0000000002 y fr.comp.lang.python 0000001254 0000000760 y free.it.comp.lang.python.learner 0000000000 0000000001 y tw.bbs.comp.lang.python 0000000304 0000000304 y .""") elif action == "ACTIVE": if param == "*distutils*": self.push_lit("""\ 215 Newsgroups in form "group high low flags" gmane.comp.python.distutils.devel 0000014104 0000000001 m gmane.comp.python.distutils.cvs 0000000000 0000000001 m .""") else: self.push_lit("""\ 215 Newsgroups in form "group high low flags" .""") elif action == "OVERVIEW.FMT": self.push_lit("""\ 215 Order of fields in overview database. Subject: From: Date: Message-ID: References: Bytes: Lines: Xref:full .""") elif action == "NEWSGROUPS": assert param is not None if param == "comp.lang.python": self.push_lit("""\ 215 Descriptions in form "group description". comp.lang.python\tThe Python computer language. .""") elif param == "comp.lang.python*": self.push_lit("""\ 215 Descriptions in form "group description". comp.lang.python.announce\tAnnouncements about the Python language. (Moderated) comp.lang.python\tThe Python computer language. .""") else: self.push_lit("""\ 215 Descriptions in form "group description". .""") else: self.push_lit('501 Unknown LIST keyword') def handle_NEWNEWS(self, group, date_str, time_str): # We hard code different return messages depending on passed # argument and date syntax. if (group == "comp.lang.python" and date_str == "20100913" and time_str == "082004"): # Date was passed in RFC 3977 format (NNTP "v2") self.push_lit("""\ 230 list of newsarticles (NNTP v2) created after Mon Sep 13 08:20:04 2010 follows <a4929a40-6328-491a-aaaf-cb79ed7309a2@q2g2000vbk.googlegroups.com> <f30c0419-f549-4218-848f-d7d0131da931@y3g2000vbm.googlegroups.com> .""") elif (group == "comp.lang.python" and date_str == "100913" and time_str == "082004"): # Date was passed in RFC 977 format (NNTP "v1") self.push_lit("""\ 230 list of newsarticles (NNTP v1) created after Mon Sep 13 08:20:04 2010 follows <a4929a40-6328-491a-aaaf-cb79ed7309a2@q2g2000vbk.googlegroups.com> <f30c0419-f549-4218-848f-d7d0131da931@y3g2000vbm.googlegroups.com> .""") elif (group == 'comp.lang.python' and date_str in ('20100101', '100101') and time_str == '090000'): self.push_lit('too long line' * 3000 + '\n.') else: self.push_lit("""\ 230 An empty list of newsarticles follows .""") # (Note for experiments: many servers disable NEWNEWS. # As of this writing, sicinfo3.epfl.ch doesn't.) def handle_XOVER(self, message_spec): if message_spec == "57-59": self.push_lit( "224 Overview information for 57-58 follows\n" "57\tRe: ANN: New Plone book with strong Python (and Zope) themes throughout" "\tDoug Hellmann <doug.hellmann-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>" "\tSat, 19 Jun 2010 18:04:08 -0400" "\t<4FD05F05-F98B-44DC-8111-C6009C925F0C@gmail.com>" "\t<hvalf7$ort$1@dough.gmane.org>\t7103\t16" "\tXref: news.gmane.org gmane.comp.python.authors:57" "\n" "58\tLooking for a few good bloggers" "\tDoug Hellmann <doug.hellmann-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>" "\tThu, 22 Jul 2010 09:14:14 -0400" "\t<A29863FA-F388-40C3-AA25-0FD06B09B5BF@gmail.com>" "\t\t6683\t16" "\t" "\n" # An UTF-8 overview line from fr.comp.lang.python "59\tRe: Message d'erreur incompréhensible (par moi)" "\tEric Brunel <eric.brunel@pragmadev.nospam.com>" "\tWed, 15 Sep 2010 18:09:15 +0200" "\t<eric.brunel-2B8B56.18091515092010@news.wanadoo.fr>" "\t<4c90ec87$0$32425$ba4acef3@reader.news.orange.fr>\t1641\t27" "\tXref: saria.nerim.net fr.comp.lang.python:1265" "\n" ".\n") else: self.push_lit("""\ 224 No articles .""") def handle_POST(self, *, body=None): if body is None: if self.allow_posting: self.push_lit("340 Input article; end with <CR-LF>.<CR-LF>") self.expect_body() else: self.push_lit("440 Posting not permitted") else: assert self.allow_posting self.push_lit("240 Article received OK") self.posted_body = body def handle_IHAVE(self, message_id, *, body=None): if body is None: if (self.allow_posting and message_id == "<i.am.an.article.you.will.want@example.com>"): self.push_lit("335 Send it; end with <CR-LF>.<CR-LF>") self.expect_body() else: self.push_lit("435 Article not wanted") else: assert self.allow_posting self.push_lit("235 Article transferred OK") self.posted_body = body sample_head = """\ From: "Demo User" <nobody@example.net> Subject: I am just a test article Content-Type: text/plain; charset=UTF-8; format=flowed Message-ID: <i.am.an.article.you.will.want@example.com>""" sample_body = """\ This is just a test article. ..Here is a dot-starting line. -- Signed by Andr\xe9.""" sample_article = sample_head + "\n\n" + sample_body def handle_ARTICLE(self, message_spec=None): if message_spec is None: self.push_lit("220 3000237 <45223423@example.com>") elif message_spec == "<45223423@example.com>": self.push_lit("220 0 <45223423@example.com>") elif message_spec == "3000234": self.push_lit("220 3000234 <45223423@example.com>") else: self.push_lit("430 No Such Article Found") return self.push_lit(self.sample_article) self.push_lit(".") def handle_HEAD(self, message_spec=None): if message_spec is None: self.push_lit("221 3000237 <45223423@example.com>") elif message_spec == "<45223423@example.com>": self.push_lit("221 0 <45223423@example.com>") elif message_spec == "3000234": self.push_lit("221 3000234 <45223423@example.com>") else: self.push_lit("430 No Such Article Found") return self.push_lit(self.sample_head) self.push_lit(".") def handle_BODY(self, message_spec=None): if message_spec is None: self.push_lit("222 3000237 <45223423@example.com>") elif message_spec == "<45223423@example.com>": self.push_lit("222 0 <45223423@example.com>") elif message_spec == "3000234": self.push_lit("222 3000234 <45223423@example.com>") else: self.push_lit("430 No Such Article Found") return self.push_lit(self.sample_body) self.push_lit(".") def handle_AUTHINFO(self, cred_type, data): if self._logged_in: self.push_lit('502 Already Logged In') elif cred_type == 'user': if self._user_sent: self.push_lit('482 User Credential Already Sent') else: self.push_lit('381 Password Required') self._user_sent = True elif cred_type == 'pass': self.push_lit('281 Login Successful') self._logged_in = True else: raise Exception('Unknown cred type {}'.format(cred_type)) class NNTPv2Handler(NNTPv1Handler): """A handler for RFC 3977 (NNTP "v2")""" def handle_CAPABILITIES(self): fmt = """\ 101 Capability list: VERSION 2 3 IMPLEMENTATION INN 2.5.1{} HDR LIST ACTIVE ACTIVE.TIMES DISTRIB.PATS HEADERS NEWSGROUPS OVERVIEW.FMT OVER POST READER .""" if not self._logged_in: self.push_lit(fmt.format('\n AUTHINFO USER')) else: self.push_lit(fmt.format('')) def handle_MODE(self, _): raise Exception('MODE READER sent despite READER has been advertised') def handle_OVER(self, message_spec=None): return self.handle_XOVER(message_spec) class CapsAfterLoginNNTPv2Handler(NNTPv2Handler): """A handler that allows CAPABILITIES only after login""" def handle_CAPABILITIES(self): if not self._logged_in: self.push_lit('480 You must log in.') else: super().handle_CAPABILITIES() class ModeSwitchingNNTPv2Handler(NNTPv2Handler): """A server that starts in transit mode""" def __init__(self): self._switched = False def handle_CAPABILITIES(self): fmt = """\ 101 Capability list: VERSION 2 3 IMPLEMENTATION INN 2.5.1 HDR LIST ACTIVE ACTIVE.TIMES DISTRIB.PATS HEADERS NEWSGROUPS OVERVIEW.FMT OVER POST {}READER .""" if self._switched: self.push_lit(fmt.format('')) else: self.push_lit(fmt.format('MODE-')) def handle_MODE(self, what): assert not self._switched and what == 'reader' self._switched = True self.push_lit('200 Posting allowed') class NNTPv1v2TestsMixin: def setUp(self): super().setUp() def test_welcome(self): self.assertEqual(self.server.welcome, self.handler.welcome) def test_authinfo(self): if self.nntp_version == 2: self.assertIn('AUTHINFO', self.server._caps) self.server.login('testuser', 'testpw') # if AUTHINFO is gone from _caps we also know that getcapabilities() # has been called after login as it should self.assertNotIn('AUTHINFO', self.server._caps) def test_date(self): resp, date = self.server.date() self.assertEqual(resp, "111 20100914001155") self.assertEqual(date, datetime.datetime(2010, 9, 14, 0, 11, 55)) def test_quit(self): self.assertFalse(self.sio.closed) resp = self.server.quit() self.assertEqual(resp, "205 Bye!") self.assertTrue(self.sio.closed) def test_help(self): resp, help = self.server.help() self.assertEqual(resp, "100 Legal commands") self.assertEqual(help, [ ' authinfo user Name|pass Password|generic <prog> <args>', ' date', ' help', 'Report problems to <root@example.org>', ]) def test_list(self): resp, groups = self.server.list() self.assertEqual(len(groups), 6) g = groups[1] self.assertEqual(g, GroupInfo("comp.lang.python.announce", "0000001153", "0000000993", "m")) resp, groups = self.server.list("*distutils*") self.assertEqual(len(groups), 2) g = groups[0] self.assertEqual(g, GroupInfo("gmane.comp.python.distutils.devel", "0000014104", "0000000001", "m")) def test_stat(self): resp, art_num, message_id = self.server.stat(3000234) self.assertEqual(resp, "223 3000234 <45223423@example.com>") self.assertEqual(art_num, 3000234) self.assertEqual(message_id, "<45223423@example.com>") resp, art_num, message_id = self.server.stat("<45223423@example.com>") self.assertEqual(resp, "223 0 <45223423@example.com>") self.assertEqual(art_num, 0) self.assertEqual(message_id, "<45223423@example.com>") with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.stat("<non.existent.id>") self.assertEqual(cm.exception.response, "430 No Such Article Found") with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.stat() self.assertEqual(cm.exception.response, "412 No newsgroup selected") def test_next(self): resp, art_num, message_id = self.server.next() self.assertEqual(resp, "223 3000237 <668929@example.org> retrieved") self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<668929@example.org>") def test_last(self): resp, art_num, message_id = self.server.last() self.assertEqual(resp, "223 3000234 <45223423@example.com> retrieved") self.assertEqual(art_num, 3000234) self.assertEqual(message_id, "<45223423@example.com>") def test_description(self): desc = self.server.description("comp.lang.python") self.assertEqual(desc, "The Python computer language.") desc = self.server.description("comp.lang.pythonx") self.assertEqual(desc, "") def test_descriptions(self): resp, groups = self.server.descriptions("comp.lang.python") self.assertEqual(resp, '215 Descriptions in form "group description".') self.assertEqual(groups, { "comp.lang.python": "The Python computer language.", }) resp, groups = self.server.descriptions("comp.lang.python*") self.assertEqual(groups, { "comp.lang.python": "The Python computer language.", "comp.lang.python.announce": "Announcements about the Python language. (Moderated)", }) resp, groups = self.server.descriptions("comp.lang.pythonx") self.assertEqual(groups, {}) def test_group(self): resp, count, first, last, group = self.server.group("fr.comp.lang.python") self.assertTrue(resp.startswith("211 "), resp) self.assertEqual(first, 761) self.assertEqual(last, 1265) self.assertEqual(count, 486) self.assertEqual(group, "fr.comp.lang.python") with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.group("comp.lang.python.devel") exc = cm.exception self.assertTrue(exc.response.startswith("411 No such group"), exc.response) def test_newnews(self): # NEWNEWS comp.lang.python [20]100913 082004 dt = datetime.datetime(2010, 9, 13, 8, 20, 4) resp, ids = self.server.newnews("comp.lang.python", dt) expected = ( "230 list of newsarticles (NNTP v{0}) " "created after Mon Sep 13 08:20:04 2010 follows" ).format(self.nntp_version) self.assertEqual(resp, expected) self.assertEqual(ids, [ "<a4929a40-6328-491a-aaaf-cb79ed7309a2@q2g2000vbk.googlegroups.com>", "<f30c0419-f549-4218-848f-d7d0131da931@y3g2000vbm.googlegroups.com>", ]) # NEWNEWS fr.comp.lang.python [20]100913 082004 dt = datetime.datetime(2010, 9, 13, 8, 20, 4) resp, ids = self.server.newnews("fr.comp.lang.python", dt) self.assertEqual(resp, "230 An empty list of newsarticles follows") self.assertEqual(ids, []) def _check_article_body(self, lines): self.assertEqual(len(lines), 4) self.assertEqual(lines[-1].decode('utf-8'), "-- Signed by André.") self.assertEqual(lines[-2], b"") self.assertEqual(lines[-3], b".Here is a dot-starting line.") self.assertEqual(lines[-4], b"This is just a test article.") def _check_article_head(self, lines): self.assertEqual(len(lines), 4) self.assertEqual(lines[0], b'From: "Demo User" <nobody@example.net>') self.assertEqual(lines[3], b"Message-ID: <i.am.an.article.you.will.want@example.com>") def _check_article_data(self, lines): self.assertEqual(len(lines), 9) self._check_article_head(lines[:4]) self._check_article_body(lines[-4:]) self.assertEqual(lines[4], b"") def test_article(self): # ARTICLE resp, info = self.server.article() self.assertEqual(resp, "220 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_data(lines) # ARTICLE num resp, info = self.server.article(3000234) self.assertEqual(resp, "220 3000234 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000234) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_data(lines) # ARTICLE id resp, info = self.server.article("<45223423@example.com>") self.assertEqual(resp, "220 0 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 0) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_data(lines) # Non-existent id with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.article("<non-existent@example.com>") self.assertEqual(cm.exception.response, "430 No Such Article Found") def test_article_file(self): # With a "file" argument f = io.BytesIO() resp, info = self.server.article(file=f) self.assertEqual(resp, "220 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self.assertEqual(lines, []) data = f.getvalue() self.assertTrue(data.startswith( b'From: "Demo User" <nobody@example.net>\r\n' b'Subject: I am just a test article\r\n' ), ascii(data)) self.assertTrue(data.endswith( b'This is just a test article.\r\n' b'.Here is a dot-starting line.\r\n' b'\r\n' b'-- Signed by Andr\xc3\xa9.\r\n' ), ascii(data)) def test_head(self): # HEAD resp, info = self.server.head() self.assertEqual(resp, "221 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_head(lines) # HEAD num resp, info = self.server.head(3000234) self.assertEqual(resp, "221 3000234 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000234) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_head(lines) # HEAD id resp, info = self.server.head("<45223423@example.com>") self.assertEqual(resp, "221 0 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 0) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_head(lines) # Non-existent id with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.head("<non-existent@example.com>") self.assertEqual(cm.exception.response, "430 No Such Article Found") def test_head_file(self): f = io.BytesIO() resp, info = self.server.head(file=f) self.assertEqual(resp, "221 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self.assertEqual(lines, []) data = f.getvalue() self.assertTrue(data.startswith( b'From: "Demo User" <nobody@example.net>\r\n' b'Subject: I am just a test article\r\n' ), ascii(data)) self.assertFalse(data.endswith( b'This is just a test article.\r\n' b'.Here is a dot-starting line.\r\n' b'\r\n' b'-- Signed by Andr\xc3\xa9.\r\n' ), ascii(data)) def test_body(self): # BODY resp, info = self.server.body() self.assertEqual(resp, "222 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_body(lines) # BODY num resp, info = self.server.body(3000234) self.assertEqual(resp, "222 3000234 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000234) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_body(lines) # BODY id resp, info = self.server.body("<45223423@example.com>") self.assertEqual(resp, "222 0 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 0) self.assertEqual(message_id, "<45223423@example.com>") self._check_article_body(lines) # Non-existent id with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.body("<non-existent@example.com>") self.assertEqual(cm.exception.response, "430 No Such Article Found") def test_body_file(self): f = io.BytesIO() resp, info = self.server.body(file=f) self.assertEqual(resp, "222 3000237 <45223423@example.com>") art_num, message_id, lines = info self.assertEqual(art_num, 3000237) self.assertEqual(message_id, "<45223423@example.com>") self.assertEqual(lines, []) data = f.getvalue() self.assertFalse(data.startswith( b'From: "Demo User" <nobody@example.net>\r\n' b'Subject: I am just a test article\r\n' ), ascii(data)) self.assertTrue(data.endswith( b'This is just a test article.\r\n' b'.Here is a dot-starting line.\r\n' b'\r\n' b'-- Signed by Andr\xc3\xa9.\r\n' ), ascii(data)) def check_over_xover_resp(self, resp, overviews): self.assertTrue(resp.startswith("224 "), resp) self.assertEqual(len(overviews), 3) art_num, over = overviews[0] self.assertEqual(art_num, 57) self.assertEqual(over, { "from": "Doug Hellmann <doug.hellmann-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>", "subject": "Re: ANN: New Plone book with strong Python (and Zope) themes throughout", "date": "Sat, 19 Jun 2010 18:04:08 -0400", "message-id": "<4FD05F05-F98B-44DC-8111-C6009C925F0C@gmail.com>", "references": "<hvalf7$ort$1@dough.gmane.org>", ":bytes": "7103", ":lines": "16", "xref": "news.gmane.org gmane.comp.python.authors:57" }) art_num, over = overviews[1] self.assertEqual(over["xref"], None) art_num, over = overviews[2] self.assertEqual(over["subject"], "Re: Message d'erreur incompréhensible (par moi)") def test_xover(self): resp, overviews = self.server.xover(57, 59) self.check_over_xover_resp(resp, overviews) def test_over(self): # In NNTP "v1", this will fallback on XOVER resp, overviews = self.server.over((57, 59)) self.check_over_xover_resp(resp, overviews) sample_post = ( b'From: "Demo User" <nobody@example.net>\r\n' b'Subject: I am just a test article\r\n' b'Content-Type: text/plain; charset=UTF-8; format=flowed\r\n' b'Message-ID: <i.am.an.article.you.will.want@example.com>\r\n' b'\r\n' b'This is just a test article.\r\n' b'.Here is a dot-starting line.\r\n' b'\r\n' b'-- Signed by Andr\xc3\xa9.\r\n' ) def _check_posted_body(self): # Check the raw body as received by the server lines = self.handler.posted_body # One additional line for the "." terminator self.assertEqual(len(lines), 10) self.assertEqual(lines[-1], b'.\r\n') self.assertEqual(lines[-2], b'-- Signed by Andr\xc3\xa9.\r\n') self.assertEqual(lines[-3], b'\r\n') self.assertEqual(lines[-4], b'..Here is a dot-starting line.\r\n') self.assertEqual(lines[0], b'From: "Demo User" <nobody@example.net>\r\n') def _check_post_ihave_sub(self, func, *args, file_factory): # First the prepared post with CRLF endings post = self.sample_post func_args = args + (file_factory(post),) self.handler.posted_body = None resp = func(*func_args) self._check_posted_body() # Then the same post with "normal" line endings - they should be # converted by NNTP.post and NNTP.ihave. post = self.sample_post.replace(b"\r\n", b"\n") func_args = args + (file_factory(post),) self.handler.posted_body = None resp = func(*func_args) self._check_posted_body() return resp def check_post_ihave(self, func, success_resp, *args): # With a bytes object resp = self._check_post_ihave_sub(func, *args, file_factory=bytes) self.assertEqual(resp, success_resp) # With a bytearray object resp = self._check_post_ihave_sub(func, *args, file_factory=bytearray) self.assertEqual(resp, success_resp) # With a file object resp = self._check_post_ihave_sub(func, *args, file_factory=io.BytesIO) self.assertEqual(resp, success_resp) # With an iterable of terminated lines def iterlines(b): return iter(b.splitlines(keepends=True)) resp = self._check_post_ihave_sub(func, *args, file_factory=iterlines) self.assertEqual(resp, success_resp) # With an iterable of non-terminated lines def iterlines(b): return iter(b.splitlines(keepends=False)) resp = self._check_post_ihave_sub(func, *args, file_factory=iterlines) self.assertEqual(resp, success_resp) def test_post(self): self.check_post_ihave(self.server.post, "240 Article received OK") self.handler.allow_posting = False with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.post(self.sample_post) self.assertEqual(cm.exception.response, "440 Posting not permitted") def test_ihave(self): self.check_post_ihave(self.server.ihave, "235 Article transferred OK", "<i.am.an.article.you.will.want@example.com>") with self.assertRaises(nntplib.NNTPTemporaryError) as cm: self.server.ihave("<another.message.id>", self.sample_post) self.assertEqual(cm.exception.response, "435 Article not wanted") def test_too_long_lines(self): dt = datetime.datetime(2010, 1, 1, 9, 0, 0) self.assertRaises(nntplib.NNTPDataError, self.server.newnews, "comp.lang.python", dt) class NNTPv1Tests(NNTPv1v2TestsMixin, MockedNNTPTestsMixin, unittest.TestCase): """Tests an NNTP v1 server (no capabilities).""" nntp_version = 1 handler_class = NNTPv1Handler def test_caps(self): caps = self.server.getcapabilities() self.assertEqual(caps, {}) self.assertEqual(self.server.nntp_version, 1) self.assertEqual(self.server.nntp_implementation, None) class NNTPv2Tests(NNTPv1v2TestsMixin, MockedNNTPTestsMixin, unittest.TestCase): """Tests an NNTP v2 server (with capabilities).""" nntp_version = 2 handler_class = NNTPv2Handler def test_caps(self): caps = self.server.getcapabilities() self.assertEqual(caps, { 'VERSION': ['2', '3'], 'IMPLEMENTATION': ['INN', '2.5.1'], 'AUTHINFO': ['USER'], 'HDR': [], 'LIST': ['ACTIVE', 'ACTIVE.TIMES', 'DISTRIB.PATS', 'HEADERS', 'NEWSGROUPS', 'OVERVIEW.FMT'], 'OVER': [], 'POST': [], 'READER': [], }) self.assertEqual(self.server.nntp_version, 3) self.assertEqual(self.server.nntp_implementation, 'INN 2.5.1') class CapsAfterLoginNNTPv2Tests(MockedNNTPTestsMixin, unittest.TestCase): """Tests a probably NNTP v2 server with capabilities only after login.""" nntp_version = 2 handler_class = CapsAfterLoginNNTPv2Handler def test_caps_only_after_login(self): self.assertEqual(self.server._caps, {}) self.server.login('testuser', 'testpw') self.assertIn('VERSION', self.server._caps) class SendReaderNNTPv2Tests(MockedNNTPWithReaderModeMixin, unittest.TestCase): """Same tests as for v2 but we tell NTTP to send MODE READER to a server that isn't in READER mode by default.""" nntp_version = 2 handler_class = ModeSwitchingNNTPv2Handler def test_we_are_in_reader_mode_after_connect(self): self.assertIn('READER', self.server._caps) class MiscTests(unittest.TestCase): def test_decode_header(self): def gives(a, b): self.assertEqual(nntplib.decode_header(a), b) gives("" , "") gives("a plain header", "a plain header") gives(" with extra spaces ", " with extra spaces ") gives("=?ISO-8859-15?Q?D=E9buter_en_Python?=", "Débuter en Python") gives("=?utf-8?q?Re=3A_=5Bsqlite=5D_probl=C3=A8me_avec_ORDER_BY_sur_des_cha?=" " =?utf-8?q?=C3=AEnes_de_caract=C3=A8res_accentu=C3=A9es?=", "Re: [sqlite] problème avec ORDER BY sur des chaînes de caractères accentuées") gives("Re: =?UTF-8?B?cHJvYmzDqG1lIGRlIG1hdHJpY2U=?=", "Re: problème de matrice") # A natively utf-8 header (found in the real world!) gives("Re: Message d'erreur incompréhensible (par moi)", "Re: Message d'erreur incompréhensible (par moi)") def test_parse_overview_fmt(self): # The minimal (default) response lines = ["Subject:", "From:", "Date:", "Message-ID:", "References:", ":bytes", ":lines"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines"]) # The minimal response using alternative names lines = ["Subject:", "From:", "Date:", "Message-ID:", "References:", "Bytes:", "Lines:"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines"]) # Variations in casing lines = ["subject:", "FROM:", "DaTe:", "message-ID:", "References:", "BYTES:", "Lines:"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines"]) # First example from RFC 3977 lines = ["Subject:", "From:", "Date:", "Message-ID:", "References:", ":bytes", ":lines", "Xref:full", "Distribution:full"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines", "xref", "distribution"]) # Second example from RFC 3977 lines = ["Subject:", "From:", "Date:", "Message-ID:", "References:", "Bytes:", "Lines:", "Xref:FULL", "Distribution:FULL"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines", "xref", "distribution"]) # A classic response from INN lines = ["Subject:", "From:", "Date:", "Message-ID:", "References:", "Bytes:", "Lines:", "Xref:full"] self.assertEqual(nntplib._parse_overview_fmt(lines), ["subject", "from", "date", "message-id", "references", ":bytes", ":lines", "xref"]) def test_parse_overview(self): fmt = nntplib._DEFAULT_OVERVIEW_FMT + ["xref"] # First example from RFC 3977 lines = [ '3000234\tI am just a test article\t"Demo User" ' '<nobody@example.com>\t6 Oct 1998 04:38:40 -0500\t' '<45223423@example.com>\t<45454@example.net>\t1234\t' '17\tXref: news.example.com misc.test:3000363', ] overview = nntplib._parse_overview(lines, fmt) (art_num, fields), = overview self.assertEqual(art_num, 3000234) self.assertEqual(fields, { 'subject': 'I am just a test article', 'from': '"Demo User" <nobody@example.com>', 'date': '6 Oct 1998 04:38:40 -0500', 'message-id': '<45223423@example.com>', 'references': '<45454@example.net>', ':bytes': '1234', ':lines': '17', 'xref': 'news.example.com misc.test:3000363', }) # Second example; here the "Xref" field is totally absent (including # the header name) and comes out as None lines = [ '3000234\tI am just a test article\t"Demo User" ' '<nobody@example.com>\t6 Oct 1998 04:38:40 -0500\t' '<45223423@example.com>\t<45454@example.net>\t1234\t' '17\t\t', ] overview = nntplib._parse_overview(lines, fmt) (art_num, fields), = overview self.assertEqual(fields['xref'], None) # Third example; the "Xref" is an empty string, while "references" # is a single space. lines = [ '3000234\tI am just a test article\t"Demo User" ' '<nobody@example.com>\t6 Oct 1998 04:38:40 -0500\t' '<45223423@example.com>\t \t1234\t' '17\tXref: \t', ] overview = nntplib._parse_overview(lines, fmt) (art_num, fields), = overview self.assertEqual(fields['references'], ' ') self.assertEqual(fields['xref'], '') def test_parse_datetime(self): def gives(a, b, *c): self.assertEqual(nntplib._parse_datetime(a, b), datetime.datetime(*c)) # Output of DATE command gives("19990623135624", None, 1999, 6, 23, 13, 56, 24) # Variations gives("19990623", "135624", 1999, 6, 23, 13, 56, 24) gives("990623", "135624", 1999, 6, 23, 13, 56, 24) gives("090623", "135624", 2009, 6, 23, 13, 56, 24) def test_unparse_datetime(self): # Test non-legacy mode # 1) with a datetime def gives(y, M, d, h, m, s, date_str, time_str): dt = datetime.datetime(y, M, d, h, m, s) self.assertEqual(nntplib._unparse_datetime(dt), (date_str, time_str)) self.assertEqual(nntplib._unparse_datetime(dt, False), (date_str, time_str)) gives(1999, 6, 23, 13, 56, 24, "19990623", "135624") gives(2000, 6, 23, 13, 56, 24, "20000623", "135624") gives(2010, 6, 5, 1, 2, 3, "20100605", "010203") # 2) with a date def gives(y, M, d, date_str, time_str): dt = datetime.date(y, M, d) self.assertEqual(nntplib._unparse_datetime(dt), (date_str, time_str)) self.assertEqual(nntplib._unparse_datetime(dt, False), (date_str, time_str)) gives(1999, 6, 23, "19990623", "000000") gives(2000, 6, 23, "20000623", "000000") gives(2010, 6, 5, "20100605", "000000") def test_unparse_datetime_legacy(self): # Test legacy mode (RFC 977) # 1) with a datetime def gives(y, M, d, h, m, s, date_str, time_str): dt = datetime.datetime(y, M, d, h, m, s) self.assertEqual(nntplib._unparse_datetime(dt, True), (date_str, time_str)) gives(1999, 6, 23, 13, 56, 24, "990623", "135624") gives(2000, 6, 23, 13, 56, 24, "000623", "135624") gives(2010, 6, 5, 1, 2, 3, "100605", "010203") # 2) with a date def gives(y, M, d, date_str, time_str): dt = datetime.date(y, M, d) self.assertEqual(nntplib._unparse_datetime(dt, True), (date_str, time_str)) gives(1999, 6, 23, "990623", "000000") gives(2000, 6, 23, "000623", "000000") gives(2010, 6, 5, "100605", "000000") @unittest.skipUnless(ssl, 'requires SSL support') def test_ssl_support(self): self.assertTrue(hasattr(nntplib, 'NNTP_SSL')) def test_main(): tests = [MiscTests, NNTPv1Tests, NNTPv2Tests, CapsAfterLoginNNTPv2Tests, SendReaderNNTPv2Tests, NetworkedNNTPTests, NetworkedNNTP_SSLTests] support.run_unittest(*tests) if __name__ == "__main__": test_main()