xref: /aosp_15_r20/external/curl/tests/http/test_31_vsftpds.py (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*6236dae4SAndroid Build Coastguard Worker# -*- coding: utf-8 -*-
3*6236dae4SAndroid Build Coastguard Worker#***************************************************************************
4*6236dae4SAndroid Build Coastguard Worker#                                  _   _ ____  _
5*6236dae4SAndroid Build Coastguard Worker#  Project                     ___| | | |  _ \| |
6*6236dae4SAndroid Build Coastguard Worker#                             / __| | | | |_) | |
7*6236dae4SAndroid Build Coastguard Worker#                            | (__| |_| |  _ <| |___
8*6236dae4SAndroid Build Coastguard Worker#                             \___|\___/|_| \_\_____|
9*6236dae4SAndroid Build Coastguard Worker#
10*6236dae4SAndroid Build Coastguard Worker# Copyright (C) Daniel Stenberg, <[email protected]>, et al.
11*6236dae4SAndroid Build Coastguard Worker#
12*6236dae4SAndroid Build Coastguard Worker# This software is licensed as described in the file COPYING, which
13*6236dae4SAndroid Build Coastguard Worker# you should have received as part of this distribution. The terms
14*6236dae4SAndroid Build Coastguard Worker# are also available at https://curl.se/docs/copyright.html.
15*6236dae4SAndroid Build Coastguard Worker#
16*6236dae4SAndroid Build Coastguard Worker# You may opt to use, copy, modify, merge, publish, distribute and/or sell
17*6236dae4SAndroid Build Coastguard Worker# copies of the Software, and permit persons to whom the Software is
18*6236dae4SAndroid Build Coastguard Worker# furnished to do so, under the terms of the COPYING file.
19*6236dae4SAndroid Build Coastguard Worker#
20*6236dae4SAndroid Build Coastguard Worker# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
21*6236dae4SAndroid Build Coastguard Worker# KIND, either express or implied.
22*6236dae4SAndroid Build Coastguard Worker#
23*6236dae4SAndroid Build Coastguard Worker# SPDX-License-Identifier: curl
24*6236dae4SAndroid Build Coastguard Worker#
25*6236dae4SAndroid Build Coastguard Worker###########################################################################
26*6236dae4SAndroid Build Coastguard Worker#
27*6236dae4SAndroid Build Coastguard Workerimport difflib
28*6236dae4SAndroid Build Coastguard Workerimport filecmp
29*6236dae4SAndroid Build Coastguard Workerimport logging
30*6236dae4SAndroid Build Coastguard Workerimport os
31*6236dae4SAndroid Build Coastguard Workerimport shutil
32*6236dae4SAndroid Build Coastguard Workerimport pytest
33*6236dae4SAndroid Build Coastguard Worker
34*6236dae4SAndroid Build Coastguard Workerfrom testenv import Env, CurlClient, VsFTPD
35*6236dae4SAndroid Build Coastguard Worker
36*6236dae4SAndroid Build Coastguard Worker
37*6236dae4SAndroid Build Coastguard Workerlog = logging.getLogger(__name__)
38*6236dae4SAndroid Build Coastguard Worker
39*6236dae4SAndroid Build Coastguard Worker
40*6236dae4SAndroid Build Coastguard Worker@pytest.mark.skipif(condition=not Env.has_vsftpd(), reason="missing vsftpd")
41*6236dae4SAndroid Build Coastguard Workerclass TestVsFTPD:
42*6236dae4SAndroid Build Coastguard Worker
43*6236dae4SAndroid Build Coastguard Worker    SUPPORTS_SSL = True
44*6236dae4SAndroid Build Coastguard Worker
45*6236dae4SAndroid Build Coastguard Worker    @pytest.fixture(autouse=True, scope='class')
46*6236dae4SAndroid Build Coastguard Worker    def vsftpds(self, env):
47*6236dae4SAndroid Build Coastguard Worker        if not TestVsFTPD.SUPPORTS_SSL:
48*6236dae4SAndroid Build Coastguard Worker            pytest.skip('vsftpd does not seem to support SSL')
49*6236dae4SAndroid Build Coastguard Worker        vsftpds = VsFTPD(env=env, with_ssl=True)
50*6236dae4SAndroid Build Coastguard Worker        if not vsftpds.start():
51*6236dae4SAndroid Build Coastguard Worker            vsftpds.stop()
52*6236dae4SAndroid Build Coastguard Worker            TestVsFTPD.SUPPORTS_SSL = False
53*6236dae4SAndroid Build Coastguard Worker            pytest.skip('vsftpd does not seem to support SSL')
54*6236dae4SAndroid Build Coastguard Worker        yield vsftpds
55*6236dae4SAndroid Build Coastguard Worker        vsftpds.stop()
56*6236dae4SAndroid Build Coastguard Worker
57*6236dae4SAndroid Build Coastguard Worker    def _make_docs_file(self, docs_dir: str, fname: str, fsize: int):
58*6236dae4SAndroid Build Coastguard Worker        fpath = os.path.join(docs_dir, fname)
59*6236dae4SAndroid Build Coastguard Worker        data1k = 1024*'x'
60*6236dae4SAndroid Build Coastguard Worker        flen = 0
61*6236dae4SAndroid Build Coastguard Worker        with open(fpath, 'w') as fd:
62*6236dae4SAndroid Build Coastguard Worker            while flen < fsize:
63*6236dae4SAndroid Build Coastguard Worker                fd.write(data1k)
64*6236dae4SAndroid Build Coastguard Worker                flen += len(data1k)
65*6236dae4SAndroid Build Coastguard Worker        return flen
66*6236dae4SAndroid Build Coastguard Worker
67*6236dae4SAndroid Build Coastguard Worker    @pytest.fixture(autouse=True, scope='class')
68*6236dae4SAndroid Build Coastguard Worker    def _class_scope(self, env, vsftpds):
69*6236dae4SAndroid Build Coastguard Worker        if os.path.exists(vsftpds.docs_dir):
70*6236dae4SAndroid Build Coastguard Worker            shutil.rmtree(vsftpds.docs_dir)
71*6236dae4SAndroid Build Coastguard Worker        if not os.path.exists(vsftpds.docs_dir):
72*6236dae4SAndroid Build Coastguard Worker            os.makedirs(vsftpds.docs_dir)
73*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpds.docs_dir, fname='data-1k', fsize=1024)
74*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpds.docs_dir, fname='data-10k', fsize=10*1024)
75*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpds.docs_dir, fname='data-1m', fsize=1024*1024)
76*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpds.docs_dir, fname='data-10m', fsize=10*1024*1024)
77*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-1k", fsize=1024)
78*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-100k", fsize=100*1024)
79*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-1m", fsize=1024*1024)
80*6236dae4SAndroid Build Coastguard Worker
81*6236dae4SAndroid Build Coastguard Worker    def test_31_01_list_dir(self, env: Env, vsftpds: VsFTPD, repeat):
82*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
83*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/'
84*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True)
85*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=1, http_status=226)
86*6236dae4SAndroid Build Coastguard Worker        lines = open(os.path.join(curl.run_dir, 'download_#1.data')).readlines()
87*6236dae4SAndroid Build Coastguard Worker        assert len(lines) == 4, f'list: {lines}'
88*6236dae4SAndroid Build Coastguard Worker
89*6236dae4SAndroid Build Coastguard Worker    # download 1 file, no SSL
90*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
91*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
92*6236dae4SAndroid Build Coastguard Worker    ])
93*6236dae4SAndroid Build Coastguard Worker    def test_31_02_download_1(self, env: Env, vsftpds: VsFTPD, docname, repeat):
94*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
95*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpds.docs_dir, f'{docname}')
96*6236dae4SAndroid Build Coastguard Worker        count = 1
97*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}?[0-{count-1}]'
98*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True)
99*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
100*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
101*6236dae4SAndroid Build Coastguard Worker
102*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
103*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
104*6236dae4SAndroid Build Coastguard Worker    ])
105*6236dae4SAndroid Build Coastguard Worker    def test_31_03_download_10_serial(self, env: Env, vsftpds: VsFTPD, docname, repeat):
106*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
107*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpds.docs_dir, f'{docname}')
108*6236dae4SAndroid Build Coastguard Worker        count = 10
109*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}?[0-{count-1}]'
110*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True)
111*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
112*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
113*6236dae4SAndroid Build Coastguard Worker
114*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
115*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
116*6236dae4SAndroid Build Coastguard Worker    ])
117*6236dae4SAndroid Build Coastguard Worker    def test_31_04_download_10_parallel(self, env: Env, vsftpds: VsFTPD, docname, repeat):
118*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
119*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpds.docs_dir, f'{docname}')
120*6236dae4SAndroid Build Coastguard Worker        count = 10
121*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}?[0-{count-1}]'
122*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True, extra_args=[
123*6236dae4SAndroid Build Coastguard Worker            '--parallel'
124*6236dae4SAndroid Build Coastguard Worker        ])
125*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
126*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
127*6236dae4SAndroid Build Coastguard Worker
128*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
129*6236dae4SAndroid Build Coastguard Worker        'upload-1k', 'upload-100k', 'upload-1m'
130*6236dae4SAndroid Build Coastguard Worker    ])
131*6236dae4SAndroid Build Coastguard Worker    def test_31_05_upload_1(self, env: Env, vsftpds: VsFTPD, docname, repeat):
132*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
133*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
134*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpds.docs_dir, docname)
135*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
136*6236dae4SAndroid Build Coastguard Worker        count = 1
137*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/'
138*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_upload(urls=[url], fupload=f'{srcfile}', with_stats=True)
139*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
140*6236dae4SAndroid Build Coastguard Worker        self.check_upload(env, vsftpds, docname=docname)
141*6236dae4SAndroid Build Coastguard Worker
142*6236dae4SAndroid Build Coastguard Worker    def _rmf(self, path):
143*6236dae4SAndroid Build Coastguard Worker        if os.path.exists(path):
144*6236dae4SAndroid Build Coastguard Worker            return os.remove(path)
145*6236dae4SAndroid Build Coastguard Worker
146*6236dae4SAndroid Build Coastguard Worker    # check with `tcpdump` if curl causes any TCP RST packets
147*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.skipif(condition=not Env.tcpdump(), reason="tcpdump not available")
148*6236dae4SAndroid Build Coastguard Worker    def test_31_06_shutdownh_download(self, env: Env, vsftpds: VsFTPD, repeat):
149*6236dae4SAndroid Build Coastguard Worker        docname = 'data-1k'
150*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
151*6236dae4SAndroid Build Coastguard Worker        count = 1
152*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}?[0-{count-1}]'
153*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True, with_tcpdump=True)
154*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
155*6236dae4SAndroid Build Coastguard Worker        # vsftp closes control connection without niceties,
156*6236dae4SAndroid Build Coastguard Worker        # disregard RST packets it sent from its port to curl
157*6236dae4SAndroid Build Coastguard Worker        assert len(r.tcpdump.stats_excluding(src_port=env.ftps_port)) == 0, 'Unexpected TCP RSTs packets'
158*6236dae4SAndroid Build Coastguard Worker
159*6236dae4SAndroid Build Coastguard Worker    # check with `tcpdump` if curl causes any TCP RST packets
160*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.skipif(condition=not Env.tcpdump(), reason="tcpdump not available")
161*6236dae4SAndroid Build Coastguard Worker    def test_31_07_shutdownh_upload(self, env: Env, vsftpds: VsFTPD, repeat):
162*6236dae4SAndroid Build Coastguard Worker        docname = 'upload-1k'
163*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
164*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
165*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpds.docs_dir, docname)
166*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
167*6236dae4SAndroid Build Coastguard Worker        count = 1
168*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/'
169*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_upload(urls=[url], fupload=f'{srcfile}', with_stats=True, with_tcpdump=True)
170*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
171*6236dae4SAndroid Build Coastguard Worker        # vsftp closes control connection without niceties,
172*6236dae4SAndroid Build Coastguard Worker        # disregard RST packets it sent from its port to curl
173*6236dae4SAndroid Build Coastguard Worker        assert len(r.tcpdump.stats_excluding(src_port=env.ftps_port)) == 0, 'Unexpected TCP RSTs packets'
174*6236dae4SAndroid Build Coastguard Worker
175*6236dae4SAndroid Build Coastguard Worker    def test_31_08_upload_ascii(self, env: Env, vsftpds: VsFTPD):
176*6236dae4SAndroid Build Coastguard Worker        docname = 'upload-ascii'
177*6236dae4SAndroid Build Coastguard Worker        line_length = 21
178*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
179*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpds.docs_dir, docname)
180*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname=docname, fsize=100*1024,
181*6236dae4SAndroid Build Coastguard Worker                           line_length=line_length)
182*6236dae4SAndroid Build Coastguard Worker        srcsize = os.path.getsize(srcfile)
183*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
184*6236dae4SAndroid Build Coastguard Worker        count = 1
185*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
186*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/'
187*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_upload(urls=[url], fupload=f'{srcfile}', with_stats=True,
188*6236dae4SAndroid Build Coastguard Worker                                extra_args=['--use-ascii'])
189*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
190*6236dae4SAndroid Build Coastguard Worker        # expect the uploaded file to be number of converted newlines larger
191*6236dae4SAndroid Build Coastguard Worker        dstsize = os.path.getsize(dstfile)
192*6236dae4SAndroid Build Coastguard Worker        newlines = len(open(srcfile).readlines())
193*6236dae4SAndroid Build Coastguard Worker        assert (srcsize + newlines) == dstsize, \
194*6236dae4SAndroid Build Coastguard Worker            f'expected source with {newlines} lines to be that much larger,'\
195*6236dae4SAndroid Build Coastguard Worker            f'instead srcsize={srcsize}, upload size={dstsize}, diff={dstsize-srcsize}'
196*6236dae4SAndroid Build Coastguard Worker
197*6236dae4SAndroid Build Coastguard Worker    def test_31_08_active_download(self, env: Env, vsftpds: VsFTPD):
198*6236dae4SAndroid Build Coastguard Worker        docname = 'data-10k'
199*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
200*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpds.docs_dir, f'{docname}')
201*6236dae4SAndroid Build Coastguard Worker        count = 1
202*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}?[0-{count-1}]'
203*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_get(urls=[url], with_stats=True, extra_args=[
204*6236dae4SAndroid Build Coastguard Worker            '--ftp-port', '127.0.0.1'
205*6236dae4SAndroid Build Coastguard Worker        ])
206*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
207*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
208*6236dae4SAndroid Build Coastguard Worker
209*6236dae4SAndroid Build Coastguard Worker    def test_31_09_active_upload(self, env: Env, vsftpds: VsFTPD):
210*6236dae4SAndroid Build Coastguard Worker        docname = 'upload-1k'
211*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
212*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
213*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpds.docs_dir, docname)
214*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
215*6236dae4SAndroid Build Coastguard Worker        count = 1
216*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/'
217*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_upload(urls=[url], fupload=f'{srcfile}', with_stats=True, extra_args=[
218*6236dae4SAndroid Build Coastguard Worker            '--ftp-port', '127.0.0.1'
219*6236dae4SAndroid Build Coastguard Worker        ])
220*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
221*6236dae4SAndroid Build Coastguard Worker        self.check_upload(env, vsftpds, docname=docname)
222*6236dae4SAndroid Build Coastguard Worker
223*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("indata", [
224*6236dae4SAndroid Build Coastguard Worker        '1234567890', ''
225*6236dae4SAndroid Build Coastguard Worker    ])
226*6236dae4SAndroid Build Coastguard Worker    def test_31_10_upload_stdin(self, env: Env, vsftpds: VsFTPD, indata):
227*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
228*6236dae4SAndroid Build Coastguard Worker        docname = "upload_31_10"
229*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpds.docs_dir, docname)
230*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
231*6236dae4SAndroid Build Coastguard Worker        count = 1
232*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpds.port}/{docname}'
233*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_ssl_upload(urls=[url], updata=indata, with_stats=True)
234*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
235*6236dae4SAndroid Build Coastguard Worker        assert os.path.exists(dstfile)
236*6236dae4SAndroid Build Coastguard Worker        destdata = open(dstfile).readlines()
237*6236dae4SAndroid Build Coastguard Worker        expdata = [indata] if len(indata) else []
238*6236dae4SAndroid Build Coastguard Worker        assert expdata == destdata, f'exected: {expdata}, got: {destdata}'
239*6236dae4SAndroid Build Coastguard Worker
240*6236dae4SAndroid Build Coastguard Worker    def check_downloads(self, client, srcfile: str, count: int,
241*6236dae4SAndroid Build Coastguard Worker                        complete: bool = True):
242*6236dae4SAndroid Build Coastguard Worker        for i in range(count):
243*6236dae4SAndroid Build Coastguard Worker            dfile = client.download_file(i)
244*6236dae4SAndroid Build Coastguard Worker            assert os.path.exists(dfile)
245*6236dae4SAndroid Build Coastguard Worker            if complete and not filecmp.cmp(srcfile, dfile, shallow=False):
246*6236dae4SAndroid Build Coastguard Worker                diff = "".join(difflib.unified_diff(a=open(srcfile).readlines(),
247*6236dae4SAndroid Build Coastguard Worker                                                    b=open(dfile).readlines(),
248*6236dae4SAndroid Build Coastguard Worker                                                    fromfile=srcfile,
249*6236dae4SAndroid Build Coastguard Worker                                                    tofile=dfile,
250*6236dae4SAndroid Build Coastguard Worker                                                    n=1))
251*6236dae4SAndroid Build Coastguard Worker                assert False, f'download {dfile} differs:\n{diff}'
252*6236dae4SAndroid Build Coastguard Worker
253*6236dae4SAndroid Build Coastguard Worker    def check_upload(self, env, vsftpd: VsFTPD, docname):
254*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
255*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpd.docs_dir, docname)
256*6236dae4SAndroid Build Coastguard Worker        assert os.path.exists(srcfile)
257*6236dae4SAndroid Build Coastguard Worker        assert os.path.exists(dstfile)
258*6236dae4SAndroid Build Coastguard Worker        if not filecmp.cmp(srcfile, dstfile, shallow=False):
259*6236dae4SAndroid Build Coastguard Worker            diff = "".join(difflib.unified_diff(a=open(srcfile).readlines(),
260*6236dae4SAndroid Build Coastguard Worker                                                b=open(dstfile).readlines(),
261*6236dae4SAndroid Build Coastguard Worker                                                fromfile=srcfile,
262*6236dae4SAndroid Build Coastguard Worker                                                tofile=dstfile,
263*6236dae4SAndroid Build Coastguard Worker                                                n=1))
264*6236dae4SAndroid Build Coastguard Worker            assert False, f'upload {dstfile} differs:\n{diff}'
265