mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-11 23:42:38 +00:00
Merge remote-tracking branch 'origin/master' into 3581.unicode_to_argv.1
This commit is contained in:
commit
3beacc9c8f
@ -91,6 +91,9 @@ workflows:
|
|||||||
- "build-porting-depgraph":
|
- "build-porting-depgraph":
|
||||||
<<: *DOCKERHUB_CONTEXT
|
<<: *DOCKERHUB_CONTEXT
|
||||||
|
|
||||||
|
- "typechecks":
|
||||||
|
<<: *DOCKERHUB_CONTEXT
|
||||||
|
|
||||||
images:
|
images:
|
||||||
# Build the Docker images used by the ci jobs. This makes the ci jobs
|
# Build the Docker images used by the ci jobs. This makes the ci jobs
|
||||||
# faster and takes various spurious failures out of the critical path.
|
# faster and takes various spurious failures out of the critical path.
|
||||||
@ -475,6 +478,18 @@ jobs:
|
|||||||
. /tmp/venv/bin/activate
|
. /tmp/venv/bin/activate
|
||||||
./misc/python3/depgraph.sh
|
./misc/python3/depgraph.sh
|
||||||
|
|
||||||
|
typechecks:
|
||||||
|
docker:
|
||||||
|
- <<: *DOCKERHUB_AUTH
|
||||||
|
image: "tahoelafsci/ubuntu:18.04-py3"
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- "checkout"
|
||||||
|
- run:
|
||||||
|
name: "Validate Types"
|
||||||
|
command: |
|
||||||
|
/tmp/venv/bin/tox -e typechecks
|
||||||
|
|
||||||
build-image: &BUILD_IMAGE
|
build-image: &BUILD_IMAGE
|
||||||
# This is a template for a job to build a Docker image that has as much of
|
# This is a template for a job to build a Docker image that has as much of
|
||||||
# the setup as we can manage already done and baked in. This cuts down on
|
# the setup as we can manage already done and baked in. This cuts down on
|
||||||
|
2
.github/CONTRIBUTING.rst
vendored
2
.github/CONTRIBUTING.rst
vendored
@ -17,4 +17,4 @@ Examples of contributions include:
|
|||||||
* `Patch reviews <https://tahoe-lafs.org/trac/tahoe-lafs/wiki/PatchReviewProcess>`_
|
* `Patch reviews <https://tahoe-lafs.org/trac/tahoe-lafs/wiki/PatchReviewProcess>`_
|
||||||
|
|
||||||
Before authoring or reviewing a patch,
|
Before authoring or reviewing a patch,
|
||||||
please familiarize yourself with the `coding standard <https://tahoe-lafs.org/trac/tahoe-lafs/wiki/CodingStandards>`_.
|
please familiarize yourself with the `Coding Standards <https://tahoe-lafs.org/trac/tahoe-lafs/wiki/CodingStandards>`_ and the `Contributor Code of Conduct <../docs/CODE_OF_CONDUCT.md>`_.
|
||||||
|
62
.github/workflows/ci.yml
vendored
62
.github/workflows/ci.yml
vendored
@ -30,17 +30,37 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
args: install vcpython27
|
args: install vcpython27
|
||||||
|
|
||||||
|
# See https://github.com/actions/checkout. A fetch-depth of 0
|
||||||
|
# fetches all tags and branches.
|
||||||
- name: Check out Tahoe-LAFS sources
|
- name: Check out Tahoe-LAFS sources
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
- name: Fetch all history for all tags and branches
|
fetch-depth: 0
|
||||||
run: git fetch --prune --unshallow
|
|
||||||
|
|
||||||
- name: Set up Python ${{ matrix.python-version }}
|
- name: Set up Python ${{ matrix.python-version }}
|
||||||
uses: actions/setup-python@v1
|
uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: ${{ matrix.python-version }}
|
python-version: ${{ matrix.python-version }}
|
||||||
|
|
||||||
|
# To use pip caching with GitHub Actions in an OS-independent
|
||||||
|
# manner, we need `pip cache dir` command, which became
|
||||||
|
# available since pip v20.1+. At the time of writing this,
|
||||||
|
# GitHub Actions offers pip v20.3.3 for both ubuntu-latest and
|
||||||
|
# windows-latest, and pip v20.3.1 for macos-latest.
|
||||||
|
- name: Get pip cache directory
|
||||||
|
id: pip-cache
|
||||||
|
run: |
|
||||||
|
echo "::set-output name=dir::$(pip cache dir)"
|
||||||
|
|
||||||
|
# See https://github.com/actions/cache
|
||||||
|
- name: Use pip cache
|
||||||
|
uses: actions/cache@v2
|
||||||
|
with:
|
||||||
|
path: ${{ steps.pip-cache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/setup.py') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pip-
|
||||||
|
|
||||||
- name: Install Python packages
|
- name: Install Python packages
|
||||||
run: |
|
run: |
|
||||||
pip install --upgrade codecov tox setuptools
|
pip install --upgrade codecov tox setuptools
|
||||||
@ -103,15 +123,27 @@ jobs:
|
|||||||
|
|
||||||
- name: Check out Tahoe-LAFS sources
|
- name: Check out Tahoe-LAFS sources
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
- name: Fetch all history for all tags and branches
|
fetch-depth: 0
|
||||||
run: git fetch --prune --unshallow
|
|
||||||
|
|
||||||
- name: Set up Python ${{ matrix.python-version }}
|
- name: Set up Python ${{ matrix.python-version }}
|
||||||
uses: actions/setup-python@v1
|
uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: ${{ matrix.python-version }}
|
python-version: ${{ matrix.python-version }}
|
||||||
|
|
||||||
|
- name: Get pip cache directory
|
||||||
|
id: pip-cache
|
||||||
|
run: |
|
||||||
|
echo "::set-output name=dir::$(pip cache dir)"
|
||||||
|
|
||||||
|
- name: Use pip cache
|
||||||
|
uses: actions/cache@v2
|
||||||
|
with:
|
||||||
|
path: ${{ steps.pip-cache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/setup.py') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pip-
|
||||||
|
|
||||||
- name: Install Python packages
|
- name: Install Python packages
|
||||||
run: |
|
run: |
|
||||||
pip install --upgrade tox
|
pip install --upgrade tox
|
||||||
@ -155,15 +187,27 @@ jobs:
|
|||||||
|
|
||||||
- name: Check out Tahoe-LAFS sources
|
- name: Check out Tahoe-LAFS sources
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
- name: Fetch all history for all tags and branches
|
fetch-depth: 0
|
||||||
run: git fetch --prune --unshallow
|
|
||||||
|
|
||||||
- name: Set up Python ${{ matrix.python-version }}
|
- name: Set up Python ${{ matrix.python-version }}
|
||||||
uses: actions/setup-python@v1
|
uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: ${{ matrix.python-version }}
|
python-version: ${{ matrix.python-version }}
|
||||||
|
|
||||||
|
- name: Get pip cache directory
|
||||||
|
id: pip-cache
|
||||||
|
run: |
|
||||||
|
echo "::set-output name=dir::$(pip cache dir)"
|
||||||
|
|
||||||
|
- name: Use pip cache
|
||||||
|
uses: actions/cache@v2
|
||||||
|
with:
|
||||||
|
path: ${{ steps.pip-cache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-pip-${{ hashFiles('**/setup.py') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pip-
|
||||||
|
|
||||||
- name: Install Python packages
|
- name: Install Python packages
|
||||||
run: |
|
run: |
|
||||||
pip install --upgrade tox
|
pip install --upgrade tox
|
||||||
|
7
CREDITS
7
CREDITS
@ -206,4 +206,9 @@ D: various bug-fixes and features
|
|||||||
|
|
||||||
N: Viktoriia Savchuk
|
N: Viktoriia Savchuk
|
||||||
W: https://twitter.com/viktoriiasvchk
|
W: https://twitter.com/viktoriiasvchk
|
||||||
D: Developer community focused improvements on the README file.
|
D: Developer community focused improvements on the README file.
|
||||||
|
|
||||||
|
N: Lukas Pirl
|
||||||
|
E: tahoe@lukas-pirl.de
|
||||||
|
W: http://lukas-pirl.de
|
||||||
|
D: Buildslaves (Debian, Fedora, CentOS; 2016-2021)
|
||||||
|
@ -173,7 +173,9 @@ from PyPI with ``venv/bin/pip install tahoe-lafs``. After installation, run
|
|||||||
Install From a Source Tarball
|
Install From a Source Tarball
|
||||||
-----------------------------
|
-----------------------------
|
||||||
|
|
||||||
You can also install directly from the source tarball URL::
|
You can also install directly from the source tarball URL. To verify
|
||||||
|
signatures, first see verifying_signatures_ and replace the URL in the
|
||||||
|
following instructions with the local filename.
|
||||||
|
|
||||||
% virtualenv venv
|
% virtualenv venv
|
||||||
New python executable in ~/venv/bin/python2.7
|
New python executable in ~/venv/bin/python2.7
|
||||||
@ -189,6 +191,40 @@ You can also install directly from the source tarball URL::
|
|||||||
tahoe-lafs: 1.14.0
|
tahoe-lafs: 1.14.0
|
||||||
...
|
...
|
||||||
|
|
||||||
|
.. _verifying_signatures:
|
||||||
|
|
||||||
|
Verifying Signatures
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
First download the source tarball and then any signatures. There are several
|
||||||
|
developers who are able to produce signatures for a release. A release may
|
||||||
|
have multiple signatures. All should be valid and you should confirm at least
|
||||||
|
one of them (ideally, confirm all).
|
||||||
|
|
||||||
|
This statement, signed by the existing Tahoe release-signing key, attests to
|
||||||
|
those developers authorized to sign a Tahoe release:
|
||||||
|
|
||||||
|
.. include:: developer-release-signatures
|
||||||
|
:code:
|
||||||
|
|
||||||
|
Signatures are made available beside the release. So for example, a release
|
||||||
|
like ``https://tahoe-lafs.org/downloads/tahoe-lafs-1.16.0.tar.bz2`` might
|
||||||
|
have signatures ``tahoe-lafs-1.16.0.tar.bz2.meejah.asc`` and
|
||||||
|
``tahoe-lafs-1.16.0.tar.bz2.warner.asc``.
|
||||||
|
|
||||||
|
To verify the signatures using GnuPG::
|
||||||
|
|
||||||
|
% gpg --verify tahoe-lafs-1.16.0.tar.bz2.meejah.asc tahoe-lafs-1.16.0.tar.bz2
|
||||||
|
gpg: Signature made XXX
|
||||||
|
gpg: using RSA key 9D5A2BD5688ECB889DEBCD3FC2602803128069A7
|
||||||
|
gpg: Good signature from "meejah <meejah@meejah.ca>" [full]
|
||||||
|
% gpg --verify tahoe-lafs-1.16.0.tar.bz2.warner.asc tahoe-lafs-1.16.0.tar.bz2
|
||||||
|
gpg: Signature made XXX
|
||||||
|
gpg: using RSA key 967EFE06699872411A77DF36D43B4C9C73225AAF
|
||||||
|
gpg: Good signature from "Brian Warner <warner@lothar.com>" [full]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Extras
|
Extras
|
||||||
------
|
------
|
||||||
|
|
||||||
|
@ -67,12 +67,12 @@ Here's how it works:
|
|||||||
A "storage grid" is made up of a number of storage servers. A storage server
|
A "storage grid" is made up of a number of storage servers. A storage server
|
||||||
has direct attached storage (typically one or more hard disks). A "gateway"
|
has direct attached storage (typically one or more hard disks). A "gateway"
|
||||||
communicates with storage nodes, and uses them to provide access to the
|
communicates with storage nodes, and uses them to provide access to the
|
||||||
grid over protocols such as HTTP(S), SFTP or FTP.
|
grid over protocols such as HTTP(S) and SFTP.
|
||||||
|
|
||||||
Note that you can find "client" used to refer to gateway nodes (which act as
|
Note that you can find "client" used to refer to gateway nodes (which act as
|
||||||
a client to storage servers), and also to processes or programs connecting to
|
a client to storage servers), and also to processes or programs connecting to
|
||||||
a gateway node and performing operations on the grid -- for example, a CLI
|
a gateway node and performing operations on the grid -- for example, a CLI
|
||||||
command, Web browser, SFTP client, or FTP client.
|
command, Web browser, or SFTP client.
|
||||||
|
|
||||||
Users do not rely on storage servers to provide *confidentiality* nor
|
Users do not rely on storage servers to provide *confidentiality* nor
|
||||||
*integrity* for their data -- instead all of the data is encrypted and
|
*integrity* for their data -- instead all of the data is encrypted and
|
||||||
|
@ -28,7 +28,7 @@ import os
|
|||||||
# Add any Sphinx extension module names here, as strings. They can be
|
# Add any Sphinx extension module names here, as strings. They can be
|
||||||
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
|
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
|
||||||
# ones.
|
# ones.
|
||||||
extensions = []
|
extensions = ['recommonmark']
|
||||||
|
|
||||||
# Add any paths that contain templates here, relative to this directory.
|
# Add any paths that contain templates here, relative to this directory.
|
||||||
templates_path = ['_templates']
|
templates_path = ['_templates']
|
||||||
@ -36,7 +36,7 @@ templates_path = ['_templates']
|
|||||||
# The suffix(es) of source filenames.
|
# The suffix(es) of source filenames.
|
||||||
# You can specify multiple suffix as a list of string:
|
# You can specify multiple suffix as a list of string:
|
||||||
# source_suffix = ['.rst', '.md']
|
# source_suffix = ['.rst', '.md']
|
||||||
source_suffix = '.rst'
|
source_suffix = ['.rst', '.md']
|
||||||
|
|
||||||
# The encoding of source files.
|
# The encoding of source files.
|
||||||
#source_encoding = 'utf-8-sig'
|
#source_encoding = 'utf-8-sig'
|
||||||
|
@ -81,7 +81,6 @@ Client/server nodes provide one or more of the following services:
|
|||||||
|
|
||||||
* web-API service
|
* web-API service
|
||||||
* SFTP service
|
* SFTP service
|
||||||
* FTP service
|
|
||||||
* helper service
|
* helper service
|
||||||
* storage service.
|
* storage service.
|
||||||
|
|
||||||
@ -708,12 +707,12 @@ CLI
|
|||||||
file store, uploading/downloading files, and creating/running Tahoe
|
file store, uploading/downloading files, and creating/running Tahoe
|
||||||
nodes. See :doc:`frontends/CLI` for details.
|
nodes. See :doc:`frontends/CLI` for details.
|
||||||
|
|
||||||
SFTP, FTP
|
SFTP
|
||||||
|
|
||||||
Tahoe can also run both SFTP and FTP servers, and map a username/password
|
Tahoe can also run SFTP servers, and map a username/password
|
||||||
pair to a top-level Tahoe directory. See :doc:`frontends/FTP-and-SFTP`
|
pair to a top-level Tahoe directory. See :doc:`frontends/FTP-and-SFTP`
|
||||||
for instructions on configuring these services, and the ``[sftpd]`` and
|
for instructions on configuring this service, and the ``[sftpd]``
|
||||||
``[ftpd]`` sections of ``tahoe.cfg``.
|
section of ``tahoe.cfg``.
|
||||||
|
|
||||||
|
|
||||||
Storage Server Configuration
|
Storage Server Configuration
|
||||||
|
1
docs/contributing.rst
Normal file
1
docs/contributing.rst
Normal file
@ -0,0 +1 @@
|
|||||||
|
.. include:: ../.github/CONTRIBUTING.rst
|
42
docs/developer-release-signatures
Normal file
42
docs/developer-release-signatures
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
-----BEGIN PGP SIGNED MESSAGE-----
|
||||||
|
Hash: SHA512
|
||||||
|
|
||||||
|
|
||||||
|
January 20, 2021
|
||||||
|
|
||||||
|
Any of the following core Tahoe contributers may sign a release. Each
|
||||||
|
release MUST be signed by at least one developer but MAY have
|
||||||
|
additional signatures. Each developer independently produces a
|
||||||
|
signature which is made available beside Tahoe releases after 1.15.0
|
||||||
|
|
||||||
|
This statement is signed by the existing Tahoe release key. Any future
|
||||||
|
such statements may be signed by it OR by any two developers (for
|
||||||
|
example, to add or remove developers from the list).
|
||||||
|
|
||||||
|
meejah
|
||||||
|
0xC2602803128069A7
|
||||||
|
9D5A 2BD5 688E CB88 9DEB CD3F C260 2803 1280 69A7
|
||||||
|
https://meejah.ca/meejah.asc
|
||||||
|
|
||||||
|
jean-paul calderone (exarkun)
|
||||||
|
0xE27B085EDEAA4B1B
|
||||||
|
96B9 C5DA B2EA 9EB6 7941 9DB7 E27B 085E DEAA 4B1B
|
||||||
|
https://twistedmatrix.com/~exarkun/E27B085EDEAA4B1B.asc
|
||||||
|
|
||||||
|
brian warner (lothar)
|
||||||
|
0x863333C265497810
|
||||||
|
5810 F125 7F8C F753 7753 895A 8633 33C2 6549 7810
|
||||||
|
https://www.lothar.com/warner-gpg.html
|
||||||
|
|
||||||
|
|
||||||
|
-----BEGIN PGP SIGNATURE-----
|
||||||
|
|
||||||
|
iQEzBAEBCgAdFiEE405i0G0Oac/KQXn/veDTHWhmanoFAmAHIyIACgkQveDTHWhm
|
||||||
|
anqhqQf/YSbMXL+gwFhAZsjX39EVlbr/Ik7WPPkJW7v1oHybTnwFpFIc52COU1x/
|
||||||
|
sqRfk4OyYtz9IBgOPXoWgXu9R4qdK6vYKxEsekcGT9C5l0OyDz8YWXEWgbGK5mvI
|
||||||
|
aEub9WucD8r2uOQnnW6DtznFuEpvOjtf/+2BU767+bvLsbViW88ocbuLfCqLdOgD
|
||||||
|
WZT9j3M+Y2Dc56DAJzP/4fkrUSVIofZStYp5u9HBjburgcYIp0g/cyc4xXRoi6Mp
|
||||||
|
lFTRFv3MIjmoamzSQseoIgP6fi8QRqPrffPrsyqAp+06mJnPhxxFqxtO/ZErmpSa
|
||||||
|
+BGrLBxdWa8IF9U1A4Fs5nuAzAKMEg==
|
||||||
|
=E9J+
|
||||||
|
-----END PGP SIGNATURE-----
|
@ -1,22 +1,21 @@
|
|||||||
.. -*- coding: utf-8-with-signature -*-
|
.. -*- coding: utf-8-with-signature -*-
|
||||||
|
|
||||||
=================================
|
========================
|
||||||
Tahoe-LAFS SFTP and FTP Frontends
|
Tahoe-LAFS SFTP Frontend
|
||||||
=================================
|
========================
|
||||||
|
|
||||||
1. `SFTP/FTP Background`_
|
1. `SFTP Background`_
|
||||||
2. `Tahoe-LAFS Support`_
|
2. `Tahoe-LAFS Support`_
|
||||||
3. `Creating an Account File`_
|
3. `Creating an Account File`_
|
||||||
4. `Running An Account Server (accounts.url)`_
|
4. `Running An Account Server (accounts.url)`_
|
||||||
5. `Configuring SFTP Access`_
|
5. `Configuring SFTP Access`_
|
||||||
6. `Configuring FTP Access`_
|
6. `Dependencies`_
|
||||||
7. `Dependencies`_
|
7. `Immutable and Mutable Files`_
|
||||||
8. `Immutable and Mutable Files`_
|
8. `Known Issues`_
|
||||||
9. `Known Issues`_
|
|
||||||
|
|
||||||
|
|
||||||
SFTP/FTP Background
|
SFTP Background
|
||||||
===================
|
===============
|
||||||
|
|
||||||
FTP is the venerable internet file-transfer protocol, first developed in
|
FTP is the venerable internet file-transfer protocol, first developed in
|
||||||
1971. The FTP server usually listens on port 21. A separate connection is
|
1971. The FTP server usually listens on port 21. A separate connection is
|
||||||
@ -33,20 +32,18 @@ Both FTP and SFTP were developed assuming a UNIX-like server, with accounts
|
|||||||
and passwords, octal file modes (user/group/other, read/write/execute), and
|
and passwords, octal file modes (user/group/other, read/write/execute), and
|
||||||
ctime/mtime timestamps.
|
ctime/mtime timestamps.
|
||||||
|
|
||||||
We recommend SFTP over FTP, because the protocol is better, and the server
|
Previous versions of Tahoe-LAFS supported FTP, but now only the superior SFTP
|
||||||
implementation in Tahoe-LAFS is more complete. See `Known Issues`_, below,
|
frontend is supported. See `Known Issues`_, below, for details on the
|
||||||
for details.
|
limitations of SFTP.
|
||||||
|
|
||||||
Tahoe-LAFS Support
|
Tahoe-LAFS Support
|
||||||
==================
|
==================
|
||||||
|
|
||||||
All Tahoe-LAFS client nodes can run a frontend SFTP server, allowing regular
|
All Tahoe-LAFS client nodes can run a frontend SFTP server, allowing regular
|
||||||
SFTP clients (like ``/usr/bin/sftp``, the ``sshfs`` FUSE plugin, and many
|
SFTP clients (like ``/usr/bin/sftp``, the ``sshfs`` FUSE plugin, and many
|
||||||
others) to access the file store. They can also run an FTP server, so FTP
|
others) to access the file store.
|
||||||
clients (like ``/usr/bin/ftp``, ``ncftp``, and others) can too. These
|
|
||||||
frontends sit at the same level as the web-API interface.
|
|
||||||
|
|
||||||
Since Tahoe-LAFS does not use user accounts or passwords, the SFTP/FTP
|
Since Tahoe-LAFS does not use user accounts or passwords, the SFTP
|
||||||
servers must be configured with a way to first authenticate a user (confirm
|
servers must be configured with a way to first authenticate a user (confirm
|
||||||
that a prospective client has a legitimate claim to whatever authorities we
|
that a prospective client has a legitimate claim to whatever authorities we
|
||||||
might grant a particular user), and second to decide what directory cap
|
might grant a particular user), and second to decide what directory cap
|
||||||
@ -173,39 +170,6 @@ clients and with the sshfs filesystem, see wiki:SftpFrontend_
|
|||||||
|
|
||||||
.. _wiki:SftpFrontend: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/SftpFrontend
|
.. _wiki:SftpFrontend: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/SftpFrontend
|
||||||
|
|
||||||
Configuring FTP Access
|
|
||||||
======================
|
|
||||||
|
|
||||||
To enable the FTP server with an accounts file, add the following lines to
|
|
||||||
the BASEDIR/tahoe.cfg file::
|
|
||||||
|
|
||||||
[ftpd]
|
|
||||||
enabled = true
|
|
||||||
port = tcp:8021:interface=127.0.0.1
|
|
||||||
accounts.file = private/accounts
|
|
||||||
|
|
||||||
The FTP server will listen on the given port number and on the loopback
|
|
||||||
interface only. The "accounts.file" pathname will be interpreted relative to
|
|
||||||
the node's BASEDIR.
|
|
||||||
|
|
||||||
To enable the FTP server with an account server instead, provide the URL of
|
|
||||||
that server in an "accounts.url" directive::
|
|
||||||
|
|
||||||
[ftpd]
|
|
||||||
enabled = true
|
|
||||||
port = tcp:8021:interface=127.0.0.1
|
|
||||||
accounts.url = https://example.com/login
|
|
||||||
|
|
||||||
You can provide both accounts.file and accounts.url, although it probably
|
|
||||||
isn't very useful except for testing.
|
|
||||||
|
|
||||||
FTP provides no security, and so your password or caps could be eavesdropped
|
|
||||||
if you connect to the FTP server remotely. The examples above include
|
|
||||||
":interface=127.0.0.1" in the "port" option, which causes the server to only
|
|
||||||
accept connections from localhost.
|
|
||||||
|
|
||||||
Public key authentication is not supported for FTP.
|
|
||||||
|
|
||||||
Dependencies
|
Dependencies
|
||||||
============
|
============
|
||||||
|
|
||||||
@ -216,7 +180,7 @@ separately: debian puts it in the "python-twisted-conch" package.
|
|||||||
Immutable and Mutable Files
|
Immutable and Mutable Files
|
||||||
===========================
|
===========================
|
||||||
|
|
||||||
All files created via SFTP (and FTP) are immutable files. However, files can
|
All files created via SFTP are immutable files. However, files can
|
||||||
only be created in writeable directories, which allows the directory entry to
|
only be created in writeable directories, which allows the directory entry to
|
||||||
be relinked to a different file. Normally, when the path of an immutable file
|
be relinked to a different file. Normally, when the path of an immutable file
|
||||||
is opened for writing by SFTP, the directory entry is relinked to another
|
is opened for writing by SFTP, the directory entry is relinked to another
|
||||||
@ -256,18 +220,3 @@ See also wiki:SftpFrontend_.
|
|||||||
|
|
||||||
.. _ticket #1059: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1059
|
.. _ticket #1059: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1059
|
||||||
.. _ticket #1089: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1089
|
.. _ticket #1089: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1089
|
||||||
|
|
||||||
Known Issues in the FTP Frontend
|
|
||||||
--------------------------------
|
|
||||||
|
|
||||||
Mutable files are not supported by the FTP frontend (`ticket #680`_).
|
|
||||||
|
|
||||||
Non-ASCII filenames are not supported by FTP (`ticket #682`_).
|
|
||||||
|
|
||||||
The FTP frontend sometimes fails to report errors, for example if an upload
|
|
||||||
fails because it does meet the "servers of happiness" threshold (`ticket
|
|
||||||
#1081`_).
|
|
||||||
|
|
||||||
.. _ticket #680: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/680
|
|
||||||
.. _ticket #682: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/682
|
|
||||||
.. _ticket #1081: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1081
|
|
||||||
|
@ -2032,10 +2032,11 @@ potential for surprises when the file store structure is changed.
|
|||||||
|
|
||||||
Tahoe-LAFS provides a mutable file store, but the ways that the store can
|
Tahoe-LAFS provides a mutable file store, but the ways that the store can
|
||||||
change are limited. The only things that can change are:
|
change are limited. The only things that can change are:
|
||||||
* the mapping from child names to child objects inside mutable directories
|
|
||||||
(by adding a new child, removing an existing child, or changing an
|
* the mapping from child names to child objects inside mutable directories
|
||||||
existing child to point to a different object)
|
(by adding a new child, removing an existing child, or changing an
|
||||||
* the contents of mutable files
|
existing child to point to a different object)
|
||||||
|
* the contents of mutable files
|
||||||
|
|
||||||
Obviously if you query for information about the file store and then act
|
Obviously if you query for information about the file store and then act
|
||||||
to change it (such as by getting a listing of the contents of a mutable
|
to change it (such as by getting a listing of the contents of a mutable
|
||||||
@ -2157,7 +2158,7 @@ When modifying the file, be careful to update it atomically, otherwise a
|
|||||||
request may arrive while the file is only halfway written, and the partial
|
request may arrive while the file is only halfway written, and the partial
|
||||||
file may be incorrectly parsed.
|
file may be incorrectly parsed.
|
||||||
|
|
||||||
The blacklist is applied to all access paths (including SFTP, FTP, and CLI
|
The blacklist is applied to all access paths (including SFTP and CLI
|
||||||
operations), not just the web-API. The blacklist also applies to directories.
|
operations), not just the web-API. The blacklist also applies to directories.
|
||||||
If a directory is blacklisted, the gateway will refuse access to both that
|
If a directory is blacklisted, the gateway will refuse access to both that
|
||||||
directory and any child files/directories underneath it, when accessed via
|
directory and any child files/directories underneath it, when accessed via
|
||||||
|
@ -122,7 +122,7 @@ Who should consider using a Helper?
|
|||||||
* clients who experience problems with TCP connection fairness: if other
|
* clients who experience problems with TCP connection fairness: if other
|
||||||
programs or machines in the same home are getting less than their fair
|
programs or machines in the same home are getting less than their fair
|
||||||
share of upload bandwidth. If the connection is being shared fairly, then
|
share of upload bandwidth. If the connection is being shared fairly, then
|
||||||
a Tahoe upload that is happening at the same time as a single FTP upload
|
a Tahoe upload that is happening at the same time as a single SFTP upload
|
||||||
should get half the bandwidth.
|
should get half the bandwidth.
|
||||||
* clients who have been given the helper.furl by someone who is running a
|
* clients who have been given the helper.furl by someone who is running a
|
||||||
Helper and is willing to let them use it
|
Helper and is willing to let them use it
|
||||||
|
@ -23,8 +23,9 @@ Contents:
|
|||||||
frontends/download-status
|
frontends/download-status
|
||||||
|
|
||||||
known_issues
|
known_issues
|
||||||
../.github/CONTRIBUTING
|
contributing
|
||||||
CODE_OF_CONDUCT
|
CODE_OF_CONDUCT
|
||||||
|
release-checklist
|
||||||
|
|
||||||
servers
|
servers
|
||||||
helper
|
helper
|
||||||
|
@ -23,7 +23,7 @@ Known Issues in Tahoe-LAFS v1.10.3, released 30-Mar-2016
|
|||||||
* `Disclosure of file through embedded hyperlinks or JavaScript in that file`_
|
* `Disclosure of file through embedded hyperlinks or JavaScript in that file`_
|
||||||
* `Command-line arguments are leaked to other local users`_
|
* `Command-line arguments are leaked to other local users`_
|
||||||
* `Capabilities may be leaked to web browser phishing filter / "safe browsing" servers`_
|
* `Capabilities may be leaked to web browser phishing filter / "safe browsing" servers`_
|
||||||
* `Known issues in the FTP and SFTP frontends`_
|
* `Known issues in the SFTP frontend`_
|
||||||
* `Traffic analysis based on sizes of files/directories, storage indices, and timing`_
|
* `Traffic analysis based on sizes of files/directories, storage indices, and timing`_
|
||||||
* `Privacy leak via Google Chart API link in map-update timing web page`_
|
* `Privacy leak via Google Chart API link in map-update timing web page`_
|
||||||
|
|
||||||
@ -213,8 +213,8 @@ To disable the filter in Chrome:
|
|||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
Known issues in the FTP and SFTP frontends
|
Known issues in the SFTP frontend
|
||||||
------------------------------------------
|
---------------------------------
|
||||||
|
|
||||||
These are documented in :doc:`frontends/FTP-and-SFTP` and on `the
|
These are documented in :doc:`frontends/FTP-and-SFTP` and on `the
|
||||||
SftpFrontend page`_ on the wiki.
|
SftpFrontend page`_ on the wiki.
|
||||||
|
@ -40,23 +40,31 @@ Create Branch and Apply Updates
|
|||||||
- Create a branch for release-candidates (e.g. `XXXX.release-1.15.0.rc0`)
|
- Create a branch for release-candidates (e.g. `XXXX.release-1.15.0.rc0`)
|
||||||
- run `tox -e news` to produce a new NEWS.txt file (this does a commit)
|
- run `tox -e news` to produce a new NEWS.txt file (this does a commit)
|
||||||
- create the news for the release
|
- create the news for the release
|
||||||
|
|
||||||
- newsfragments/<ticket number>.minor
|
- newsfragments/<ticket number>.minor
|
||||||
- commit it
|
- commit it
|
||||||
|
|
||||||
- manually fix NEWS.txt
|
- manually fix NEWS.txt
|
||||||
|
|
||||||
- proper title for latest release ("Release 1.15.0" instead of "Release ...post1432")
|
- proper title for latest release ("Release 1.15.0" instead of "Release ...post1432")
|
||||||
- double-check date (maybe release will be in the future)
|
- double-check date (maybe release will be in the future)
|
||||||
- spot-check the release notes (these come from the newsfragments
|
- spot-check the release notes (these come from the newsfragments
|
||||||
files though so don't do heavy editing)
|
files though so don't do heavy editing)
|
||||||
- commit these changes
|
- commit these changes
|
||||||
|
|
||||||
- update "relnotes.txt"
|
- update "relnotes.txt"
|
||||||
|
|
||||||
- update all mentions of 1.14.0 -> 1.15.0
|
- update all mentions of 1.14.0 -> 1.15.0
|
||||||
- update "previous release" statement and date
|
- update "previous release" statement and date
|
||||||
- summarize major changes
|
- summarize major changes
|
||||||
- commit it
|
- commit it
|
||||||
|
|
||||||
- update "CREDITS"
|
- update "CREDITS"
|
||||||
|
|
||||||
- are there any new contributors in this release?
|
- are there any new contributors in this release?
|
||||||
- one way: git log release-1.14.0.. | grep Author | sort | uniq
|
- one way: git log release-1.14.0.. | grep Author | sort | uniq
|
||||||
- commit it
|
- commit it
|
||||||
|
|
||||||
- update "docs/known_issues.rst" if appropriate
|
- update "docs/known_issues.rst" if appropriate
|
||||||
- update "docs/INSTALL.rst" references to the new release
|
- update "docs/INSTALL.rst" references to the new release
|
||||||
- Push the branch to github
|
- Push the branch to github
|
||||||
@ -82,25 +90,36 @@ they will need to evaluate which contributors' signatures they trust.
|
|||||||
|
|
||||||
- (all steps above are completed)
|
- (all steps above are completed)
|
||||||
- sign the release
|
- sign the release
|
||||||
|
|
||||||
- git tag -s -u 0xE34E62D06D0E69CFCA4179FFBDE0D31D68666A7A -m "release Tahoe-LAFS-1.15.0rc0" tahoe-lafs-1.15.0rc0
|
- git tag -s -u 0xE34E62D06D0E69CFCA4179FFBDE0D31D68666A7A -m "release Tahoe-LAFS-1.15.0rc0" tahoe-lafs-1.15.0rc0
|
||||||
- (replace the key-id above with your own)
|
- (replace the key-id above with your own)
|
||||||
|
|
||||||
- build all code locally
|
- build all code locally
|
||||||
- these should all pass:
|
- these should all pass:
|
||||||
|
|
||||||
- tox -e py27,codechecks,docs,integration
|
- tox -e py27,codechecks,docs,integration
|
||||||
|
|
||||||
- these can fail (ideally they should not of course):
|
- these can fail (ideally they should not of course):
|
||||||
|
|
||||||
- tox -e deprecations,upcoming-deprecations
|
- tox -e deprecations,upcoming-deprecations
|
||||||
|
|
||||||
- build tarballs
|
- build tarballs
|
||||||
|
|
||||||
- tox -e tarballs
|
- tox -e tarballs
|
||||||
- confirm it at least exists:
|
- confirm it at least exists:
|
||||||
- ls dist/ | grep 1.15.0rc0
|
- ls dist/ | grep 1.15.0rc0
|
||||||
|
|
||||||
- inspect and test the tarballs
|
- inspect and test the tarballs
|
||||||
|
|
||||||
- install each in a fresh virtualenv
|
- install each in a fresh virtualenv
|
||||||
- run `tahoe` command
|
- run `tahoe` command
|
||||||
|
|
||||||
- when satisfied, sign the tarballs:
|
- when satisfied, sign the tarballs:
|
||||||
- gpg --pinentry=loopback --armor --sign dist/tahoe_lafs-1.15.0rc0-py2-none-any.whl
|
|
||||||
- gpg --pinentry=loopback --armor --sign dist/tahoe_lafs-1.15.0rc0.tar.bz2
|
- gpg --pinentry=loopback --armor --detach-sign dist/tahoe_lafs-1.15.0rc0-py2-none-any.whl
|
||||||
- gpg --pinentry=loopback --armor --sign dist/tahoe_lafs-1.15.0rc0.tar.gz
|
- gpg --pinentry=loopback --armor --detach-sign dist/tahoe_lafs-1.15.0rc0.tar.bz2
|
||||||
- gpg --pinentry=loopback --armor --sign dist/tahoe_lafs-1.15.0rc0.zip
|
- gpg --pinentry=loopback --armor --detach-sign dist/tahoe_lafs-1.15.0rc0.tar.gz
|
||||||
|
- gpg --pinentry=loopback --armor --detach-sign dist/tahoe_lafs-1.15.0rc0.zip
|
||||||
|
|
||||||
|
|
||||||
Privileged Contributor
|
Privileged Contributor
|
||||||
@ -118,6 +137,12 @@ Did anyone contribute a hack since the last release? If so, then
|
|||||||
https://tahoe-lafs.org/hacktahoelafs/ needs to be updated.
|
https://tahoe-lafs.org/hacktahoelafs/ needs to be updated.
|
||||||
|
|
||||||
|
|
||||||
|
Sign Git Tag
|
||||||
|
````````````
|
||||||
|
|
||||||
|
- git tag -s -u 0xE34E62D06D0E69CFCA4179FFBDE0D31D68666A7A -m "release Tahoe-LAFS-X.Y.Z" tahoe-lafs-X.Y.Z
|
||||||
|
|
||||||
|
|
||||||
Upload Artifacts
|
Upload Artifacts
|
||||||
````````````````
|
````````````````
|
||||||
|
|
||||||
@ -129,6 +154,7 @@ need to be uploaded to https://tahoe-lafs.org in `~source/downloads`
|
|||||||
https://tahoe-lafs.org/downloads/ on the Web.
|
https://tahoe-lafs.org/downloads/ on the Web.
|
||||||
- scp dist/*1.15.0* username@tahoe-lafs.org:/home/source/downloads
|
- scp dist/*1.15.0* username@tahoe-lafs.org:/home/source/downloads
|
||||||
- the following developers have access to do this:
|
- the following developers have access to do this:
|
||||||
|
|
||||||
- exarkun
|
- exarkun
|
||||||
- meejah
|
- meejah
|
||||||
- warner
|
- warner
|
||||||
@ -137,8 +163,9 @@ For the actual release, the tarball and signature files need to be
|
|||||||
uploaded to PyPI as well.
|
uploaded to PyPI as well.
|
||||||
|
|
||||||
- how to do this?
|
- how to do this?
|
||||||
- (original guide says only "twine upload dist/*")
|
- (original guide says only `twine upload dist/*`)
|
||||||
- the following developers have access to do this:
|
- the following developers have access to do this:
|
||||||
|
|
||||||
- warner
|
- warner
|
||||||
- exarkun (partial?)
|
- exarkun (partial?)
|
||||||
- meejah (partial?)
|
- meejah (partial?)
|
||||||
|
@ -207,10 +207,10 @@ create a new directory and lose the capability to it, then you cannot
|
|||||||
access that directory ever again.
|
access that directory ever again.
|
||||||
|
|
||||||
|
|
||||||
The SFTP and FTP frontends
|
The SFTP frontend
|
||||||
--------------------------
|
-----------------
|
||||||
|
|
||||||
You can access your Tahoe-LAFS grid via any SFTP_ or FTP_ client. See
|
You can access your Tahoe-LAFS grid via any SFTP_ client. See
|
||||||
:doc:`frontends/FTP-and-SFTP` for how to set this up. On most Unix
|
:doc:`frontends/FTP-and-SFTP` for how to set this up. On most Unix
|
||||||
platforms, you can also use SFTP to plug Tahoe-LAFS into your computer's
|
platforms, you can also use SFTP to plug Tahoe-LAFS into your computer's
|
||||||
local filesystem via ``sshfs``, but see the `FAQ about performance
|
local filesystem via ``sshfs``, but see the `FAQ about performance
|
||||||
@ -220,7 +220,6 @@ The SftpFrontend_ page on the wiki has more information about using SFTP with
|
|||||||
Tahoe-LAFS.
|
Tahoe-LAFS.
|
||||||
|
|
||||||
.. _SFTP: https://en.wikipedia.org/wiki/SSH_file_transfer_protocol
|
.. _SFTP: https://en.wikipedia.org/wiki/SSH_file_transfer_protocol
|
||||||
.. _FTP: https://en.wikipedia.org/wiki/File_Transfer_Protocol
|
|
||||||
.. _FAQ about performance problems: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/FAQ#Q23_FUSE
|
.. _FAQ about performance problems: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/FAQ#Q23_FUSE
|
||||||
.. _SftpFrontend: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/SftpFrontend
|
.. _SftpFrontend: https://tahoe-lafs.org/trac/tahoe-lafs/wiki/SftpFrontend
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@ from os import mkdir, listdir, environ
|
|||||||
from os.path import join, exists
|
from os.path import join, exists
|
||||||
from tempfile import mkdtemp, mktemp
|
from tempfile import mkdtemp, mktemp
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
from json import loads
|
||||||
|
|
||||||
from foolscap.furl import (
|
from foolscap.furl import (
|
||||||
decode_furl,
|
decode_furl,
|
||||||
@ -37,6 +38,10 @@ from util import (
|
|||||||
_tahoe_runner_optional_coverage,
|
_tahoe_runner_optional_coverage,
|
||||||
await_client_ready,
|
await_client_ready,
|
||||||
TahoeProcess,
|
TahoeProcess,
|
||||||
|
cli,
|
||||||
|
_run_node,
|
||||||
|
generate_ssh_key,
|
||||||
|
block_with_timeout,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -152,7 +157,7 @@ def flog_gatherer(reactor, temp_dir, flog_binary, request):
|
|||||||
)
|
)
|
||||||
print("Waiting for flogtool to complete")
|
print("Waiting for flogtool to complete")
|
||||||
try:
|
try:
|
||||||
pytest_twisted.blockon(flog_protocol.done)
|
block_with_timeout(flog_protocol.done, reactor)
|
||||||
except ProcessTerminated as e:
|
except ProcessTerminated as e:
|
||||||
print("flogtool exited unexpectedly: {}".format(str(e)))
|
print("flogtool exited unexpectedly: {}".format(str(e)))
|
||||||
print("Flogtool completed")
|
print("Flogtool completed")
|
||||||
@ -293,7 +298,7 @@ log_gatherer.furl = {log_furl}
|
|||||||
def cleanup():
|
def cleanup():
|
||||||
try:
|
try:
|
||||||
transport.signalProcess('TERM')
|
transport.signalProcess('TERM')
|
||||||
pytest_twisted.blockon(protocol.exited)
|
block_with_timeout(protocol.exited, reactor)
|
||||||
except ProcessExitedAlready:
|
except ProcessExitedAlready:
|
||||||
pass
|
pass
|
||||||
request.addfinalizer(cleanup)
|
request.addfinalizer(cleanup)
|
||||||
@ -347,8 +352,50 @@ def alice(reactor, temp_dir, introducer_furl, flog_gatherer, storage_nodes, requ
|
|||||||
reactor, request, temp_dir, introducer_furl, flog_gatherer, "alice",
|
reactor, request, temp_dir, introducer_furl, flog_gatherer, "alice",
|
||||||
web_port="tcp:9980:interface=localhost",
|
web_port="tcp:9980:interface=localhost",
|
||||||
storage=False,
|
storage=False,
|
||||||
|
# We're going to kill this ourselves, so no need for finalizer to
|
||||||
|
# do it:
|
||||||
|
finalize=False,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
await_client_ready(process)
|
||||||
|
|
||||||
|
# 1. Create a new RW directory cap:
|
||||||
|
cli(process, "create-alias", "test")
|
||||||
|
rwcap = loads(cli(process, "list-aliases", "--json"))["test"]["readwrite"]
|
||||||
|
|
||||||
|
# 2. Enable SFTP on the node:
|
||||||
|
host_ssh_key_path = join(process.node_dir, "private", "ssh_host_rsa_key")
|
||||||
|
accounts_path = join(process.node_dir, "private", "accounts")
|
||||||
|
with open(join(process.node_dir, "tahoe.cfg"), "a") as f:
|
||||||
|
f.write("""\
|
||||||
|
[sftpd]
|
||||||
|
enabled = true
|
||||||
|
port = tcp:8022:interface=127.0.0.1
|
||||||
|
host_pubkey_file = {ssh_key_path}.pub
|
||||||
|
host_privkey_file = {ssh_key_path}
|
||||||
|
accounts.file = {accounts_path}
|
||||||
|
""".format(ssh_key_path=host_ssh_key_path, accounts_path=accounts_path))
|
||||||
|
generate_ssh_key(host_ssh_key_path)
|
||||||
|
|
||||||
|
# 3. Add a SFTP access file with username/password and SSH key auth.
|
||||||
|
|
||||||
|
# The client SSH key path is typically going to be somewhere else (~/.ssh,
|
||||||
|
# typically), but for convenience sake for testing we'll put it inside node.
|
||||||
|
client_ssh_key_path = join(process.node_dir, "private", "ssh_client_rsa_key")
|
||||||
|
generate_ssh_key(client_ssh_key_path)
|
||||||
|
# Pub key format is "ssh-rsa <thekey> <username>". We want the key.
|
||||||
|
ssh_public_key = open(client_ssh_key_path + ".pub").read().strip().split()[1]
|
||||||
|
with open(accounts_path, "w") as f:
|
||||||
|
f.write("""\
|
||||||
|
alice password {rwcap}
|
||||||
|
|
||||||
|
alice2 ssh-rsa {ssh_public_key} {rwcap}
|
||||||
|
""".format(rwcap=rwcap, ssh_public_key=ssh_public_key))
|
||||||
|
|
||||||
|
# 4. Restart the node with new SFTP config.
|
||||||
|
process.kill()
|
||||||
|
pytest_twisted.blockon(_run_node(reactor, process.node_dir, request, None))
|
||||||
|
|
||||||
await_client_ready(process)
|
await_client_ready(process)
|
||||||
return process
|
return process
|
||||||
|
|
||||||
@ -490,7 +537,13 @@ def tor_network(reactor, temp_dir, chutney, request):
|
|||||||
path=join(chutney_dir),
|
path=join(chutney_dir),
|
||||||
env=env,
|
env=env,
|
||||||
)
|
)
|
||||||
pytest_twisted.blockon(proto.done)
|
try:
|
||||||
|
block_with_timeout(proto.done, reactor)
|
||||||
|
except ProcessTerminated:
|
||||||
|
# If this doesn't exit cleanly, that's fine, that shouldn't fail
|
||||||
|
# the test suite.
|
||||||
|
pass
|
||||||
|
|
||||||
request.addfinalizer(cleanup)
|
request.addfinalizer(cleanup)
|
||||||
|
|
||||||
return chut
|
return chut
|
||||||
|
162
integration/test_sftp.py
Normal file
162
integration/test_sftp.py
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
"""
|
||||||
|
It's possible to create/rename/delete files and directories in Tahoe-LAFS using
|
||||||
|
SFTP.
|
||||||
|
|
||||||
|
These tests use Paramiko, rather than Twisted's Conch, because:
|
||||||
|
|
||||||
|
1. It's a different implementation, so we're not testing Conch against
|
||||||
|
itself.
|
||||||
|
|
||||||
|
2. Its API is much simpler to use.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
|
from posixpath import join
|
||||||
|
from stat import S_ISDIR
|
||||||
|
|
||||||
|
from paramiko import SSHClient
|
||||||
|
from paramiko.client import AutoAddPolicy
|
||||||
|
from paramiko.sftp_client import SFTPClient
|
||||||
|
from paramiko.ssh_exception import AuthenticationException
|
||||||
|
from paramiko.rsakey import RSAKey
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from .util import generate_ssh_key, run_in_thread
|
||||||
|
|
||||||
|
|
||||||
|
def connect_sftp(connect_args={"username": "alice", "password": "password"}):
|
||||||
|
"""Create an SFTP client."""
|
||||||
|
client = SSHClient()
|
||||||
|
client.set_missing_host_key_policy(AutoAddPolicy)
|
||||||
|
client.connect("localhost", port=8022, look_for_keys=False,
|
||||||
|
allow_agent=False, **connect_args)
|
||||||
|
sftp = SFTPClient.from_transport(client.get_transport())
|
||||||
|
|
||||||
|
def rmdir(path, delete_root=True):
|
||||||
|
for f in sftp.listdir_attr(path=path):
|
||||||
|
childpath = join(path, f.filename)
|
||||||
|
if S_ISDIR(f.st_mode):
|
||||||
|
rmdir(childpath)
|
||||||
|
else:
|
||||||
|
sftp.remove(childpath)
|
||||||
|
if delete_root:
|
||||||
|
sftp.rmdir(path)
|
||||||
|
|
||||||
|
# Delete any files left over from previous tests :(
|
||||||
|
rmdir("/", delete_root=False)
|
||||||
|
|
||||||
|
return sftp
|
||||||
|
|
||||||
|
|
||||||
|
@run_in_thread
|
||||||
|
def test_bad_account_password_ssh_key(alice, tmpdir):
|
||||||
|
"""
|
||||||
|
Can't login with unknown username, wrong password, or wrong SSH pub key.
|
||||||
|
"""
|
||||||
|
# Wrong password, wrong username:
|
||||||
|
for u, p in [("alice", "wrong"), ("someuser", "password")]:
|
||||||
|
with pytest.raises(AuthenticationException):
|
||||||
|
connect_sftp(connect_args={
|
||||||
|
"username": u, "password": p,
|
||||||
|
})
|
||||||
|
|
||||||
|
another_key = join(str(tmpdir), "ssh_key")
|
||||||
|
generate_ssh_key(another_key)
|
||||||
|
good_key = RSAKey(filename=join(alice.node_dir, "private", "ssh_client_rsa_key"))
|
||||||
|
bad_key = RSAKey(filename=another_key)
|
||||||
|
|
||||||
|
# Wrong key:
|
||||||
|
with pytest.raises(AuthenticationException):
|
||||||
|
connect_sftp(connect_args={
|
||||||
|
"username": "alice2", "pkey": bad_key,
|
||||||
|
})
|
||||||
|
|
||||||
|
# Wrong username:
|
||||||
|
with pytest.raises(AuthenticationException):
|
||||||
|
connect_sftp(connect_args={
|
||||||
|
"username": "someoneelse", "pkey": good_key,
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
@run_in_thread
|
||||||
|
def test_ssh_key_auth(alice):
|
||||||
|
"""It's possible to login authenticating with SSH public key."""
|
||||||
|
key = RSAKey(filename=join(alice.node_dir, "private", "ssh_client_rsa_key"))
|
||||||
|
sftp = connect_sftp(connect_args={
|
||||||
|
"username": "alice2", "pkey": key
|
||||||
|
})
|
||||||
|
assert sftp.listdir() == []
|
||||||
|
|
||||||
|
|
||||||
|
@run_in_thread
|
||||||
|
def test_read_write_files(alice):
|
||||||
|
"""It's possible to upload and download files."""
|
||||||
|
sftp = connect_sftp()
|
||||||
|
with sftp.file("myfile", "wb") as f:
|
||||||
|
f.write(b"abc")
|
||||||
|
f.write(b"def")
|
||||||
|
|
||||||
|
with sftp.file("myfile", "rb") as f:
|
||||||
|
assert f.read(4) == b"abcd"
|
||||||
|
assert f.read(2) == b"ef"
|
||||||
|
assert f.read(1) == b""
|
||||||
|
|
||||||
|
|
||||||
|
@run_in_thread
|
||||||
|
def test_directories(alice):
|
||||||
|
"""
|
||||||
|
It's possible to create, list directories, and create and remove files in
|
||||||
|
them.
|
||||||
|
"""
|
||||||
|
sftp = connect_sftp()
|
||||||
|
assert sftp.listdir() == []
|
||||||
|
|
||||||
|
sftp.mkdir("childdir")
|
||||||
|
assert sftp.listdir() == ["childdir"]
|
||||||
|
|
||||||
|
with sftp.file("myfile", "wb") as f:
|
||||||
|
f.write(b"abc")
|
||||||
|
assert sorted(sftp.listdir()) == ["childdir", "myfile"]
|
||||||
|
|
||||||
|
sftp.chdir("childdir")
|
||||||
|
assert sftp.listdir() == []
|
||||||
|
|
||||||
|
with sftp.file("myfile2", "wb") as f:
|
||||||
|
f.write(b"def")
|
||||||
|
assert sftp.listdir() == ["myfile2"]
|
||||||
|
|
||||||
|
sftp.chdir(None) # root
|
||||||
|
with sftp.file("childdir/myfile2", "rb") as f:
|
||||||
|
assert f.read() == b"def"
|
||||||
|
|
||||||
|
sftp.remove("myfile")
|
||||||
|
assert sftp.listdir() == ["childdir"]
|
||||||
|
|
||||||
|
sftp.rmdir("childdir")
|
||||||
|
assert sftp.listdir() == []
|
||||||
|
|
||||||
|
|
||||||
|
@run_in_thread
|
||||||
|
def test_rename(alice):
|
||||||
|
"""Directories and files can be renamed."""
|
||||||
|
sftp = connect_sftp()
|
||||||
|
sftp.mkdir("dir")
|
||||||
|
|
||||||
|
filepath = join("dir", "file")
|
||||||
|
with sftp.file(filepath, "wb") as f:
|
||||||
|
f.write(b"abc")
|
||||||
|
|
||||||
|
sftp.rename(filepath, join("dir", "file2"))
|
||||||
|
sftp.rename("dir", "dir2")
|
||||||
|
|
||||||
|
with sftp.file(join("dir2", "file2"), "rb") as f:
|
||||||
|
assert f.read() == b"abc"
|
@ -127,12 +127,12 @@ def test_deep_stats(alice):
|
|||||||
dircap_uri,
|
dircap_uri,
|
||||||
data={
|
data={
|
||||||
u"t": u"upload",
|
u"t": u"upload",
|
||||||
u"when_done": u".",
|
|
||||||
},
|
},
|
||||||
files={
|
files={
|
||||||
u"file": FILE_CONTENTS,
|
u"file": FILE_CONTENTS,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
resp.raise_for_status()
|
||||||
|
|
||||||
# confirm the file is in the directory
|
# confirm the file is in the directory
|
||||||
resp = requests.get(
|
resp = requests.get(
|
||||||
@ -175,6 +175,7 @@ def test_deep_stats(alice):
|
|||||||
time.sleep(.5)
|
time.sleep(.5)
|
||||||
|
|
||||||
|
|
||||||
|
@util.run_in_thread
|
||||||
def test_status(alice):
|
def test_status(alice):
|
||||||
"""
|
"""
|
||||||
confirm we get something sensible from /status and the various sub-types
|
confirm we get something sensible from /status and the various sub-types
|
||||||
|
@ -5,6 +5,7 @@ from os import mkdir, environ
|
|||||||
from os.path import exists, join
|
from os.path import exists, join
|
||||||
from six.moves import StringIO
|
from six.moves import StringIO
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
from subprocess import check_output
|
||||||
|
|
||||||
from twisted.python.filepath import (
|
from twisted.python.filepath import (
|
||||||
FilePath,
|
FilePath,
|
||||||
@ -12,9 +13,13 @@ from twisted.python.filepath import (
|
|||||||
from twisted.internet.defer import Deferred, succeed
|
from twisted.internet.defer import Deferred, succeed
|
||||||
from twisted.internet.protocol import ProcessProtocol
|
from twisted.internet.protocol import ProcessProtocol
|
||||||
from twisted.internet.error import ProcessExitedAlready, ProcessDone
|
from twisted.internet.error import ProcessExitedAlready, ProcessDone
|
||||||
|
from twisted.internet.threads import deferToThread
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
|
from paramiko.rsakey import RSAKey
|
||||||
|
from boltons.funcutils import wraps
|
||||||
|
|
||||||
from allmydata.util.configutil import (
|
from allmydata.util.configutil import (
|
||||||
get_config,
|
get_config,
|
||||||
set_config,
|
set_config,
|
||||||
@ -25,6 +30,12 @@ from allmydata import client
|
|||||||
import pytest_twisted
|
import pytest_twisted
|
||||||
|
|
||||||
|
|
||||||
|
def block_with_timeout(deferred, reactor, timeout=120):
|
||||||
|
"""Block until Deferred has result, but timeout instead of waiting forever."""
|
||||||
|
deferred.addTimeout(timeout, reactor)
|
||||||
|
return pytest_twisted.blockon(deferred)
|
||||||
|
|
||||||
|
|
||||||
class _ProcessExitedProtocol(ProcessProtocol):
|
class _ProcessExitedProtocol(ProcessProtocol):
|
||||||
"""
|
"""
|
||||||
Internal helper that .callback()s on self.done when the process
|
Internal helper that .callback()s on self.done when the process
|
||||||
@ -123,11 +134,12 @@ def _cleanup_tahoe_process(tahoe_transport, exited):
|
|||||||
|
|
||||||
:return: After the process has exited.
|
:return: After the process has exited.
|
||||||
"""
|
"""
|
||||||
|
from twisted.internet import reactor
|
||||||
try:
|
try:
|
||||||
print("signaling {} with TERM".format(tahoe_transport.pid))
|
print("signaling {} with TERM".format(tahoe_transport.pid))
|
||||||
tahoe_transport.signalProcess('TERM')
|
tahoe_transport.signalProcess('TERM')
|
||||||
print("signaled, blocking on exit")
|
print("signaled, blocking on exit")
|
||||||
pytest_twisted.blockon(exited)
|
block_with_timeout(exited, reactor)
|
||||||
print("exited, goodbye")
|
print("exited, goodbye")
|
||||||
except ProcessExitedAlready:
|
except ProcessExitedAlready:
|
||||||
pass
|
pass
|
||||||
@ -175,11 +187,15 @@ class TahoeProcess(object):
|
|||||||
u"portnum",
|
u"portnum",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def kill(self):
|
||||||
|
"""Kill the process, block until it's done."""
|
||||||
|
_cleanup_tahoe_process(self.transport, self.transport.exited)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "<TahoeProcess in '{}'>".format(self._node_dir)
|
return "<TahoeProcess in '{}'>".format(self._node_dir)
|
||||||
|
|
||||||
|
|
||||||
def _run_node(reactor, node_dir, request, magic_text):
|
def _run_node(reactor, node_dir, request, magic_text, finalize=True):
|
||||||
"""
|
"""
|
||||||
Run a tahoe process from its node_dir.
|
Run a tahoe process from its node_dir.
|
||||||
|
|
||||||
@ -203,7 +219,8 @@ def _run_node(reactor, node_dir, request, magic_text):
|
|||||||
)
|
)
|
||||||
transport.exited = protocol.exited
|
transport.exited = protocol.exited
|
||||||
|
|
||||||
request.addfinalizer(partial(_cleanup_tahoe_process, transport, protocol.exited))
|
if finalize:
|
||||||
|
request.addfinalizer(partial(_cleanup_tahoe_process, transport, protocol.exited))
|
||||||
|
|
||||||
# XXX abusing the Deferred; should use .when_magic_seen() pattern
|
# XXX abusing the Deferred; should use .when_magic_seen() pattern
|
||||||
|
|
||||||
@ -222,7 +239,8 @@ def _create_node(reactor, request, temp_dir, introducer_furl, flog_gatherer, nam
|
|||||||
magic_text=None,
|
magic_text=None,
|
||||||
needed=2,
|
needed=2,
|
||||||
happy=3,
|
happy=3,
|
||||||
total=4):
|
total=4,
|
||||||
|
finalize=True):
|
||||||
"""
|
"""
|
||||||
Helper to create a single node, run it and return the instance
|
Helper to create a single node, run it and return the instance
|
||||||
spawnProcess returned (ITransport)
|
spawnProcess returned (ITransport)
|
||||||
@ -270,7 +288,7 @@ def _create_node(reactor, request, temp_dir, introducer_furl, flog_gatherer, nam
|
|||||||
d = Deferred()
|
d = Deferred()
|
||||||
d.callback(None)
|
d.callback(None)
|
||||||
d.addCallback(lambda _: created_d)
|
d.addCallback(lambda _: created_d)
|
||||||
d.addCallback(lambda _: _run_node(reactor, node_dir, request, magic_text))
|
d.addCallback(lambda _: _run_node(reactor, node_dir, request, magic_text, finalize=finalize))
|
||||||
return d
|
return d
|
||||||
|
|
||||||
|
|
||||||
@ -390,17 +408,13 @@ def await_file_vanishes(path, timeout=10):
|
|||||||
raise FileShouldVanishException(path, timeout)
|
raise FileShouldVanishException(path, timeout)
|
||||||
|
|
||||||
|
|
||||||
def cli(request, reactor, node_dir, *argv):
|
def cli(node, *argv):
|
||||||
"""
|
"""
|
||||||
Run a tahoe CLI subcommand for a given node, optionally running
|
Run a tahoe CLI subcommand for a given node in a blocking manner, returning
|
||||||
under coverage if '--coverage' was supplied.
|
the output.
|
||||||
"""
|
"""
|
||||||
proto = _CollectOutputProtocol()
|
arguments = ["tahoe", '--node-directory', node.node_dir]
|
||||||
_tahoe_runner_optional_coverage(
|
return check_output(arguments + list(argv))
|
||||||
proto, reactor, request,
|
|
||||||
['--node-directory', node_dir] + list(argv),
|
|
||||||
)
|
|
||||||
return proto.done
|
|
||||||
|
|
||||||
|
|
||||||
def node_url(node_dir, uri_fragment):
|
def node_url(node_dir, uri_fragment):
|
||||||
@ -505,3 +519,36 @@ def await_client_ready(tahoe, timeout=10, liveness=60*2):
|
|||||||
tahoe,
|
tahoe,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def generate_ssh_key(path):
|
||||||
|
"""Create a new SSH private/public key pair."""
|
||||||
|
key = RSAKey.generate(2048)
|
||||||
|
key.write_private_key_file(path)
|
||||||
|
with open(path + ".pub", "wb") as f:
|
||||||
|
f.write(b"%s %s" % (key.get_name(), key.get_base64()))
|
||||||
|
|
||||||
|
|
||||||
|
def run_in_thread(f):
|
||||||
|
"""Decorator for integration tests that runs code in a thread.
|
||||||
|
|
||||||
|
Because we're using pytest_twisted, tests that rely on the reactor are
|
||||||
|
expected to return a Deferred and use async APIs so the reactor can run.
|
||||||
|
|
||||||
|
In the case of the integration test suite, it launches nodes in the
|
||||||
|
background using Twisted APIs. The nodes stdout and stderr is read via
|
||||||
|
Twisted code. If the reactor doesn't run, reads don't happen, and
|
||||||
|
eventually the buffers fill up, and the nodes block when they try to flush
|
||||||
|
logs.
|
||||||
|
|
||||||
|
We can switch to Twisted APIs (treq instead of requests etc.), but
|
||||||
|
sometimes it's easier or expedient to just have a blocking test. So this
|
||||||
|
decorator allows you to run the test in a thread, and the reactor can keep
|
||||||
|
running in the main thread.
|
||||||
|
|
||||||
|
See https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3597 for tracking bug.
|
||||||
|
"""
|
||||||
|
@wraps(f)
|
||||||
|
def test(*args, **kwargs):
|
||||||
|
return deferToThread(lambda: f(*args, **kwargs))
|
||||||
|
return test
|
||||||
|
3
mypy.ini
Normal file
3
mypy.ini
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[mypy]
|
||||||
|
ignore_missing_imports = True
|
||||||
|
plugins=mypy_zope:plugin
|
1
newsfragments/3399.feature
Normal file
1
newsfragments/3399.feature
Normal file
@ -0,0 +1 @@
|
|||||||
|
Added 'typechecks' environment for tox running mypy and performing static typechecks.
|
0
newsfragments/3536.minor
Normal file
0
newsfragments/3536.minor
Normal file
0
newsfragments/3576.minor
Normal file
0
newsfragments/3576.minor
Normal file
0
newsfragments/3577.minor
Normal file
0
newsfragments/3577.minor
Normal file
0
newsfragments/3579.minor
Normal file
0
newsfragments/3579.minor
Normal file
0
newsfragments/3580.minor
Normal file
0
newsfragments/3580.minor
Normal file
1
newsfragments/3583.removed
Normal file
1
newsfragments/3583.removed
Normal file
@ -0,0 +1 @@
|
|||||||
|
FTP is no longer supported by Tahoe-LAFS. Please use the SFTP support instead.
|
1
newsfragments/3584.bugfix
Normal file
1
newsfragments/3584.bugfix
Normal file
@ -0,0 +1 @@
|
|||||||
|
SFTP public key auth likely works more consistently, and SFTP in general was previously broken.
|
1
newsfragments/3587.minor
Normal file
1
newsfragments/3587.minor
Normal file
@ -0,0 +1 @@
|
|||||||
|
|
0
newsfragments/3589.minor
Normal file
0
newsfragments/3589.minor
Normal file
1
newsfragments/3590.bugfix
Normal file
1
newsfragments/3590.bugfix
Normal file
@ -0,0 +1 @@
|
|||||||
|
Fixed issue where redirecting old-style URIs (/uri/?uri=...) didn't work.
|
0
newsfragments/3591.minor
Normal file
0
newsfragments/3591.minor
Normal file
0
newsfragments/3593.minor
Normal file
0
newsfragments/3593.minor
Normal file
0
newsfragments/3594.minor
Normal file
0
newsfragments/3594.minor
Normal file
0
newsfragments/3595.minor
Normal file
0
newsfragments/3595.minor
Normal file
0
newsfragments/3599.minor
Normal file
0
newsfragments/3599.minor
Normal file
6
setup.py
6
setup.py
@ -63,12 +63,8 @@ install_requires = [
|
|||||||
# version of cryptography will *really* be installed.
|
# version of cryptography will *really* be installed.
|
||||||
"cryptography >= 2.6",
|
"cryptography >= 2.6",
|
||||||
|
|
||||||
# * We need Twisted 10.1.0 for the FTP frontend in order for
|
|
||||||
# Twisted's FTP server to support asynchronous close.
|
|
||||||
# * The SFTP frontend depends on Twisted 11.0.0 to fix the SSH server
|
# * The SFTP frontend depends on Twisted 11.0.0 to fix the SSH server
|
||||||
# rekeying bug <https://twistedmatrix.com/trac/ticket/4395>
|
# rekeying bug <https://twistedmatrix.com/trac/ticket/4395>
|
||||||
# * The FTP frontend depends on Twisted >= 11.1.0 for
|
|
||||||
# filepath.Permissions
|
|
||||||
# * The SFTP frontend and manhole depend on the conch extra. However, we
|
# * The SFTP frontend and manhole depend on the conch extra. However, we
|
||||||
# can't explicitly declare that without an undesirable dependency on gmpy,
|
# can't explicitly declare that without an undesirable dependency on gmpy,
|
||||||
# as explained in ticket #2740.
|
# as explained in ticket #2740.
|
||||||
@ -399,6 +395,8 @@ setup(name="tahoe-lafs", # also set in __init__.py
|
|||||||
"html5lib",
|
"html5lib",
|
||||||
"junitxml",
|
"junitxml",
|
||||||
"tenacity",
|
"tenacity",
|
||||||
|
"paramiko",
|
||||||
|
"pytest-timeout",
|
||||||
] + tor_requires + i2p_requires,
|
] + tor_requires + i2p_requires,
|
||||||
"tor": tor_requires,
|
"tor": tor_requires,
|
||||||
"i2p": i2p_requires,
|
"i2p": i2p_requires,
|
||||||
|
@ -14,7 +14,9 @@ __all__ = [
|
|||||||
|
|
||||||
__version__ = "unknown"
|
__version__ = "unknown"
|
||||||
try:
|
try:
|
||||||
from allmydata._version import __version__
|
# type ignored as it fails in CI
|
||||||
|
# (https://app.circleci.com/pipelines/github/tahoe-lafs/tahoe-lafs/1647/workflows/60ae95d4-abe8-492c-8a03-1ad3b9e42ed3/jobs/40972)
|
||||||
|
from allmydata._version import __version__ # type: ignore
|
||||||
except ImportError:
|
except ImportError:
|
||||||
# We're running in a tree that hasn't run update_version, and didn't
|
# We're running in a tree that hasn't run update_version, and didn't
|
||||||
# come with a _version.py, so we don't know what our version is.
|
# come with a _version.py, so we don't know what our version is.
|
||||||
@ -24,7 +26,9 @@ except ImportError:
|
|||||||
full_version = "unknown"
|
full_version = "unknown"
|
||||||
branch = "unknown"
|
branch = "unknown"
|
||||||
try:
|
try:
|
||||||
from allmydata._version import full_version, branch
|
# type ignored as it fails in CI
|
||||||
|
# (https://app.circleci.com/pipelines/github/tahoe-lafs/tahoe-lafs/1647/workflows/60ae95d4-abe8-492c-8a03-1ad3b9e42ed3/jobs/40972)
|
||||||
|
from allmydata._version import full_version, branch # type: ignore
|
||||||
except ImportError:
|
except ImportError:
|
||||||
# We're running in a tree that hasn't run update_version, and didn't
|
# We're running in a tree that hasn't run update_version, and didn't
|
||||||
# come with a _version.py, so we don't know what our full version or
|
# come with a _version.py, so we don't know what our full version or
|
||||||
|
@ -86,12 +86,6 @@ _client_config = configutil.ValidConfiguration(
|
|||||||
"shares.total",
|
"shares.total",
|
||||||
"storage.plugins",
|
"storage.plugins",
|
||||||
),
|
),
|
||||||
"ftpd": (
|
|
||||||
"accounts.file",
|
|
||||||
"accounts.url",
|
|
||||||
"enabled",
|
|
||||||
"port",
|
|
||||||
),
|
|
||||||
"storage": (
|
"storage": (
|
||||||
"debug_discard",
|
"debug_discard",
|
||||||
"enabled",
|
"enabled",
|
||||||
@ -656,7 +650,6 @@ class _Client(node.Node, pollmixin.PollMixin):
|
|||||||
raise ValueError("config error: helper is enabled, but tub "
|
raise ValueError("config error: helper is enabled, but tub "
|
||||||
"is not listening ('tub.port=' is empty)")
|
"is not listening ('tub.port=' is empty)")
|
||||||
self.init_helper()
|
self.init_helper()
|
||||||
self.init_ftp_server()
|
|
||||||
self.init_sftp_server()
|
self.init_sftp_server()
|
||||||
|
|
||||||
# If the node sees an exit_trigger file, it will poll every second to see
|
# If the node sees an exit_trigger file, it will poll every second to see
|
||||||
@ -1032,18 +1025,6 @@ class _Client(node.Node, pollmixin.PollMixin):
|
|||||||
)
|
)
|
||||||
ws.setServiceParent(self)
|
ws.setServiceParent(self)
|
||||||
|
|
||||||
def init_ftp_server(self):
|
|
||||||
if self.config.get_config("ftpd", "enabled", False, boolean=True):
|
|
||||||
accountfile = self.config.get_config("ftpd", "accounts.file", None)
|
|
||||||
if accountfile:
|
|
||||||
accountfile = self.config.get_config_path(accountfile)
|
|
||||||
accounturl = self.config.get_config("ftpd", "accounts.url", None)
|
|
||||||
ftp_portstr = self.config.get_config("ftpd", "port", "8021")
|
|
||||||
|
|
||||||
from allmydata.frontends import ftpd
|
|
||||||
s = ftpd.FTPServer(self, accountfile, accounturl, ftp_portstr)
|
|
||||||
s.setServiceParent(self)
|
|
||||||
|
|
||||||
def init_sftp_server(self):
|
def init_sftp_server(self):
|
||||||
if self.config.get_config("sftpd", "enabled", False, boolean=True):
|
if self.config.get_config("sftpd", "enabled", False, boolean=True):
|
||||||
accountfile = self.config.get_config("sftpd", "accounts.file", None)
|
accountfile = self.config.get_config("sftpd", "accounts.file", None)
|
||||||
|
@ -57,6 +57,10 @@ class CRSEncoder(object):
|
|||||||
|
|
||||||
return defer.succeed((shares, desired_share_ids))
|
return defer.succeed((shares, desired_share_ids))
|
||||||
|
|
||||||
|
def encode_proposal(self, data, desired_share_ids=None):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
|
||||||
@implementer(ICodecDecoder)
|
@implementer(ICodecDecoder)
|
||||||
class CRSDecoder(object):
|
class CRSDecoder(object):
|
||||||
|
|
||||||
|
@ -568,7 +568,7 @@ class DirectoryNode(object):
|
|||||||
d = self.get_child_and_metadata(childnamex)
|
d = self.get_child_and_metadata(childnamex)
|
||||||
return d
|
return d
|
||||||
|
|
||||||
def set_uri(self, namex, writecap, readcap, metadata=None, overwrite=True):
|
def set_uri(self, namex, writecap, readcap=None, metadata=None, overwrite=True):
|
||||||
precondition(isinstance(writecap, (bytes, type(None))), writecap)
|
precondition(isinstance(writecap, (bytes, type(None))), writecap)
|
||||||
precondition(isinstance(readcap, (bytes, type(None))), readcap)
|
precondition(isinstance(readcap, (bytes, type(None))), readcap)
|
||||||
|
|
||||||
|
@ -4,8 +4,8 @@ from zope.interface import implementer
|
|||||||
from twisted.web.client import getPage
|
from twisted.web.client import getPage
|
||||||
from twisted.internet import defer
|
from twisted.internet import defer
|
||||||
from twisted.cred import error, checkers, credentials
|
from twisted.cred import error, checkers, credentials
|
||||||
from twisted.conch import error as conch_error
|
|
||||||
from twisted.conch.ssh import keys
|
from twisted.conch.ssh import keys
|
||||||
|
from twisted.conch.checkers import SSHPublicKeyChecker, InMemorySSHKeyDB
|
||||||
|
|
||||||
from allmydata.util import base32
|
from allmydata.util import base32
|
||||||
from allmydata.util.fileutil import abspath_expanduser_unicode
|
from allmydata.util.fileutil import abspath_expanduser_unicode
|
||||||
@ -29,7 +29,7 @@ class AccountFileChecker(object):
|
|||||||
def __init__(self, client, accountfile):
|
def __init__(self, client, accountfile):
|
||||||
self.client = client
|
self.client = client
|
||||||
self.passwords = {}
|
self.passwords = {}
|
||||||
self.pubkeys = {}
|
pubkeys = {}
|
||||||
self.rootcaps = {}
|
self.rootcaps = {}
|
||||||
with open(abspath_expanduser_unicode(accountfile), "r") as f:
|
with open(abspath_expanduser_unicode(accountfile), "r") as f:
|
||||||
for line in f:
|
for line in f:
|
||||||
@ -40,12 +40,14 @@ class AccountFileChecker(object):
|
|||||||
if passwd.startswith("ssh-"):
|
if passwd.startswith("ssh-"):
|
||||||
bits = rest.split()
|
bits = rest.split()
|
||||||
keystring = " ".join([passwd] + bits[:-1])
|
keystring = " ".join([passwd] + bits[:-1])
|
||||||
|
key = keys.Key.fromString(keystring)
|
||||||
rootcap = bits[-1]
|
rootcap = bits[-1]
|
||||||
self.pubkeys[name] = keystring
|
pubkeys[name] = [key]
|
||||||
else:
|
else:
|
||||||
self.passwords[name] = passwd
|
self.passwords[name] = passwd
|
||||||
rootcap = rest
|
rootcap = rest
|
||||||
self.rootcaps[name] = rootcap
|
self.rootcaps[name] = rootcap
|
||||||
|
self._pubkeychecker = SSHPublicKeyChecker(InMemorySSHKeyDB(pubkeys))
|
||||||
|
|
||||||
def _avatarId(self, username):
|
def _avatarId(self, username):
|
||||||
return FTPAvatarID(username, self.rootcaps[username])
|
return FTPAvatarID(username, self.rootcaps[username])
|
||||||
@ -57,11 +59,9 @@ class AccountFileChecker(object):
|
|||||||
|
|
||||||
def requestAvatarId(self, creds):
|
def requestAvatarId(self, creds):
|
||||||
if credentials.ISSHPrivateKey.providedBy(creds):
|
if credentials.ISSHPrivateKey.providedBy(creds):
|
||||||
# Re-using twisted.conch.checkers.SSHPublicKeyChecker here, rather
|
d = defer.maybeDeferred(self._pubkeychecker.requestAvatarId, creds)
|
||||||
# than re-implementing all of the ISSHPrivateKey checking logic,
|
d.addCallback(self._avatarId)
|
||||||
# would be better. That would require Twisted 14.1.0 or newer,
|
return d
|
||||||
# though.
|
|
||||||
return self._checkKey(creds)
|
|
||||||
elif credentials.IUsernameHashedPassword.providedBy(creds):
|
elif credentials.IUsernameHashedPassword.providedBy(creds):
|
||||||
return self._checkPassword(creds)
|
return self._checkPassword(creds)
|
||||||
elif credentials.IUsernamePassword.providedBy(creds):
|
elif credentials.IUsernamePassword.providedBy(creds):
|
||||||
@ -86,28 +86,6 @@ class AccountFileChecker(object):
|
|||||||
d.addCallback(self._cbPasswordMatch, str(creds.username))
|
d.addCallback(self._cbPasswordMatch, str(creds.username))
|
||||||
return d
|
return d
|
||||||
|
|
||||||
def _checkKey(self, creds):
|
|
||||||
"""
|
|
||||||
Determine whether some key-based credentials correctly authenticates a
|
|
||||||
user.
|
|
||||||
|
|
||||||
Returns a Deferred that fires with the username if so or with an
|
|
||||||
UnauthorizedLogin failure otherwise.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Is the public key indicated by the given credentials allowed to
|
|
||||||
# authenticate the username in those credentials?
|
|
||||||
if creds.blob == self.pubkeys.get(creds.username):
|
|
||||||
if creds.signature is None:
|
|
||||||
return defer.fail(conch_error.ValidPublicKey())
|
|
||||||
|
|
||||||
# Is the signature in the given credentials the correct
|
|
||||||
# signature for the data in those credentials?
|
|
||||||
key = keys.Key.fromString(creds.blob)
|
|
||||||
if key.verify(creds.signature, creds.sigData):
|
|
||||||
return defer.succeed(self._avatarId(creds.username))
|
|
||||||
|
|
||||||
return defer.fail(error.UnauthorizedLogin())
|
|
||||||
|
|
||||||
@implementer(checkers.ICredentialsChecker)
|
@implementer(checkers.ICredentialsChecker)
|
||||||
class AccountURLChecker(object):
|
class AccountURLChecker(object):
|
||||||
|
@ -1,340 +0,0 @@
|
|||||||
from six import ensure_str
|
|
||||||
|
|
||||||
from types import NoneType
|
|
||||||
|
|
||||||
from zope.interface import implementer
|
|
||||||
from twisted.application import service, strports
|
|
||||||
from twisted.internet import defer
|
|
||||||
from twisted.internet.interfaces import IConsumer
|
|
||||||
from twisted.cred import portal
|
|
||||||
from twisted.python import filepath
|
|
||||||
from twisted.protocols import ftp
|
|
||||||
|
|
||||||
from allmydata.interfaces import IDirectoryNode, ExistingChildError, \
|
|
||||||
NoSuchChildError
|
|
||||||
from allmydata.immutable.upload import FileHandle
|
|
||||||
from allmydata.util.fileutil import EncryptedTemporaryFile
|
|
||||||
from allmydata.util.assertutil import precondition
|
|
||||||
|
|
||||||
@implementer(ftp.IReadFile)
|
|
||||||
class ReadFile(object):
|
|
||||||
def __init__(self, node):
|
|
||||||
self.node = node
|
|
||||||
def send(self, consumer):
|
|
||||||
d = self.node.read(consumer)
|
|
||||||
return d # when consumed
|
|
||||||
|
|
||||||
@implementer(IConsumer)
|
|
||||||
class FileWriter(object):
|
|
||||||
|
|
||||||
def registerProducer(self, producer, streaming):
|
|
||||||
if not streaming:
|
|
||||||
raise NotImplementedError("Non-streaming producer not supported.")
|
|
||||||
# we write the data to a temporary file, since Tahoe can't do
|
|
||||||
# streaming upload yet.
|
|
||||||
self.f = EncryptedTemporaryFile()
|
|
||||||
return None
|
|
||||||
|
|
||||||
def unregisterProducer(self):
|
|
||||||
# the upload actually happens in WriteFile.close()
|
|
||||||
pass
|
|
||||||
|
|
||||||
def write(self, data):
|
|
||||||
self.f.write(data)
|
|
||||||
|
|
||||||
@implementer(ftp.IWriteFile)
|
|
||||||
class WriteFile(object):
|
|
||||||
|
|
||||||
def __init__(self, parent, childname, convergence):
|
|
||||||
self.parent = parent
|
|
||||||
self.childname = childname
|
|
||||||
self.convergence = convergence
|
|
||||||
|
|
||||||
def receive(self):
|
|
||||||
self.c = FileWriter()
|
|
||||||
return defer.succeed(self.c)
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
u = FileHandle(self.c.f, self.convergence)
|
|
||||||
d = self.parent.add_file(self.childname, u)
|
|
||||||
return d
|
|
||||||
|
|
||||||
|
|
||||||
class NoParentError(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
# filepath.Permissions was added in Twisted-11.1.0, which we require. Twisted
|
|
||||||
# <15.0.0 expected an int, and only does '&' on it. Twisted >=15.0.0 expects
|
|
||||||
# a filepath.Permissions. This satisfies both.
|
|
||||||
|
|
||||||
class IntishPermissions(filepath.Permissions):
|
|
||||||
def __init__(self, statModeInt):
|
|
||||||
self._tahoe_statModeInt = statModeInt
|
|
||||||
filepath.Permissions.__init__(self, statModeInt)
|
|
||||||
def __and__(self, other):
|
|
||||||
return self._tahoe_statModeInt & other
|
|
||||||
|
|
||||||
@implementer(ftp.IFTPShell)
|
|
||||||
class Handler(object):
|
|
||||||
def __init__(self, client, rootnode, username, convergence):
|
|
||||||
self.client = client
|
|
||||||
self.root = rootnode
|
|
||||||
self.username = username
|
|
||||||
self.convergence = convergence
|
|
||||||
|
|
||||||
def makeDirectory(self, path):
|
|
||||||
d = self._get_root(path)
|
|
||||||
d.addCallback(lambda root_and_path:
|
|
||||||
self._get_or_create_directories(root_and_path[0], root_and_path[1]))
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _get_or_create_directories(self, node, path):
|
|
||||||
if not IDirectoryNode.providedBy(node):
|
|
||||||
# unfortunately it is too late to provide the name of the
|
|
||||||
# blocking directory in the error message.
|
|
||||||
raise ftp.FileExistsError("cannot create directory because there "
|
|
||||||
"is a file in the way")
|
|
||||||
if not path:
|
|
||||||
return defer.succeed(node)
|
|
||||||
d = node.get(path[0])
|
|
||||||
def _maybe_create(f):
|
|
||||||
f.trap(NoSuchChildError)
|
|
||||||
return node.create_subdirectory(path[0])
|
|
||||||
d.addErrback(_maybe_create)
|
|
||||||
d.addCallback(self._get_or_create_directories, path[1:])
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _get_parent(self, path):
|
|
||||||
# fire with (parentnode, childname)
|
|
||||||
path = [unicode(p) for p in path]
|
|
||||||
if not path:
|
|
||||||
raise NoParentError
|
|
||||||
childname = path[-1]
|
|
||||||
d = self._get_root(path)
|
|
||||||
def _got_root(root_and_path):
|
|
||||||
(root, path) = root_and_path
|
|
||||||
if not path:
|
|
||||||
raise NoParentError
|
|
||||||
return root.get_child_at_path(path[:-1])
|
|
||||||
d.addCallback(_got_root)
|
|
||||||
def _got_parent(parent):
|
|
||||||
return (parent, childname)
|
|
||||||
d.addCallback(_got_parent)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _remove_thing(self, path, must_be_directory=False, must_be_file=False):
|
|
||||||
d = defer.maybeDeferred(self._get_parent, path)
|
|
||||||
def _convert_error(f):
|
|
||||||
f.trap(NoParentError)
|
|
||||||
raise ftp.PermissionDeniedError("cannot delete root directory")
|
|
||||||
d.addErrback(_convert_error)
|
|
||||||
def _got_parent(parent_and_childname):
|
|
||||||
(parent, childname) = parent_and_childname
|
|
||||||
d = parent.get(childname)
|
|
||||||
def _got_child(child):
|
|
||||||
if must_be_directory and not IDirectoryNode.providedBy(child):
|
|
||||||
raise ftp.IsNotADirectoryError("rmdir called on a file")
|
|
||||||
if must_be_file and IDirectoryNode.providedBy(child):
|
|
||||||
raise ftp.IsADirectoryError("rmfile called on a directory")
|
|
||||||
return parent.delete(childname)
|
|
||||||
d.addCallback(_got_child)
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
return d
|
|
||||||
d.addCallback(_got_parent)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def removeDirectory(self, path):
|
|
||||||
return self._remove_thing(path, must_be_directory=True)
|
|
||||||
|
|
||||||
def removeFile(self, path):
|
|
||||||
return self._remove_thing(path, must_be_file=True)
|
|
||||||
|
|
||||||
def rename(self, fromPath, toPath):
|
|
||||||
# the target directory must already exist
|
|
||||||
d = self._get_parent(fromPath)
|
|
||||||
def _got_from_parent(fromparent_and_childname):
|
|
||||||
(fromparent, childname) = fromparent_and_childname
|
|
||||||
d = self._get_parent(toPath)
|
|
||||||
d.addCallback(lambda toparent_and_tochildname:
|
|
||||||
fromparent.move_child_to(childname,
|
|
||||||
toparent_and_tochildname[0], toparent_and_tochildname[1],
|
|
||||||
overwrite=False))
|
|
||||||
return d
|
|
||||||
d.addCallback(_got_from_parent)
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def access(self, path):
|
|
||||||
# we allow access to everything that exists. We are required to raise
|
|
||||||
# an error for paths that don't exist: FTP clients (at least ncftp)
|
|
||||||
# uses this to decide whether to mkdir or not.
|
|
||||||
d = self._get_node_and_metadata_for_path(path)
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
d.addCallback(lambda res: None)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _convert_error(self, f):
|
|
||||||
if f.check(NoSuchChildError):
|
|
||||||
childname = f.value.args[0].encode("utf-8")
|
|
||||||
msg = "'%s' doesn't exist" % childname
|
|
||||||
raise ftp.FileNotFoundError(msg)
|
|
||||||
if f.check(ExistingChildError):
|
|
||||||
msg = f.value.args[0].encode("utf-8")
|
|
||||||
raise ftp.FileExistsError(msg)
|
|
||||||
return f
|
|
||||||
|
|
||||||
def _get_root(self, path):
|
|
||||||
# return (root, remaining_path)
|
|
||||||
path = [unicode(p) for p in path]
|
|
||||||
if path and path[0] == "uri":
|
|
||||||
d = defer.maybeDeferred(self.client.create_node_from_uri,
|
|
||||||
str(path[1]))
|
|
||||||
d.addCallback(lambda root: (root, path[2:]))
|
|
||||||
else:
|
|
||||||
d = defer.succeed((self.root,path))
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _get_node_and_metadata_for_path(self, path):
|
|
||||||
d = self._get_root(path)
|
|
||||||
def _got_root(root_and_path):
|
|
||||||
(root,path) = root_and_path
|
|
||||||
if path:
|
|
||||||
return root.get_child_and_metadata_at_path(path)
|
|
||||||
else:
|
|
||||||
return (root,{})
|
|
||||||
d.addCallback(_got_root)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _populate_row(self, keys, childnode_and_metadata):
|
|
||||||
(childnode, metadata) = childnode_and_metadata
|
|
||||||
values = []
|
|
||||||
isdir = bool(IDirectoryNode.providedBy(childnode))
|
|
||||||
for key in keys:
|
|
||||||
if key == "size":
|
|
||||||
if isdir:
|
|
||||||
value = 0
|
|
||||||
else:
|
|
||||||
value = childnode.get_size() or 0
|
|
||||||
elif key == "directory":
|
|
||||||
value = isdir
|
|
||||||
elif key == "permissions":
|
|
||||||
# Twisted-14.0.2 (and earlier) expected an int, and used it
|
|
||||||
# in a rendering function that did (mode & NUMBER).
|
|
||||||
# Twisted-15.0.0 expects a
|
|
||||||
# twisted.python.filepath.Permissions , and calls its
|
|
||||||
# .shorthand() method. This provides both.
|
|
||||||
value = IntishPermissions(0o600)
|
|
||||||
elif key == "hardlinks":
|
|
||||||
value = 1
|
|
||||||
elif key == "modified":
|
|
||||||
# follow sftpd convention (i.e. linkmotime in preference to mtime)
|
|
||||||
if "linkmotime" in metadata.get("tahoe", {}):
|
|
||||||
value = metadata["tahoe"]["linkmotime"]
|
|
||||||
else:
|
|
||||||
value = metadata.get("mtime", 0)
|
|
||||||
elif key == "owner":
|
|
||||||
value = self.username
|
|
||||||
elif key == "group":
|
|
||||||
value = self.username
|
|
||||||
else:
|
|
||||||
value = "??"
|
|
||||||
values.append(value)
|
|
||||||
return values
|
|
||||||
|
|
||||||
def stat(self, path, keys=()):
|
|
||||||
# for files only, I think
|
|
||||||
d = self._get_node_and_metadata_for_path(path)
|
|
||||||
def _render(node_and_metadata):
|
|
||||||
(node, metadata) = node_and_metadata
|
|
||||||
assert not IDirectoryNode.providedBy(node)
|
|
||||||
return self._populate_row(keys, (node,metadata))
|
|
||||||
d.addCallback(_render)
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def list(self, path, keys=()):
|
|
||||||
# the interface claims that path is a list of unicodes, but in
|
|
||||||
# practice it is not
|
|
||||||
d = self._get_node_and_metadata_for_path(path)
|
|
||||||
def _list(node_and_metadata):
|
|
||||||
(node, metadata) = node_and_metadata
|
|
||||||
if IDirectoryNode.providedBy(node):
|
|
||||||
return node.list()
|
|
||||||
return { path[-1]: (node, metadata) } # need last-edge metadata
|
|
||||||
d.addCallback(_list)
|
|
||||||
def _render(children):
|
|
||||||
results = []
|
|
||||||
for (name, childnode) in children.iteritems():
|
|
||||||
# the interface claims that the result should have a unicode
|
|
||||||
# object as the name, but it fails unless you give it a
|
|
||||||
# bytestring
|
|
||||||
results.append( (name.encode("utf-8"),
|
|
||||||
self._populate_row(keys, childnode) ) )
|
|
||||||
return results
|
|
||||||
d.addCallback(_render)
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def openForReading(self, path):
|
|
||||||
d = self._get_node_and_metadata_for_path(path)
|
|
||||||
d.addCallback(lambda node_and_metadata: ReadFile(node_and_metadata[0]))
|
|
||||||
d.addErrback(self._convert_error)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def openForWriting(self, path):
|
|
||||||
path = [unicode(p) for p in path]
|
|
||||||
if not path:
|
|
||||||
raise ftp.PermissionDeniedError("cannot STOR to root directory")
|
|
||||||
childname = path[-1]
|
|
||||||
d = self._get_root(path)
|
|
||||||
def _got_root(root_and_path):
|
|
||||||
(root, path) = root_and_path
|
|
||||||
if not path:
|
|
||||||
raise ftp.PermissionDeniedError("cannot STOR to root directory")
|
|
||||||
return root.get_child_at_path(path[:-1])
|
|
||||||
d.addCallback(_got_root)
|
|
||||||
def _got_parent(parent):
|
|
||||||
return WriteFile(parent, childname, self.convergence)
|
|
||||||
d.addCallback(_got_parent)
|
|
||||||
return d
|
|
||||||
|
|
||||||
from allmydata.frontends.auth import AccountURLChecker, AccountFileChecker, NeedRootcapLookupScheme
|
|
||||||
|
|
||||||
|
|
||||||
@implementer(portal.IRealm)
|
|
||||||
class Dispatcher(object):
|
|
||||||
def __init__(self, client):
|
|
||||||
self.client = client
|
|
||||||
|
|
||||||
def requestAvatar(self, avatarID, mind, interface):
|
|
||||||
assert interface == ftp.IFTPShell
|
|
||||||
rootnode = self.client.create_node_from_uri(avatarID.rootcap)
|
|
||||||
convergence = self.client.convergence
|
|
||||||
s = Handler(self.client, rootnode, avatarID.username, convergence)
|
|
||||||
def logout(): pass
|
|
||||||
return (interface, s, None)
|
|
||||||
|
|
||||||
|
|
||||||
class FTPServer(service.MultiService):
|
|
||||||
def __init__(self, client, accountfile, accounturl, ftp_portstr):
|
|
||||||
precondition(isinstance(accountfile, (unicode, NoneType)), accountfile)
|
|
||||||
service.MultiService.__init__(self)
|
|
||||||
|
|
||||||
r = Dispatcher(client)
|
|
||||||
p = portal.Portal(r)
|
|
||||||
|
|
||||||
if accountfile:
|
|
||||||
c = AccountFileChecker(self, accountfile)
|
|
||||||
p.registerChecker(c)
|
|
||||||
if accounturl:
|
|
||||||
c = AccountURLChecker(self, accounturl)
|
|
||||||
p.registerChecker(c)
|
|
||||||
if not accountfile and not accounturl:
|
|
||||||
# we could leave this anonymous, with just the /uri/CAP form
|
|
||||||
raise NeedRootcapLookupScheme("must provide some translation")
|
|
||||||
|
|
||||||
f = ftp.FTPFactory(p)
|
|
||||||
# strports requires a native string.
|
|
||||||
ftp_portstr = ensure_str(ftp_portstr)
|
|
||||||
s = strports.service(ftp_portstr, f)
|
|
||||||
s.setServiceParent(self)
|
|
@ -1,6 +1,17 @@
|
|||||||
|
"""
|
||||||
|
Ported to Python 3.
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
import six
|
import six
|
||||||
import heapq, traceback, array, stat, struct
|
import heapq, traceback, stat, struct
|
||||||
from types import NoneType
|
|
||||||
from stat import S_IFREG, S_IFDIR
|
from stat import S_IFREG, S_IFDIR
|
||||||
from time import time, strftime, localtime
|
from time import time, strftime, localtime
|
||||||
|
|
||||||
@ -45,6 +56,17 @@ from allmydata.util.log import NOISY, OPERATIONAL, WEIRD, \
|
|||||||
if six.PY3:
|
if six.PY3:
|
||||||
long = int
|
long = int
|
||||||
|
|
||||||
|
|
||||||
|
def createSFTPError(errorCode, errorMessage):
|
||||||
|
"""
|
||||||
|
SFTPError that can accept both Unicode and bytes.
|
||||||
|
|
||||||
|
Twisted expects _native_ strings for the SFTPError message, but we often do
|
||||||
|
Unicode by default even on Python 2.
|
||||||
|
"""
|
||||||
|
return SFTPError(errorCode, six.ensure_str(errorMessage))
|
||||||
|
|
||||||
|
|
||||||
def eventually_callback(d):
|
def eventually_callback(d):
|
||||||
return lambda res: eventually(d.callback, res)
|
return lambda res: eventually(d.callback, res)
|
||||||
|
|
||||||
@ -53,9 +75,9 @@ def eventually_errback(d):
|
|||||||
|
|
||||||
|
|
||||||
def _utf8(x):
|
def _utf8(x):
|
||||||
if isinstance(x, unicode):
|
|
||||||
return x.encode('utf-8')
|
|
||||||
if isinstance(x, str):
|
if isinstance(x, str):
|
||||||
|
return x.encode('utf-8')
|
||||||
|
if isinstance(x, bytes):
|
||||||
return x
|
return x
|
||||||
return repr(x)
|
return repr(x)
|
||||||
|
|
||||||
@ -64,7 +86,7 @@ def _to_sftp_time(t):
|
|||||||
"""SFTP times are unsigned 32-bit integers representing UTC seconds
|
"""SFTP times are unsigned 32-bit integers representing UTC seconds
|
||||||
(ignoring leap seconds) since the Unix epoch, January 1 1970 00:00 UTC.
|
(ignoring leap seconds) since the Unix epoch, January 1 1970 00:00 UTC.
|
||||||
A Tahoe time is the corresponding float."""
|
A Tahoe time is the corresponding float."""
|
||||||
return long(t) & long(0xFFFFFFFF)
|
return int(t) & int(0xFFFFFFFF)
|
||||||
|
|
||||||
|
|
||||||
def _convert_error(res, request):
|
def _convert_error(res, request):
|
||||||
@ -73,7 +95,7 @@ def _convert_error(res, request):
|
|||||||
|
|
||||||
if not isinstance(res, Failure):
|
if not isinstance(res, Failure):
|
||||||
logged_res = res
|
logged_res = res
|
||||||
if isinstance(res, str): logged_res = "<data of length %r>" % (len(res),)
|
if isinstance(res, (bytes, str)): logged_res = "<data of length %r>" % (len(res),)
|
||||||
logmsg("SUCCESS %r %r" % (request, logged_res,), level=OPERATIONAL)
|
logmsg("SUCCESS %r %r" % (request, logged_res,), level=OPERATIONAL)
|
||||||
return res
|
return res
|
||||||
|
|
||||||
@ -92,10 +114,10 @@ def _convert_error(res, request):
|
|||||||
raise err
|
raise err
|
||||||
if err.check(NoSuchChildError):
|
if err.check(NoSuchChildError):
|
||||||
childname = _utf8(err.value.args[0])
|
childname = _utf8(err.value.args[0])
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, childname)
|
raise createSFTPError(FX_NO_SUCH_FILE, childname)
|
||||||
if err.check(NotWriteableError) or err.check(ChildOfWrongTypeError):
|
if err.check(NotWriteableError) or err.check(ChildOfWrongTypeError):
|
||||||
msg = _utf8(err.value.args[0])
|
msg = _utf8(err.value.args[0])
|
||||||
raise SFTPError(FX_PERMISSION_DENIED, msg)
|
raise createSFTPError(FX_PERMISSION_DENIED, msg)
|
||||||
if err.check(ExistingChildError):
|
if err.check(ExistingChildError):
|
||||||
# Versions of SFTP after v3 (which is what twisted.conch implements)
|
# Versions of SFTP after v3 (which is what twisted.conch implements)
|
||||||
# define a specific error code for this case: FX_FILE_ALREADY_EXISTS.
|
# define a specific error code for this case: FX_FILE_ALREADY_EXISTS.
|
||||||
@ -104,16 +126,16 @@ def _convert_error(res, request):
|
|||||||
# to translate the error to the equivalent of POSIX EEXIST, which is
|
# to translate the error to the equivalent of POSIX EEXIST, which is
|
||||||
# necessary for some picky programs (such as gedit).
|
# necessary for some picky programs (such as gedit).
|
||||||
msg = _utf8(err.value.args[0])
|
msg = _utf8(err.value.args[0])
|
||||||
raise SFTPError(FX_FAILURE, msg)
|
raise createSFTPError(FX_FAILURE, msg)
|
||||||
if err.check(NotImplementedError):
|
if err.check(NotImplementedError):
|
||||||
raise SFTPError(FX_OP_UNSUPPORTED, _utf8(err.value))
|
raise createSFTPError(FX_OP_UNSUPPORTED, _utf8(err.value))
|
||||||
if err.check(EOFError):
|
if err.check(EOFError):
|
||||||
raise SFTPError(FX_EOF, "end of file reached")
|
raise createSFTPError(FX_EOF, "end of file reached")
|
||||||
if err.check(defer.FirstError):
|
if err.check(defer.FirstError):
|
||||||
_convert_error(err.value.subFailure, request)
|
_convert_error(err.value.subFailure, request)
|
||||||
|
|
||||||
# We assume that the error message is not anonymity-sensitive.
|
# We assume that the error message is not anonymity-sensitive.
|
||||||
raise SFTPError(FX_FAILURE, _utf8(err.value))
|
raise createSFTPError(FX_FAILURE, _utf8(err.value))
|
||||||
|
|
||||||
|
|
||||||
def _repr_flags(flags):
|
def _repr_flags(flags):
|
||||||
@ -146,7 +168,7 @@ def _lsLine(name, attrs):
|
|||||||
# Since we now depend on Twisted v10.1, consider calling Twisted's version.
|
# Since we now depend on Twisted v10.1, consider calling Twisted's version.
|
||||||
|
|
||||||
mode = st_mode
|
mode = st_mode
|
||||||
perms = array.array('c', '-'*10)
|
perms = ["-"] * 10
|
||||||
ft = stat.S_IFMT(mode)
|
ft = stat.S_IFMT(mode)
|
||||||
if stat.S_ISDIR(ft): perms[0] = 'd'
|
if stat.S_ISDIR(ft): perms[0] = 'd'
|
||||||
elif stat.S_ISREG(ft): perms[0] = '-'
|
elif stat.S_ISREG(ft): perms[0] = '-'
|
||||||
@ -165,7 +187,7 @@ def _lsLine(name, attrs):
|
|||||||
if mode&stat.S_IXOTH: perms[9] = 'x'
|
if mode&stat.S_IXOTH: perms[9] = 'x'
|
||||||
# suid/sgid never set
|
# suid/sgid never set
|
||||||
|
|
||||||
l = perms.tostring()
|
l = "".join(perms)
|
||||||
l += str(st_nlink).rjust(5) + ' '
|
l += str(st_nlink).rjust(5) + ' '
|
||||||
un = str(st_uid)
|
un = str(st_uid)
|
||||||
l += un.ljust(9)
|
l += un.ljust(9)
|
||||||
@ -182,6 +204,7 @@ def _lsLine(name, attrs):
|
|||||||
l += strftime("%b %d %Y ", localtime(st_mtime))
|
l += strftime("%b %d %Y ", localtime(st_mtime))
|
||||||
else:
|
else:
|
||||||
l += strftime("%b %d %H:%M ", localtime(st_mtime))
|
l += strftime("%b %d %H:%M ", localtime(st_mtime))
|
||||||
|
l = l.encode("utf-8")
|
||||||
l += name
|
l += name
|
||||||
return l
|
return l
|
||||||
|
|
||||||
@ -223,7 +246,7 @@ def _populate_attrs(childnode, metadata, size=None):
|
|||||||
if childnode and size is None:
|
if childnode and size is None:
|
||||||
size = childnode.get_size()
|
size = childnode.get_size()
|
||||||
if size is not None:
|
if size is not None:
|
||||||
_assert(isinstance(size, (int, long)) and not isinstance(size, bool), size=size)
|
_assert(isinstance(size, int) and not isinstance(size, bool), size=size)
|
||||||
attrs['size'] = size
|
attrs['size'] = size
|
||||||
perms = S_IFREG | 0o666
|
perms = S_IFREG | 0o666
|
||||||
|
|
||||||
@ -255,7 +278,7 @@ def _attrs_to_metadata(attrs):
|
|||||||
|
|
||||||
for key in attrs:
|
for key in attrs:
|
||||||
if key == "mtime" or key == "ctime" or key == "createtime":
|
if key == "mtime" or key == "ctime" or key == "createtime":
|
||||||
metadata[key] = long(attrs[key])
|
metadata[key] = int(attrs[key])
|
||||||
elif key.startswith("ext_"):
|
elif key.startswith("ext_"):
|
||||||
metadata[key] = str(attrs[key])
|
metadata[key] = str(attrs[key])
|
||||||
|
|
||||||
@ -267,7 +290,7 @@ def _attrs_to_metadata(attrs):
|
|||||||
|
|
||||||
|
|
||||||
def _direntry_for(filenode_or_parent, childname, filenode=None):
|
def _direntry_for(filenode_or_parent, childname, filenode=None):
|
||||||
precondition(isinstance(childname, (unicode, NoneType)), childname=childname)
|
precondition(isinstance(childname, (str, type(None))), childname=childname)
|
||||||
|
|
||||||
if childname is None:
|
if childname is None:
|
||||||
filenode_or_parent = filenode
|
filenode_or_parent = filenode
|
||||||
@ -275,7 +298,7 @@ def _direntry_for(filenode_or_parent, childname, filenode=None):
|
|||||||
if filenode_or_parent:
|
if filenode_or_parent:
|
||||||
rw_uri = filenode_or_parent.get_write_uri()
|
rw_uri = filenode_or_parent.get_write_uri()
|
||||||
if rw_uri and childname:
|
if rw_uri and childname:
|
||||||
return rw_uri + "/" + childname.encode('utf-8')
|
return rw_uri + b"/" + childname.encode('utf-8')
|
||||||
else:
|
else:
|
||||||
return rw_uri
|
return rw_uri
|
||||||
|
|
||||||
@ -327,7 +350,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
if size < self.current_size or size < self.downloaded:
|
if size < self.current_size or size < self.downloaded:
|
||||||
self.f.truncate(size)
|
self.f.truncate(size)
|
||||||
if size > self.current_size:
|
if size > self.current_size:
|
||||||
self.overwrite(self.current_size, "\x00" * (size - self.current_size))
|
self.overwrite(self.current_size, b"\x00" * (size - self.current_size))
|
||||||
self.current_size = size
|
self.current_size = size
|
||||||
|
|
||||||
# make the invariant self.download_size <= self.current_size be true again
|
# make the invariant self.download_size <= self.current_size be true again
|
||||||
@ -335,7 +358,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
self.download_size = size
|
self.download_size = size
|
||||||
|
|
||||||
if self.downloaded >= self.download_size:
|
if self.downloaded >= self.download_size:
|
||||||
self.download_done("size changed")
|
self.download_done(b"size changed")
|
||||||
|
|
||||||
def registerProducer(self, p, streaming):
|
def registerProducer(self, p, streaming):
|
||||||
if noisy: self.log(".registerProducer(%r, streaming=%r)" % (p, streaming), level=NOISY)
|
if noisy: self.log(".registerProducer(%r, streaming=%r)" % (p, streaming), level=NOISY)
|
||||||
@ -410,21 +433,21 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
milestone = end
|
milestone = end
|
||||||
|
|
||||||
while len(self.milestones) > 0:
|
while len(self.milestones) > 0:
|
||||||
(next, d) = self.milestones[0]
|
(next_, d) = self.milestones[0]
|
||||||
if next > milestone:
|
if next_ > milestone:
|
||||||
return
|
return
|
||||||
if noisy: self.log("MILESTONE %r %r" % (next, d), level=NOISY)
|
if noisy: self.log("MILESTONE %r %r" % (next_, d), level=NOISY)
|
||||||
heapq.heappop(self.milestones)
|
heapq.heappop(self.milestones)
|
||||||
eventually_callback(d)("reached")
|
eventually_callback(d)(b"reached")
|
||||||
|
|
||||||
if milestone >= self.download_size:
|
if milestone >= self.download_size:
|
||||||
self.download_done("reached download size")
|
self.download_done(b"reached download size")
|
||||||
|
|
||||||
def overwrite(self, offset, data):
|
def overwrite(self, offset, data):
|
||||||
if noisy: self.log(".overwrite(%r, <data of length %r>)" % (offset, len(data)), level=NOISY)
|
if noisy: self.log(".overwrite(%r, <data of length %r>)" % (offset, len(data)), level=NOISY)
|
||||||
if self.is_closed:
|
if self.is_closed:
|
||||||
self.log("overwrite called on a closed OverwriteableFileConsumer", level=WEIRD)
|
self.log("overwrite called on a closed OverwriteableFileConsumer", level=WEIRD)
|
||||||
raise SFTPError(FX_BAD_MESSAGE, "cannot write to a closed file handle")
|
raise createSFTPError(FX_BAD_MESSAGE, "cannot write to a closed file handle")
|
||||||
|
|
||||||
if offset > self.current_size:
|
if offset > self.current_size:
|
||||||
# Normally writing at an offset beyond the current end-of-file
|
# Normally writing at an offset beyond the current end-of-file
|
||||||
@ -435,7 +458,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
# the gap between the current EOF and the offset.
|
# the gap between the current EOF and the offset.
|
||||||
|
|
||||||
self.f.seek(self.current_size)
|
self.f.seek(self.current_size)
|
||||||
self.f.write("\x00" * (offset - self.current_size))
|
self.f.write(b"\x00" * (offset - self.current_size))
|
||||||
start = self.current_size
|
start = self.current_size
|
||||||
else:
|
else:
|
||||||
self.f.seek(offset)
|
self.f.seek(offset)
|
||||||
@ -455,7 +478,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
if noisy: self.log(".read(%r, %r), current_size = %r" % (offset, length, self.current_size), level=NOISY)
|
if noisy: self.log(".read(%r, %r), current_size = %r" % (offset, length, self.current_size), level=NOISY)
|
||||||
if self.is_closed:
|
if self.is_closed:
|
||||||
self.log("read called on a closed OverwriteableFileConsumer", level=WEIRD)
|
self.log("read called on a closed OverwriteableFileConsumer", level=WEIRD)
|
||||||
raise SFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
raise createSFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
||||||
|
|
||||||
# Note that the overwrite method is synchronous. When a write request is processed
|
# Note that the overwrite method is synchronous. When a write request is processed
|
||||||
# (e.g. a writeChunk request on the async queue of GeneralSFTPFile), overwrite will
|
# (e.g. a writeChunk request on the async queue of GeneralSFTPFile), overwrite will
|
||||||
@ -509,7 +532,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
return d
|
return d
|
||||||
|
|
||||||
def download_done(self, res):
|
def download_done(self, res):
|
||||||
_assert(isinstance(res, (str, Failure)), res=res)
|
_assert(isinstance(res, (bytes, Failure)), res=res)
|
||||||
# Only the first call to download_done counts, but we log subsequent calls
|
# Only the first call to download_done counts, but we log subsequent calls
|
||||||
# (multiple calls are normal).
|
# (multiple calls are normal).
|
||||||
if self.done_status is not None:
|
if self.done_status is not None:
|
||||||
@ -526,8 +549,8 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
eventually_callback(self.done)(None)
|
eventually_callback(self.done)(None)
|
||||||
|
|
||||||
while len(self.milestones) > 0:
|
while len(self.milestones) > 0:
|
||||||
(next, d) = self.milestones[0]
|
(next_, d) = self.milestones[0]
|
||||||
if noisy: self.log("MILESTONE FINISH %r %r %r" % (next, d, res), level=NOISY)
|
if noisy: self.log("MILESTONE FINISH %r %r %r" % (next_, d, res), level=NOISY)
|
||||||
heapq.heappop(self.milestones)
|
heapq.heappop(self.milestones)
|
||||||
# The callback means that the milestone has been reached if
|
# The callback means that the milestone has been reached if
|
||||||
# it is ever going to be. Note that the file may have been
|
# it is ever going to be. Note that the file may have been
|
||||||
@ -541,7 +564,7 @@ class OverwriteableFileConsumer(PrefixingLogMixin):
|
|||||||
self.f.close()
|
self.f.close()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.log("suppressed %r from close of temporary file %r" % (e, self.f), level=WEIRD)
|
self.log("suppressed %r from close of temporary file %r" % (e, self.f), level=WEIRD)
|
||||||
self.download_done("closed")
|
self.download_done(b"closed")
|
||||||
return self.done_status
|
return self.done_status
|
||||||
|
|
||||||
def unregisterProducer(self):
|
def unregisterProducer(self):
|
||||||
@ -565,7 +588,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
PrefixingLogMixin.__init__(self, facility="tahoe.sftp", prefix=userpath)
|
PrefixingLogMixin.__init__(self, facility="tahoe.sftp", prefix=userpath)
|
||||||
if noisy: self.log(".__init__(%r, %r, %r)" % (userpath, filenode, metadata), level=NOISY)
|
if noisy: self.log(".__init__(%r, %r, %r)" % (userpath, filenode, metadata), level=NOISY)
|
||||||
|
|
||||||
precondition(isinstance(userpath, str) and IFileNode.providedBy(filenode),
|
precondition(isinstance(userpath, bytes) and IFileNode.providedBy(filenode),
|
||||||
userpath=userpath, filenode=filenode)
|
userpath=userpath, filenode=filenode)
|
||||||
self.filenode = filenode
|
self.filenode = filenode
|
||||||
self.metadata = metadata
|
self.metadata = metadata
|
||||||
@ -577,7 +600,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
d = defer.Deferred()
|
d = defer.Deferred()
|
||||||
@ -594,7 +617,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
# i.e. we respond with an EOF error iff offset is already at EOF.
|
# i.e. we respond with an EOF error iff offset is already at EOF.
|
||||||
|
|
||||||
if offset >= len(data):
|
if offset >= len(data):
|
||||||
eventually_errback(d)(Failure(SFTPError(FX_EOF, "read at or past end of file")))
|
eventually_errback(d)(Failure(createSFTPError(FX_EOF, "read at or past end of file")))
|
||||||
else:
|
else:
|
||||||
eventually_callback(d)(data[offset:offset+length]) # truncated if offset+length > len(data)
|
eventually_callback(d)(data[offset:offset+length]) # truncated if offset+length > len(data)
|
||||||
return data
|
return data
|
||||||
@ -605,7 +628,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
def writeChunk(self, offset, data):
|
def writeChunk(self, offset, data):
|
||||||
self.log(".writeChunk(%r, <data of length %r>) denied" % (offset, len(data)), level=OPERATIONAL)
|
self.log(".writeChunk(%r, <data of length %r>) denied" % (offset, len(data)), level=OPERATIONAL)
|
||||||
|
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
@ -619,7 +642,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot get attributes for a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot get attributes for a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
d = defer.execute(_populate_attrs, self.filenode, self.metadata)
|
d = defer.execute(_populate_attrs, self.filenode, self.metadata)
|
||||||
@ -628,7 +651,7 @@ class ShortReadOnlySFTPFile(PrefixingLogMixin):
|
|||||||
|
|
||||||
def setAttrs(self, attrs):
|
def setAttrs(self, attrs):
|
||||||
self.log(".setAttrs(%r) denied" % (attrs,), level=OPERATIONAL)
|
self.log(".setAttrs(%r) denied" % (attrs,), level=OPERATIONAL)
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
|
|
||||||
@ -649,7 +672,7 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
if noisy: self.log(".__init__(%r, %r = %r, %r, <convergence censored>)" %
|
if noisy: self.log(".__init__(%r, %r = %r, %r, <convergence censored>)" %
|
||||||
(userpath, flags, _repr_flags(flags), close_notify), level=NOISY)
|
(userpath, flags, _repr_flags(flags), close_notify), level=NOISY)
|
||||||
|
|
||||||
precondition(isinstance(userpath, str), userpath=userpath)
|
precondition(isinstance(userpath, bytes), userpath=userpath)
|
||||||
self.userpath = userpath
|
self.userpath = userpath
|
||||||
self.flags = flags
|
self.flags = flags
|
||||||
self.close_notify = close_notify
|
self.close_notify = close_notify
|
||||||
@ -668,11 +691,11 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
# not be set before then.
|
# not be set before then.
|
||||||
self.consumer = None
|
self.consumer = None
|
||||||
|
|
||||||
def open(self, parent=None, childname=None, filenode=None, metadata=None):
|
def open(self, parent=None, childname=None, filenode=None, metadata=None): # noqa: F811
|
||||||
self.log(".open(parent=%r, childname=%r, filenode=%r, metadata=%r)" %
|
self.log(".open(parent=%r, childname=%r, filenode=%r, metadata=%r)" %
|
||||||
(parent, childname, filenode, metadata), level=OPERATIONAL)
|
(parent, childname, filenode, metadata), level=OPERATIONAL)
|
||||||
|
|
||||||
precondition(isinstance(childname, (unicode, NoneType)), childname=childname)
|
precondition(isinstance(childname, (str, type(None))), childname=childname)
|
||||||
precondition(filenode is None or IFileNode.providedBy(filenode), filenode=filenode)
|
precondition(filenode is None or IFileNode.providedBy(filenode), filenode=filenode)
|
||||||
precondition(not self.closed, sftpfile=self)
|
precondition(not self.closed, sftpfile=self)
|
||||||
|
|
||||||
@ -689,7 +712,7 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
if (self.flags & FXF_TRUNC) or not filenode:
|
if (self.flags & FXF_TRUNC) or not filenode:
|
||||||
# We're either truncating or creating the file, so we don't need the old contents.
|
# We're either truncating or creating the file, so we don't need the old contents.
|
||||||
self.consumer = OverwriteableFileConsumer(0, tempfile_maker)
|
self.consumer = OverwriteableFileConsumer(0, tempfile_maker)
|
||||||
self.consumer.download_done("download not needed")
|
self.consumer.download_done(b"download not needed")
|
||||||
else:
|
else:
|
||||||
self.async_.addCallback(lambda ignored: filenode.get_best_readable_version())
|
self.async_.addCallback(lambda ignored: filenode.get_best_readable_version())
|
||||||
|
|
||||||
@ -703,7 +726,7 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
d = version.read(self.consumer, 0, None)
|
d = version.read(self.consumer, 0, None)
|
||||||
def _finished(res):
|
def _finished(res):
|
||||||
if not isinstance(res, Failure):
|
if not isinstance(res, Failure):
|
||||||
res = "download finished"
|
res = b"download finished"
|
||||||
self.consumer.download_done(res)
|
self.consumer.download_done(res)
|
||||||
d.addBoth(_finished)
|
d.addBoth(_finished)
|
||||||
# It is correct to drop d here.
|
# It is correct to drop d here.
|
||||||
@ -723,7 +746,7 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
def rename(self, new_userpath, new_parent, new_childname):
|
def rename(self, new_userpath, new_parent, new_childname):
|
||||||
self.log(".rename(%r, %r, %r)" % (new_userpath, new_parent, new_childname), level=OPERATIONAL)
|
self.log(".rename(%r, %r, %r)" % (new_userpath, new_parent, new_childname), level=OPERATIONAL)
|
||||||
|
|
||||||
precondition(isinstance(new_userpath, str) and isinstance(new_childname, unicode),
|
precondition(isinstance(new_userpath, bytes) and isinstance(new_childname, str),
|
||||||
new_userpath=new_userpath, new_childname=new_childname)
|
new_userpath=new_userpath, new_childname=new_childname)
|
||||||
self.userpath = new_userpath
|
self.userpath = new_userpath
|
||||||
self.parent = new_parent
|
self.parent = new_parent
|
||||||
@ -751,11 +774,11 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
if not (self.flags & FXF_READ):
|
if not (self.flags & FXF_READ):
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "file handle was not opened for reading")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "file handle was not opened for reading")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot read from a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
d = defer.Deferred()
|
d = defer.Deferred()
|
||||||
@ -773,11 +796,11 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
self.log(".writeChunk(%r, <data of length %r>)" % (offset, len(data)), level=OPERATIONAL)
|
self.log(".writeChunk(%r, <data of length %r>)" % (offset, len(data)), level=OPERATIONAL)
|
||||||
|
|
||||||
if not (self.flags & FXF_WRITE):
|
if not (self.flags & FXF_WRITE):
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot write to a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot write to a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
self.has_changed = True
|
self.has_changed = True
|
||||||
@ -893,7 +916,7 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot get attributes for a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot get attributes for a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
# Optimization for read-only handles, when we already know the metadata.
|
# Optimization for read-only handles, when we already know the metadata.
|
||||||
@ -917,16 +940,16 @@ class GeneralSFTPFile(PrefixingLogMixin):
|
|||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
if not (self.flags & FXF_WRITE):
|
if not (self.flags & FXF_WRITE):
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "file handle was not opened for writing")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
if self.closed:
|
if self.closed:
|
||||||
def _closed(): raise SFTPError(FX_BAD_MESSAGE, "cannot set attributes for a closed file handle")
|
def _closed(): raise createSFTPError(FX_BAD_MESSAGE, "cannot set attributes for a closed file handle")
|
||||||
return defer.execute(_closed)
|
return defer.execute(_closed)
|
||||||
|
|
||||||
size = attrs.get("size", None)
|
size = attrs.get("size", None)
|
||||||
if size is not None and (not isinstance(size, (int, long)) or size < 0):
|
if size is not None and (not isinstance(size, int) or size < 0):
|
||||||
def _bad(): raise SFTPError(FX_BAD_MESSAGE, "new size is not a valid nonnegative integer")
|
def _bad(): raise createSFTPError(FX_BAD_MESSAGE, "new size is not a valid nonnegative integer")
|
||||||
return defer.execute(_bad)
|
return defer.execute(_bad)
|
||||||
|
|
||||||
d = defer.Deferred()
|
d = defer.Deferred()
|
||||||
@ -1012,7 +1035,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def logout(self):
|
def logout(self):
|
||||||
self.log(".logout()", level=OPERATIONAL)
|
self.log(".logout()", level=OPERATIONAL)
|
||||||
|
|
||||||
for files in self._heisenfiles.itervalues():
|
for files in self._heisenfiles.values():
|
||||||
for f in files:
|
for f in files:
|
||||||
f.abandon()
|
f.abandon()
|
||||||
|
|
||||||
@ -1039,7 +1062,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
request = "._abandon_any_heisenfiles(%r, %r)" % (userpath, direntry)
|
request = "._abandon_any_heisenfiles(%r, %r)" % (userpath, direntry)
|
||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
precondition(isinstance(userpath, str), userpath=userpath)
|
precondition(isinstance(userpath, bytes), userpath=userpath)
|
||||||
|
|
||||||
# First we synchronously mark all heisenfiles matching the userpath or direntry
|
# First we synchronously mark all heisenfiles matching the userpath or direntry
|
||||||
# as abandoned, and remove them from the two heisenfile dicts. Then we .sync()
|
# as abandoned, and remove them from the two heisenfile dicts. Then we .sync()
|
||||||
@ -1088,8 +1111,8 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
(from_userpath, from_parent, from_childname, to_userpath, to_parent, to_childname, overwrite))
|
(from_userpath, from_parent, from_childname, to_userpath, to_parent, to_childname, overwrite))
|
||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
precondition((isinstance(from_userpath, str) and isinstance(from_childname, unicode) and
|
precondition((isinstance(from_userpath, bytes) and isinstance(from_childname, str) and
|
||||||
isinstance(to_userpath, str) and isinstance(to_childname, unicode)),
|
isinstance(to_userpath, bytes) and isinstance(to_childname, str)),
|
||||||
from_userpath=from_userpath, from_childname=from_childname, to_userpath=to_userpath, to_childname=to_childname)
|
from_userpath=from_userpath, from_childname=from_childname, to_userpath=to_userpath, to_childname=to_childname)
|
||||||
|
|
||||||
if noisy: self.log("all_heisenfiles = %r\nself._heisenfiles = %r" % (all_heisenfiles, self._heisenfiles), level=NOISY)
|
if noisy: self.log("all_heisenfiles = %r\nself._heisenfiles = %r" % (all_heisenfiles, self._heisenfiles), level=NOISY)
|
||||||
@ -1118,7 +1141,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
# does not mean that they were not committed; it is used to determine whether
|
# does not mean that they were not committed; it is used to determine whether
|
||||||
# a NoSuchChildError from the rename attempt should be suppressed). If overwrite
|
# a NoSuchChildError from the rename attempt should be suppressed). If overwrite
|
||||||
# is False and there were already heisenfiles at the destination userpath or
|
# is False and there were already heisenfiles at the destination userpath or
|
||||||
# direntry, we return a Deferred that fails with SFTPError(FX_PERMISSION_DENIED).
|
# direntry, we return a Deferred that fails with createSFTPError(FX_PERMISSION_DENIED).
|
||||||
|
|
||||||
from_direntry = _direntry_for(from_parent, from_childname)
|
from_direntry = _direntry_for(from_parent, from_childname)
|
||||||
to_direntry = _direntry_for(to_parent, to_childname)
|
to_direntry = _direntry_for(to_parent, to_childname)
|
||||||
@ -1127,7 +1150,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
(from_direntry, to_direntry, len(all_heisenfiles), len(self._heisenfiles), request), level=NOISY)
|
(from_direntry, to_direntry, len(all_heisenfiles), len(self._heisenfiles), request), level=NOISY)
|
||||||
|
|
||||||
if not overwrite and (to_userpath in self._heisenfiles or to_direntry in all_heisenfiles):
|
if not overwrite and (to_userpath in self._heisenfiles or to_direntry in all_heisenfiles):
|
||||||
def _existing(): raise SFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + to_userpath)
|
def _existing(): raise createSFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + str(to_userpath, "utf-8"))
|
||||||
if noisy: self.log("existing", level=NOISY)
|
if noisy: self.log("existing", level=NOISY)
|
||||||
return defer.execute(_existing)
|
return defer.execute(_existing)
|
||||||
|
|
||||||
@ -1161,7 +1184,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
request = "._update_attrs_for_heisenfiles(%r, %r, %r)" % (userpath, direntry, attrs)
|
request = "._update_attrs_for_heisenfiles(%r, %r, %r)" % (userpath, direntry, attrs)
|
||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
_assert(isinstance(userpath, str) and isinstance(direntry, str),
|
_assert(isinstance(userpath, bytes) and isinstance(direntry, bytes),
|
||||||
userpath=userpath, direntry=direntry)
|
userpath=userpath, direntry=direntry)
|
||||||
|
|
||||||
files = []
|
files = []
|
||||||
@ -1194,7 +1217,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
request = "._sync_heisenfiles(%r, %r, ignore=%r)" % (userpath, direntry, ignore)
|
request = "._sync_heisenfiles(%r, %r, ignore=%r)" % (userpath, direntry, ignore)
|
||||||
self.log(request, level=OPERATIONAL)
|
self.log(request, level=OPERATIONAL)
|
||||||
|
|
||||||
_assert(isinstance(userpath, str) and isinstance(direntry, (str, NoneType)),
|
_assert(isinstance(userpath, bytes) and isinstance(direntry, (bytes, type(None))),
|
||||||
userpath=userpath, direntry=direntry)
|
userpath=userpath, direntry=direntry)
|
||||||
|
|
||||||
files = []
|
files = []
|
||||||
@ -1219,7 +1242,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _remove_heisenfile(self, userpath, parent, childname, file_to_remove):
|
def _remove_heisenfile(self, userpath, parent, childname, file_to_remove):
|
||||||
if noisy: self.log("._remove_heisenfile(%r, %r, %r, %r)" % (userpath, parent, childname, file_to_remove), level=NOISY)
|
if noisy: self.log("._remove_heisenfile(%r, %r, %r, %r)" % (userpath, parent, childname, file_to_remove), level=NOISY)
|
||||||
|
|
||||||
_assert(isinstance(userpath, str) and isinstance(childname, (unicode, NoneType)),
|
_assert(isinstance(userpath, bytes) and isinstance(childname, (str, type(None))),
|
||||||
userpath=userpath, childname=childname)
|
userpath=userpath, childname=childname)
|
||||||
|
|
||||||
direntry = _direntry_for(parent, childname)
|
direntry = _direntry_for(parent, childname)
|
||||||
@ -1246,8 +1269,8 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
(existing_file, userpath, flags, _repr_flags(flags), parent, childname, filenode, metadata),
|
(existing_file, userpath, flags, _repr_flags(flags), parent, childname, filenode, metadata),
|
||||||
level=NOISY)
|
level=NOISY)
|
||||||
|
|
||||||
_assert((isinstance(userpath, str) and isinstance(childname, (unicode, NoneType)) and
|
_assert((isinstance(userpath, bytes) and isinstance(childname, (str, type(None))) and
|
||||||
(metadata is None or 'no-write' in metadata)),
|
(metadata is None or 'no-write' in metadata)),
|
||||||
userpath=userpath, childname=childname, metadata=metadata)
|
userpath=userpath, childname=childname, metadata=metadata)
|
||||||
|
|
||||||
writing = (flags & (FXF_WRITE | FXF_CREAT)) != 0
|
writing = (flags & (FXF_WRITE | FXF_CREAT)) != 0
|
||||||
@ -1280,17 +1303,17 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
|
|
||||||
if not (flags & (FXF_READ | FXF_WRITE)):
|
if not (flags & (FXF_READ | FXF_WRITE)):
|
||||||
def _bad_readwrite():
|
def _bad_readwrite():
|
||||||
raise SFTPError(FX_BAD_MESSAGE, "invalid file open flags: at least one of FXF_READ and FXF_WRITE must be set")
|
raise createSFTPError(FX_BAD_MESSAGE, "invalid file open flags: at least one of FXF_READ and FXF_WRITE must be set")
|
||||||
return defer.execute(_bad_readwrite)
|
return defer.execute(_bad_readwrite)
|
||||||
|
|
||||||
if (flags & FXF_EXCL) and not (flags & FXF_CREAT):
|
if (flags & FXF_EXCL) and not (flags & FXF_CREAT):
|
||||||
def _bad_exclcreat():
|
def _bad_exclcreat():
|
||||||
raise SFTPError(FX_BAD_MESSAGE, "invalid file open flags: FXF_EXCL cannot be set without FXF_CREAT")
|
raise createSFTPError(FX_BAD_MESSAGE, "invalid file open flags: FXF_EXCL cannot be set without FXF_CREAT")
|
||||||
return defer.execute(_bad_exclcreat)
|
return defer.execute(_bad_exclcreat)
|
||||||
|
|
||||||
path = self._path_from_string(pathstring)
|
path = self._path_from_string(pathstring)
|
||||||
if not path:
|
if not path:
|
||||||
def _emptypath(): raise SFTPError(FX_NO_SUCH_FILE, "path cannot be empty")
|
def _emptypath(): raise createSFTPError(FX_NO_SUCH_FILE, "path cannot be empty")
|
||||||
return defer.execute(_emptypath)
|
return defer.execute(_emptypath)
|
||||||
|
|
||||||
# The combination of flags is potentially valid.
|
# The combination of flags is potentially valid.
|
||||||
@ -1349,20 +1372,20 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _got_root(root_and_path):
|
def _got_root(root_and_path):
|
||||||
(root, path) = root_and_path
|
(root, path) = root_and_path
|
||||||
if root.is_unknown():
|
if root.is_unknown():
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open an unknown cap (or child of an unknown object). "
|
"cannot open an unknown cap (or child of an unknown object). "
|
||||||
"Upgrading the gateway to a later Tahoe-LAFS version may help")
|
"Upgrading the gateway to a later Tahoe-LAFS version may help")
|
||||||
if not path:
|
if not path:
|
||||||
# case 1
|
# case 1
|
||||||
if noisy: self.log("case 1: root = %r, path[:-1] = %r" % (root, path[:-1]), level=NOISY)
|
if noisy: self.log("case 1: root = %r, path[:-1] = %r" % (root, path[:-1]), level=NOISY)
|
||||||
if not IFileNode.providedBy(root):
|
if not IFileNode.providedBy(root):
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open a directory cap")
|
"cannot open a directory cap")
|
||||||
if (flags & FXF_WRITE) and root.is_readonly():
|
if (flags & FXF_WRITE) and root.is_readonly():
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot write to a non-writeable filecap without a parent directory")
|
"cannot write to a non-writeable filecap without a parent directory")
|
||||||
if flags & FXF_EXCL:
|
if flags & FXF_EXCL:
|
||||||
raise SFTPError(FX_FAILURE,
|
raise createSFTPError(FX_FAILURE,
|
||||||
"cannot create a file exclusively when it already exists")
|
"cannot create a file exclusively when it already exists")
|
||||||
|
|
||||||
# The file does not need to be added to all_heisenfiles, because it is not
|
# The file does not need to be added to all_heisenfiles, because it is not
|
||||||
@ -1389,7 +1412,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _got_parent(parent):
|
def _got_parent(parent):
|
||||||
if noisy: self.log("_got_parent(%r)" % (parent,), level=NOISY)
|
if noisy: self.log("_got_parent(%r)" % (parent,), level=NOISY)
|
||||||
if parent.is_unknown():
|
if parent.is_unknown():
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open a child of an unknown object. "
|
"cannot open a child of an unknown object. "
|
||||||
"Upgrading the gateway to a later Tahoe-LAFS version may help")
|
"Upgrading the gateway to a later Tahoe-LAFS version may help")
|
||||||
|
|
||||||
@ -1404,13 +1427,13 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
# which is consistent with what might happen on a POSIX filesystem.
|
# which is consistent with what might happen on a POSIX filesystem.
|
||||||
|
|
||||||
if parent_readonly:
|
if parent_readonly:
|
||||||
raise SFTPError(FX_FAILURE,
|
raise createSFTPError(FX_FAILURE,
|
||||||
"cannot create a file exclusively when the parent directory is read-only")
|
"cannot create a file exclusively when the parent directory is read-only")
|
||||||
|
|
||||||
# 'overwrite=False' ensures failure if the link already exists.
|
# 'overwrite=False' ensures failure if the link already exists.
|
||||||
# FIXME: should use a single call to set_uri and return (child, metadata) (#1035)
|
# FIXME: should use a single call to set_uri and return (child, metadata) (#1035)
|
||||||
|
|
||||||
zero_length_lit = "URI:LIT:"
|
zero_length_lit = b"URI:LIT:"
|
||||||
if noisy: self.log("%r.set_uri(%r, None, readcap=%r, overwrite=False)" %
|
if noisy: self.log("%r.set_uri(%r, None, readcap=%r, overwrite=False)" %
|
||||||
(parent, zero_length_lit, childname), level=NOISY)
|
(parent, zero_length_lit, childname), level=NOISY)
|
||||||
d3.addCallback(lambda ign: parent.set_uri(childname, None, readcap=zero_length_lit,
|
d3.addCallback(lambda ign: parent.set_uri(childname, None, readcap=zero_length_lit,
|
||||||
@ -1436,14 +1459,14 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
metadata['no-write'] = _no_write(parent_readonly, filenode, current_metadata)
|
metadata['no-write'] = _no_write(parent_readonly, filenode, current_metadata)
|
||||||
|
|
||||||
if filenode.is_unknown():
|
if filenode.is_unknown():
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open an unknown cap. Upgrading the gateway "
|
"cannot open an unknown cap. Upgrading the gateway "
|
||||||
"to a later Tahoe-LAFS version may help")
|
"to a later Tahoe-LAFS version may help")
|
||||||
if not IFileNode.providedBy(filenode):
|
if not IFileNode.providedBy(filenode):
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open a directory as if it were a file")
|
"cannot open a directory as if it were a file")
|
||||||
if (flags & FXF_WRITE) and metadata['no-write']:
|
if (flags & FXF_WRITE) and metadata['no-write']:
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot open a non-writeable file for writing")
|
"cannot open a non-writeable file for writing")
|
||||||
|
|
||||||
return self._make_file(file, userpath, flags, parent=parent, childname=childname,
|
return self._make_file(file, userpath, flags, parent=parent, childname=childname,
|
||||||
@ -1453,10 +1476,10 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
f.trap(NoSuchChildError)
|
f.trap(NoSuchChildError)
|
||||||
|
|
||||||
if not (flags & FXF_CREAT):
|
if not (flags & FXF_CREAT):
|
||||||
raise SFTPError(FX_NO_SUCH_FILE,
|
raise createSFTPError(FX_NO_SUCH_FILE,
|
||||||
"the file does not exist, and was not opened with the creation (CREAT) flag")
|
"the file does not exist, and was not opened with the creation (CREAT) flag")
|
||||||
if parent_readonly:
|
if parent_readonly:
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot create a file when the parent directory is read-only")
|
"cannot create a file when the parent directory is read-only")
|
||||||
|
|
||||||
return self._make_file(file, userpath, flags, parent=parent, childname=childname)
|
return self._make_file(file, userpath, flags, parent=parent, childname=childname)
|
||||||
@ -1495,9 +1518,9 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
(to_parent, to_childname) = to_pair
|
(to_parent, to_childname) = to_pair
|
||||||
|
|
||||||
if from_childname is None:
|
if from_childname is None:
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, "cannot rename a source object specified by URI")
|
raise createSFTPError(FX_NO_SUCH_FILE, "cannot rename a source object specified by URI")
|
||||||
if to_childname is None:
|
if to_childname is None:
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, "cannot rename to a destination specified by URI")
|
raise createSFTPError(FX_NO_SUCH_FILE, "cannot rename to a destination specified by URI")
|
||||||
|
|
||||||
# <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-6.5>
|
# <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-6.5>
|
||||||
# "It is an error if there already exists a file with the name specified
|
# "It is an error if there already exists a file with the name specified
|
||||||
@ -1512,7 +1535,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
d2.addCallback(lambda ign: to_parent.get(to_childname))
|
d2.addCallback(lambda ign: to_parent.get(to_childname))
|
||||||
def _expect_fail(res):
|
def _expect_fail(res):
|
||||||
if not isinstance(res, Failure):
|
if not isinstance(res, Failure):
|
||||||
raise SFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + to_userpath)
|
raise createSFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + str(to_userpath, "utf-8"))
|
||||||
|
|
||||||
# It is OK if we fail for errors other than NoSuchChildError, since that probably
|
# It is OK if we fail for errors other than NoSuchChildError, since that probably
|
||||||
# indicates some problem accessing the destination directory.
|
# indicates some problem accessing the destination directory.
|
||||||
@ -1537,7 +1560,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
if not isinstance(err, Failure) or (renamed and err.check(NoSuchChildError)):
|
if not isinstance(err, Failure) or (renamed and err.check(NoSuchChildError)):
|
||||||
return None
|
return None
|
||||||
if not overwrite and err.check(ExistingChildError):
|
if not overwrite and err.check(ExistingChildError):
|
||||||
raise SFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + to_userpath)
|
raise createSFTPError(FX_PERMISSION_DENIED, "cannot rename to existing path " + str(to_userpath, "utf-8"))
|
||||||
|
|
||||||
return err
|
return err
|
||||||
d3.addBoth(_check)
|
d3.addBoth(_check)
|
||||||
@ -1555,7 +1578,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
path = self._path_from_string(pathstring)
|
path = self._path_from_string(pathstring)
|
||||||
metadata = _attrs_to_metadata(attrs)
|
metadata = _attrs_to_metadata(attrs)
|
||||||
if 'no-write' in metadata:
|
if 'no-write' in metadata:
|
||||||
def _denied(): raise SFTPError(FX_PERMISSION_DENIED, "cannot create a directory that is initially read-only")
|
def _denied(): raise createSFTPError(FX_PERMISSION_DENIED, "cannot create a directory that is initially read-only")
|
||||||
return defer.execute(_denied)
|
return defer.execute(_denied)
|
||||||
|
|
||||||
d = self._get_root(path)
|
d = self._get_root(path)
|
||||||
@ -1567,7 +1590,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _get_or_create_directories(self, node, path, metadata):
|
def _get_or_create_directories(self, node, path, metadata):
|
||||||
if not IDirectoryNode.providedBy(node):
|
if not IDirectoryNode.providedBy(node):
|
||||||
# TODO: provide the name of the blocking file in the error message.
|
# TODO: provide the name of the blocking file in the error message.
|
||||||
def _blocked(): raise SFTPError(FX_FAILURE, "cannot create directory because there "
|
def _blocked(): raise createSFTPError(FX_FAILURE, "cannot create directory because there "
|
||||||
"is a file in the way") # close enough
|
"is a file in the way") # close enough
|
||||||
return defer.execute(_blocked)
|
return defer.execute(_blocked)
|
||||||
|
|
||||||
@ -1605,7 +1628,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _got_parent(parent_and_childname):
|
def _got_parent(parent_and_childname):
|
||||||
(parent, childname) = parent_and_childname
|
(parent, childname) = parent_and_childname
|
||||||
if childname is None:
|
if childname is None:
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, "cannot remove an object specified by URI")
|
raise createSFTPError(FX_NO_SUCH_FILE, "cannot remove an object specified by URI")
|
||||||
|
|
||||||
direntry = _direntry_for(parent, childname)
|
direntry = _direntry_for(parent, childname)
|
||||||
d2 = defer.succeed(False)
|
d2 = defer.succeed(False)
|
||||||
@ -1636,18 +1659,18 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
d.addCallback(_got_parent_or_node)
|
d.addCallback(_got_parent_or_node)
|
||||||
def _list(dirnode):
|
def _list(dirnode):
|
||||||
if dirnode.is_unknown():
|
if dirnode.is_unknown():
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot list an unknown cap as a directory. Upgrading the gateway "
|
"cannot list an unknown cap as a directory. Upgrading the gateway "
|
||||||
"to a later Tahoe-LAFS version may help")
|
"to a later Tahoe-LAFS version may help")
|
||||||
if not IDirectoryNode.providedBy(dirnode):
|
if not IDirectoryNode.providedBy(dirnode):
|
||||||
raise SFTPError(FX_PERMISSION_DENIED,
|
raise createSFTPError(FX_PERMISSION_DENIED,
|
||||||
"cannot list a file as if it were a directory")
|
"cannot list a file as if it were a directory")
|
||||||
|
|
||||||
d2 = dirnode.list()
|
d2 = dirnode.list()
|
||||||
def _render(children):
|
def _render(children):
|
||||||
parent_readonly = dirnode.is_readonly()
|
parent_readonly = dirnode.is_readonly()
|
||||||
results = []
|
results = []
|
||||||
for filename, (child, metadata) in children.iteritems():
|
for filename, (child, metadata) in list(children.items()):
|
||||||
# The file size may be cached or absent.
|
# The file size may be cached or absent.
|
||||||
metadata['no-write'] = _no_write(parent_readonly, child, metadata)
|
metadata['no-write'] = _no_write(parent_readonly, child, metadata)
|
||||||
attrs = _populate_attrs(child, metadata)
|
attrs = _populate_attrs(child, metadata)
|
||||||
@ -1727,7 +1750,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
if "size" in attrs:
|
if "size" in attrs:
|
||||||
# this would require us to download and re-upload the truncated/extended
|
# this would require us to download and re-upload the truncated/extended
|
||||||
# file contents
|
# file contents
|
||||||
def _unsupported(): raise SFTPError(FX_OP_UNSUPPORTED, "setAttrs wth size attribute unsupported")
|
def _unsupported(): raise createSFTPError(FX_OP_UNSUPPORTED, "setAttrs wth size attribute unsupported")
|
||||||
return defer.execute(_unsupported)
|
return defer.execute(_unsupported)
|
||||||
|
|
||||||
path = self._path_from_string(pathstring)
|
path = self._path_from_string(pathstring)
|
||||||
@ -1744,7 +1767,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
if childname is None:
|
if childname is None:
|
||||||
if updated_heisenfiles:
|
if updated_heisenfiles:
|
||||||
return None
|
return None
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, userpath)
|
raise createSFTPError(FX_NO_SUCH_FILE, userpath)
|
||||||
else:
|
else:
|
||||||
desired_metadata = _attrs_to_metadata(attrs)
|
desired_metadata = _attrs_to_metadata(attrs)
|
||||||
if noisy: self.log("desired_metadata = %r" % (desired_metadata,), level=NOISY)
|
if noisy: self.log("desired_metadata = %r" % (desired_metadata,), level=NOISY)
|
||||||
@ -1767,7 +1790,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def readLink(self, pathstring):
|
def readLink(self, pathstring):
|
||||||
self.log(".readLink(%r)" % (pathstring,), level=OPERATIONAL)
|
self.log(".readLink(%r)" % (pathstring,), level=OPERATIONAL)
|
||||||
|
|
||||||
def _unsupported(): raise SFTPError(FX_OP_UNSUPPORTED, "readLink")
|
def _unsupported(): raise createSFTPError(FX_OP_UNSUPPORTED, "readLink")
|
||||||
return defer.execute(_unsupported)
|
return defer.execute(_unsupported)
|
||||||
|
|
||||||
def makeLink(self, linkPathstring, targetPathstring):
|
def makeLink(self, linkPathstring, targetPathstring):
|
||||||
@ -1776,7 +1799,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
# If this is implemented, note the reversal of arguments described in point 7 of
|
# If this is implemented, note the reversal of arguments described in point 7 of
|
||||||
# <http://www.openbsd.org/cgi-bin/cvsweb/src/usr.bin/ssh/PROTOCOL?rev=1.15>.
|
# <http://www.openbsd.org/cgi-bin/cvsweb/src/usr.bin/ssh/PROTOCOL?rev=1.15>.
|
||||||
|
|
||||||
def _unsupported(): raise SFTPError(FX_OP_UNSUPPORTED, "makeLink")
|
def _unsupported(): raise createSFTPError(FX_OP_UNSUPPORTED, "makeLink")
|
||||||
return defer.execute(_unsupported)
|
return defer.execute(_unsupported)
|
||||||
|
|
||||||
def extendedRequest(self, extensionName, extensionData):
|
def extendedRequest(self, extensionName, extensionData):
|
||||||
@ -1785,8 +1808,8 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
# We implement the three main OpenSSH SFTP extensions; see
|
# We implement the three main OpenSSH SFTP extensions; see
|
||||||
# <http://www.openbsd.org/cgi-bin/cvsweb/src/usr.bin/ssh/PROTOCOL?rev=1.15>
|
# <http://www.openbsd.org/cgi-bin/cvsweb/src/usr.bin/ssh/PROTOCOL?rev=1.15>
|
||||||
|
|
||||||
if extensionName == 'posix-rename@openssh.com':
|
if extensionName == b'posix-rename@openssh.com':
|
||||||
def _bad(): raise SFTPError(FX_BAD_MESSAGE, "could not parse posix-rename@openssh.com request")
|
def _bad(): raise createSFTPError(FX_BAD_MESSAGE, "could not parse posix-rename@openssh.com request")
|
||||||
|
|
||||||
if 4 > len(extensionData): return defer.execute(_bad)
|
if 4 > len(extensionData): return defer.execute(_bad)
|
||||||
(fromPathLen,) = struct.unpack('>L', extensionData[0:4])
|
(fromPathLen,) = struct.unpack('>L', extensionData[0:4])
|
||||||
@ -1803,11 +1826,11 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
# an error, or an FXP_EXTENDED_REPLY. But it happens to do the right thing
|
# an error, or an FXP_EXTENDED_REPLY. But it happens to do the right thing
|
||||||
# (respond with an FXP_STATUS message) if we return a Failure with code FX_OK.
|
# (respond with an FXP_STATUS message) if we return a Failure with code FX_OK.
|
||||||
def _succeeded(ign):
|
def _succeeded(ign):
|
||||||
raise SFTPError(FX_OK, "request succeeded")
|
raise createSFTPError(FX_OK, "request succeeded")
|
||||||
d.addCallback(_succeeded)
|
d.addCallback(_succeeded)
|
||||||
return d
|
return d
|
||||||
|
|
||||||
if extensionName == 'statvfs@openssh.com' or extensionName == 'fstatvfs@openssh.com':
|
if extensionName == b'statvfs@openssh.com' or extensionName == b'fstatvfs@openssh.com':
|
||||||
# f_bsize and f_frsize should be the same to avoid a bug in 'df'
|
# f_bsize and f_frsize should be the same to avoid a bug in 'df'
|
||||||
return defer.succeed(struct.pack('>11Q',
|
return defer.succeed(struct.pack('>11Q',
|
||||||
1024, # uint64 f_bsize /* file system block size */
|
1024, # uint64 f_bsize /* file system block size */
|
||||||
@ -1823,7 +1846,7 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
65535, # uint64 f_namemax /* maximum filename length */
|
65535, # uint64 f_namemax /* maximum filename length */
|
||||||
))
|
))
|
||||||
|
|
||||||
def _unsupported(): raise SFTPError(FX_OP_UNSUPPORTED, "unsupported %r request <data of length %r>" %
|
def _unsupported(): raise createSFTPError(FX_OP_UNSUPPORTED, "unsupported %r request <data of length %r>" %
|
||||||
(extensionName, len(extensionData)))
|
(extensionName, len(extensionData)))
|
||||||
return defer.execute(_unsupported)
|
return defer.execute(_unsupported)
|
||||||
|
|
||||||
@ -1838,29 +1861,29 @@ class SFTPUserHandler(ConchUser, PrefixingLogMixin):
|
|||||||
def _path_from_string(self, pathstring):
|
def _path_from_string(self, pathstring):
|
||||||
if noisy: self.log("CONVERT %r" % (pathstring,), level=NOISY)
|
if noisy: self.log("CONVERT %r" % (pathstring,), level=NOISY)
|
||||||
|
|
||||||
_assert(isinstance(pathstring, str), pathstring=pathstring)
|
_assert(isinstance(pathstring, bytes), pathstring=pathstring)
|
||||||
|
|
||||||
# The home directory is the root directory.
|
# The home directory is the root directory.
|
||||||
pathstring = pathstring.strip("/")
|
pathstring = pathstring.strip(b"/")
|
||||||
if pathstring == "" or pathstring == ".":
|
if pathstring == b"" or pathstring == b".":
|
||||||
path_utf8 = []
|
path_utf8 = []
|
||||||
else:
|
else:
|
||||||
path_utf8 = pathstring.split("/")
|
path_utf8 = pathstring.split(b"/")
|
||||||
|
|
||||||
# <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-6.2>
|
# <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-6.2>
|
||||||
# "Servers SHOULD interpret a path name component ".." as referring to
|
# "Servers SHOULD interpret a path name component ".." as referring to
|
||||||
# the parent directory, and "." as referring to the current directory."
|
# the parent directory, and "." as referring to the current directory."
|
||||||
path = []
|
path = []
|
||||||
for p_utf8 in path_utf8:
|
for p_utf8 in path_utf8:
|
||||||
if p_utf8 == "..":
|
if p_utf8 == b"..":
|
||||||
# ignore excess .. components at the root
|
# ignore excess .. components at the root
|
||||||
if len(path) > 0:
|
if len(path) > 0:
|
||||||
path = path[:-1]
|
path = path[:-1]
|
||||||
elif p_utf8 != ".":
|
elif p_utf8 != b".":
|
||||||
try:
|
try:
|
||||||
p = p_utf8.decode('utf-8', 'strict')
|
p = p_utf8.decode('utf-8', 'strict')
|
||||||
except UnicodeError:
|
except UnicodeError:
|
||||||
raise SFTPError(FX_NO_SUCH_FILE, "path could not be decoded as UTF-8")
|
raise createSFTPError(FX_NO_SUCH_FILE, "path could not be decoded as UTF-8")
|
||||||
path.append(p)
|
path.append(p)
|
||||||
|
|
||||||
if noisy: self.log(" PATH %r" % (path,), level=NOISY)
|
if noisy: self.log(" PATH %r" % (path,), level=NOISY)
|
||||||
@ -1979,9 +2002,9 @@ class SFTPServer(service.MultiService):
|
|||||||
|
|
||||||
def __init__(self, client, accountfile, accounturl,
|
def __init__(self, client, accountfile, accounturl,
|
||||||
sftp_portstr, pubkey_file, privkey_file):
|
sftp_portstr, pubkey_file, privkey_file):
|
||||||
precondition(isinstance(accountfile, (unicode, NoneType)), accountfile)
|
precondition(isinstance(accountfile, (str, type(None))), accountfile)
|
||||||
precondition(isinstance(pubkey_file, unicode), pubkey_file)
|
precondition(isinstance(pubkey_file, str), pubkey_file)
|
||||||
precondition(isinstance(privkey_file, unicode), privkey_file)
|
precondition(isinstance(privkey_file, str), privkey_file)
|
||||||
service.MultiService.__init__(self)
|
service.MultiService.__init__(self)
|
||||||
|
|
||||||
r = Dispatcher(client)
|
r = Dispatcher(client)
|
||||||
@ -2012,5 +2035,5 @@ class SFTPServer(service.MultiService):
|
|||||||
f = SSHFactory()
|
f = SSHFactory()
|
||||||
f.portal = p
|
f.portal = p
|
||||||
|
|
||||||
s = strports.service(sftp_portstr, f)
|
s = strports.service(six.ensure_str(sftp_portstr), f)
|
||||||
s.setServiceParent(self)
|
s.setServiceParent(self)
|
||||||
|
@ -19,7 +19,7 @@ from twisted.protocols import basic
|
|||||||
from allmydata.interfaces import IImmutableFileNode, ICheckable
|
from allmydata.interfaces import IImmutableFileNode, ICheckable
|
||||||
from allmydata.uri import LiteralFileURI
|
from allmydata.uri import LiteralFileURI
|
||||||
|
|
||||||
@implementer(IImmutableFileNode, ICheckable)
|
|
||||||
class _ImmutableFileNodeBase(object):
|
class _ImmutableFileNodeBase(object):
|
||||||
|
|
||||||
def get_write_uri(self):
|
def get_write_uri(self):
|
||||||
@ -56,6 +56,7 @@ class _ImmutableFileNodeBase(object):
|
|||||||
return not self == other
|
return not self == other
|
||||||
|
|
||||||
|
|
||||||
|
@implementer(IImmutableFileNode, ICheckable)
|
||||||
class LiteralFileNode(_ImmutableFileNodeBase):
|
class LiteralFileNode(_ImmutableFileNodeBase):
|
||||||
|
|
||||||
def __init__(self, filecap):
|
def __init__(self, filecap):
|
||||||
|
@ -141,7 +141,7 @@ class CHKCheckerAndUEBFetcher(object):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(interfaces.RICHKUploadHelper)
|
@implementer(interfaces.RICHKUploadHelper)
|
||||||
class CHKUploadHelper(Referenceable, upload.CHKUploader):
|
class CHKUploadHelper(Referenceable, upload.CHKUploader): # type: ignore # warner/foolscap#78
|
||||||
"""I am the helper-server -side counterpart to AssistedUploader. I handle
|
"""I am the helper-server -side counterpart to AssistedUploader. I handle
|
||||||
peer selection, encoding, and share pushing. I read ciphertext from the
|
peer selection, encoding, and share pushing. I read ciphertext from the
|
||||||
remote AssistedUploader.
|
remote AssistedUploader.
|
||||||
@ -499,10 +499,13 @@ class LocalCiphertextReader(AskUntilSuccessMixin):
|
|||||||
# ??. I'm not sure if it makes sense to forward the close message.
|
# ??. I'm not sure if it makes sense to forward the close message.
|
||||||
return self.call("close")
|
return self.call("close")
|
||||||
|
|
||||||
|
# https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3561
|
||||||
|
def set_upload_status(self, upload_status):
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
@implementer(interfaces.RIHelper, interfaces.IStatsProducer)
|
@implementer(interfaces.RIHelper, interfaces.IStatsProducer)
|
||||||
class Helper(Referenceable):
|
class Helper(Referenceable): # type: ignore # warner/foolscap#78
|
||||||
"""
|
"""
|
||||||
:ivar dict[bytes, CHKUploadHelper] _active_uploads: For any uploads which
|
:ivar dict[bytes, CHKUploadHelper] _active_uploads: For any uploads which
|
||||||
have been started but not finished, a mapping from storage index to the
|
have been started but not finished, a mapping from storage index to the
|
||||||
|
@ -13,19 +13,30 @@ if PY2:
|
|||||||
from past.builtins import long, unicode
|
from past.builtins import long, unicode
|
||||||
from six import ensure_str
|
from six import ensure_str
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import List
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
import os, time, weakref, itertools
|
import os, time, weakref, itertools
|
||||||
|
|
||||||
|
import attr
|
||||||
|
|
||||||
from zope.interface import implementer
|
from zope.interface import implementer
|
||||||
from twisted.python import failure
|
from twisted.python import failure
|
||||||
from twisted.internet import defer
|
from twisted.internet import defer
|
||||||
from twisted.application import service
|
from twisted.application import service
|
||||||
from foolscap.api import Referenceable, Copyable, RemoteCopy, fireEventually
|
from foolscap.api import Referenceable, Copyable, RemoteCopy
|
||||||
|
|
||||||
from allmydata.crypto import aes
|
from allmydata.crypto import aes
|
||||||
from allmydata.util.hashutil import file_renewal_secret_hash, \
|
from allmydata.util.hashutil import file_renewal_secret_hash, \
|
||||||
file_cancel_secret_hash, bucket_renewal_secret_hash, \
|
file_cancel_secret_hash, bucket_renewal_secret_hash, \
|
||||||
bucket_cancel_secret_hash, plaintext_hasher, \
|
bucket_cancel_secret_hash, plaintext_hasher, \
|
||||||
storage_index_hash, plaintext_segment_hasher, convergence_hasher
|
storage_index_hash, plaintext_segment_hasher, convergence_hasher
|
||||||
from allmydata.util.deferredutil import timeout_call
|
from allmydata.util.deferredutil import (
|
||||||
|
timeout_call,
|
||||||
|
until,
|
||||||
|
)
|
||||||
from allmydata import hashtree, uri
|
from allmydata import hashtree, uri
|
||||||
from allmydata.storage.server import si_b2a
|
from allmydata.storage.server import si_b2a
|
||||||
from allmydata.immutable import encode
|
from allmydata.immutable import encode
|
||||||
@ -386,6 +397,9 @@ class PeerSelector(object):
|
|||||||
)
|
)
|
||||||
return self.happiness_mappings
|
return self.happiness_mappings
|
||||||
|
|
||||||
|
def add_peers(self, peerids=None):
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
class _QueryStatistics(object):
|
class _QueryStatistics(object):
|
||||||
|
|
||||||
@ -897,13 +911,45 @@ class Tahoe2ServerSelector(log.PrefixingLogMixin):
|
|||||||
raise UploadUnhappinessError(msg)
|
raise UploadUnhappinessError(msg)
|
||||||
|
|
||||||
|
|
||||||
|
@attr.s
|
||||||
|
class _Accum(object):
|
||||||
|
"""
|
||||||
|
Accumulate up to some known amount of ciphertext.
|
||||||
|
|
||||||
|
:ivar remaining: The number of bytes still expected.
|
||||||
|
:ivar ciphertext: The bytes accumulated so far.
|
||||||
|
"""
|
||||||
|
remaining = attr.ib(validator=attr.validators.instance_of(int)) # type: int
|
||||||
|
ciphertext = attr.ib(default=attr.Factory(list)) # type: List[bytes]
|
||||||
|
|
||||||
|
def extend(self,
|
||||||
|
size, # type: int
|
||||||
|
ciphertext, # type: List[bytes]
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Accumulate some more ciphertext.
|
||||||
|
|
||||||
|
:param size: The amount of data the new ciphertext represents towards
|
||||||
|
the goal. This may be more than the actual size of the given
|
||||||
|
ciphertext if the source has run out of data.
|
||||||
|
|
||||||
|
:param ciphertext: The new ciphertext to accumulate.
|
||||||
|
"""
|
||||||
|
self.remaining -= size
|
||||||
|
self.ciphertext.extend(ciphertext)
|
||||||
|
|
||||||
|
|
||||||
@implementer(IEncryptedUploadable)
|
@implementer(IEncryptedUploadable)
|
||||||
class EncryptAnUploadable(object):
|
class EncryptAnUploadable(object):
|
||||||
"""This is a wrapper that takes an IUploadable and provides
|
"""This is a wrapper that takes an IUploadable and provides
|
||||||
IEncryptedUploadable."""
|
IEncryptedUploadable."""
|
||||||
CHUNKSIZE = 50*1024
|
CHUNKSIZE = 50*1024
|
||||||
|
|
||||||
def __init__(self, original, log_parent=None, progress=None):
|
def __init__(self, original, log_parent=None, progress=None, chunk_size=None):
|
||||||
|
"""
|
||||||
|
:param chunk_size: The number of bytes to read from the uploadable at a
|
||||||
|
time, or None for some default.
|
||||||
|
"""
|
||||||
precondition(original.default_params_set,
|
precondition(original.default_params_set,
|
||||||
"set_default_encoding_parameters not called on %r before wrapping with EncryptAnUploadable" % (original,))
|
"set_default_encoding_parameters not called on %r before wrapping with EncryptAnUploadable" % (original,))
|
||||||
self.original = IUploadable(original)
|
self.original = IUploadable(original)
|
||||||
@ -917,6 +963,8 @@ class EncryptAnUploadable(object):
|
|||||||
self._ciphertext_bytes_read = 0
|
self._ciphertext_bytes_read = 0
|
||||||
self._status = None
|
self._status = None
|
||||||
self._progress = progress
|
self._progress = progress
|
||||||
|
if chunk_size is not None:
|
||||||
|
self.CHUNKSIZE = chunk_size
|
||||||
|
|
||||||
def set_upload_status(self, upload_status):
|
def set_upload_status(self, upload_status):
|
||||||
self._status = IUploadStatus(upload_status)
|
self._status = IUploadStatus(upload_status)
|
||||||
@ -1023,47 +1071,53 @@ class EncryptAnUploadable(object):
|
|||||||
# and size
|
# and size
|
||||||
d.addCallback(lambda ignored: self.get_size())
|
d.addCallback(lambda ignored: self.get_size())
|
||||||
d.addCallback(lambda ignored: self._get_encryptor())
|
d.addCallback(lambda ignored: self._get_encryptor())
|
||||||
# then fetch and encrypt the plaintext. The unusual structure here
|
|
||||||
# (passing a Deferred *into* a function) is needed to avoid
|
accum = _Accum(length)
|
||||||
# overflowing the stack: Deferreds don't optimize out tail recursion.
|
|
||||||
# We also pass in a list, to which _read_encrypted will append
|
def action():
|
||||||
# ciphertext.
|
"""
|
||||||
ciphertext = []
|
Read some bytes into the accumulator.
|
||||||
d2 = defer.Deferred()
|
"""
|
||||||
d.addCallback(lambda ignored:
|
return self._read_encrypted(accum, hash_only)
|
||||||
self._read_encrypted(length, ciphertext, hash_only, d2))
|
|
||||||
d.addCallback(lambda ignored: d2)
|
def condition():
|
||||||
|
"""
|
||||||
|
Check to see if the accumulator has all the data.
|
||||||
|
"""
|
||||||
|
return accum.remaining == 0
|
||||||
|
|
||||||
|
d.addCallback(lambda ignored: until(action, condition))
|
||||||
|
d.addCallback(lambda ignored: accum.ciphertext)
|
||||||
return d
|
return d
|
||||||
|
|
||||||
def _read_encrypted(self, remaining, ciphertext, hash_only, fire_when_done):
|
def _read_encrypted(self,
|
||||||
if not remaining:
|
ciphertext_accum, # type: _Accum
|
||||||
fire_when_done.callback(ciphertext)
|
hash_only, # type: bool
|
||||||
return None
|
):
|
||||||
|
# type: (...) -> defer.Deferred
|
||||||
|
"""
|
||||||
|
Read the next chunk of plaintext, encrypt it, and extend the accumulator
|
||||||
|
with the resulting ciphertext.
|
||||||
|
"""
|
||||||
# tolerate large length= values without consuming a lot of RAM by
|
# tolerate large length= values without consuming a lot of RAM by
|
||||||
# reading just a chunk (say 50kB) at a time. This only really matters
|
# reading just a chunk (say 50kB) at a time. This only really matters
|
||||||
# when hash_only==True (i.e. resuming an interrupted upload), since
|
# when hash_only==True (i.e. resuming an interrupted upload), since
|
||||||
# that's the case where we will be skipping over a lot of data.
|
# that's the case where we will be skipping over a lot of data.
|
||||||
size = min(remaining, self.CHUNKSIZE)
|
size = min(ciphertext_accum.remaining, self.CHUNKSIZE)
|
||||||
remaining = remaining - size
|
|
||||||
# read a chunk of plaintext..
|
# read a chunk of plaintext..
|
||||||
d = defer.maybeDeferred(self.original.read, size)
|
d = defer.maybeDeferred(self.original.read, size)
|
||||||
# N.B.: if read() is synchronous, then since everything else is
|
|
||||||
# actually synchronous too, we'd blow the stack unless we stall for a
|
|
||||||
# tick. Once you accept a Deferred from IUploadable.read(), you must
|
|
||||||
# be prepared to have it fire immediately too.
|
|
||||||
d.addCallback(fireEventually)
|
|
||||||
def _good(plaintext):
|
def _good(plaintext):
|
||||||
# and encrypt it..
|
# and encrypt it..
|
||||||
# o/' over the fields we go, hashing all the way, sHA! sHA! sHA! o/'
|
# o/' over the fields we go, hashing all the way, sHA! sHA! sHA! o/'
|
||||||
ct = self._hash_and_encrypt_plaintext(plaintext, hash_only)
|
ct = self._hash_and_encrypt_plaintext(plaintext, hash_only)
|
||||||
ciphertext.extend(ct)
|
# Intentionally tell the accumulator about the expected size, not
|
||||||
self._read_encrypted(remaining, ciphertext, hash_only,
|
# the actual size. If we run out of data we still want remaining
|
||||||
fire_when_done)
|
# to drop otherwise it will never reach 0 and the loop will never
|
||||||
def _err(why):
|
# end.
|
||||||
fire_when_done.errback(why)
|
ciphertext_accum.extend(size, ct)
|
||||||
d.addCallback(_good)
|
d.addCallback(_good)
|
||||||
d.addErrback(_err)
|
return d
|
||||||
return None
|
|
||||||
|
|
||||||
def _hash_and_encrypt_plaintext(self, data, hash_only):
|
def _hash_and_encrypt_plaintext(self, data, hash_only):
|
||||||
assert isinstance(data, (tuple, list)), type(data)
|
assert isinstance(data, (tuple, list)), type(data)
|
||||||
@ -1424,7 +1478,7 @@ class LiteralUploader(object):
|
|||||||
return self._status
|
return self._status
|
||||||
|
|
||||||
@implementer(RIEncryptedUploadable)
|
@implementer(RIEncryptedUploadable)
|
||||||
class RemoteEncryptedUploadable(Referenceable):
|
class RemoteEncryptedUploadable(Referenceable): # type: ignore # warner/foolscap#78
|
||||||
|
|
||||||
def __init__(self, encrypted_uploadable, upload_status):
|
def __init__(self, encrypted_uploadable, upload_status):
|
||||||
self._eu = IEncryptedUploadable(encrypted_uploadable)
|
self._eu = IEncryptedUploadable(encrypted_uploadable)
|
||||||
|
@ -681,7 +681,7 @@ class IURI(Interface):
|
|||||||
passing into init_from_string."""
|
passing into init_from_string."""
|
||||||
|
|
||||||
|
|
||||||
class IVerifierURI(Interface, IURI):
|
class IVerifierURI(IURI):
|
||||||
def init_from_string(uri):
|
def init_from_string(uri):
|
||||||
"""Accept a string (as created by my to_string() method) and populate
|
"""Accept a string (as created by my to_string() method) and populate
|
||||||
this instance with its data. I am not normally called directly,
|
this instance with its data. I am not normally called directly,
|
||||||
@ -748,7 +748,7 @@ class IProgress(Interface):
|
|||||||
"Current amount of progress (in percentage)"
|
"Current amount of progress (in percentage)"
|
||||||
)
|
)
|
||||||
|
|
||||||
def set_progress(self, value):
|
def set_progress(value):
|
||||||
"""
|
"""
|
||||||
Sets the current amount of progress.
|
Sets the current amount of progress.
|
||||||
|
|
||||||
@ -756,7 +756,7 @@ class IProgress(Interface):
|
|||||||
set_progress_total.
|
set_progress_total.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def set_progress_total(self, value):
|
def set_progress_total(value):
|
||||||
"""
|
"""
|
||||||
Sets the total amount of expected progress
|
Sets the total amount of expected progress
|
||||||
|
|
||||||
@ -859,12 +859,6 @@ class IPeerSelector(Interface):
|
|||||||
peer selection begins.
|
peer selection begins.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def confirm_share_allocation(peerid, shnum):
|
|
||||||
"""
|
|
||||||
Confirm that an allocated peer=>share pairing has been
|
|
||||||
successfully established.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def add_peers(peerids=set):
|
def add_peers(peerids=set):
|
||||||
"""
|
"""
|
||||||
Update my internal state to include the peers in peerids as
|
Update my internal state to include the peers in peerids as
|
||||||
@ -1824,11 +1818,6 @@ class IEncoder(Interface):
|
|||||||
willing to receive data.
|
willing to receive data.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def set_size(size):
|
|
||||||
"""Specify the number of bytes that will be encoded. This must be
|
|
||||||
peformed before get_serialized_params() can be called.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def set_encrypted_uploadable(u):
|
def set_encrypted_uploadable(u):
|
||||||
"""Provide a source of encrypted upload data. 'u' must implement
|
"""Provide a source of encrypted upload data. 'u' must implement
|
||||||
IEncryptedUploadable.
|
IEncryptedUploadable.
|
||||||
|
@ -178,9 +178,9 @@ class IntroducerClient(service.Service, Referenceable):
|
|||||||
kwargs["facility"] = "tahoe.introducer.client"
|
kwargs["facility"] = "tahoe.introducer.client"
|
||||||
return log.msg(*args, **kwargs)
|
return log.msg(*args, **kwargs)
|
||||||
|
|
||||||
def subscribe_to(self, service_name, cb, *args, **kwargs):
|
def subscribe_to(self, service_name, callback, *args, **kwargs):
|
||||||
obs = self._local_subscribers.setdefault(service_name, ObserverList())
|
obs = self._local_subscribers.setdefault(service_name, ObserverList())
|
||||||
obs.subscribe(lambda key_s, ann: cb(key_s, ann, *args, **kwargs))
|
obs.subscribe(lambda key_s, ann: callback(key_s, ann, *args, **kwargs))
|
||||||
self._maybe_subscribe()
|
self._maybe_subscribe()
|
||||||
for index,(ann,key_s,when) in list(self._inbound_announcements.items()):
|
for index,(ann,key_s,when) in list(self._inbound_announcements.items()):
|
||||||
precondition(isinstance(key_s, bytes), key_s)
|
precondition(isinstance(key_s, bytes), key_s)
|
||||||
|
@ -73,7 +73,7 @@ class IIntroducerClient(Interface):
|
|||||||
publish their services to the rest of the world, and I help them learn
|
publish their services to the rest of the world, and I help them learn
|
||||||
about services available on other nodes."""
|
about services available on other nodes."""
|
||||||
|
|
||||||
def publish(service_name, ann, signing_key=None):
|
def publish(service_name, ann, signing_key):
|
||||||
"""Publish the given announcement dictionary (which must be
|
"""Publish the given announcement dictionary (which must be
|
||||||
JSON-serializable), plus some additional keys, to the world.
|
JSON-serializable), plus some additional keys, to the world.
|
||||||
|
|
||||||
@ -83,8 +83,7 @@ class IIntroducerClient(Interface):
|
|||||||
the signing_key, if present, otherwise it is derived from the
|
the signing_key, if present, otherwise it is derived from the
|
||||||
'anonymous-storage-FURL' key.
|
'anonymous-storage-FURL' key.
|
||||||
|
|
||||||
If signing_key= is set to an instance of SigningKey, it will be
|
signing_key (a SigningKey) will be used to sign the announcement."""
|
||||||
used to sign the announcement."""
|
|
||||||
|
|
||||||
def subscribe_to(service_name, callback, *args, **kwargs):
|
def subscribe_to(service_name, callback, *args, **kwargs):
|
||||||
"""Call this if you will eventually want to use services with the
|
"""Call this if you will eventually want to use services with the
|
||||||
|
@ -15,6 +15,12 @@ from past.builtins import long
|
|||||||
from six import ensure_text
|
from six import ensure_text
|
||||||
|
|
||||||
import time, os.path, textwrap
|
import time, os.path, textwrap
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Any, Dict, Union
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from zope.interface import implementer
|
from zope.interface import implementer
|
||||||
from twisted.application import service
|
from twisted.application import service
|
||||||
from twisted.internet import defer
|
from twisted.internet import defer
|
||||||
@ -147,10 +153,12 @@ class IntroducerService(service.MultiService, Referenceable):
|
|||||||
name = "introducer"
|
name = "introducer"
|
||||||
# v1 is the original protocol, added in 1.0 (but only advertised starting
|
# v1 is the original protocol, added in 1.0 (but only advertised starting
|
||||||
# in 1.3), removed in 1.12. v2 is the new signed protocol, added in 1.10
|
# in 1.3), removed in 1.12. v2 is the new signed protocol, added in 1.10
|
||||||
VERSION = { #"http://allmydata.org/tahoe/protocols/introducer/v1": { },
|
# TODO: reconcile bytes/str for keys
|
||||||
|
VERSION = {
|
||||||
|
#"http://allmydata.org/tahoe/protocols/introducer/v1": { },
|
||||||
b"http://allmydata.org/tahoe/protocols/introducer/v2": { },
|
b"http://allmydata.org/tahoe/protocols/introducer/v2": { },
|
||||||
b"application-version": allmydata.__full_version__.encode("utf-8"),
|
b"application-version": allmydata.__full_version__.encode("utf-8"),
|
||||||
}
|
} # type: Dict[Union[bytes, str], Any]
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
service.MultiService.__init__(self)
|
service.MultiService.__init__(self)
|
||||||
|
@ -564,7 +564,7 @@ class MutableFileNode(object):
|
|||||||
return d
|
return d
|
||||||
|
|
||||||
|
|
||||||
def upload(self, new_contents, servermap):
|
def upload(self, new_contents, servermap, progress=None):
|
||||||
"""
|
"""
|
||||||
I overwrite the contents of the best recoverable version of this
|
I overwrite the contents of the best recoverable version of this
|
||||||
mutable file with new_contents, using servermap instead of
|
mutable file with new_contents, using servermap instead of
|
||||||
@ -951,7 +951,7 @@ class MutableFileVersion(object):
|
|||||||
return self._servermap.size_of_version(self._version)
|
return self._servermap.size_of_version(self._version)
|
||||||
|
|
||||||
|
|
||||||
def download_to_data(self, fetch_privkey=False, progress=None):
|
def download_to_data(self, fetch_privkey=False, progress=None): # type: ignore # fixme
|
||||||
"""
|
"""
|
||||||
I return a Deferred that fires with the contents of this
|
I return a Deferred that fires with the contents of this
|
||||||
readable object as a byte string.
|
readable object as a byte string.
|
||||||
@ -1205,3 +1205,7 @@ class MutableFileVersion(object):
|
|||||||
self._servermap,
|
self._servermap,
|
||||||
mode=mode)
|
mode=mode)
|
||||||
return u.update()
|
return u.update()
|
||||||
|
|
||||||
|
# https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3562
|
||||||
|
def get_servermap(self):
|
||||||
|
raise NotImplementedError
|
||||||
|
@ -23,6 +23,11 @@ from base64 import b32decode, b32encode
|
|||||||
from errno import ENOENT, EPERM
|
from errno import ENOENT, EPERM
|
||||||
from warnings import warn
|
from warnings import warn
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Union
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
import attr
|
import attr
|
||||||
|
|
||||||
# On Python 2 this will be the backported package.
|
# On Python 2 this will be the backported package.
|
||||||
@ -273,6 +278,11 @@ def _error_about_old_config_files(basedir, generated_files):
|
|||||||
raise e
|
raise e
|
||||||
|
|
||||||
|
|
||||||
|
def ensure_text_and_abspath_expanduser_unicode(basedir):
|
||||||
|
# type: (Union[bytes, str]) -> str
|
||||||
|
return abspath_expanduser_unicode(ensure_text(basedir))
|
||||||
|
|
||||||
|
|
||||||
@attr.s
|
@attr.s
|
||||||
class _Config(object):
|
class _Config(object):
|
||||||
"""
|
"""
|
||||||
@ -300,8 +310,8 @@ class _Config(object):
|
|||||||
config = attr.ib(validator=attr.validators.instance_of(configparser.ConfigParser))
|
config = attr.ib(validator=attr.validators.instance_of(configparser.ConfigParser))
|
||||||
portnum_fname = attr.ib()
|
portnum_fname = attr.ib()
|
||||||
_basedir = attr.ib(
|
_basedir = attr.ib(
|
||||||
converter=lambda basedir: abspath_expanduser_unicode(ensure_text(basedir)),
|
converter=ensure_text_and_abspath_expanduser_unicode,
|
||||||
)
|
) # type: str
|
||||||
config_path = attr.ib(
|
config_path = attr.ib(
|
||||||
validator=attr.validators.optional(
|
validator=attr.validators.optional(
|
||||||
attr.validators.instance_of(FilePath),
|
attr.validators.instance_of(FilePath),
|
||||||
@ -927,7 +937,6 @@ class Node(service.MultiService):
|
|||||||
"""
|
"""
|
||||||
NODETYPE = "unknown NODETYPE"
|
NODETYPE = "unknown NODETYPE"
|
||||||
CERTFILE = "node.pem"
|
CERTFILE = "node.pem"
|
||||||
GENERATED_FILES = []
|
|
||||||
|
|
||||||
def __init__(self, config, main_tub, control_tub, i2p_provider, tor_provider):
|
def __init__(self, config, main_tub, control_tub, i2p_provider, tor_provider):
|
||||||
"""
|
"""
|
||||||
|
@ -1,5 +1,10 @@
|
|||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.python import usage
|
from twisted.python import usage
|
||||||
from allmydata.scripts.common import BaseOptions
|
from allmydata.scripts.common import BaseOptions
|
||||||
|
|
||||||
@ -79,8 +84,8 @@ def do_admin(options):
|
|||||||
|
|
||||||
|
|
||||||
subCommands = [
|
subCommands = [
|
||||||
["admin", None, AdminCommand, "admin subcommands: use 'tahoe admin' for a list"],
|
("admin", None, AdminCommand, "admin subcommands: use 'tahoe admin' for a list"),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
dispatch = {
|
dispatch = {
|
||||||
"admin": do_admin,
|
"admin": do_admin,
|
||||||
|
@ -1,6 +1,12 @@
|
|||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
|
|
||||||
import os.path, re, fnmatch
|
import os.path, re, fnmatch
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands, Parameters
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.python import usage
|
from twisted.python import usage
|
||||||
from allmydata.scripts.common import get_aliases, get_default_nodedir, \
|
from allmydata.scripts.common import get_aliases, get_default_nodedir, \
|
||||||
DEFAULT_ALIAS, BaseOptions
|
DEFAULT_ALIAS, BaseOptions
|
||||||
@ -19,7 +25,7 @@ class FileStoreOptions(BaseOptions):
|
|||||||
"This overrides the URL found in the --node-directory ."],
|
"This overrides the URL found in the --node-directory ."],
|
||||||
["dir-cap", None, None,
|
["dir-cap", None, None,
|
||||||
"Specify which dirnode URI should be used as the 'tahoe' alias."]
|
"Specify which dirnode URI should be used as the 'tahoe' alias."]
|
||||||
]
|
] # type: Parameters
|
||||||
|
|
||||||
def postOptions(self):
|
def postOptions(self):
|
||||||
self["quiet"] = self.parent["quiet"]
|
self["quiet"] = self.parent["quiet"]
|
||||||
@ -455,25 +461,25 @@ class DeepCheckOptions(FileStoreOptions):
|
|||||||
Optionally repair any problems found."""
|
Optionally repair any problems found."""
|
||||||
|
|
||||||
subCommands = [
|
subCommands = [
|
||||||
["mkdir", None, MakeDirectoryOptions, "Create a new directory."],
|
("mkdir", None, MakeDirectoryOptions, "Create a new directory."),
|
||||||
["add-alias", None, AddAliasOptions, "Add a new alias cap."],
|
("add-alias", None, AddAliasOptions, "Add a new alias cap."),
|
||||||
["create-alias", None, CreateAliasOptions, "Create a new alias cap."],
|
("create-alias", None, CreateAliasOptions, "Create a new alias cap."),
|
||||||
["list-aliases", None, ListAliasesOptions, "List all alias caps."],
|
("list-aliases", None, ListAliasesOptions, "List all alias caps."),
|
||||||
["ls", None, ListOptions, "List a directory."],
|
("ls", None, ListOptions, "List a directory."),
|
||||||
["get", None, GetOptions, "Retrieve a file from the grid."],
|
("get", None, GetOptions, "Retrieve a file from the grid."),
|
||||||
["put", None, PutOptions, "Upload a file into the grid."],
|
("put", None, PutOptions, "Upload a file into the grid."),
|
||||||
["cp", None, CpOptions, "Copy one or more files or directories."],
|
("cp", None, CpOptions, "Copy one or more files or directories."),
|
||||||
["unlink", None, UnlinkOptions, "Unlink a file or directory on the grid."],
|
("unlink", None, UnlinkOptions, "Unlink a file or directory on the grid."),
|
||||||
["mv", None, MvOptions, "Move a file within the grid."],
|
("mv", None, MvOptions, "Move a file within the grid."),
|
||||||
["ln", None, LnOptions, "Make an additional link to an existing file or directory."],
|
("ln", None, LnOptions, "Make an additional link to an existing file or directory."),
|
||||||
["backup", None, BackupOptions, "Make target dir look like local dir."],
|
("backup", None, BackupOptions, "Make target dir look like local dir."),
|
||||||
["webopen", None, WebopenOptions, "Open a web browser to a grid file or directory."],
|
("webopen", None, WebopenOptions, "Open a web browser to a grid file or directory."),
|
||||||
["manifest", None, ManifestOptions, "List all files/directories in a subtree."],
|
("manifest", None, ManifestOptions, "List all files/directories in a subtree."),
|
||||||
["stats", None, StatsOptions, "Print statistics about all files/directories in a subtree."],
|
("stats", None, StatsOptions, "Print statistics about all files/directories in a subtree."),
|
||||||
["check", None, CheckOptions, "Check a single file or directory."],
|
("check", None, CheckOptions, "Check a single file or directory."),
|
||||||
["deep-check", None, DeepCheckOptions, "Check all files/directories reachable from a starting point."],
|
("deep-check", None, DeepCheckOptions, "Check all files/directories reachable from a starting point."),
|
||||||
["status", None, TahoeStatusCommand, "Various status information."],
|
("status", None, TahoeStatusCommand, "Various status information."),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
def mkdir(options):
|
def mkdir(options):
|
||||||
from allmydata.scripts import tahoe_mkdir
|
from allmydata.scripts import tahoe_mkdir
|
||||||
|
@ -4,6 +4,12 @@ import os, sys, urllib, textwrap
|
|||||||
import codecs
|
import codecs
|
||||||
from os.path import join
|
from os.path import join
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Optional
|
||||||
|
from .types_ import Parameters
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from yaml import (
|
from yaml import (
|
||||||
safe_dump,
|
safe_dump,
|
||||||
)
|
)
|
||||||
@ -41,8 +47,8 @@ class BaseOptions(usage.Options):
|
|||||||
def opt_version(self):
|
def opt_version(self):
|
||||||
raise usage.UsageError("--version not allowed on subcommands")
|
raise usage.UsageError("--version not allowed on subcommands")
|
||||||
|
|
||||||
description = None
|
description = None # type: Optional[str]
|
||||||
description_unwrapped = None
|
description_unwrapped = None # type: Optional[str]
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
width = int(os.environ.get('COLUMNS', '80'))
|
width = int(os.environ.get('COLUMNS', '80'))
|
||||||
@ -65,7 +71,7 @@ class BasedirOptions(BaseOptions):
|
|||||||
optParameters = [
|
optParameters = [
|
||||||
["basedir", "C", None, "Specify which Tahoe base directory should be used. [default: %s]"
|
["basedir", "C", None, "Specify which Tahoe base directory should be used. [default: %s]"
|
||||||
% quote_local_unicode_path(_default_nodedir)],
|
% quote_local_unicode_path(_default_nodedir)],
|
||||||
]
|
] # type: Parameters
|
||||||
|
|
||||||
def parseArgs(self, basedir=None):
|
def parseArgs(self, basedir=None):
|
||||||
# This finds the node-directory option correctly even if we are in a subcommand.
|
# This finds the node-directory option correctly even if we are in a subcommand.
|
||||||
@ -102,7 +108,7 @@ class NoDefaultBasedirOptions(BasedirOptions):
|
|||||||
|
|
||||||
optParameters = [
|
optParameters = [
|
||||||
["basedir", "C", None, "Specify which Tahoe base directory should be used."],
|
["basedir", "C", None, "Specify which Tahoe base directory should be used."],
|
||||||
]
|
] # type: Parameters
|
||||||
|
|
||||||
# This is overridden in order to ensure we get a "Wrong number of arguments."
|
# This is overridden in order to ensure we get a "Wrong number of arguments."
|
||||||
# error when more than one argument is given.
|
# error when more than one argument is given.
|
||||||
|
@ -3,6 +3,11 @@ from __future__ import print_function
|
|||||||
import os
|
import os
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.internet import reactor, defer
|
from twisted.internet import reactor, defer
|
||||||
from twisted.python.usage import UsageError
|
from twisted.python.usage import UsageError
|
||||||
from twisted.python.filepath import (
|
from twisted.python.filepath import (
|
||||||
@ -492,10 +497,10 @@ def create_introducer(config):
|
|||||||
|
|
||||||
|
|
||||||
subCommands = [
|
subCommands = [
|
||||||
["create-node", None, CreateNodeOptions, "Create a node that acts as a client, server or both."],
|
("create-node", None, CreateNodeOptions, "Create a node that acts as a client, server or both."),
|
||||||
["create-client", None, CreateClientOptions, "Create a client node (with storage initially disabled)."],
|
("create-client", None, CreateClientOptions, "Create a client node (with storage initially disabled)."),
|
||||||
["create-introducer", None, CreateIntroducerOptions, "Create an introducer node."],
|
("create-introducer", None, CreateIntroducerOptions, "Create an introducer node."),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
dispatch = {
|
dispatch = {
|
||||||
"create-node": create_node,
|
"create-node": create_node,
|
||||||
|
@ -1,5 +1,10 @@
|
|||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from future.utils import bchr
|
from future.utils import bchr
|
||||||
|
|
||||||
# do not import any allmydata modules at this level. Do that from inside
|
# do not import any allmydata modules at this level. Do that from inside
|
||||||
@ -1053,8 +1058,8 @@ def do_debug(options):
|
|||||||
|
|
||||||
|
|
||||||
subCommands = [
|
subCommands = [
|
||||||
["debug", None, DebugCommand, "debug subcommands: use 'tahoe debug' for a list."],
|
("debug", None, DebugCommand, "debug subcommands: use 'tahoe debug' for a list."),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
dispatch = {
|
dispatch = {
|
||||||
"debug": do_debug,
|
"debug": do_debug,
|
||||||
|
@ -4,6 +4,11 @@ import os, sys
|
|||||||
from six.moves import StringIO
|
from six.moves import StringIO
|
||||||
import six
|
import six
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.python import usage
|
from twisted.python import usage
|
||||||
from twisted.internet import defer, task, threads
|
from twisted.internet import defer, task, threads
|
||||||
|
|
||||||
@ -40,8 +45,8 @@ _control_node_dispatch = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
process_control_commands = [
|
process_control_commands = [
|
||||||
["run", None, tahoe_run.RunOptions, "run a node without daemonizing"],
|
("run", None, tahoe_run.RunOptions, "run a node without daemonizing"),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
|
|
||||||
class Options(usage.Options):
|
class Options(usage.Options):
|
||||||
@ -98,7 +103,7 @@ class Options(usage.Options):
|
|||||||
|
|
||||||
create_dispatch = {}
|
create_dispatch = {}
|
||||||
for module in (create_node,):
|
for module in (create_node,):
|
||||||
create_dispatch.update(module.dispatch)
|
create_dispatch.update(module.dispatch) # type: ignore
|
||||||
|
|
||||||
def parse_options(argv, config=None):
|
def parse_options(argv, config=None):
|
||||||
if not config:
|
if not config:
|
||||||
|
@ -2,6 +2,11 @@ from __future__ import print_function
|
|||||||
|
|
||||||
import json
|
import json
|
||||||
|
|
||||||
|
try:
|
||||||
|
from allmydata.scripts.types_ import SubCommands
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.python import usage
|
from twisted.python import usage
|
||||||
from twisted.internet import defer, reactor
|
from twisted.internet import defer, reactor
|
||||||
|
|
||||||
@ -103,7 +108,7 @@ def invite(options):
|
|||||||
subCommands = [
|
subCommands = [
|
||||||
("invite", None, InviteOptions,
|
("invite", None, InviteOptions,
|
||||||
"Invite a new node to this grid"),
|
"Invite a new node to this grid"),
|
||||||
]
|
] # type: SubCommands
|
||||||
|
|
||||||
dispatch = {
|
dispatch = {
|
||||||
"invite": invite,
|
"invite": invite,
|
||||||
|
12
src/allmydata/scripts/types_.py
Normal file
12
src/allmydata/scripts/types_.py
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
from typing import List, Tuple, Type, Sequence, Any
|
||||||
|
from allmydata.scripts.common import BaseOptions
|
||||||
|
|
||||||
|
|
||||||
|
# Historically, subcommands were implemented as lists, but due to a
|
||||||
|
# [designed contraint in mypy](https://stackoverflow.com/a/52559625/70170),
|
||||||
|
# a Tuple is required.
|
||||||
|
SubCommand = Tuple[str, None, Type[BaseOptions], str]
|
||||||
|
|
||||||
|
SubCommands = List[SubCommand]
|
||||||
|
|
||||||
|
Parameters = List[Sequence[Any]]
|
@ -23,7 +23,7 @@ from allmydata.interfaces import IStatsProducer
|
|||||||
@implementer(IStatsProducer)
|
@implementer(IStatsProducer)
|
||||||
class CPUUsageMonitor(service.MultiService):
|
class CPUUsageMonitor(service.MultiService):
|
||||||
HISTORY_LENGTH = 15
|
HISTORY_LENGTH = 15
|
||||||
POLL_INTERVAL = 60
|
POLL_INTERVAL = 60 # type: float
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
service.MultiService.__init__(self)
|
service.MultiService.__init__(self)
|
||||||
|
@ -19,7 +19,7 @@ import os, time, struct
|
|||||||
try:
|
try:
|
||||||
import cPickle as pickle
|
import cPickle as pickle
|
||||||
except ImportError:
|
except ImportError:
|
||||||
import pickle
|
import pickle # type: ignore
|
||||||
from twisted.internet import reactor
|
from twisted.internet import reactor
|
||||||
from twisted.application import service
|
from twisted.application import service
|
||||||
from allmydata.storage.common import si_b2a
|
from allmydata.storage.common import si_b2a
|
||||||
|
@ -202,7 +202,7 @@ class ShareFile(object):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(RIBucketWriter)
|
@implementer(RIBucketWriter)
|
||||||
class BucketWriter(Referenceable):
|
class BucketWriter(Referenceable): # type: ignore # warner/foolscap#78
|
||||||
|
|
||||||
def __init__(self, ss, incominghome, finalhome, max_size, lease_info, canary):
|
def __init__(self, ss, incominghome, finalhome, max_size, lease_info, canary):
|
||||||
self.ss = ss
|
self.ss = ss
|
||||||
@ -301,7 +301,7 @@ class BucketWriter(Referenceable):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(RIBucketReader)
|
@implementer(RIBucketReader)
|
||||||
class BucketReader(Referenceable):
|
class BucketReader(Referenceable): # type: ignore # warner/foolscap#78
|
||||||
|
|
||||||
def __init__(self, ss, sharefname, storage_index=None, shnum=None):
|
def __init__(self, ss, sharefname, storage_index=None, shnum=None):
|
||||||
self.ss = ss
|
self.ss = ss
|
||||||
|
@ -581,7 +581,7 @@ class StorageServer(service.MultiService, Referenceable):
|
|||||||
for share in six.viewvalues(shares):
|
for share in six.viewvalues(shares):
|
||||||
share.add_or_renew_lease(lease_info)
|
share.add_or_renew_lease(lease_info)
|
||||||
|
|
||||||
def slot_testv_and_readv_and_writev(
|
def slot_testv_and_readv_and_writev( # type: ignore # warner/foolscap#78
|
||||||
self,
|
self,
|
||||||
storage_index,
|
storage_index,
|
||||||
secrets,
|
secrets,
|
||||||
|
@ -1,5 +1,43 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
## Copyright (C) 2021 Valentin Lab
|
||||||
|
##
|
||||||
|
## Redistribution and use in source and binary forms, with or without
|
||||||
|
## modification, are permitted provided that the following conditions
|
||||||
|
## are met:
|
||||||
|
##
|
||||||
|
## 1. Redistributions of source code must retain the above copyright
|
||||||
|
## notice, this list of conditions and the following disclaimer.
|
||||||
|
##
|
||||||
|
## 2. Redistributions in binary form must reproduce the above
|
||||||
|
## copyright notice, this list of conditions and the following
|
||||||
|
## disclaimer in the documentation and/or other materials provided
|
||||||
|
## with the distribution.
|
||||||
|
##
|
||||||
|
## 3. Neither the name of the copyright holder nor the names of its
|
||||||
|
## contributors may be used to endorse or promote products derived
|
||||||
|
## from this software without specific prior written permission.
|
||||||
|
##
|
||||||
|
## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
## "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
## LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
## FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
## COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
||||||
|
## INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
## (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
## SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
## HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||||||
|
## STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
## ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
||||||
|
## OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
##
|
||||||
|
|
||||||
## issue: https://bugs.python.org/issue19264
|
## issue: https://bugs.python.org/issue19264
|
||||||
|
|
||||||
|
# See allmydata/windows/fixups.py
|
||||||
|
import sys
|
||||||
|
assert sys.platform == "win32"
|
||||||
|
|
||||||
import os
|
import os
|
||||||
import ctypes
|
import ctypes
|
||||||
import subprocess
|
import subprocess
|
||||||
|
@ -37,6 +37,11 @@ a mean of 10kB and a max of 100MB, so filesize=min(int(1.0/random(.0002)),1e8)
|
|||||||
import os, sys, httplib, binascii
|
import os, sys, httplib, binascii
|
||||||
import urllib, json, random, time, urlparse
|
import urllib, json, random, time, urlparse
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Dict
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
# Python 2 compatibility
|
# Python 2 compatibility
|
||||||
from future.utils import PY2
|
from future.utils import PY2
|
||||||
if PY2:
|
if PY2:
|
||||||
@ -49,13 +54,13 @@ if sys.argv[1] == "--stats":
|
|||||||
DELAY = 10
|
DELAY = 10
|
||||||
MAXSAMPLES = 6
|
MAXSAMPLES = 6
|
||||||
totals = []
|
totals = []
|
||||||
last_stats = {}
|
last_stats = {} # type: Dict[str, float]
|
||||||
while True:
|
while True:
|
||||||
stats = {}
|
stats = {} # type: Dict[str, float]
|
||||||
for sf in statsfiles:
|
for sf in statsfiles:
|
||||||
for line in open(sf, "r").readlines():
|
for line in open(sf, "r").readlines():
|
||||||
name, value = line.split(":")
|
name, str_value = line.split(":")
|
||||||
value = int(value.strip())
|
value = int(str_value.strip())
|
||||||
if name not in stats:
|
if name not in stats:
|
||||||
stats[name] = 0
|
stats[name] = 0
|
||||||
stats[name] += float(value)
|
stats[name] += float(value)
|
||||||
|
@ -508,13 +508,13 @@ if __name__ == '__main__':
|
|||||||
mode = "upload"
|
mode = "upload"
|
||||||
if len(sys.argv) > 1:
|
if len(sys.argv) > 1:
|
||||||
mode = sys.argv[1]
|
mode = sys.argv[1]
|
||||||
if sys.maxint == 2147483647:
|
if sys.maxsize == 2147483647:
|
||||||
bits = "32"
|
bits = "32"
|
||||||
elif sys.maxint == 9223372036854775807:
|
elif sys.maxsize == 9223372036854775807:
|
||||||
bits = "64"
|
bits = "64"
|
||||||
else:
|
else:
|
||||||
bits = "?"
|
bits = "?"
|
||||||
print("%s-bit system (sys.maxint=%d)" % (bits, sys.maxint))
|
print("%s-bit system (sys.maxsize=%d)" % (bits, sys.maxsize))
|
||||||
# put the logfile and stats.out in _test_memory/ . These stick around.
|
# put the logfile and stats.out in _test_memory/ . These stick around.
|
||||||
# put the nodes and other files in _test_memory/test/ . These are
|
# put the nodes and other files in _test_memory/test/ . These are
|
||||||
# removed each time we run.
|
# removed each time we run.
|
||||||
|
@ -428,7 +428,7 @@ class DummyProducer(object):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
@implementer(IImmutableFileNode)
|
@implementer(IImmutableFileNode)
|
||||||
class FakeCHKFileNode(object):
|
class FakeCHKFileNode(object): # type: ignore # incomplete implementation
|
||||||
"""I provide IImmutableFileNode, but all of my data is stored in a
|
"""I provide IImmutableFileNode, but all of my data is stored in a
|
||||||
class-level dictionary."""
|
class-level dictionary."""
|
||||||
|
|
||||||
@ -566,7 +566,7 @@ def create_chk_filenode(contents, all_contents):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(IMutableFileNode, ICheckable)
|
@implementer(IMutableFileNode, ICheckable)
|
||||||
class FakeMutableFileNode(object):
|
class FakeMutableFileNode(object): # type: ignore # incomplete implementation
|
||||||
"""I provide IMutableFileNode, but all of my data is stored in a
|
"""I provide IMutableFileNode, but all of my data is stored in a
|
||||||
class-level dictionary."""
|
class-level dictionary."""
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ class Marker(object):
|
|||||||
|
|
||||||
fireNow = partial(defer.succeed, None)
|
fireNow = partial(defer.succeed, None)
|
||||||
|
|
||||||
@implementer(IRemoteReference)
|
@implementer(IRemoteReference) # type: ignore # warner/foolscap#79
|
||||||
class LocalWrapper(object):
|
class LocalWrapper(object):
|
||||||
"""
|
"""
|
||||||
A ``LocalWrapper`` presents the remote reference interface to a local
|
A ``LocalWrapper`` presents the remote reference interface to a local
|
||||||
@ -213,9 +213,12 @@ class NoNetworkServer(object):
|
|||||||
return _StorageServer(lambda: self.rref)
|
return _StorageServer(lambda: self.rref)
|
||||||
def get_version(self):
|
def get_version(self):
|
||||||
return self.rref.version
|
return self.rref.version
|
||||||
|
def start_connecting(self, trigger_cb):
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
@implementer(IStorageBroker)
|
@implementer(IStorageBroker)
|
||||||
class NoNetworkStorageBroker(object):
|
class NoNetworkStorageBroker(object): # type: ignore # missing many methods
|
||||||
def get_servers_for_psi(self, peer_selection_index):
|
def get_servers_for_psi(self, peer_selection_index):
|
||||||
def _permuted(server):
|
def _permuted(server):
|
||||||
seed = server.get_permutation_seed()
|
seed = server.get_permutation_seed()
|
||||||
@ -259,7 +262,7 @@ def create_no_network_client(basedir):
|
|||||||
return defer.succeed(client)
|
return defer.succeed(client)
|
||||||
|
|
||||||
|
|
||||||
class _NoNetworkClient(_Client):
|
class _NoNetworkClient(_Client): # type: ignore # tahoe-lafs/ticket/3573
|
||||||
"""
|
"""
|
||||||
Overrides all _Client networking functionality to do nothing.
|
Overrides all _Client networking functionality to do nothing.
|
||||||
"""
|
"""
|
||||||
|
@ -47,8 +47,9 @@ class RIDummy(RemoteInterface):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
# type ignored due to missing stubs for Twisted
|
||||||
@implementer(IFoolscapStoragePlugin)
|
# https://twistedmatrix.com/trac/ticket/9717
|
||||||
|
@implementer(IFoolscapStoragePlugin) # type: ignore
|
||||||
@attr.s
|
@attr.s
|
||||||
class DummyStorage(object):
|
class DummyStorage(object):
|
||||||
name = attr.ib()
|
name = attr.ib()
|
||||||
@ -107,7 +108,7 @@ class GetCounter(Resource, object):
|
|||||||
|
|
||||||
@implementer(RIDummy)
|
@implementer(RIDummy)
|
||||||
@attr.s(frozen=True)
|
@attr.s(frozen=True)
|
||||||
class DummyStorageServer(object):
|
class DummyStorageServer(object): # type: ignore # warner/foolscap#78
|
||||||
get_anonymous_storage_server = attr.ib()
|
get_anonymous_storage_server = attr.ib()
|
||||||
|
|
||||||
def remote_just_some_method(self):
|
def remote_just_some_method(self):
|
||||||
@ -116,7 +117,7 @@ class DummyStorageServer(object):
|
|||||||
|
|
||||||
@implementer(IStorageServer)
|
@implementer(IStorageServer)
|
||||||
@attr.s
|
@attr.s
|
||||||
class DummyStorageClient(object):
|
class DummyStorageClient(object): # type: ignore # incomplete implementation
|
||||||
get_rref = attr.ib()
|
get_rref = attr.ib()
|
||||||
configuration = attr.ib()
|
configuration = attr.ib()
|
||||||
announcement = attr.ib()
|
announcement = attr.ib()
|
||||||
|
@ -62,7 +62,7 @@ class FakeClient(object):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(IServer)
|
@implementer(IServer)
|
||||||
class FakeServer(object):
|
class FakeServer(object): # type: ignore # incomplete implementation
|
||||||
|
|
||||||
def get_name(self):
|
def get_name(self):
|
||||||
return "fake name"
|
return "fake name"
|
||||||
@ -75,7 +75,7 @@ class FakeServer(object):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(ICheckResults)
|
@implementer(ICheckResults)
|
||||||
class FakeCheckResults(object):
|
class FakeCheckResults(object): # type: ignore # incomplete implementation
|
||||||
|
|
||||||
def __init__(self, si=None,
|
def __init__(self, si=None,
|
||||||
healthy=False, recoverable=False,
|
healthy=False, recoverable=False,
|
||||||
@ -106,7 +106,7 @@ class FakeCheckResults(object):
|
|||||||
|
|
||||||
|
|
||||||
@implementer(ICheckAndRepairResults)
|
@implementer(ICheckAndRepairResults)
|
||||||
class FakeCheckAndRepairResults(object):
|
class FakeCheckAndRepairResults(object): # type: ignore # incomplete implementation
|
||||||
|
|
||||||
def __init__(self, si=None,
|
def __init__(self, si=None,
|
||||||
repair_attempted=False,
|
repair_attempted=False,
|
||||||
|
@ -51,7 +51,6 @@ from allmydata.nodemaker import (
|
|||||||
NodeMaker,
|
NodeMaker,
|
||||||
)
|
)
|
||||||
from allmydata.node import OldConfigError, UnescapedHashError, create_node_dir
|
from allmydata.node import OldConfigError, UnescapedHashError, create_node_dir
|
||||||
from allmydata.frontends.auth import NeedRootcapLookupScheme
|
|
||||||
from allmydata import client
|
from allmydata import client
|
||||||
from allmydata.storage_client import (
|
from allmydata.storage_client import (
|
||||||
StorageClientConfig,
|
StorageClientConfig,
|
||||||
@ -424,88 +423,8 @@ class Basic(testutil.ReallyEqualMixin, unittest.TestCase):
|
|||||||
expected = fileutil.abspath_expanduser_unicode(u"relative", abs_basedir)
|
expected = fileutil.abspath_expanduser_unicode(u"relative", abs_basedir)
|
||||||
self.failUnlessReallyEqual(w.staticdir, expected)
|
self.failUnlessReallyEqual(w.staticdir, expected)
|
||||||
|
|
||||||
# TODO: also test config options for SFTP.
|
# TODO: also test config options for SFTP. See Git history for deleted FTP
|
||||||
|
# tests that could be used as basis for these tests.
|
||||||
@defer.inlineCallbacks
|
|
||||||
def test_ftp_create(self):
|
|
||||||
"""
|
|
||||||
configuration for sftpd results in it being started
|
|
||||||
"""
|
|
||||||
root = FilePath(self.mktemp())
|
|
||||||
root.makedirs()
|
|
||||||
accounts = root.child(b"sftp-accounts")
|
|
||||||
accounts.touch()
|
|
||||||
|
|
||||||
data = FilePath(__file__).sibling(b"data")
|
|
||||||
privkey = data.child(b"openssh-rsa-2048.txt")
|
|
||||||
pubkey = data.child(b"openssh-rsa-2048.pub.txt")
|
|
||||||
|
|
||||||
basedir = u"client.Basic.test_ftp_create"
|
|
||||||
create_node_dir(basedir, "testing")
|
|
||||||
with open(os.path.join(basedir, "tahoe.cfg"), "w") as f:
|
|
||||||
f.write((
|
|
||||||
'[sftpd]\n'
|
|
||||||
'enabled = true\n'
|
|
||||||
'accounts.file = {}\n'
|
|
||||||
'host_pubkey_file = {}\n'
|
|
||||||
'host_privkey_file = {}\n'
|
|
||||||
).format(accounts.path, pubkey.path, privkey.path))
|
|
||||||
|
|
||||||
client_node = yield client.create_client(
|
|
||||||
basedir,
|
|
||||||
)
|
|
||||||
sftp = client_node.getServiceNamed("frontend:sftp")
|
|
||||||
self.assertIs(sftp.parent, client_node)
|
|
||||||
|
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
|
||||||
def test_ftp_auth_keyfile(self):
|
|
||||||
"""
|
|
||||||
ftpd accounts.file is parsed properly
|
|
||||||
"""
|
|
||||||
basedir = u"client.Basic.test_ftp_auth_keyfile"
|
|
||||||
os.mkdir(basedir)
|
|
||||||
fileutil.write(os.path.join(basedir, "tahoe.cfg"),
|
|
||||||
(BASECONFIG +
|
|
||||||
"[ftpd]\n"
|
|
||||||
"enabled = true\n"
|
|
||||||
"port = tcp:0:interface=127.0.0.1\n"
|
|
||||||
"accounts.file = private/accounts\n"))
|
|
||||||
os.mkdir(os.path.join(basedir, "private"))
|
|
||||||
fileutil.write(os.path.join(basedir, "private", "accounts"), "\n")
|
|
||||||
c = yield client.create_client(basedir) # just make sure it can be instantiated
|
|
||||||
del c
|
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
|
||||||
def test_ftp_auth_url(self):
|
|
||||||
"""
|
|
||||||
ftpd accounts.url is parsed properly
|
|
||||||
"""
|
|
||||||
basedir = u"client.Basic.test_ftp_auth_url"
|
|
||||||
os.mkdir(basedir)
|
|
||||||
fileutil.write(os.path.join(basedir, "tahoe.cfg"),
|
|
||||||
(BASECONFIG +
|
|
||||||
"[ftpd]\n"
|
|
||||||
"enabled = true\n"
|
|
||||||
"port = tcp:0:interface=127.0.0.1\n"
|
|
||||||
"accounts.url = http://0.0.0.0/\n"))
|
|
||||||
c = yield client.create_client(basedir) # just make sure it can be instantiated
|
|
||||||
del c
|
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
|
||||||
def test_ftp_auth_no_accountfile_or_url(self):
|
|
||||||
"""
|
|
||||||
ftpd requires some way to look up accounts
|
|
||||||
"""
|
|
||||||
basedir = u"client.Basic.test_ftp_auth_no_accountfile_or_url"
|
|
||||||
os.mkdir(basedir)
|
|
||||||
fileutil.write(os.path.join(basedir, "tahoe.cfg"),
|
|
||||||
(BASECONFIG +
|
|
||||||
"[ftpd]\n"
|
|
||||||
"enabled = true\n"
|
|
||||||
"port = tcp:0:interface=127.0.0.1\n"))
|
|
||||||
with self.assertRaises(NeedRootcapLookupScheme):
|
|
||||||
yield client.create_client(basedir)
|
|
||||||
|
|
||||||
@defer.inlineCallbacks
|
@defer.inlineCallbacks
|
||||||
def _storage_dir_test(self, basedir, storage_path, expected_path):
|
def _storage_dir_test(self, basedir, storage_path, expected_path):
|
||||||
|
@ -74,3 +74,58 @@ class DeferredUtilTests(unittest.TestCase, deferredutil.WaitForDelayedCallsMixin
|
|||||||
d = defer.succeed(None)
|
d = defer.succeed(None)
|
||||||
d.addBoth(self.wait_for_delayed_calls)
|
d.addBoth(self.wait_for_delayed_calls)
|
||||||
return d
|
return d
|
||||||
|
|
||||||
|
|
||||||
|
class UntilTests(unittest.TestCase):
|
||||||
|
"""
|
||||||
|
Tests for ``deferredutil.until``.
|
||||||
|
"""
|
||||||
|
def test_exception(self):
|
||||||
|
"""
|
||||||
|
If the action raises an exception, the ``Deferred`` returned by ``until``
|
||||||
|
fires with a ``Failure``.
|
||||||
|
"""
|
||||||
|
self.assertFailure(
|
||||||
|
deferredutil.until(lambda: 1/0, lambda: True),
|
||||||
|
ZeroDivisionError,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_stops_on_condition(self):
|
||||||
|
"""
|
||||||
|
The action is called repeatedly until ``condition`` returns ``True``.
|
||||||
|
"""
|
||||||
|
calls = []
|
||||||
|
def action():
|
||||||
|
calls.append(None)
|
||||||
|
|
||||||
|
def condition():
|
||||||
|
return len(calls) == 3
|
||||||
|
|
||||||
|
self.assertIs(
|
||||||
|
self.successResultOf(
|
||||||
|
deferredutil.until(action, condition),
|
||||||
|
),
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
self.assertEqual(3, len(calls))
|
||||||
|
|
||||||
|
def test_waits_for_deferred(self):
|
||||||
|
"""
|
||||||
|
If the action returns a ``Deferred`` then it is called again when the
|
||||||
|
``Deferred`` fires.
|
||||||
|
"""
|
||||||
|
counter = [0]
|
||||||
|
r1 = defer.Deferred()
|
||||||
|
r2 = defer.Deferred()
|
||||||
|
results = [r1, r2]
|
||||||
|
def action():
|
||||||
|
counter[0] += 1
|
||||||
|
return results.pop(0)
|
||||||
|
|
||||||
|
def condition():
|
||||||
|
return False
|
||||||
|
|
||||||
|
deferredutil.until(action, condition)
|
||||||
|
self.assertEqual([1], counter)
|
||||||
|
r1.callback(None)
|
||||||
|
self.assertEqual([2], counter)
|
||||||
|
@ -1561,7 +1561,7 @@ class Packing(testutil.ReallyEqualMixin, unittest.TestCase):
|
|||||||
kids, fn.get_writekey(), deep_immutable=True)
|
kids, fn.get_writekey(), deep_immutable=True)
|
||||||
|
|
||||||
@implementer(IMutableFileNode)
|
@implementer(IMutableFileNode)
|
||||||
class FakeMutableFile(object):
|
class FakeMutableFile(object): # type: ignore # incomplete implementation
|
||||||
counter = 0
|
counter = 0
|
||||||
def __init__(self, initial_contents=b""):
|
def __init__(self, initial_contents=b""):
|
||||||
data = self._get_initial_contents(initial_contents)
|
data = self._get_initial_contents(initial_contents)
|
||||||
@ -1622,7 +1622,7 @@ class FakeNodeMaker(NodeMaker):
|
|||||||
def create_mutable_file(self, contents=b"", keysize=None, version=None):
|
def create_mutable_file(self, contents=b"", keysize=None, version=None):
|
||||||
return defer.succeed(FakeMutableFile(contents))
|
return defer.succeed(FakeMutableFile(contents))
|
||||||
|
|
||||||
class FakeClient2(_Client):
|
class FakeClient2(_Client): # type: ignore # tahoe-lafs/ticket/3573
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.nodemaker = FakeNodeMaker(None, None, None,
|
self.nodemaker = FakeNodeMaker(None, None, None,
|
||||||
None, None,
|
None, None,
|
||||||
|
@ -1,106 +0,0 @@
|
|||||||
|
|
||||||
from twisted.trial import unittest
|
|
||||||
|
|
||||||
from allmydata.frontends import ftpd
|
|
||||||
from allmydata.immutable import upload
|
|
||||||
from allmydata.mutable import publish
|
|
||||||
from allmydata.test.no_network import GridTestMixin
|
|
||||||
from allmydata.test.common_util import ReallyEqualMixin
|
|
||||||
|
|
||||||
class Handler(GridTestMixin, ReallyEqualMixin, unittest.TestCase):
|
|
||||||
"""
|
|
||||||
This is a no-network unit test of ftpd.Handler and the abstractions
|
|
||||||
it uses.
|
|
||||||
"""
|
|
||||||
|
|
||||||
FALL_OF_BERLIN_WALL = 626644800
|
|
||||||
TURN_OF_MILLENIUM = 946684800
|
|
||||||
|
|
||||||
def _set_up(self, basedir, num_clients=1, num_servers=10):
|
|
||||||
self.basedir = "ftp/" + basedir
|
|
||||||
self.set_up_grid(num_clients=num_clients, num_servers=num_servers,
|
|
||||||
oneshare=True)
|
|
||||||
|
|
||||||
self.client = self.g.clients[0]
|
|
||||||
self.username = "alice"
|
|
||||||
self.convergence = ""
|
|
||||||
|
|
||||||
d = self.client.create_dirnode()
|
|
||||||
def _created_root(node):
|
|
||||||
self.root = node
|
|
||||||
self.root_uri = node.get_uri()
|
|
||||||
self.handler = ftpd.Handler(self.client, self.root, self.username,
|
|
||||||
self.convergence)
|
|
||||||
d.addCallback(_created_root)
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _set_metadata(self, name, metadata):
|
|
||||||
"""Set metadata for `name', avoiding MetadataSetter's timestamp reset
|
|
||||||
behavior."""
|
|
||||||
def _modifier(old_contents, servermap, first_time):
|
|
||||||
children = self.root._unpack_contents(old_contents)
|
|
||||||
children[name] = (children[name][0], metadata)
|
|
||||||
return self.root._pack_contents(children)
|
|
||||||
|
|
||||||
return self.root._node.modify(_modifier)
|
|
||||||
|
|
||||||
def _set_up_tree(self):
|
|
||||||
# add immutable file at root
|
|
||||||
immutable = upload.Data("immutable file contents", None)
|
|
||||||
d = self.root.add_file(u"immutable", immutable)
|
|
||||||
|
|
||||||
# `mtime' and `linkmotime' both set
|
|
||||||
md_both = {'mtime': self.FALL_OF_BERLIN_WALL,
|
|
||||||
'tahoe': {'linkmotime': self.TURN_OF_MILLENIUM}}
|
|
||||||
d.addCallback(lambda _: self._set_metadata(u"immutable", md_both))
|
|
||||||
|
|
||||||
# add link to root from root
|
|
||||||
d.addCallback(lambda _: self.root.set_node(u"loop", self.root))
|
|
||||||
|
|
||||||
# `mtime' set, but no `linkmotime'
|
|
||||||
md_just_mtime = {'mtime': self.FALL_OF_BERLIN_WALL, 'tahoe': {}}
|
|
||||||
d.addCallback(lambda _: self._set_metadata(u"loop", md_just_mtime))
|
|
||||||
|
|
||||||
# add mutable file at root
|
|
||||||
mutable = publish.MutableData("mutable file contents")
|
|
||||||
d.addCallback(lambda _: self.client.create_mutable_file(mutable))
|
|
||||||
d.addCallback(lambda node: self.root.set_node(u"mutable", node))
|
|
||||||
|
|
||||||
# neither `mtime' nor `linkmotime' set
|
|
||||||
d.addCallback(lambda _: self._set_metadata(u"mutable", {}))
|
|
||||||
|
|
||||||
return d
|
|
||||||
|
|
||||||
def _compareDirLists(self, actual, expected):
|
|
||||||
actual_list = sorted(actual)
|
|
||||||
expected_list = sorted(expected)
|
|
||||||
|
|
||||||
self.failUnlessReallyEqual(len(actual_list), len(expected_list),
|
|
||||||
"%r is wrong length, expecting %r" % (
|
|
||||||
actual_list, expected_list))
|
|
||||||
for (a, b) in zip(actual_list, expected_list):
|
|
||||||
(name, meta) = a
|
|
||||||
(expected_name, expected_meta) = b
|
|
||||||
self.failUnlessReallyEqual(name, expected_name)
|
|
||||||
self.failUnlessReallyEqual(meta, expected_meta)
|
|
||||||
|
|
||||||
def test_list(self):
|
|
||||||
keys = ("size", "directory", "permissions", "hardlinks", "modified",
|
|
||||||
"owner", "group", "unexpected")
|
|
||||||
d = self._set_up("list")
|
|
||||||
|
|
||||||
d.addCallback(lambda _: self._set_up_tree())
|
|
||||||
d.addCallback(lambda _: self.handler.list("", keys=keys))
|
|
||||||
|
|
||||||
expected_root = [
|
|
||||||
('loop',
|
|
||||||
[0, True, ftpd.IntishPermissions(0o600), 1, self.FALL_OF_BERLIN_WALL, 'alice', 'alice', '??']),
|
|
||||||
('immutable',
|
|
||||||
[23, False, ftpd.IntishPermissions(0o600), 1, self.TURN_OF_MILLENIUM, 'alice', 'alice', '??']),
|
|
||||||
('mutable',
|
|
||||||
# timestamp should be 0 if no timestamp metadata is present
|
|
||||||
[0, False, ftpd.IntishPermissions(0o600), 1, 0, 'alice', 'alice', '??'])]
|
|
||||||
|
|
||||||
d.addCallback(lambda root: self._compareDirLists(root, expected_root))
|
|
||||||
|
|
||||||
return d
|
|
@ -19,6 +19,12 @@ from functools import (
|
|||||||
)
|
)
|
||||||
import attr
|
import attr
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import List
|
||||||
|
from allmydata.introducer.client import IntroducerClient
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.internet import defer
|
from twisted.internet import defer
|
||||||
from twisted.trial import unittest
|
from twisted.trial import unittest
|
||||||
from twisted.application import service
|
from twisted.application import service
|
||||||
@ -125,7 +131,7 @@ class FakeCHKCheckerAndUEBFetcher(object):
|
|||||||
))
|
))
|
||||||
|
|
||||||
class FakeClient(service.MultiService):
|
class FakeClient(service.MultiService):
|
||||||
introducer_clients = []
|
introducer_clients = [] # type: List[IntroducerClient]
|
||||||
DEFAULT_ENCODING_PARAMETERS = {"k":25,
|
DEFAULT_ENCODING_PARAMETERS = {"k":25,
|
||||||
"happy": 75,
|
"happy": 75,
|
||||||
"n": 100,
|
"n": 100,
|
||||||
|
@ -564,7 +564,7 @@ class TestMissingPorts(unittest.TestCase):
|
|||||||
config = config_from_string(self.basedir, "portnum", config_data)
|
config = config_from_string(self.basedir, "portnum", config_data)
|
||||||
with self.assertRaises(PortAssignmentRequired):
|
with self.assertRaises(PortAssignmentRequired):
|
||||||
_tub_portlocation(config, None, None)
|
_tub_portlocation(config, None, None)
|
||||||
test_listen_on_zero_with_host.todo = native_str(
|
test_listen_on_zero_with_host.todo = native_str( # type: ignore
|
||||||
"https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3563"
|
"https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3563"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -14,6 +14,7 @@ from testtools.matchers import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
BLACKLIST = {
|
BLACKLIST = {
|
||||||
|
"allmydata.scripts.types_",
|
||||||
"allmydata.test.check_load",
|
"allmydata.test.check_load",
|
||||||
"allmydata.test._win_subprocess",
|
"allmydata.test._win_subprocess",
|
||||||
"allmydata.windows.registry",
|
"allmydata.windows.registry",
|
||||||
|
@ -44,7 +44,7 @@ class Python3PortingEffortTests(SynchronousTestCase):
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
test_finished_porting.todo = native_str(
|
test_finished_porting.todo = native_str( # type: ignore
|
||||||
"https://tahoe-lafs.org/trac/tahoe-lafs/milestone/Support%20Python%203 should be completed",
|
"https://tahoe-lafs.org/trac/tahoe-lafs/milestone/Support%20Python%203 should be completed",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -105,7 +105,8 @@ from allmydata.interfaces import (
|
|||||||
|
|
||||||
SOME_FURL = "pb://abcde@nowhere/fake"
|
SOME_FURL = "pb://abcde@nowhere/fake"
|
||||||
|
|
||||||
class NativeStorageServerWithVersion(NativeStorageServer):
|
|
||||||
|
class NativeStorageServerWithVersion(NativeStorageServer): # type: ignore # tahoe-lafs/ticket/3573
|
||||||
def __init__(self, version):
|
def __init__(self, version):
|
||||||
# note: these instances won't work for anything other than
|
# note: these instances won't work for anything other than
|
||||||
# get_available_space() because we don't upcall
|
# get_available_space() because we don't upcall
|
||||||
@ -569,7 +570,7 @@ class SpyEndpoint(object):
|
|||||||
return d
|
return d
|
||||||
|
|
||||||
|
|
||||||
@implementer(IConnectionHintHandler)
|
@implementer(IConnectionHintHandler) # type: ignore # warner/foolscap#78
|
||||||
@attr.s
|
@attr.s
|
||||||
class SpyHandler(object):
|
class SpyHandler(object):
|
||||||
"""
|
"""
|
||||||
|
@ -14,6 +14,17 @@ if PY2:
|
|||||||
|
|
||||||
import os, shutil
|
import os, shutil
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
from base64 import (
|
||||||
|
b64encode,
|
||||||
|
)
|
||||||
|
|
||||||
|
from hypothesis import (
|
||||||
|
given,
|
||||||
|
)
|
||||||
|
from hypothesis.strategies import (
|
||||||
|
just,
|
||||||
|
integers,
|
||||||
|
)
|
||||||
|
|
||||||
from twisted.trial import unittest
|
from twisted.trial import unittest
|
||||||
from twisted.python.failure import Failure
|
from twisted.python.failure import Failure
|
||||||
@ -877,6 +888,34 @@ def is_happy_enough(servertoshnums, h, k):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class FileHandleTests(unittest.TestCase):
|
||||||
|
"""
|
||||||
|
Tests for ``FileHandle``.
|
||||||
|
"""
|
||||||
|
def test_get_encryption_key_convergent(self):
|
||||||
|
"""
|
||||||
|
When ``FileHandle`` is initialized with a convergence secret,
|
||||||
|
``FileHandle.get_encryption_key`` returns a deterministic result that
|
||||||
|
is a function of that secret.
|
||||||
|
"""
|
||||||
|
secret = b"\x42" * 16
|
||||||
|
handle = upload.FileHandle(BytesIO(b"hello world"), secret)
|
||||||
|
handle.set_default_encoding_parameters({
|
||||||
|
"k": 3,
|
||||||
|
"happy": 5,
|
||||||
|
"n": 10,
|
||||||
|
# Remember this is the *max* segment size. In reality, the data
|
||||||
|
# size is much smaller so the actual segment size incorporated
|
||||||
|
# into the encryption key is also smaller.
|
||||||
|
"max_segment_size": 128 * 1024,
|
||||||
|
})
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
b64encode(self.successResultOf(handle.get_encryption_key())),
|
||||||
|
b"oBcuR/wKdCgCV2GKKXqiNg==",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class EncodingParameters(GridTestMixin, unittest.TestCase, SetDEPMixin,
|
class EncodingParameters(GridTestMixin, unittest.TestCase, SetDEPMixin,
|
||||||
ShouldFailMixin):
|
ShouldFailMixin):
|
||||||
|
|
||||||
@ -2029,6 +2068,91 @@ class EncodingParameters(GridTestMixin, unittest.TestCase, SetDEPMixin,
|
|||||||
f.close()
|
f.close()
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class EncryptAnUploadableTests(unittest.TestCase):
|
||||||
|
"""
|
||||||
|
Tests for ``EncryptAnUploadable``.
|
||||||
|
"""
|
||||||
|
def test_same_length(self):
|
||||||
|
"""
|
||||||
|
``EncryptAnUploadable.read_encrypted`` returns ciphertext of the same
|
||||||
|
length as the underlying plaintext.
|
||||||
|
"""
|
||||||
|
plaintext = b"hello world"
|
||||||
|
uploadable = upload.FileHandle(BytesIO(plaintext), None)
|
||||||
|
uploadable.set_default_encoding_parameters({
|
||||||
|
# These values shouldn't matter.
|
||||||
|
"k": 3,
|
||||||
|
"happy": 5,
|
||||||
|
"n": 10,
|
||||||
|
"max_segment_size": 128 * 1024,
|
||||||
|
})
|
||||||
|
encrypter = upload.EncryptAnUploadable(uploadable)
|
||||||
|
ciphertext = b"".join(self.successResultOf(encrypter.read_encrypted(1024, False)))
|
||||||
|
self.assertEqual(len(ciphertext), len(plaintext))
|
||||||
|
|
||||||
|
@given(just(b"hello world"), integers(min_value=0, max_value=len(b"hello world")))
|
||||||
|
def test_known_result(self, plaintext, split_at):
|
||||||
|
"""
|
||||||
|
``EncryptAnUploadable.read_encrypted`` returns a known-correct ciphertext
|
||||||
|
string for certain inputs. The ciphertext is independent of the read
|
||||||
|
sizes.
|
||||||
|
"""
|
||||||
|
convergence = b"\x42" * 16
|
||||||
|
uploadable = upload.FileHandle(BytesIO(plaintext), convergence)
|
||||||
|
uploadable.set_default_encoding_parameters({
|
||||||
|
# The convergence key is a function of k, n, and max_segment_size
|
||||||
|
# (among other things). The value for happy doesn't matter
|
||||||
|
# though.
|
||||||
|
"k": 3,
|
||||||
|
"happy": 5,
|
||||||
|
"n": 10,
|
||||||
|
"max_segment_size": 128 * 1024,
|
||||||
|
})
|
||||||
|
encrypter = upload.EncryptAnUploadable(uploadable)
|
||||||
|
def read(n):
|
||||||
|
return b"".join(self.successResultOf(encrypter.read_encrypted(n, False)))
|
||||||
|
|
||||||
|
# Read the string in one or two pieces to make sure underlying state
|
||||||
|
# is maintained properly.
|
||||||
|
first = read(split_at)
|
||||||
|
second = read(len(plaintext) - split_at)
|
||||||
|
third = read(1)
|
||||||
|
ciphertext = first + second + third
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
b"Jd2LHCRXozwrEJc=",
|
||||||
|
b64encode(ciphertext),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_large_read(self):
|
||||||
|
"""
|
||||||
|
``EncryptAnUploadable.read_encrypted`` succeeds even when the requested
|
||||||
|
data length is much larger than the chunk size.
|
||||||
|
"""
|
||||||
|
convergence = b"\x42" * 16
|
||||||
|
# 4kB of plaintext
|
||||||
|
plaintext = b"\xde\xad\xbe\xef" * 1024
|
||||||
|
uploadable = upload.FileHandle(BytesIO(plaintext), convergence)
|
||||||
|
uploadable.set_default_encoding_parameters({
|
||||||
|
"k": 3,
|
||||||
|
"happy": 5,
|
||||||
|
"n": 10,
|
||||||
|
"max_segment_size": 128 * 1024,
|
||||||
|
})
|
||||||
|
# Make the chunk size very small so we don't have to operate on a huge
|
||||||
|
# amount of data to exercise the relevant codepath.
|
||||||
|
encrypter = upload.EncryptAnUploadable(uploadable, chunk_size=1)
|
||||||
|
d = encrypter.read_encrypted(len(plaintext), False)
|
||||||
|
ciphertext = self.successResultOf(d)
|
||||||
|
self.assertEqual(
|
||||||
|
list(map(len, ciphertext)),
|
||||||
|
# Chunk size was specified as 1 above so we will get the whole
|
||||||
|
# plaintext in one byte chunks.
|
||||||
|
[1] * len(plaintext),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
# TODO:
|
# TODO:
|
||||||
# upload with exactly 75 servers (shares_of_happiness)
|
# upload with exactly 75 servers (shares_of_happiness)
|
||||||
# have a download fail
|
# have a download fail
|
||||||
|
@ -1,54 +0,0 @@
|
|||||||
import json
|
|
||||||
|
|
||||||
from twisted.trial import unittest
|
|
||||||
from twisted.internet.defer import inlineCallbacks
|
|
||||||
|
|
||||||
from eliot import log_call
|
|
||||||
|
|
||||||
from autobahn.twisted.testing import create_memory_agent, MemoryReactorClockResolver, create_pumper
|
|
||||||
|
|
||||||
from allmydata.web.logs import TokenAuthenticatedWebSocketServerProtocol
|
|
||||||
|
|
||||||
|
|
||||||
class TestStreamingLogs(unittest.TestCase):
|
|
||||||
"""
|
|
||||||
Test websocket streaming of logs
|
|
||||||
"""
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.reactor = MemoryReactorClockResolver()
|
|
||||||
self.pumper = create_pumper()
|
|
||||||
self.agent = create_memory_agent(self.reactor, self.pumper, TokenAuthenticatedWebSocketServerProtocol)
|
|
||||||
return self.pumper.start()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
return self.pumper.stop()
|
|
||||||
|
|
||||||
@inlineCallbacks
|
|
||||||
def test_one_log(self):
|
|
||||||
"""
|
|
||||||
write a single Eliot log and see it streamed via websocket
|
|
||||||
"""
|
|
||||||
|
|
||||||
proto = yield self.agent.open(
|
|
||||||
transport_config=u"ws://localhost:1234/ws",
|
|
||||||
options={},
|
|
||||||
)
|
|
||||||
|
|
||||||
messages = []
|
|
||||||
def got_message(msg, is_binary=False):
|
|
||||||
messages.append(json.loads(msg))
|
|
||||||
proto.on("message", got_message)
|
|
||||||
|
|
||||||
@log_call(action_type=u"test:cli:some-exciting-action")
|
|
||||||
def do_a_thing():
|
|
||||||
pass
|
|
||||||
|
|
||||||
do_a_thing()
|
|
||||||
|
|
||||||
proto.transport.loseConnection()
|
|
||||||
yield proto.is_closed
|
|
||||||
|
|
||||||
self.assertEqual(len(messages), 2)
|
|
||||||
self.assertEqual("started", messages[0]["action_status"])
|
|
||||||
self.assertEqual("succeeded", messages[1]["action_status"])
|
|
@ -1,3 +1,15 @@
|
|||||||
|
"""
|
||||||
|
Ported to Python 3.
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
import json
|
import json
|
||||||
from os.path import join
|
from os.path import join
|
||||||
|
|
||||||
@ -213,7 +225,7 @@ class IntroducerRootTests(unittest.TestCase):
|
|||||||
resource = IntroducerRoot(introducer_node)
|
resource = IntroducerRoot(introducer_node)
|
||||||
response = json.loads(
|
response = json.loads(
|
||||||
self.successResultOf(
|
self.successResultOf(
|
||||||
render(resource, {"t": [b"json"]}),
|
render(resource, {b"t": [b"json"]}),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
"""
|
"""
|
||||||
Tests for ``allmydata.web.logs``.
|
Tests for ``allmydata.web.logs``.
|
||||||
|
|
||||||
|
Ported to Python 3.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from __future__ import (
|
from __future__ import (
|
||||||
@ -9,6 +11,19 @@ from __future__ import (
|
|||||||
division,
|
division,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
|
import json
|
||||||
|
|
||||||
|
from twisted.trial import unittest
|
||||||
|
from twisted.internet.defer import inlineCallbacks
|
||||||
|
|
||||||
|
from eliot import log_call
|
||||||
|
|
||||||
|
from autobahn.twisted.testing import create_memory_agent, MemoryReactorClockResolver, create_pumper
|
||||||
|
|
||||||
from testtools.matchers import (
|
from testtools.matchers import (
|
||||||
Equals,
|
Equals,
|
||||||
)
|
)
|
||||||
@ -37,6 +52,7 @@ from ..common import (
|
|||||||
|
|
||||||
from ...web.logs import (
|
from ...web.logs import (
|
||||||
create_log_resources,
|
create_log_resources,
|
||||||
|
TokenAuthenticatedWebSocketServerProtocol,
|
||||||
)
|
)
|
||||||
|
|
||||||
class StreamingEliotLogsTests(SyncTestCase):
|
class StreamingEliotLogsTests(SyncTestCase):
|
||||||
@ -57,3 +73,47 @@ class StreamingEliotLogsTests(SyncTestCase):
|
|||||||
self.client.get(b"http:///v1"),
|
self.client.get(b"http:///v1"),
|
||||||
succeeded(has_response_code(Equals(OK))),
|
succeeded(has_response_code(Equals(OK))),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestStreamingLogs(unittest.TestCase):
|
||||||
|
"""
|
||||||
|
Test websocket streaming of logs
|
||||||
|
"""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.reactor = MemoryReactorClockResolver()
|
||||||
|
self.pumper = create_pumper()
|
||||||
|
self.agent = create_memory_agent(self.reactor, self.pumper, TokenAuthenticatedWebSocketServerProtocol)
|
||||||
|
return self.pumper.start()
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
return self.pumper.stop()
|
||||||
|
|
||||||
|
@inlineCallbacks
|
||||||
|
def test_one_log(self):
|
||||||
|
"""
|
||||||
|
write a single Eliot log and see it streamed via websocket
|
||||||
|
"""
|
||||||
|
|
||||||
|
proto = yield self.agent.open(
|
||||||
|
transport_config=u"ws://localhost:1234/ws",
|
||||||
|
options={},
|
||||||
|
)
|
||||||
|
|
||||||
|
messages = []
|
||||||
|
def got_message(msg, is_binary=False):
|
||||||
|
messages.append(json.loads(msg))
|
||||||
|
proto.on("message", got_message)
|
||||||
|
|
||||||
|
@log_call(action_type=u"test:cli:some-exciting-action")
|
||||||
|
def do_a_thing():
|
||||||
|
pass
|
||||||
|
|
||||||
|
do_a_thing()
|
||||||
|
|
||||||
|
proto.transport.loseConnection()
|
||||||
|
yield proto.is_closed
|
||||||
|
|
||||||
|
self.assertEqual(len(messages), 2)
|
||||||
|
self.assertEqual("started", messages[0]["action_status"])
|
||||||
|
self.assertEqual("succeeded", messages[1]["action_status"])
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
"""
|
"""
|
||||||
Tests for ``allmydata.web.private``.
|
Tests for ``allmydata.web.private``.
|
||||||
|
|
||||||
|
Ported to Python 3.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from __future__ import (
|
from __future__ import (
|
||||||
@ -9,6 +11,10 @@ from __future__ import (
|
|||||||
division,
|
division,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
from testtools.matchers import (
|
from testtools.matchers import (
|
||||||
Equals,
|
Equals,
|
||||||
)
|
)
|
||||||
@ -56,6 +62,7 @@ class PrivacyTests(SyncTestCase):
|
|||||||
return super(PrivacyTests, self).setUp()
|
return super(PrivacyTests, self).setUp()
|
||||||
|
|
||||||
def _authorization(self, scheme, value):
|
def _authorization(self, scheme, value):
|
||||||
|
value = str(value, "utf-8")
|
||||||
return Headers({
|
return Headers({
|
||||||
u"authorization": [u"{} {}".format(scheme, value)],
|
u"authorization": [u"{} {}".format(scheme, value)],
|
||||||
})
|
})
|
||||||
@ -90,7 +97,7 @@ class PrivacyTests(SyncTestCase):
|
|||||||
self.assertThat(
|
self.assertThat(
|
||||||
self.client.head(
|
self.client.head(
|
||||||
b"http:///foo/bar",
|
b"http:///foo/bar",
|
||||||
headers=self._authorization(SCHEME, u"foo bar"),
|
headers=self._authorization(str(SCHEME, "utf-8"), b"foo bar"),
|
||||||
),
|
),
|
||||||
succeeded(has_response_code(Equals(UNAUTHORIZED))),
|
succeeded(has_response_code(Equals(UNAUTHORIZED))),
|
||||||
)
|
)
|
||||||
@ -103,7 +110,7 @@ class PrivacyTests(SyncTestCase):
|
|||||||
self.assertThat(
|
self.assertThat(
|
||||||
self.client.head(
|
self.client.head(
|
||||||
b"http:///foo/bar",
|
b"http:///foo/bar",
|
||||||
headers=self._authorization(SCHEME, self.token),
|
headers=self._authorization(str(SCHEME, "utf-8"), self.token),
|
||||||
),
|
),
|
||||||
# It's a made up URL so we don't get a 200, either, but a 404.
|
# It's a made up URL so we don't get a 200, either, but a 404.
|
||||||
succeeded(has_response_code(Equals(NOT_FOUND))),
|
succeeded(has_response_code(Equals(NOT_FOUND))),
|
||||||
|
@ -1,6 +1,18 @@
|
|||||||
|
"""
|
||||||
|
Ported to Python 3.
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
import time
|
import time
|
||||||
|
|
||||||
from urllib import (
|
from urllib.parse import (
|
||||||
quote,
|
quote,
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -77,7 +89,7 @@ class RenderSlashUri(unittest.TestCase):
|
|||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
response_body,
|
response_body,
|
||||||
"Invalid capability",
|
b"Invalid capability",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -92,7 +104,7 @@ class RenderServiceRow(unittest.TestCase):
|
|||||||
ann = {"anonymous-storage-FURL": "pb://w2hqnbaa25yw4qgcvghl5psa3srpfgw3@tcp:127.0.0.1:51309/vucto2z4fxment3vfxbqecblbf6zyp6x",
|
ann = {"anonymous-storage-FURL": "pb://w2hqnbaa25yw4qgcvghl5psa3srpfgw3@tcp:127.0.0.1:51309/vucto2z4fxment3vfxbqecblbf6zyp6x",
|
||||||
"permutation-seed-base32": "w2hqnbaa25yw4qgcvghl5psa3srpfgw3",
|
"permutation-seed-base32": "w2hqnbaa25yw4qgcvghl5psa3srpfgw3",
|
||||||
}
|
}
|
||||||
srv = NativeStorageServer("server_id", ann, None, {}, EMPTY_CLIENT_CONFIG)
|
srv = NativeStorageServer(b"server_id", ann, None, {}, EMPTY_CLIENT_CONFIG)
|
||||||
srv.get_connection_status = lambda: ConnectionStatus(False, "summary", {}, 0, 0)
|
srv.get_connection_status = lambda: ConnectionStatus(False, "summary", {}, 0, 0)
|
||||||
|
|
||||||
class FakeClient(_Client):
|
class FakeClient(_Client):
|
||||||
@ -103,7 +115,7 @@ class RenderServiceRow(unittest.TestCase):
|
|||||||
tub_maker=None,
|
tub_maker=None,
|
||||||
node_config=EMPTY_CLIENT_CONFIG,
|
node_config=EMPTY_CLIENT_CONFIG,
|
||||||
)
|
)
|
||||||
self.storage_broker.test_add_server("test-srv", srv)
|
self.storage_broker.test_add_server(b"test-srv", srv)
|
||||||
|
|
||||||
root = RootElement(FakeClient(), time.time)
|
root = RootElement(FakeClient(), time.time)
|
||||||
req = DummyRequest(b"")
|
req = DummyRequest(b"")
|
||||||
|
@ -189,7 +189,7 @@ class FakeHistory(object):
|
|||||||
def list_all_helper_statuses(self):
|
def list_all_helper_statuses(self):
|
||||||
return []
|
return []
|
||||||
|
|
||||||
class FakeDisplayableServer(StubServer):
|
class FakeDisplayableServer(StubServer): # type: ignore # tahoe-lafs/ticket/3573
|
||||||
def __init__(self, serverid, nickname, connected,
|
def __init__(self, serverid, nickname, connected,
|
||||||
last_connect_time, last_loss_time, last_rx_time):
|
last_connect_time, last_loss_time, last_rx_time):
|
||||||
StubServer.__init__(self, serverid)
|
StubServer.__init__(self, serverid)
|
||||||
@ -255,7 +255,7 @@ class FakeStorageServer(service.MultiService):
|
|||||||
def on_status_changed(self, cb):
|
def on_status_changed(self, cb):
|
||||||
cb(self)
|
cb(self)
|
||||||
|
|
||||||
class FakeClient(_Client):
|
class FakeClient(_Client): # type: ignore # tahoe-lafs/ticket/3573
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
# don't upcall to Client.__init__, since we only want to initialize a
|
# don't upcall to Client.__init__, since we only want to initialize a
|
||||||
# minimal subset
|
# minimal subset
|
||||||
@ -4757,6 +4757,31 @@ class Web(WebMixin, WebErrorMixin, testutil.StallMixin, testutil.ReallyEqualMixi
|
|||||||
op_url = self.webish_url + "/operations/134?t=status&output=JSON"
|
op_url = self.webish_url + "/operations/134?t=status&output=JSON"
|
||||||
yield self.assertHTTPError(op_url, 404, "unknown/expired handle '134'")
|
yield self.assertHTTPError(op_url, 404, "unknown/expired handle '134'")
|
||||||
|
|
||||||
|
@inlineCallbacks
|
||||||
|
def test_uri_redirect(self):
|
||||||
|
"""URI redirects don't cause failure.
|
||||||
|
|
||||||
|
Unit test reproducer for https://tahoe-lafs.org/trac/tahoe-lafs/ticket/3590
|
||||||
|
"""
|
||||||
|
def req(method, path, **kwargs):
|
||||||
|
return treq.request(method, self.webish_url + path, persistent=False,
|
||||||
|
**kwargs)
|
||||||
|
|
||||||
|
response = yield req("POST", "/uri?format=sdmf&t=mkdir")
|
||||||
|
dircap = yield response.content()
|
||||||
|
assert dircap.startswith('URI:DIR2:')
|
||||||
|
dircap_uri = "/uri/?uri={}&t=json".format(urllib.quote(dircap))
|
||||||
|
|
||||||
|
response = yield req(
|
||||||
|
"GET",
|
||||||
|
dircap_uri,
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
response.request.absoluteURI,
|
||||||
|
self.webish_url + "/uri/{}?t=json".format(urllib.quote(dircap)))
|
||||||
|
if response.code >= 400:
|
||||||
|
raise Error(response.code, response=response.content())
|
||||||
|
|
||||||
def test_incident(self):
|
def test_incident(self):
|
||||||
d = self.POST("/report_incident", details="eek")
|
d = self.POST("/report_incident", details="eek")
|
||||||
def _done(res):
|
def _done(res):
|
||||||
|
@ -1,6 +1,16 @@
|
|||||||
"""
|
"""
|
||||||
Tests for ``allmydata.webish``.
|
Tests for ``allmydata.webish``.
|
||||||
|
|
||||||
|
Ported to Python 3.
|
||||||
"""
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
from uuid import (
|
from uuid import (
|
||||||
uuid4,
|
uuid4,
|
||||||
@ -96,7 +106,7 @@ class TahoeLAFSRequestTests(SyncTestCase):
|
|||||||
])
|
])
|
||||||
self._fields_test(
|
self._fields_test(
|
||||||
b"POST",
|
b"POST",
|
||||||
{b"content-type": b"multipart/form-data; boundary={}".format(boundary)},
|
{b"content-type": b"multipart/form-data; boundary=" + bytes(boundary, 'ascii')},
|
||||||
form_data.encode("ascii"),
|
form_data.encode("ascii"),
|
||||||
AfterPreprocessing(
|
AfterPreprocessing(
|
||||||
lambda fs: {
|
lambda fs: {
|
||||||
@ -105,8 +115,8 @@ class TahoeLAFSRequestTests(SyncTestCase):
|
|||||||
in fs.keys()
|
in fs.keys()
|
||||||
},
|
},
|
||||||
Equals({
|
Equals({
|
||||||
b"foo": b"bar",
|
"foo": "bar",
|
||||||
b"baz": b"some file contents",
|
"baz": b"some file contents",
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
@ -1,3 +1,13 @@
|
|||||||
|
"""Ported to Python 3.
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from future.utils import PY2
|
||||||
|
if PY2:
|
||||||
|
from future.builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
from zope.interface import implementer
|
from zope.interface import implementer
|
||||||
from twisted.internet import defer
|
from twisted.internet import defer
|
||||||
|
@ -22,6 +22,11 @@ from past.builtins import unicode, long
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Type
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from zope.interface import implementer
|
from zope.interface import implementer
|
||||||
from twisted.python.components import registerAdapter
|
from twisted.python.components import registerAdapter
|
||||||
|
|
||||||
@ -489,7 +494,7 @@ class MDMFVerifierURI(_BaseURI):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
|
|
||||||
@implementer(IURI, IDirnodeURI)
|
@implementer(IDirnodeURI)
|
||||||
class _DirectoryBaseURI(_BaseURI):
|
class _DirectoryBaseURI(_BaseURI):
|
||||||
def __init__(self, filenode_uri=None):
|
def __init__(self, filenode_uri=None):
|
||||||
self._filenode_uri = filenode_uri
|
self._filenode_uri = filenode_uri
|
||||||
@ -536,7 +541,7 @@ class _DirectoryBaseURI(_BaseURI):
|
|||||||
return self._filenode_uri.get_storage_index()
|
return self._filenode_uri.get_storage_index()
|
||||||
|
|
||||||
|
|
||||||
@implementer(IDirectoryURI)
|
@implementer(IURI, IDirectoryURI)
|
||||||
class DirectoryURI(_DirectoryBaseURI):
|
class DirectoryURI(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2:'
|
BASE_STRING=b'URI:DIR2:'
|
||||||
@ -555,7 +560,7 @@ class DirectoryURI(_DirectoryBaseURI):
|
|||||||
return ReadonlyDirectoryURI(self._filenode_uri.get_readonly())
|
return ReadonlyDirectoryURI(self._filenode_uri.get_readonly())
|
||||||
|
|
||||||
|
|
||||||
@implementer(IReadonlyDirectoryURI)
|
@implementer(IURI, IReadonlyDirectoryURI)
|
||||||
class ReadonlyDirectoryURI(_DirectoryBaseURI):
|
class ReadonlyDirectoryURI(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2-RO:'
|
BASE_STRING=b'URI:DIR2-RO:'
|
||||||
@ -574,6 +579,7 @@ class ReadonlyDirectoryURI(_DirectoryBaseURI):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
|
|
||||||
|
@implementer(IURI, IDirnodeURI)
|
||||||
class _ImmutableDirectoryBaseURI(_DirectoryBaseURI):
|
class _ImmutableDirectoryBaseURI(_DirectoryBaseURI):
|
||||||
def __init__(self, filenode_uri=None):
|
def __init__(self, filenode_uri=None):
|
||||||
if filenode_uri:
|
if filenode_uri:
|
||||||
@ -611,7 +617,7 @@ class LiteralDirectoryURI(_ImmutableDirectoryBaseURI):
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
@implementer(IDirectoryURI)
|
@implementer(IURI, IDirectoryURI)
|
||||||
class MDMFDirectoryURI(_DirectoryBaseURI):
|
class MDMFDirectoryURI(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2-MDMF:'
|
BASE_STRING=b'URI:DIR2-MDMF:'
|
||||||
@ -633,7 +639,7 @@ class MDMFDirectoryURI(_DirectoryBaseURI):
|
|||||||
return MDMFDirectoryURIVerifier(self._filenode_uri.get_verify_cap())
|
return MDMFDirectoryURIVerifier(self._filenode_uri.get_verify_cap())
|
||||||
|
|
||||||
|
|
||||||
@implementer(IReadonlyDirectoryURI)
|
@implementer(IURI, IReadonlyDirectoryURI)
|
||||||
class ReadonlyMDMFDirectoryURI(_DirectoryBaseURI):
|
class ReadonlyMDMFDirectoryURI(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2-MDMF-RO:'
|
BASE_STRING=b'URI:DIR2-MDMF-RO:'
|
||||||
@ -671,7 +677,7 @@ def wrap_dirnode_cap(filecap):
|
|||||||
raise AssertionError("cannot interpret as a directory cap: %s" % filecap.__class__)
|
raise AssertionError("cannot interpret as a directory cap: %s" % filecap.__class__)
|
||||||
|
|
||||||
|
|
||||||
@implementer(IVerifierURI)
|
@implementer(IURI, IVerifierURI)
|
||||||
class MDMFDirectoryURIVerifier(_DirectoryBaseURI):
|
class MDMFDirectoryURIVerifier(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2-MDMF-Verifier:'
|
BASE_STRING=b'URI:DIR2-MDMF-Verifier:'
|
||||||
@ -696,12 +702,12 @@ class MDMFDirectoryURIVerifier(_DirectoryBaseURI):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
|
|
||||||
@implementer(IVerifierURI)
|
@implementer(IURI, IVerifierURI)
|
||||||
class DirectoryURIVerifier(_DirectoryBaseURI):
|
class DirectoryURIVerifier(_DirectoryBaseURI):
|
||||||
|
|
||||||
BASE_STRING=b'URI:DIR2-Verifier:'
|
BASE_STRING=b'URI:DIR2-Verifier:'
|
||||||
BASE_STRING_RE=re.compile(b'^'+BASE_STRING)
|
BASE_STRING_RE=re.compile(b'^'+BASE_STRING)
|
||||||
INNER_URI_CLASS=SSKVerifierURI
|
INNER_URI_CLASS=SSKVerifierURI # type: Type[IVerifierURI]
|
||||||
|
|
||||||
def __init__(self, filenode_uri=None):
|
def __init__(self, filenode_uri=None):
|
||||||
if filenode_uri:
|
if filenode_uri:
|
||||||
|
@ -37,6 +37,7 @@ PORTED_MODULES = [
|
|||||||
"allmydata.crypto.util",
|
"allmydata.crypto.util",
|
||||||
"allmydata.deep_stats",
|
"allmydata.deep_stats",
|
||||||
"allmydata.dirnode",
|
"allmydata.dirnode",
|
||||||
|
"allmydata.frontends.sftpd",
|
||||||
"allmydata.hashtree",
|
"allmydata.hashtree",
|
||||||
"allmydata.immutable.checker",
|
"allmydata.immutable.checker",
|
||||||
"allmydata.immutable.downloader",
|
"allmydata.immutable.downloader",
|
||||||
@ -83,6 +84,7 @@ PORTED_MODULES = [
|
|||||||
"allmydata.storage.shares",
|
"allmydata.storage.shares",
|
||||||
"allmydata.test.no_network",
|
"allmydata.test.no_network",
|
||||||
"allmydata.test.mutable.util",
|
"allmydata.test.mutable.util",
|
||||||
|
"allmydata.unknown",
|
||||||
"allmydata.uri",
|
"allmydata.uri",
|
||||||
"allmydata.util._python3",
|
"allmydata.util._python3",
|
||||||
"allmydata.util.abbreviate",
|
"allmydata.util.abbreviate",
|
||||||
@ -113,6 +115,8 @@ PORTED_MODULES = [
|
|||||||
"allmydata.util.spans",
|
"allmydata.util.spans",
|
||||||
"allmydata.util.statistics",
|
"allmydata.util.statistics",
|
||||||
"allmydata.util.time_format",
|
"allmydata.util.time_format",
|
||||||
|
"allmydata.web.logs",
|
||||||
|
"allmydata.webish",
|
||||||
]
|
]
|
||||||
|
|
||||||
PORTED_TEST_MODULES = [
|
PORTED_TEST_MODULES = [
|
||||||
@ -167,6 +171,7 @@ PORTED_TEST_MODULES = [
|
|||||||
"allmydata.test.test_pipeline",
|
"allmydata.test.test_pipeline",
|
||||||
"allmydata.test.test_python3",
|
"allmydata.test.test_python3",
|
||||||
"allmydata.test.test_repairer",
|
"allmydata.test.test_repairer",
|
||||||
|
"allmydata.test.test_sftp",
|
||||||
"allmydata.test.test_spans",
|
"allmydata.test.test_spans",
|
||||||
"allmydata.test.test_statistics",
|
"allmydata.test.test_statistics",
|
||||||
"allmydata.test.test_stats",
|
"allmydata.test.test_stats",
|
||||||
@ -185,6 +190,11 @@ PORTED_TEST_MODULES = [
|
|||||||
"allmydata.test.test_util",
|
"allmydata.test.test_util",
|
||||||
"allmydata.test.web.test_common",
|
"allmydata.test.web.test_common",
|
||||||
"allmydata.test.web.test_grid",
|
"allmydata.test.web.test_grid",
|
||||||
"allmydata.test.web.test_util",
|
"allmydata.test.web.test_introducer",
|
||||||
|
"allmydata.test.web.test_logs",
|
||||||
|
"allmydata.test.web.test_private",
|
||||||
|
"allmydata.test.web.test_root",
|
||||||
"allmydata.test.web.test_status",
|
"allmydata.test.web.test_status",
|
||||||
|
"allmydata.test.web.test_util",
|
||||||
|
"allmydata.test.web.test_webish",
|
||||||
]
|
]
|
||||||
|
@ -15,7 +15,18 @@ if PY2:
|
|||||||
|
|
||||||
import time
|
import time
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import (
|
||||||
|
Callable,
|
||||||
|
Any,
|
||||||
|
)
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from foolscap.api import eventually
|
from foolscap.api import eventually
|
||||||
|
from eliot.twisted import (
|
||||||
|
inline_callbacks,
|
||||||
|
)
|
||||||
from twisted.internet import defer, reactor, error
|
from twisted.internet import defer, reactor, error
|
||||||
from twisted.python.failure import Failure
|
from twisted.python.failure import Failure
|
||||||
|
|
||||||
@ -201,3 +212,22 @@ class WaitForDelayedCallsMixin(PollMixin):
|
|||||||
d.addErrback(log.err, "error while waiting for delayed calls")
|
d.addErrback(log.err, "error while waiting for delayed calls")
|
||||||
d.addBoth(lambda ign: res)
|
d.addBoth(lambda ign: res)
|
||||||
return d
|
return d
|
||||||
|
|
||||||
|
@inline_callbacks
|
||||||
|
def until(
|
||||||
|
action, # type: Callable[[], defer.Deferred[Any]]
|
||||||
|
condition, # type: Callable[[], bool]
|
||||||
|
):
|
||||||
|
# type: (...) -> defer.Deferred[None]
|
||||||
|
"""
|
||||||
|
Run a Deferred-returning function until a condition is true.
|
||||||
|
|
||||||
|
:param action: The action to run.
|
||||||
|
:param condition: The predicate signaling stop.
|
||||||
|
|
||||||
|
:return: A Deferred that fires after the condition signals stop.
|
||||||
|
"""
|
||||||
|
while True:
|
||||||
|
yield action()
|
||||||
|
if condition():
|
||||||
|
break
|
||||||
|
@ -311,7 +311,7 @@ def precondition_abspath(path):
|
|||||||
|
|
||||||
_getfullpathname = None
|
_getfullpathname = None
|
||||||
try:
|
try:
|
||||||
from nt import _getfullpathname
|
from nt import _getfullpathname # type: ignore
|
||||||
except ImportError:
|
except ImportError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -14,6 +14,12 @@ if PY2:
|
|||||||
from builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
from builtins import filter, map, zip, ascii, chr, hex, input, next, oct, open, pow, round, super, bytes, dict, list, object, range, str, max, min # noqa: F401
|
||||||
|
|
||||||
import time
|
import time
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import List
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.internet import task
|
from twisted.internet import task
|
||||||
|
|
||||||
class TimeoutError(Exception):
|
class TimeoutError(Exception):
|
||||||
@ -23,7 +29,7 @@ class PollComplete(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
class PollMixin(object):
|
class PollMixin(object):
|
||||||
_poll_should_ignore_these_errors = []
|
_poll_should_ignore_these_errors = [] # type: List[Exception]
|
||||||
|
|
||||||
def poll(self, check_f, pollinterval=0.01, timeout=1000):
|
def poll(self, check_f, pollinterval=0.01, timeout=1000):
|
||||||
# Return a Deferred, then call check_f periodically until it returns
|
# Return a Deferred, then call check_f periodically until it returns
|
||||||
|
@ -6,6 +6,11 @@ Can eventually be merged back into allmydata.web.common.
|
|||||||
|
|
||||||
from past.builtins import unicode
|
from past.builtins import unicode
|
||||||
|
|
||||||
|
try:
|
||||||
|
from typing import Optional
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
from twisted.web import resource, http
|
from twisted.web import resource, http
|
||||||
|
|
||||||
from allmydata.util import abbreviate
|
from allmydata.util import abbreviate
|
||||||
@ -55,7 +60,7 @@ class MultiFormatResource(resource.Resource, object):
|
|||||||
format if nothing else is given as the ``formatDefault``.
|
format if nothing else is given as the ``formatDefault``.
|
||||||
"""
|
"""
|
||||||
formatArgument = "t"
|
formatArgument = "t"
|
||||||
formatDefault = None
|
formatDefault = None # type: Optional[str]
|
||||||
|
|
||||||
def render(self, req):
|
def render(self, req):
|
||||||
"""
|
"""
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user