urlreader

URLReader: a wrapper around macOS’s NSURLSession, etc. for PyObjC apps


License
MIT
Install
pip install urlreader==0.1.2

Documentation

Tests

URLReader

URLReader is a wrapper around macOS’s NSURLSession, etc.

Scope & Limitations

URLReader originated from an effort to improve the UI responsiveness of Mechanic, a package manager for RoboFont. Because of this original use-case, URLReader is meant to be used in PyObjC apps or scripts that need to download and possibly cache relatively small bits of additional data. Technically there is nothing preventing you from using URLReader for other use-cases, larger downloads, etc. but depending what these might be you might be better served by using NSURLSession directly.

Being powered by the system NSURLSession, URLReader only works on macOS 10.9+.

Install

URLReader is available through the Python Packaging Index, so you can pip install urlreader. We recommend using virtual environments, in which case you can add urlreader in your requirements.txt or Pipfile etc.

You can also install directly from source, in which case you can run:

$ python setup.py install

In case you want run the test suite:

$ python setup.py test

URLReader has a small test suite which spins a simple HTTP server in a separate process and runs against it. The test suite is also hooked up to Github’s actions, so it runs whenever there’s a push on the repo.

Basic usage

In its most basic form, URLReader takes a URL, fetches its contents in the background (so it won’t block your UI) and sends them back on the main thread to a callback you provide. The callback is a regular Python function with three arguments: url , data and error (of types NSURL, NSData and NSError). Like this:

def callback(url, data, error):
    if url and data and not error:
        print(f"Received {url} contents, {len(data)} bytes")

URLReader().fetch("http://example.org/", callback)

Timeout

You can set a custom timeout for requests (which by default is 10 seconds):

URLReader(timeout=2) # in seconds

Notice that this is the total response time, not how long it takes for the initial request to make it to the server.

Quote URL path and force HTTPS

Sometimes people have spaces in their URL paths, like, say, /Foo Bar, but forget to quote them. The NSURLSession reading machinery really doesn’t like that. By default URLReader quotes the path component of a URL. This behavior can be turned off, if needed:

URLReader(quote_url_path=False)

Also, sometimes people try to open http URLs, but their apps’ App Security Policy won’t allow that. If the remote server also supports https, then you can tell URLReader to promote the URLs from http to https by calling:

URLReader(force_https=True)

This setting won’t affect non-HTTP requests. Technically there is nothing in URL-reader that is HTTP-only, and NSURLSession does support other protocols. I haven’t tested it with these, but I’m not aware of anything special that would need to change.

Usage from scripts

You can use URLReader in synchronous mode for one-off scripts if you need to. It will block until the response is fully returned to your callback instead of calling it asynchronously like it normally would:

URLReader(wait_until_done=True)

Caching

By default, URLReader follows the caching policy set by the protocol, i.e. NSURLRequestUseProtocolCachePolicy which means it will do whatever the response HTTP caching headers tell it to do (if the request is HTTP.)

In some cases, however, you might want to cache the response so it’s available to your code even when offline. In these situations, URLReader switches the request cache policy to NSURLRequestReturnCacheDataElseLoad which means it will first check its cache and return data from there no matter what the HTTP headers say.

On top of the standard NSURLRequestReturnCacheDataElseLoad behavior, URLReader adds a little twist: if the original response required one or more redirects, which wouldn’t be cached therefore not available offline, NSURLCache isn’t able to fetch the response from its cache. URLReader adds a cache entry for the response URL before the redirects so they are fully accessible offline.

This all happens with the same code:

URLReader(use_cache=True) # this switches on the caching mechanism described above

By default the cache will be written in ~/Library/Caches/<your app bundle identifier> but that can be configured like this:

URLReader(use_cache=True, cache_location="/my/cache/path") # cache_location can be either a string path or an NSURL

Once a URL is fetched by an URLReader with caching enabled, it stays in the cache indefinitely. You can force-reset a cached entry by setting invalidate_cache=True on the fetch method:

reader.fetch(url, callback, invalidate_cache=True)

Or you can remove the cached entry without replacing it with a new one:

reader.invalidate_cache_for_url(url)

Flushing the cache, removing all the cached entries, is also possible:

reader.flush_cache()

Callback

Sometimes you just need to fetch some values, quick. You could use a lambda:

URLReader().fetch('http://example.org',
    lambda url, data, error: print(f'Received {len(data)} bytes'))

Just remember the lambda would be called asynchronously and the call would return right away. So it’d be no good in a script without an NSRunLoop. In that case, this would work:

URLReader(wait_until_done=True).fetch('http://example.org',
    lambda url, data, error: print(f'Received {len(data)} bytes'))

This would make the call block until the data is received and the lambda has exited. But then again, if you just want to fetch a URL in a blocking fashion with no caches or other accoutrements, either requests.get(url) or urllib.request.urlopen(url) are probably a better fit.

And that’s it

Thanks for reading, hope this code is useful somehow.


URLReader was written by Antonio Cavedoni and is licensed under the terms of the MIT License.