# Installation
### Versioning
This library uses [SimVer](http://simver.org/) versioning, where a change in the major version number indicates a
breaking change and a change in the minor version number indicates a non-breaking change (such as an additional
feature or bug fix).
### Changelog
The changelog is available [here](CHANGELOG.md).
### Requirements.
This library has been tested on Python 3.6 and higher.
### pip install
```sh
pip install canopy
```
You may need to run `pip` with root permission: `sudo pip install canopy`.
From a Jupyter Notebook you can run `!pip install canopy`.
### Setuptools
Install via [Setuptools](http://pypi.python.org/pypi/setuptools).
```sh
python setup.py install --user
```
(or `sudo python setup.py install` to install the package for all users)
### Running Tests
Unit tests can be run with:
```
pytest canopy
```
Integration tests can be run with:
```
pytest integration_tests
```
To run the integration tests you'll need to ensure you have an environment variable called `CANOPY_PYTHON_INTEGRATION_TEST_CREDENTIALS`
containing the string `<client_id>|<client_secret>|<username>|<tenant_name>|<password>`.
# Getting Started
## Example Usage
See the [Canopy Python Examples](https://github.com/CanopySimulations/canopy-python-examples) repository for example usage.
## Introduction
This package is designed for customers of [Canopy Simulations](https://www.canopysimulations.com/) who would like
to access the Canopy API from Python, for example using Jupyter Notebooks.
Currently the library is split into two parts:
- The client generated using the OpenAPI toolset is located in the "canopy/openapi" folder.
We don't have a great deal of control over how this code looks, but it should give a fairly complete interface to the main API.
- One folder up from that in the "canopy" folder we are adding helper functions which wrap common use cases in simple functions.
You can also use these functions as a reference to using the OpenAPI generated code.
When using the library you generally start by creating a `canopy.Session` object.
The session object manages authentication, and the caching of user settings.
Calling `session.authentication.authenticate()` before calling OpenAPI generated client functions ensures that you are
authenticated and that any expired access tokens are refreshed.
Our helper functions will handle calling `authenticate` before making any calls, so if you are only using our
helper functions you won't need to call it yourself.
The `session` should generally be created once per application. It will automatically dispose itself when the application
shuts down. Alternatively you can enclose it in an `async with` or a `with` block if you need to create multiple sessions,
as shown in the examples below.
If you are using the OpenAPI generated code then you can pass the `session.async_client` or `session.sync_client` into the OpenAPI
generated API client instance as the `api_client` parameter as shown below. Passing in `async_client` will cause it to use
`asyncio`, and you will need to `await` the calls. Passing in `sync_client` will cause the calls to complete synchronously.
Our helper functions all use `asyncio` for efficient parallelisation of downloads, and must therefore be awaited.
The following example shows how to create a session and request some output channels from a study using our helper function:
```python
import canopy
import asyncio
async with canopy.Session(client_id='<your_client_id>', username='<your_username>') as session:
study_data = await canopy.load_study(session, '<study_id>', 'DynamicLap', ['sRun', 'vCar'])
# Using the OpenAPI generated client directly:
study_api = canopy.openapi.StudyApi(session.async_client)
job_result = await study_api.study_get_study_job_metadata(
session.authentication.tenant_id,
'<study_id>',
0)
# Using asyncio.ensure_future() to enable us to perform multiple calls in parallel
job_result_task = asyncio.ensure_future(study_api.study_get_study_job_metadata(
session.authentication.tenant_id,
'<study_id>',
0))
job_result_2 = await job_result_task
```
When running this code you will be prompted for your client secret and your password if
it is the first time `session.authentication.authenticate()` has been called for this session instance. Alternatively
you can pass the client secret and password into the Session class (after fetching them from a secure location) to
avoid being prompted.
If you can't use `asyncio` and `async/await` you can instead instantiate the session object synchronously
and use the `canopy.run` method when calling our async helper methods.
You can pass `session.sync_client` into the OpenAPI client classes instead of `session.async_client` to make them
return results synchronously.
```python
import canopy
with canopy.Session(client_id='<your_client_id>', username='<your_username>') as session:
# Note we are using canopy.run(..) to force the async method to run synchronously.
# This is a wrapper for asyncio.get_event_loop().run_until_complete(..).
study_data = canopy.run(canopy.load_study(session, '<study_id>', 'DynamicLap', ['sRun', 'vCar']))
# Using the OpenAPI generated client synchronously by passing in sync_client:
study_api = canopy.openapi.StudyApi(session.sync_client)
job_result = study_api.study_get_study_job_metadata(
session.authentication.tenant_id,
'<study_id>',
0)
# You can still run synchronous OpenAPI client methods asynchronously using threads if you need to:
job_result_thread = study_api.study_get_study_job_metadata(
session.authentication.tenant_id,
'<study_id>',
0,
async_req=True)
job_result_2 = job_result_thread.get()
```
## Proxy Servers
You can configure your proxy server by passing in a `proxy` argument to the `canopy.Session` object:
```python
async with canopy.Session(authentication_data, proxy=canopy.ProxyConfiguration('http://some.proxy.com', 'user', 'pass')) as session:
```
# Updating the OpenAPI Client
This needs to be tidied up, improved, and automated.
Additional options can be found here: https://openapi-generator.tech/docs/generators/openapi/
- e.g. enumUnknownDefaultCase could be useful if the remaining exposed enums change in future.
You can use the Dockerfile in this repository to create a docker image to generate the new API stubs:
```sh
docker image build -t canopy-python-gen:1 .
docker container run -i -t --mount type=bind,src='<path>/<to>/canopy/canopy-python',dst=/canopy/repo canopy-python-gen:1 /bin/bash
```
```sh
java -jar openapi-generator-cli.jar generate -g python-legacy -i https://api.canopysimulations.com/swagger/v1/swagger.json -o ./gen --package-name "canopy.openapi"
rm -r repo/canopy/openapi
rm -r repo/docs
cp -r gen/canopy/openapi repo/canopy
cp -r gen/docs repo
cp -r gen/README.md repo/OPENAPI_README.md
```
To regenerate the `asyncio` files execute:
```sh
rm -r gen
java -jar openapi-generator-cli.jar generate -g python-legacy -i https://api.canopysimulations.com/swagger/v1/swagger.json -o ./gen --package-name "canopy.openapi" --library asyncio
mv gen/canopy/openapi gen/canopy/openapi_asyncio
rm -r gen/canopy/openapi_asyncio/api
rm -r gen/canopy/openapi_asyncio/models
rm gen/canopy/openapi_asyncio/configuration.py
rm gen/canopy/openapi_asyncio/exceptions.py
sed -i 's/from canopy\.openapi import rest/from canopy.openapi_asyncio import rest/g' gen/canopy/openapi_asyncio/api_client.py
sed -i '/from canopy.*/d' gen/canopy/openapi_asyncio/__init__.py
sed -i '/# import /d' gen/canopy/openapi_asyncio/__init__.py
echo 'from canopy.openapi_asyncio.api_client import ApiClient' >> gen/canopy/openapi_asyncio/__init__.py
cp -r gen/canopy/openapi_asyncio repo/canopy
```
Note: The `openapi/configuration.py` file will need to be manually modified to add the default API host URL.
Note: The `openapi_asyncio/rest.py` file will need to be manually modified to support proxy servers after generation.
Note: The `openapi_asyncio/client_api.py` and `openapi/client_api.py` files will need to be manually modified to
support numpy array serialization after generation.
## Documentation for OpenAPI Generated Client
OpenAPI generated documentation can be found [here](OPENAPI_README.md).
Raw data
{
"_id": null,
"home_page": "https://github.com/canopysimulations/canopy-python/",
"name": "canopy",
"maintainer": "",
"docs_url": null,
"requires_python": "",
"maintainer_email": "",
"keywords": "Canopy API,Canopy Simulations,Canopy Client",
"author": "James Thurley",
"author_email": "james.thurley@canopysimulations.com",
"download_url": "https://files.pythonhosted.org/packages/22/85/e456926a09d60252872ad1c318fd558d4b85d727ae8db6eec00087dfab97/canopy-8.42.tar.gz",
"platform": null,
"description": "# Installation\n\n### Versioning\n\nThis library uses [SimVer](http://simver.org/) versioning, where a change in the major version number indicates a\nbreaking change and a change in the minor version number indicates a non-breaking change (such as an additional\nfeature or bug fix).\n\n### Changelog\n\nThe changelog is available [here](CHANGELOG.md).\n\n### Requirements.\n\nThis library has been tested on Python 3.6 and higher.\n\n### pip install\n\n```sh\npip install canopy\n```\n\nYou may need to run `pip` with root permission: `sudo pip install canopy`.\n\nFrom a Jupyter Notebook you can run `!pip install canopy`.\n\n### Setuptools\n\nInstall via [Setuptools](http://pypi.python.org/pypi/setuptools).\n\n```sh\npython setup.py install --user\n```\n(or `sudo python setup.py install` to install the package for all users)\n\n### Running Tests\n\nUnit tests can be run with:\n```\npytest canopy\n```\n\nIntegration tests can be run with:\n```\npytest integration_tests\n```\n\nTo run the integration tests you'll need to ensure you have an environment variable called `CANOPY_PYTHON_INTEGRATION_TEST_CREDENTIALS`\ncontaining the string `<client_id>|<client_secret>|<username>|<tenant_name>|<password>`.\n\n# Getting Started\n\n## Example Usage\nSee the [Canopy Python Examples](https://github.com/CanopySimulations/canopy-python-examples) repository for example usage.\n\n## Introduction\n\nThis package is designed for customers of [Canopy Simulations](https://www.canopysimulations.com/) who would like\nto access the Canopy API from Python, for example using Jupyter Notebooks.\n\nCurrently the library is split into two parts:\n\n - The client generated using the OpenAPI toolset is located in the \"canopy/openapi\" folder.\n We don't have a great deal of control over how this code looks, but it should give a fairly complete interface to the main API.\n\n - One folder up from that in the \"canopy\" folder we are adding helper functions which wrap common use cases in simple functions.\n You can also use these functions as a reference to using the OpenAPI generated code.\n\nWhen using the library you generally start by creating a `canopy.Session` object. \nThe session object manages authentication, and the caching of user settings.\nCalling `session.authentication.authenticate()` before calling OpenAPI generated client functions ensures that you are\nauthenticated and that any expired access tokens are refreshed.\nOur helper functions will handle calling `authenticate` before making any calls, so if you are only using our\nhelper functions you won't need to call it yourself.\n\nThe `session` should generally be created once per application. It will automatically dispose itself when the application\nshuts down. Alternatively you can enclose it in an `async with` or a `with` block if you need to create multiple sessions,\nas shown in the examples below. \n\nIf you are using the OpenAPI generated code then you can pass the `session.async_client` or `session.sync_client` into the OpenAPI \ngenerated API client instance as the `api_client` parameter as shown below. Passing in `async_client` will cause it to use\n`asyncio`, and you will need to `await` the calls. Passing in `sync_client` will cause the calls to complete synchronously.\n\nOur helper functions all use `asyncio` for efficient parallelisation of downloads, and must therefore be awaited.\n\nThe following example shows how to create a session and request some output channels from a study using our helper function:\n\n```python\nimport canopy\nimport asyncio\n\nasync with canopy.Session(client_id='<your_client_id>', username='<your_username>') as session:\n study_data = await canopy.load_study(session, '<study_id>', 'DynamicLap', ['sRun', 'vCar'])\n\n # Using the OpenAPI generated client directly:\n study_api = canopy.openapi.StudyApi(session.async_client)\n job_result = await study_api.study_get_study_job_metadata(\n session.authentication.tenant_id,\n '<study_id>',\n 0)\n\n # Using asyncio.ensure_future() to enable us to perform multiple calls in parallel\n job_result_task = asyncio.ensure_future(study_api.study_get_study_job_metadata(\n session.authentication.tenant_id,\n '<study_id>',\n 0))\n\n job_result_2 = await job_result_task\n\n```\n\nWhen running this code you will be prompted for your client secret and your password if \nit is the first time `session.authentication.authenticate()` has been called for this session instance. Alternatively\nyou can pass the client secret and password into the Session class (after fetching them from a secure location) to\navoid being prompted.\n\nIf you can't use `asyncio` and `async/await` you can instead instantiate the session object synchronously \nand use the `canopy.run` method when calling our async helper methods. \nYou can pass `session.sync_client` into the OpenAPI client classes instead of `session.async_client` to make them \nreturn results synchronously.\n\n```python\nimport canopy\n\nwith canopy.Session(client_id='<your_client_id>', username='<your_username>') as session:\n # Note we are using canopy.run(..) to force the async method to run synchronously.\n # This is a wrapper for asyncio.get_event_loop().run_until_complete(..).\n study_data = canopy.run(canopy.load_study(session, '<study_id>', 'DynamicLap', ['sRun', 'vCar']))\n\n # Using the OpenAPI generated client synchronously by passing in sync_client:\n study_api = canopy.openapi.StudyApi(session.sync_client)\n job_result = study_api.study_get_study_job_metadata(\n session.authentication.tenant_id,\n '<study_id>',\n 0)\n\n # You can still run synchronous OpenAPI client methods asynchronously using threads if you need to:\n job_result_thread = study_api.study_get_study_job_metadata(\n session.authentication.tenant_id,\n '<study_id>',\n 0,\n async_req=True)\n\n job_result_2 = job_result_thread.get()\n```\n\n## Proxy Servers\n\nYou can configure your proxy server by passing in a `proxy` argument to the `canopy.Session` object:\n```python\nasync with canopy.Session(authentication_data, proxy=canopy.ProxyConfiguration('http://some.proxy.com', 'user', 'pass')) as session:\n``` \n\n# Updating the OpenAPI Client\n\nThis needs to be tidied up, improved, and automated.\n\nAdditional options can be found here: https://openapi-generator.tech/docs/generators/openapi/\n - e.g. enumUnknownDefaultCase could be useful if the remaining exposed enums change in future. \n\nYou can use the Dockerfile in this repository to create a docker image to generate the new API stubs:\n\n```sh\ndocker image build -t canopy-python-gen:1 .\ndocker container run -i -t --mount type=bind,src='<path>/<to>/canopy/canopy-python',dst=/canopy/repo canopy-python-gen:1 /bin/bash\n```\n\n```sh\njava -jar openapi-generator-cli.jar generate -g python-legacy -i https://api.canopysimulations.com/swagger/v1/swagger.json -o ./gen --package-name \"canopy.openapi\"\nrm -r repo/canopy/openapi\nrm -r repo/docs\ncp -r gen/canopy/openapi repo/canopy\ncp -r gen/docs repo\ncp -r gen/README.md repo/OPENAPI_README.md\n```\n\nTo regenerate the `asyncio` files execute:\n```sh\nrm -r gen\njava -jar openapi-generator-cli.jar generate -g python-legacy -i https://api.canopysimulations.com/swagger/v1/swagger.json -o ./gen --package-name \"canopy.openapi\" --library asyncio\nmv gen/canopy/openapi gen/canopy/openapi_asyncio\nrm -r gen/canopy/openapi_asyncio/api\nrm -r gen/canopy/openapi_asyncio/models\nrm gen/canopy/openapi_asyncio/configuration.py\nrm gen/canopy/openapi_asyncio/exceptions.py\nsed -i 's/from canopy\\.openapi import rest/from canopy.openapi_asyncio import rest/g' gen/canopy/openapi_asyncio/api_client.py\nsed -i '/from canopy.*/d' gen/canopy/openapi_asyncio/__init__.py\nsed -i '/# import /d' gen/canopy/openapi_asyncio/__init__.py\necho 'from canopy.openapi_asyncio.api_client import ApiClient' >> gen/canopy/openapi_asyncio/__init__.py\ncp -r gen/canopy/openapi_asyncio repo/canopy\n```\n\nNote: The `openapi/configuration.py` file will need to be manually modified to add the default API host URL.\nNote: The `openapi_asyncio/rest.py` file will need to be manually modified to support proxy servers after generation. \nNote: The `openapi_asyncio/client_api.py` and `openapi/client_api.py` files will need to be manually modified to \nsupport numpy array serialization after generation. \n\n## Documentation for OpenAPI Generated Client\n\nOpenAPI generated documentation can be found [here](OPENAPI_README.md).",
"bugtrack_url": null,
"license": "MIT",
"summary": "Python Client for the Canopy Simulation API",
"version": "8.42",
"project_urls": {
"Homepage": "https://github.com/canopysimulations/canopy-python/"
},
"split_keywords": [
"canopy api",
"canopy simulations",
"canopy client"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "2285e456926a09d60252872ad1c318fd558d4b85d727ae8db6eec00087dfab97",
"md5": "35b9413f98d0189eba915af06fd57f84",
"sha256": "0be1de23a243c7877a582e65cf7e07311771147f8b8fb806ccada09d4c70642b"
},
"downloads": -1,
"filename": "canopy-8.42.tar.gz",
"has_sig": false,
"md5_digest": "35b9413f98d0189eba915af06fd57f84",
"packagetype": "sdist",
"python_version": "source",
"requires_python": null,
"size": 166937,
"upload_time": "2023-11-09T14:10:37",
"upload_time_iso_8601": "2023-11-09T14:10:37.168032Z",
"url": "https://files.pythonhosted.org/packages/22/85/e456926a09d60252872ad1c318fd558d4b85d727ae8db6eec00087dfab97/canopy-8.42.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2023-11-09 14:10:37",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "canopysimulations",
"github_project": "canopy-python",
"travis_ci": false,
"coveralls": false,
"github_actions": false,
"requirements": [],
"lcname": "canopy"
}