In 95% cases it is enough to use a little more than 10 client coroutines. So, lets start!
Firstly you should create :class:`aioftp.Client` instance and connect to host
:py:meth:`aioftp.Client.connect`
:py:meth:`aioftp.Client.login`
>>> client = aioftp.Client() >>> await client.connect("ftp.server.com") >>> await client.login("user", "pass")
Or just use :class:`aioftp.context` async context, which will connect, login and quit automatically
>>> async with aioftp.Client.context("ftp.server.com", user="user", password="pass") as client: ... # do
:py:meth:`aioftp.Client.upload` and :py:meth:`aioftp.Client.download` coroutines are pretty similar, except data flow direction. You can upload/download file or directory. There is "source" and "destination". When you does not specify "destination", then current working directory will be used as destination.
Lets upload some file to current directory
>>> await client.upload("test.py")
If you want specify new name, or different path to uploading/downloading path you should use "write_into" argument, which works for directory as well
>>> await client.upload("test.py", "tmp/test.py", write_into=True) >>> await client.upload("folder1", "folder2", write_into=True)
After that you get
tmp/test.py folder2/*content of folder1*
If you will not use "write_into", you will get something you probably did not expect
tmp/test.py/test.py folder2/folder1/*content of folder1*
Or you can upload path as is and then rename it (:py:meth:`aioftp.Client.rename`)
Downloading is pretty same
>>> await client.download("tmp/test.py", "foo.py", write_into=True) >>> await client.download("folder2")
For listing paths you should use :py:meth:`aioftp.Client.list` coroutine, which can list paths recursively and produce a :py:class:`list` and can be used with async for
>>> await client.list("/") [(PosixPath('/.logs'), {'unix.mode': '0755', 'unique': '801g4804045', ...
>>> await client.list("/", recursive=True) [(PosixPath('/.logs'), {'unix.mode': '0755', 'unique': '801g4804045', ...
>>> async for path, info in client.list("/", recursive=True): ... print(path) (PosixPath('/.logs'), {'unix.mode': '0755', 'unique': '801g4804045', ...
If you ommit path argument, result will be list for current working directory
>>> await c.list() [(PosixPath('test.py'), {'unique': '801g480a508', 'size': '3102', ...
In case of async for be careful, since asynchronous variation of list is lazy. It means that you can't interact with server until you leave `async for` block. If you need list and interact with server you should use eager version of list:
>>> for path, info in (await client.list()): ... await client.download(path, path.name)
If you want to mix lazy list and client interaction, you can create two client connections to server:
>>> async for path, info in client1.list(): ... await client2.download(path, path.name)
:py:meth:`aioftp.Client.list` in general use MLSD command, but some nasty servers does not support this command. Then client will try to use LIST command, and parse server response. For proper work of :py:meth:`datetime.datetime.strptime` (in part of parsing month abbreviation) locale should be setted to "C". For this reason if you use multithreaded app, and use some locale-dependent stuff, you should use :py:meth:`aioftp.setlocale` context manager when you dealing with locale in another thread. since 0.8.1 If fallback LIST parser can't parse line, then this line will be ignored, so fallback LIST implementation will never raise exception.
When you need get some path stats you should use :py:meth:`aioftp.Client.stat`
>>> await client.stat("tmp2.py") {'size': '909', 'create': '1445437246.4320722', 'type': 'file', ... >>> await client.stat(".git") {'create': '1445435702.6441028', 'type': 'dir', 'size': '4096', ...
If you need just to check path for is it file, directory or exists you can use
:py:meth:`aioftp.Client.is_file`
>>> await client.is_file("/public_html") False >>> await client.is_dir("/public_html") True >>> await client.is_file("test.py") True >>> await client.exists("test.py") True >>> await client.exists("naked-guido.png") False
:py:meth:`aioftp.Client.stat` in general use MLST command, but some nasty servers does not support this command. Then client will try to use LIST command, and parse server response. For proper work of :py:meth:`datetime.datetime.strptime` (in part of parsing month abbreviation) locale should be setted to "C". For this reason if you use multithreaded app, and use some locale-dependent stuff, you should use :py:meth:`aioftp.setlocale` context manager when you dealing with locale in another thread. since 0.8.1 If fallback LIST parser can't parse line, then this line will be ignored, so fallback LIST implementation will never raise exception. But if requested path line can't be parsed, then :py:meth:`aioftp.Client.stat` method will raise path does not exists.
For removing paths you have universal coroutine :py:meth:`aioftp.Client.remove` which can remove file or directory recursive. So, you don't need to do borring checks.
>>> await client.remove("tmp.py") >>> await client.remove("folder1")
Directories coroutines are pretty simple.
:py:meth:`aioftp.Client.get_current_directory`
:py:meth:`aioftp.Client.change_directory`
:py:meth:`aioftp.Client.make_directory`
>>> await client.get_current_directory() PosixPath('/public_html') >>> await client.change_directory("folder1") >>> await client.get_current_directory() PosixPath('/public_html/folder1') >>> await client.change_directory() >>> await client.get_current_directory() PosixPath('/public_html') >>> await client.make_directory("folder2") >>> await client.change_directory("folder2") >>> await client.get_current_directory() PosixPath('/public_html/folder2')
To change name (move) file or directory use :py:meth:`aioftp.Client.rename`.
>>> await client.list() [(PosixPath('test.py'), {'modify': '20150423090041', 'type': 'file', ... >>> await client.rename("test.py", "foo.py") >>> await client.list() [(PosixPath('foo.py'), {'modify': '20150423090041', 'type': 'file', ...
:py:meth:`aioftp.Client.quit` coroutine will send "QUIT" ftp command and close connection.
>>> await client.quit()
File read/write operations are blocking and slow. So if you want just parse/calculate something on the fly when receiving file, or generate data to upload it to file system on ftp server, then you should use :py:meth:`aioftp.Client.download_stream`, :py:meth:`aioftp.Client.upload_stream` and :py:meth:`aioftp.Client.append_stream`. All this methods based on :py:meth:`aioftp.Client.get_stream`, which return :py:class:`aioftp.DataConnectionThrottleStreamIO`. The common pattern to work with streams is:
>>> async with client.download_stream("tmp.py") as stream: ... async for block in stream.iter_by_block(): ... # do something with data
Or, if you want to abort transfer at some point
>>> stream = await client.download_stream("tmp.py") ... async for block in stream.iter_by_block(): ... # do something with data ... if something_not_interesting: ... await client.abort() ... stream.close() ... break ... else: ... await stream.finish()
Do not use async with <stream> syntax if you want to use abort, this will lead to deadlock.
For restarting upload/download at exact byte position (REST command) there is offset argument for *_stream methods:
>>> async with client.download_stream("tmp.py", offset=256) as stream: ... async for block in stream.iter_by_block(): ... # do something with data
Or if you want to restore upload/download process:
>>> while True: ... try: ... async with aioftp.Client.context(HOST, PORT) as client: ... if await client.exists(filename): ... stat = await client.stat(filename) ... size = int(stat["size"]) ... else: ... size = 0 ... file_in.seek(size) ... async with client.upload_stream(filename, offset=size) as stream: ... while True: ... data = file_in.read(block_size) ... if not data: ... break ... await stream.write(data) ... break ... except ConnectionResetError: ... pass
The idea is to seek position of source «file» for upload and start upload + offset/append. Opposite situation for download («file» append and download + offset)
Client have two types of speed limit: read_speed_limit and write_speed_limit. Throttle can be set at initialization time:
>>> client = aioftp.Client(read_speed_limit=100 * 1024) # 100 Kib/s
And can be changed after creation:
>>> client.throttle.write.limit = 250 * 1024
aioftp provides abstraction of file system operations. You can use exist ones:
- :py:class:`aioftp.PathIO` — blocking path operations
- :py:class:`aioftp.AsyncPathIO` — non-blocking path operations, this one is blocking ones just wrapped with :py:meth:`asyncio.BaseEventLoop.run_in_executor`. It's really slow, so it's better to avoid usage of this path io layer.
- :py:class:`aioftp.MemoryPathIO` — in-memory realization of file system, this one is just proof of concept and probably not too fast (as it can be).
You can specify path_io_factory when creating :py:class:`aioftp.Client` instance. Default factory is :py:class:`aioftp.PathIO`.
>>> client = aioftp.Client(path_io_factory=pathio.MemoryPathIO)
:py:class:`aioftp.Client` have socket_timeout argument, which you can use to specify global timeout for socket io operations.
>>> client = aioftp.Client(socket_timeout=1) # 1 second socket timeout
:py:class:`aioftp.Client` also have path_timeout, which is applied only for non-blocking path io layers.
>>> client = aioftp.Client( ... path_timeout=1, ... path_io_factory=pathio.AsyncPathIO ... )
Simplest way to use socks proxy with :class:`aioftp.Client` is siosocks
>>> client = aioftp.Client( ... socks_host="localhost", ... socks_port=9050, ... socks_version=5, ... )
Don't forget to install aioftp as pip install aioftp[socks], or install siosocks directly with pip install siosocks.
:py:meth:`aioftp.Client.list` and :py:meth:`aioftp.Client.stat` in general use MLSD and MLST, but some nasty servers does not support this commands. Then client will try to use LIST command, and parse server response. For proper work of :py:meth:`datetime.datetime.strptime` (in part of parsing month abbreviation) locale should be setted to "C". For this reason if you use multithreaded app, and use some locale-dependent stuff, you should use :py:meth:`aioftp.setlocale` context manager when you dealing with locale in another thread. since 0.8.1 If fallback LIST parser can't parse line, then this line will be ignored, so fallback LIST implementation will never raise exception.