Metadata-Version: 2.1
Name: urlreader
Version: 0.1.3
Summary: URLReader: a wrapper around macOS’s NSURLSession, etc. for PyObjC apps
Home-page: https://github.com/verbosus/urlreader
Author: Antonio Cavedoni
Author-email: antonio@cavedoni.org
License: UNKNOWN
Description: [<img src="https://github.com/verbosus/urlreader/workflows/Tests/badge.svg" alt="Tests">](https://github.com/verbosus/urlreader/actions)
        
        # URLReader
        
        URLReader is a wrapper around macOS’s NSURLSession, etc. 
        
        ## Scope & Limitations
        
        URLReader originated from [an effort](https://github.com/robofont-mechanic/mechanic-2/pull/18) to improve the UI responsiveness of [Mechanic](https://robofontmechanic.com/), a package manager for [RoboFont](https://www.robofont.com/). 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](https://pypi.org/project/urlreader/) 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:
        
        ```shell
        $ python setup.py install
        ```
        
        In case you want run the test suite:
        
        ```shell
        $ 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](https://github.com/verbosus/urlreader/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:
        
        ```python
        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):
        
        ```python
        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:
        
        ```python
        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:
        
        ```python
        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:
        
        ```python
        URLReader(wait_until_done=True)
        ```
        
        ## Caching
        
        By default, URLReader follows the caching policy set by the protocol, i.e. [NSURLRequestUseProtocolCachePolicy](https://developer.apple.com/documentation/foundation/nsurlrequestcachepolicy/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](https://developer.apple.com/documentation/foundation/nsurlrequestcachepolicy/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: 
        
        ```python 
        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:
        
        ```python
        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:
        
        ```python
        reader.fetch(url, callback, invalidate_cache=True)
        ```
        
        Or you can remove the cached entry without replacing it with a new one:
        
        ```python
        reader.invalidate_cache_for_url(url)
        ```
        
        Flushing the cache, removing all the cached entries, is also possible:
        
        ```python
        reader.flush_cache()
        ```
        
        ## Callback
        
        Sometimes you just need to fetch some values, quick. You could use a lambda:
        
        ```python
        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:
        
        ```python
        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](mailto:antonio@cavedoni.org) and is licensed under the terms of the [MIT License](https://github.com/verbosus/urlreader/blob/master/LICENSE).
        
Platform: UNKNOWN
Classifier: Programming Language :: Python :: 3
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: MacOS :: MacOS X
Classifier: Development Status :: 3 - Alpha
Classifier: Environment :: MacOS X
Classifier: Environment :: MacOS X :: Cocoa
Classifier: Topic :: Internet
Requires-Python: >=3.6
Description-Content-Type: text/markdown
