Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 9f5631cd90 |
+37
-29
@@ -1,68 +1,76 @@
|
|||||||
version: 2.1
|
version: 2.1
|
||||||
|
|
||||||
orbs:
|
orbs:
|
||||||
python: circleci/python@2.0.3
|
python: circleci/python@1.3.2
|
||||||
codecov: codecov/codecov@3.2.2
|
codecov: codecov/codecov@3.2.2
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
test:
|
test:
|
||||||
machine:
|
docker:
|
||||||
image: ubuntu-2204:2023.02.1
|
- image: "cimg/python:<<parameters.tag>>"
|
||||||
docker_layer_caching: true
|
- image: 'cimg/node:14.18'
|
||||||
resource_class: medium
|
- image: 'circleci/redis:6'
|
||||||
|
- image: 'cimg/postgres:12.8'
|
||||||
|
environment:
|
||||||
|
POSTGRES_DB: speckle2_test
|
||||||
|
POSTGRES_PASSWORD: speckle
|
||||||
|
POSTGRES_USER: speckle
|
||||||
|
- image: "speckle/speckle-server"
|
||||||
|
command: ["bash", "-c", "/wait && node bin/www"]
|
||||||
|
environment:
|
||||||
|
POSTGRES_URL: "localhost"
|
||||||
|
POSTGRES_USER: "speckle"
|
||||||
|
POSTGRES_PASSWORD: "speckle"
|
||||||
|
POSTGRES_DB: "speckle2_test"
|
||||||
|
REDIS_URL: "redis://localhost"
|
||||||
|
SESSION_SECRET: "keyboard cat"
|
||||||
|
STRATEGY_LOCAL: "true"
|
||||||
|
CANONICAL_URL: "http://localhost:3000"
|
||||||
|
WAIT_HOSTS: localhost:5432, localhost:6379
|
||||||
parameters:
|
parameters:
|
||||||
tag:
|
tag:
|
||||||
default: "3.11"
|
default: "3.8"
|
||||||
type: string
|
type: string
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
|
- run: python --version
|
||||||
- run:
|
- run:
|
||||||
name: Install python
|
command: python -m pip install --upgrade pip
|
||||||
command: |
|
name: upgrade pip
|
||||||
pyenv install -s << parameters.tag >>
|
- python/install-packages:
|
||||||
pyenv global << parameters.tag >>
|
pkg-manager: poetry
|
||||||
- run:
|
- run: poetry run pytest --cov --cov-report xml:reports/coverage.xml --junitxml=reports/test-results.xml
|
||||||
name: Startup the Speckle Server
|
|
||||||
command: docker compose -f docker-compose.yml up -d
|
|
||||||
- run:
|
|
||||||
name: Install Poetry
|
|
||||||
command: |
|
|
||||||
pip install poetry
|
|
||||||
- run:
|
|
||||||
name: Install packages
|
|
||||||
command: poetry install
|
|
||||||
- run:
|
|
||||||
name: Run tests
|
|
||||||
command: poetry run pytest --cov --cov-report xml:reports/coverage.xml --junitxml=reports/test-results.xml
|
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
path: reports
|
path: reports
|
||||||
|
|
||||||
- store_artifacts:
|
- store_artifacts:
|
||||||
path: reports
|
path: reports
|
||||||
|
|
||||||
- codecov/upload
|
- codecov/upload
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
docker:
|
docker:
|
||||||
- image: "cimg/python:3.8"
|
- image: "circleci/python:3.8"
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- run: python patch_version.py $CIRCLE_TAG
|
- run: python patch_version.py $CIRCLE_TAG
|
||||||
- run: poetry build
|
- run: poetry build
|
||||||
- run: poetry publish -u __token__ -p $PYPI_TOKEN
|
- run: poetry publish -u specklesystems -p $PYPI_PASSWORD
|
||||||
|
|
||||||
workflows:
|
workflows:
|
||||||
main:
|
main:
|
||||||
jobs:
|
jobs:
|
||||||
- test:
|
- test:
|
||||||
matrix:
|
matrix:
|
||||||
parameters:
|
parameters:
|
||||||
tag: ["3.11"]
|
tag: ["3.6", "3.7", "3.8", "3.9"]
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
- deploy:
|
- deploy:
|
||||||
context: pypi
|
|
||||||
requires:
|
requires:
|
||||||
- test
|
- test
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /[0-9]+(\.[0-9]+)*/
|
only: /[0-9]+(\.[0-9]+)*/
|
||||||
|
|||||||
@@ -1,27 +0,0 @@
|
|||||||
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/blob/main/containers/python-3/.devcontainer/base.Dockerfile
|
|
||||||
|
|
||||||
# [Choice] Python version: 3, 3.9, 3.8, 3.7, 3.6
|
|
||||||
ARG VARIANT="3.10"
|
|
||||||
FROM mcr.microsoft.com/vscode/devcontainers/python:${VARIANT}
|
|
||||||
|
|
||||||
# [Choice] Node.js version: none, lts/*, 16, 14, 12, 10
|
|
||||||
ARG NODE_VERSION="16"
|
|
||||||
RUN if [ "${NODE_VERSION}" != "none" ]; then su vscode -c "umask 0002 && . /usr/local/share/nvm/nvm.sh && nvm install ${NODE_VERSION} 2>&1"; fi
|
|
||||||
|
|
||||||
# [Optional] If your pip requirements rarely change, uncomment this section to add them to the image.
|
|
||||||
# COPY requirements.txt /tmp/pip-tmp/
|
|
||||||
# RUN pip3 --disable-pip-version-check --no-cache-dir install -r /tmp/pip-tmp/requirements.txt \
|
|
||||||
# && rm -rf /tmp/pip-tmp
|
|
||||||
|
|
||||||
# [Optional] Uncomment this section to install additional OS packages.
|
|
||||||
# RUN apt-get update && export DEBIAN_FRONTEND=noninteractive \
|
|
||||||
# && apt-get -y install --no-install-recommends <your-package-list-here>
|
|
||||||
|
|
||||||
# [Optional] Uncomment this line to install global node packages.
|
|
||||||
# RUN su vscode -c "source /usr/local/share/nvm/nvm.sh && npm install -g <your-package-here>" 2>&1
|
|
||||||
|
|
||||||
USER vscode
|
|
||||||
|
|
||||||
RUN curl -sSL https://raw.githubusercontent.com/python-poetry/poetry/master/get-poetry.py | python -
|
|
||||||
|
|
||||||
ENV PATH=$PATH:$HOME/.poetry/env
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
// For format details, see https://aka.ms/devcontainer.json. For config options, see the README at:
|
|
||||||
// https://github.com/microsoft/vscode-dev-containers/tree/v0.191.1/containers/python-3
|
|
||||||
{
|
|
||||||
"name": "Python 3",
|
|
||||||
// "build": {
|
|
||||||
// "dockerfile": "Dockerfile",
|
|
||||||
// "context": "..",
|
|
||||||
// "args": {
|
|
||||||
// // Update 'VARIANT' to pick a Python version: 3, 3.6, 3.7, 3.8, 3.9
|
|
||||||
// "VARIANT": "3.6",
|
|
||||||
// // Options
|
|
||||||
// "NODE_VERSION": "lts/*"
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
"dockerComposeFile": "./docker-compose.yaml",
|
|
||||||
"service": "specklepy",
|
|
||||||
"workspaceFolder": "/workspaces/specklepy",
|
|
||||||
"shutdownAction": "stopCompose",
|
|
||||||
// Set *default* container specific settings.json values on container create.
|
|
||||||
"settings": {
|
|
||||||
"python.pythonPath": "/usr/local/bin/python",
|
|
||||||
"python.languageServer": "Pylance",
|
|
||||||
"python.linting.enabled": true,
|
|
||||||
"python.linting.pylintEnabled": true,
|
|
||||||
"python.linting.pylintArgs": [
|
|
||||||
"--max-line-length=120"
|
|
||||||
],
|
|
||||||
"python.formatting.autopep8Path": "/usr/local/py-utils/bin/autopep8",
|
|
||||||
"python.formatting.blackPath": "/usr/local/py-utils/bin/black",
|
|
||||||
"python.formatting.yapfPath": "/usr/local/py-utils/bin/yapf",
|
|
||||||
"python.linting.banditPath": "/usr/local/py-utils/bin/bandit",
|
|
||||||
"python.linting.flake8Path": "/usr/local/py-utils/bin/flake8",
|
|
||||||
"python.linting.mypyPath": "/usr/local/py-utils/bin/mypy",
|
|
||||||
"python.linting.pycodestylePath": "/usr/local/py-utils/bin/pycodestyle",
|
|
||||||
"python.linting.pydocstylePath": "/usr/local/py-utils/bin/pydocstyle",
|
|
||||||
"python.linting.pylintPath": "/usr/local/py-utils/bin/pylint",
|
|
||||||
"python.testing.pytestArgs": [
|
|
||||||
"tests/",
|
|
||||||
"-s"
|
|
||||||
],
|
|
||||||
"python.testing.pytestEnabled": true,
|
|
||||||
"editor.formatOnSave": true,
|
|
||||||
},
|
|
||||||
// Add the IDs of extensions you want installed when the container is created.
|
|
||||||
"extensions": [
|
|
||||||
"ms-python.python",
|
|
||||||
"ms-python.vscode-pylance"
|
|
||||||
],
|
|
||||||
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
|
||||||
// "forwardPorts": [],
|
|
||||||
// Use 'postCreateCommand' to run commands after the container is created.
|
|
||||||
"postCreateCommand": "poetry config virtualenvs.create false && poetry install",
|
|
||||||
// Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
|
||||||
"remoteUser": "vscode"
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
version: "3.3" # optional since v1.27.0
|
|
||||||
services:
|
|
||||||
postgres:
|
|
||||||
image: cimg/postgres:14.2
|
|
||||||
environment:
|
|
||||||
POSTGRES_DB: speckle2_test
|
|
||||||
POSTGRES_PASSWORD: speckle
|
|
||||||
POSTGRES_USER: speckle
|
|
||||||
network_mode: host
|
|
||||||
redis:
|
|
||||||
image: cimg/redis:6.2
|
|
||||||
network_mode: host
|
|
||||||
speckle-server:
|
|
||||||
image: speckle/speckle-server:latest
|
|
||||||
command: ["bash", "-c", "/wait && node bin/www"]
|
|
||||||
environment:
|
|
||||||
POSTGRES_URL: "localhost"
|
|
||||||
POSTGRES_USER: "speckle"
|
|
||||||
POSTGRES_PASSWORD: "speckle"
|
|
||||||
POSTGRES_DB: "speckle2_test"
|
|
||||||
REDIS_URL: "redis://localhost"
|
|
||||||
SESSION_SECRET: "keyboard cat"
|
|
||||||
STRATEGY_LOCAL: "true"
|
|
||||||
CANONICAL_URL: "http://localhost:3000"
|
|
||||||
WAIT_HOSTS: localhost:5432, localhost:6379
|
|
||||||
DISABLE_FILE_UPLOADS: "true"
|
|
||||||
network_mode: host
|
|
||||||
|
|
||||||
specklepy:
|
|
||||||
build:
|
|
||||||
dockerfile: Dockerfile
|
|
||||||
context: .
|
|
||||||
args:
|
|
||||||
VARIANT: 3.9
|
|
||||||
NODE_VERSION: lts/*
|
|
||||||
volumes:
|
|
||||||
# Mounts the project folder to '/workspace'. While this file is in .devcontainer,
|
|
||||||
# mounts are relative to the first file in the list, which is a level up.
|
|
||||||
- ..:/workspaces/specklepy:cached
|
|
||||||
# Overrides default command so things don't shut down after the process ends.
|
|
||||||
command: /bin/sh -c "while sleep 1000; do :; done"
|
|
||||||
network_mode: host
|
|
||||||
# networks:
|
|
||||||
# default:
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
* text=auto eol=lf
|
|
||||||
*.{cmd,[cC][mM][dD]} text eol=crlf
|
|
||||||
*.{bat,[bB][aA][tT]} text eol=crlf
|
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
---
|
||||||
|
name: New issue
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title:
|
||||||
|
labels:
|
||||||
|
assignees:
|
||||||
|
---
|
||||||
|
|
||||||
|
If it's your first time here - or you forgot about them - make sure you read the [contribution guidelines](CONTRIBUTING.md), and then feel free to delete this line!
|
||||||
|
|
||||||
|
### Expected vs. Actual Behavior
|
||||||
|
|
||||||
|
Describe the problem here.
|
||||||
|
|
||||||
|
### Reproduction Steps & System Config (win, osx, web, etc.)
|
||||||
|
|
||||||
|
Let us know how we can reproduce this, and attach relevant files (if any).
|
||||||
|
|
||||||
|
### Proposed Solution (if any)
|
||||||
|
|
||||||
|
Let us know what how you would solve this.
|
||||||
|
|
||||||
|
#### Optional: Affected Projects
|
||||||
|
|
||||||
|
Does this issue propagate to other dependencies or dependents? If so, list them here!
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
---
|
|
||||||
name: Bug report
|
|
||||||
about: Help improve Speckle!
|
|
||||||
title: ''
|
|
||||||
labels: bug
|
|
||||||
assignees: ''
|
|
||||||
---
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Provide a short summary in the Title above. Examples of good Issue titles:
|
|
||||||
|
|
||||||
* "Bug: Error from server when reticulating splines"
|
|
||||||
* "Bug: Revit crashes when installing connector"
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Prerequisites
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Please answer the following questions before submitting an issue.
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
- [ ] I read the [contribution guidelines](https://github.com/specklesystems/speckle-server/blob/main/CONTRIBUTING.md)
|
|
||||||
- [ ] I checked the [documentation](https://speckle.guide/) and found no answer.
|
|
||||||
- [ ] I checked [existing issues](../issues?q=is%3Aissue) and found no similar issue. <!-- If you do find an existing issue, please show your support by liking it :+1: instead of creating a new issue -->
|
|
||||||
- [ ] I checked the [community forum](https://speckle.community/) for related discussions and found no answer.
|
|
||||||
- [ ] I'm reporting the issue to the correct repository (see also [speckle-server](https://github.com/specklesystems/speckle-server), [speckle-sharp](https://github.com/specklesystems/speckle-sharp), [specklepy](https://github.com/specklesystems/specklepy), [speckle-docs](https://github.com/specklesystems/speckle-docs), and [others](https://github.com/orgs/specklesystems/repositories))
|
|
||||||
|
|
||||||
## What package are you referring to?
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Is it related to the server (backend) only, or does this bug relate to the frontend, viewer, objectloader or any other package?
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Describe the bug
|
|
||||||
|
|
||||||
<!---
|
|
||||||
A clear and concise description of what the bug is.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## To Reproduce
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Steps to reproduce the behavior:
|
|
||||||
|
|
||||||
1. Go to '...'
|
|
||||||
2. Click on '....'
|
|
||||||
3. Scroll down to '....'
|
|
||||||
4. See error
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Expected behavior
|
|
||||||
|
|
||||||
<!---
|
|
||||||
A clear and concise description of what you expected to happen.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Screenshots
|
|
||||||
|
|
||||||
<!---
|
|
||||||
If applicable, add screenshots to help explain your problem.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## System Info
|
|
||||||
|
|
||||||
If applicable, please fill in the below details - they help a lot!
|
|
||||||
|
|
||||||
### Desktop (please complete the following information):
|
|
||||||
|
|
||||||
- OS: [e.g. iOS]
|
|
||||||
- Browser [e.g. chrome, safari]
|
|
||||||
- Version [e.g. 22]
|
|
||||||
|
|
||||||
### Smartphone (please complete the following information):
|
|
||||||
|
|
||||||
- Device: [e.g. iPhone6]
|
|
||||||
- OS: [e.g. iOS8.1]
|
|
||||||
- Browser [e.g. stock browser, safari]
|
|
||||||
- Version [e.g. 22]
|
|
||||||
|
|
||||||
## Failure Logs
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Please include any relevant log snippets or files here, or upload as a file.
|
|
||||||
|
|
||||||
If including inline, please use markdown code block syntax. https://docs.github.com/en/get-started/writing-on-github/working-with-advanced-formatting/creating-and-highlighting-code-blocks
|
|
||||||
For example:
|
|
||||||
|
|
||||||
```
|
|
||||||
your log output here
|
|
||||||
```
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Additional context
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Add any other context about the problem here.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Proposed Solution (if any)
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Let us know what how you would solve this.
|
|
||||||
-->
|
|
||||||
|
|
||||||
#### Optional: Affected Projects
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Does this issue propagate to other dependencies or dependents? If so, list them here with links!
|
|
||||||
-->
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
---
|
|
||||||
name: Feature request
|
|
||||||
about: Suggest an idea for Speckle!
|
|
||||||
title: ''
|
|
||||||
labels: enhancement, question
|
|
||||||
assignees: ''
|
|
||||||
---
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Provide a short summary in the Title above. Examples of good Issue titles:
|
|
||||||
|
|
||||||
* "Enhancement: Connector for Minecraft"
|
|
||||||
* "Enhancement: Web viewer should support tesseracts"
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Prerequisites
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Please answer the following questions before submitting an issue.
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
- [ ] I read the [contribution guidelines](https://github.com/specklesystems/speckle-server/blob/main/CONTRIBUTING.md)
|
|
||||||
- [ ] I checked the [documentation](https://speckle.guide/) and found no answer.
|
|
||||||
- [ ] I checked [existing issues](../issues?q=is%3Aissue) and found no similar issue. <!-- If you do find an existing issue, please show your support by liking it :+1: instead of creating a new issue -->
|
|
||||||
- [ ] I checked the [community forum](https://speckle.community/) for related discussions and found no answer.
|
|
||||||
- [ ] I'm requesting the feature to the correct repository (see also [speckle-server](https://github.com/specklesystems/speckle-server), [speckle-sharp](https://github.com/specklesystems/speckle-sharp), [specklepy](https://github.com/specklesystems/specklepy), [speckle-docs](https://github.com/specklesystems/speckle-docs), and [others](https://github.com/orgs/specklesystems/repositories))
|
|
||||||
|
|
||||||
## What package are you referring to?
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Is it related to the server (backend) only, or does this feature request relate to the frontend, viewer, objectloader or any other package?
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Is your feature request related to a problem? Please describe.
|
|
||||||
|
|
||||||
<!---
|
|
||||||
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Describe the solution you'd like
|
|
||||||
|
|
||||||
<!---
|
|
||||||
A clear and concise description of what you want to happen.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Describe alternatives you've considered
|
|
||||||
|
|
||||||
<!---
|
|
||||||
A clear and concise description of any alternative solutions or features you've considered.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Additional context
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Add any other context or screenshots about the feature request here.
|
|
||||||
|
|
||||||
Have you seen this feature implemented in any other software? Can you provide screenshots or links to video or documentation?
|
|
||||||
What works well about these existing features in other software? What doesn't work well?
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Related issues or community discussions
|
|
||||||
|
|
||||||
<!---
|
|
||||||
Is this feature request related to (but sufficiently distinct from) any existing issues?
|
|
||||||
Does this feature request require other features to be available beforehand?
|
|
||||||
Has this feature been discussed in the community forum, please link here? https://speckle.community/
|
|
||||||
-->
|
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
Description of PR...
|
||||||
|
|
||||||
|
## Changes
|
||||||
|
|
||||||
|
- Item 1
|
||||||
|
- Item 2
|
||||||
|
|
||||||
|
## Checklist
|
||||||
|
|
||||||
|
- [ ] Unit tests
|
||||||
|
- [ ] Documentation
|
||||||
|
|
||||||
|
## References
|
||||||
|
|
||||||
|
(optional)
|
||||||
|
|
||||||
|
Include **important** links regarding the implementation of this PR.
|
||||||
|
This usually includes and RFC or an aggregation of issues and/or individual conversations
|
||||||
|
that helped put this solution together. This helps ensure there is a good aggregation
|
||||||
|
of resources regarding the implementation.
|
||||||
|
|
||||||
|
```text
|
||||||
|
Fixes #85, Fixes #22, Fixes username/repo#123
|
||||||
|
Connects #123
|
||||||
|
```
|
||||||
@@ -1,102 +0,0 @@
|
|||||||
<!---
|
|
||||||
|
|
||||||
Provide a short summary in the Title above. Examples of good PR titles:
|
|
||||||
|
|
||||||
* "Feature: adds metrics to component"
|
|
||||||
|
|
||||||
* "Fix: resolves duplication in comment thread"
|
|
||||||
|
|
||||||
* "Update: apollo v2.34.0"
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Description & motivation
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Describe your changes, and why you're making them. What benefit will this have to others?
|
|
||||||
|
|
||||||
Is this linked to an open Github issue, a thread in Speckle community,
|
|
||||||
or another pull request? Link it here.
|
|
||||||
|
|
||||||
If it is related to a Github issue, and resolves it, please link to the issue number, e.g.:
|
|
||||||
Fixes #85, Fixes #22, Fixes username/repo#123
|
|
||||||
Connects #123
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Changes:
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
- Item 1
|
|
||||||
- Item 2
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## To-do before merge:
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
(Optional -- remove this section if not needed)
|
|
||||||
|
|
||||||
Include any notes about things that need to happen before this PR is merged, e.g.:
|
|
||||||
|
|
||||||
- [ ] Change the base branch
|
|
||||||
|
|
||||||
- [ ] Ensure PR #56 is merged
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Screenshots:
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Include a screenshot the before and after. This can be a screenshot of a plugin, web frontend, or output in a terminal.
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Validation of changes:
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
Describe what tests have been added or amended, and why these demonstrate it works and will prevent this feature being accidentally broken by future changes.
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Checklist:
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
This checklist is mostly useful as a reminder of small things that can easily be
|
|
||||||
|
|
||||||
forgotten – it is meant as a helpful tool rather than hoops to jump through.
|
|
||||||
|
|
||||||
Put an `x` between the square brackets, e.g. [x], for all the items that apply,
|
|
||||||
|
|
||||||
make notes next to any that haven't been addressed, and remove any items that are not relevant to this PR.
|
|
||||||
|
|
||||||
-->
|
|
||||||
|
|
||||||
- [ ] My pull request follows the guidelines in the [Contributing guide](https://github.com/specklesystems/speckle-server/blob/main/CONTRIBUTING.md)?
|
|
||||||
- [ ] My pull request does not duplicate any other open [Pull Requests](../../pulls) for the same update/change?
|
|
||||||
- [ ] My commits are related to the pull request and do not amend unrelated code or documentation.
|
|
||||||
- [ ] My code follows a similar style to existing code.
|
|
||||||
- [ ] I have added appropriate tests.
|
|
||||||
- [ ] I have updated or added relevant documentation.
|
|
||||||
|
|
||||||
## References
|
|
||||||
|
|
||||||
<!---
|
|
||||||
|
|
||||||
(Optional -- remove this section if not needed )
|
|
||||||
|
|
||||||
Include **important** links regarding the implementation of this PR.
|
|
||||||
|
|
||||||
This usually includes a RFC or an aggregation of issues and/or individual conversations
|
|
||||||
|
|
||||||
that helped put this solution together. This helps ensure we retain and share knowledge
|
|
||||||
|
|
||||||
regarding the implementation, and may help others understand motivation and design decisions etc..
|
|
||||||
|
|
||||||
-->
|
|
||||||
@@ -0,0 +1,78 @@
|
|||||||
|
name: Update issue Status
|
||||||
|
|
||||||
|
on:
|
||||||
|
issues:
|
||||||
|
types: [closed]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update_issue:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Get project data
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{secrets.GHPROJECT_TOKEN}}
|
||||||
|
ORGANIZATION: specklesystems
|
||||||
|
PROJECT_NUMBER: 9
|
||||||
|
run: |
|
||||||
|
gh api graphql --header 'GraphQL-Features: projects_next_graphql' -f query='
|
||||||
|
query($org: String!, $number: Int!) {
|
||||||
|
organization(login: $org){
|
||||||
|
projectNext(number: $number) {
|
||||||
|
id
|
||||||
|
fields(first:20) {
|
||||||
|
nodes {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
settings
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}' -f org=$ORGANIZATION -F number=$PROJECT_NUMBER > project_data.json
|
||||||
|
|
||||||
|
echo 'PROJECT_ID='$(jq '.data.organization.projectNext.id' project_data.json) >> $GITHUB_ENV
|
||||||
|
echo 'STATUS_FIELD_ID='$(jq '.data.organization.projectNext.fields.nodes[] | select(.name== "Status") | .id' project_data.json) >> $GITHUB_ENV
|
||||||
|
|
||||||
|
echo "$PROJECT_ID"
|
||||||
|
echo "$STATUS_FIELD_ID"
|
||||||
|
|
||||||
|
echo 'DONE_ID='$(jq '.data.organization.projectNext.fields.nodes[] | select(.name== "Status") | .settings | fromjson | .options[] | select(.name== "Done") | .id' project_data.json) >> $GITHUB_ENV
|
||||||
|
echo "$DONE_ID"
|
||||||
|
|
||||||
|
- name: Add Issue to project #it's already in the project, but we do this to get its node id!
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{secrets.GHPROJECT_TOKEN}}
|
||||||
|
ISSUE_ID: ${{ github.event.issue.node_id }}
|
||||||
|
run: |
|
||||||
|
item_id="$( gh api graphql --header 'GraphQL-Features: projects_next_graphql' -f query='
|
||||||
|
mutation($project:ID!, $id:ID!) {
|
||||||
|
addProjectNextItem(input: {projectId: $project, contentId: $id}) {
|
||||||
|
projectNextItem {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}' -f project=$PROJECT_ID -f id=$ISSUE_ID --jq '.data.addProjectNextItem.projectNextItem.id')"
|
||||||
|
|
||||||
|
echo 'ITEM_ID='$item_id >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Update Status
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{secrets.GHPROJECT_TOKEN}}
|
||||||
|
ISSUE_ID: ${{ github.event.issue.node_id }}
|
||||||
|
run: |
|
||||||
|
gh api graphql --header 'GraphQL-Features: projects_next_graphql' -f query='
|
||||||
|
mutation($project:ID!, $status:ID!, $id:ID!, $value:String!) {
|
||||||
|
set_status: updateProjectNextItemField(
|
||||||
|
input: {
|
||||||
|
projectId: $project
|
||||||
|
itemId: $id
|
||||||
|
fieldId: $status
|
||||||
|
value: $value
|
||||||
|
}
|
||||||
|
) {
|
||||||
|
projectNextItem {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}' -f project=$PROJECT_ID -f status=$STATUS_FIELD_ID -f id=$ITEM_ID -f value=${{ env.DONE_ID }}
|
||||||
|
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
name: Move new issues into Project
|
||||||
|
|
||||||
|
on:
|
||||||
|
issues:
|
||||||
|
types: [opened]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
track_issue:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Get project data
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{secrets.GHPROJECT_TOKEN}}
|
||||||
|
ORGANIZATION: specklesystems
|
||||||
|
PROJECT_NUMBER: 9
|
||||||
|
run: |
|
||||||
|
gh api graphql --header 'GraphQL-Features: projects_next_graphql' -f query='
|
||||||
|
query($org: String!, $number: Int!) {
|
||||||
|
organization(login: $org){
|
||||||
|
projectNext(number: $number) {
|
||||||
|
id
|
||||||
|
fields(first:20) {
|
||||||
|
nodes {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
settings
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}' -f org=$ORGANIZATION -F number=$PROJECT_NUMBER > project_data.json
|
||||||
|
|
||||||
|
echo 'PROJECT_ID='$(jq '.data.organization.projectNext.id' project_data.json) >> $GITHUB_ENV
|
||||||
|
echo 'STATUS_FIELD_ID='$(jq '.data.organization.projectNext.fields.nodes[] | select(.name== "Status") | .id' project_data.json) >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Add Issue to project
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{secrets.GHPROJECT_TOKEN}}
|
||||||
|
ISSUE_ID: ${{ github.event.issue.node_id }}
|
||||||
|
run: |
|
||||||
|
item_id="$( gh api graphql --header 'GraphQL-Features: projects_next_graphql' -f query='
|
||||||
|
mutation($project:ID!, $id:ID!) {
|
||||||
|
addProjectNextItem(input: {projectId: $project, contentId: $id}) {
|
||||||
|
projectNextItem {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}' -f project=$PROJECT_ID -f id=$ISSUE_ID --jq '.data.addProjectNextItem.projectNextItem.id')"
|
||||||
|
|
||||||
|
echo 'ITEM_ID='$item_id >> $GITHUB_ENV
|
||||||
+1
-1
@@ -112,4 +112,4 @@ venv.bak/
|
|||||||
|
|
||||||
# other
|
# other
|
||||||
scratch.py
|
scratch.py
|
||||||
settings.json
|
settings.json
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
repos:
|
|
||||||
- repo: https://github.com/charliermarsh/ruff-pre-commit
|
|
||||||
hooks:
|
|
||||||
- id: ruff
|
|
||||||
rev: v0.0.186
|
|
||||||
|
|
||||||
- repo: https://github.com/commitizen-tools/commitizen
|
|
||||||
hooks:
|
|
||||||
- id: commitizen
|
|
||||||
- id: commitizen-branch
|
|
||||||
stages:
|
|
||||||
- push
|
|
||||||
rev: v2.38.0
|
|
||||||
|
|
||||||
- repo: https://github.com/pycqa/isort
|
|
||||||
rev: v5.11.3
|
|
||||||
hooks:
|
|
||||||
- id: isort
|
|
||||||
|
|
||||||
- repo: https://github.com/psf/black
|
|
||||||
rev: 22.12.0
|
|
||||||
hooks:
|
|
||||||
- id: black
|
|
||||||
# It is recommended to specify the latest version of Python
|
|
||||||
# supported by your project here, or alternatively use
|
|
||||||
# pre-commit's default_language_version, see
|
|
||||||
# https://pre-commit.com/#top_level-default_language_version
|
|
||||||
# language_version: python3.11
|
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
|
||||||
rev: v4.4.0
|
|
||||||
hooks:
|
|
||||||
- id: end-of-file-fixer
|
|
||||||
- id: trailing-whitespace
|
|
||||||
Vendored
+3
-8
@@ -4,8 +4,6 @@
|
|||||||
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
|
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
|
||||||
"version": "0.2.0",
|
"version": "0.2.0",
|
||||||
"configurations": [
|
"configurations": [
|
||||||
|
|
||||||
|
|
||||||
{
|
{
|
||||||
"name": "Python: Current File",
|
"name": "Python: Current File",
|
||||||
"type": "python",
|
"type": "python",
|
||||||
@@ -15,13 +13,10 @@
|
|||||||
"justMyCode": false
|
"justMyCode": false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Pytest",
|
"name": "Python: Test debug config",
|
||||||
"type": "python",
|
"type": "python",
|
||||||
"request": "launch",
|
"request": "test",
|
||||||
"program": "pytest",
|
|
||||||
"args": [],
|
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"justMyCode": true
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -34,7 +34,7 @@ What is Speckle? Check our ](https://speckle.xyz) ⇒ creating an account at our public server
|
- [](https://speckle.xyz) ⇒ creating an account at our public server
|
||||||
- [](https://marketplace.digitalocean.com/apps/speckle-server?refcode=947a2b5d7dc1) ⇒ deploying an instance in 1 click
|
- [](https://marketplace.digitalocean.com/apps/speckle-server?refcode=947a2b5d7dc1) ⇒ deploying an instance in 1 click
|
||||||
|
|
||||||
### Resources
|
### Resources
|
||||||
|
|
||||||
@@ -47,7 +47,7 @@ Give Speckle a try in no time by:
|
|||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
Send and receive data from a Speckle Server with `operations`, interact with the Speckle API with the `SpeckleClient`, create and extend your own custom Speckle Objects with `Base`, and more!
|
Send and receive data from a Speckle Server with `operations`, interact with the Speckle API with the `SpeckleClient`, create and extend your own custom Speckle Objects with `Base`, and more!
|
||||||
|
|
||||||
Head to the [**📚 specklepy docs**](https://speckle.guide/dev/python.html) for more information and usage examples.
|
Head to the [**📚 specklepy docs**](https://speckle.guide/dev/python.html) for more information and usage examples.
|
||||||
|
|
||||||
@@ -65,12 +65,6 @@ To execute any python script run `$ poetry run python my_script.py`
|
|||||||
|
|
||||||
> Alternatively you may roll your own virtual-env with either venv, virtualenv, pyenv-virtualenv etc. Poetry will play along an recognize if it is invoked from inside a virtual environment.
|
> Alternatively you may roll your own virtual-env with either venv, virtualenv, pyenv-virtualenv etc. Poetry will play along an recognize if it is invoked from inside a virtual environment.
|
||||||
|
|
||||||
### Style guide
|
|
||||||
|
|
||||||
All our repo wide styling linting and other rules are checked and enforced by `pre-commit`, which is included in the dev dependencies.
|
|
||||||
It is recommended to set up `pre-commit` after installing the dependencies by running `$ pre-commit install`.
|
|
||||||
Commiting code that doesn't adhere to the given rules, will fail the checks in our CI system.
|
|
||||||
|
|
||||||
### Local Data Paths
|
### Local Data Paths
|
||||||
|
|
||||||
It may be helpful to know where the local accounts and object cache dbs are stored. Depending on on your OS, you can find the dbs at:
|
It may be helpful to know where the local accounts and object cache dbs are stored. Depending on on your OS, you can find the dbs at:
|
||||||
@@ -80,7 +74,7 @@ It may be helpful to know where the local accounts and object cache dbs are stor
|
|||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
Please make sure you read the [contribution guidelines](.github/CONTRIBUTING.md) and [code of conduct](.github/CODE_OF_CONDUCT.md) for an overview of the practices we try to follow.
|
Please make sure you read the [contribution guidelines](.github/CONTRIBUTING.md) for an overview of the best practices we try to follow.
|
||||||
|
|
||||||
## Community
|
## Community
|
||||||
|
|
||||||
@@ -88,7 +82,7 @@ The Speckle Community hangs out on [the forum](https://discourse.speckle.works),
|
|||||||
|
|
||||||
## Security
|
## Security
|
||||||
|
|
||||||
For any security vulnerabilities or concerns, please contact us directly at security[at]speckle.systems.
|
For any security vulnerabilities or concerns, please contact us directly at security[at]speckle.systems.
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
|||||||
-12
@@ -1,12 +0,0 @@
|
|||||||
# Security Policy
|
|
||||||
|
|
||||||
## Supported Versions
|
|
||||||
|
|
||||||
| Version | Supported |
|
|
||||||
| ------- | ------------------ |
|
|
||||||
| 2.2.+ | :white_check_mark: |
|
|
||||||
| < 2.2 | :x: |
|
|
||||||
|
|
||||||
## Reporting a Vulnerability
|
|
||||||
|
|
||||||
Hi! If you've found something off, we'd be more than happy if you would report it via security@speckle.systems. We will work together with you to correctly identify the cause and implement a fix. Thanks for helping make Speckle safer!
|
|
||||||
@@ -1,162 +0,0 @@
|
|||||||
version: "3.9"
|
|
||||||
name: "speckle-server"
|
|
||||||
|
|
||||||
services:
|
|
||||||
####
|
|
||||||
# Speckle Server dependencies
|
|
||||||
#######
|
|
||||||
postgres:
|
|
||||||
image: "postgres:14.5-alpine"
|
|
||||||
restart: always
|
|
||||||
environment:
|
|
||||||
POSTGRES_DB: speckle
|
|
||||||
POSTGRES_USER: speckle
|
|
||||||
POSTGRES_PASSWORD: speckle
|
|
||||||
volumes:
|
|
||||||
- postgres-data:/var/lib/postgresql/data/
|
|
||||||
healthcheck:
|
|
||||||
# the -U user has to match the POSTGRES_USER value
|
|
||||||
test: ["CMD-SHELL", "pg_isready -U speckle"]
|
|
||||||
interval: 5s
|
|
||||||
timeout: 5s
|
|
||||||
retries: 30
|
|
||||||
|
|
||||||
redis:
|
|
||||||
image: "redis:6.0-alpine"
|
|
||||||
restart: always
|
|
||||||
volumes:
|
|
||||||
- redis-data:/data
|
|
||||||
healthcheck:
|
|
||||||
test: ["CMD", "redis-cli", "--raw", "incr", "ping"]
|
|
||||||
interval: 5s
|
|
||||||
timeout: 5s
|
|
||||||
retries: 30
|
|
||||||
|
|
||||||
minio:
|
|
||||||
image: "minio/minio"
|
|
||||||
command: server /data --console-address ":9001"
|
|
||||||
restart: always
|
|
||||||
volumes:
|
|
||||||
- minio-data:/data
|
|
||||||
healthcheck:
|
|
||||||
test:
|
|
||||||
[
|
|
||||||
"CMD-SHELL",
|
|
||||||
"curl -s -o /dev/null http://127.0.0.1:9000/minio/index.html",
|
|
||||||
]
|
|
||||||
interval: 5s
|
|
||||||
timeout: 30s
|
|
||||||
retries: 30
|
|
||||||
start_period: 10s
|
|
||||||
|
|
||||||
####
|
|
||||||
# Speckle Server
|
|
||||||
#######
|
|
||||||
speckle-frontend:
|
|
||||||
image: speckle/speckle-frontend:latest
|
|
||||||
restart: always
|
|
||||||
ports:
|
|
||||||
- "0.0.0.0:8080:8080"
|
|
||||||
environment:
|
|
||||||
FILE_SIZE_LIMIT_MB: 100
|
|
||||||
|
|
||||||
speckle-server:
|
|
||||||
image: speckle/speckle-server:latest
|
|
||||||
restart: always
|
|
||||||
healthcheck:
|
|
||||||
test:
|
|
||||||
[
|
|
||||||
"CMD",
|
|
||||||
"node",
|
|
||||||
"-e",
|
|
||||||
"require('node:http').request({headers: {'Content-Type': 'application/json'}, port:3000, hostname:'127.0.0.1', path:'/graphql?query={serverInfo{version}}', method: 'GET' }, (res) => { body = ''; res.on('data', (chunk) => {body += chunk;}); res.on('end', () => {process.exit(res.statusCode != 200 || body.toLowerCase().includes('error'));}); }).end();",
|
|
||||||
]
|
|
||||||
interval: 10s
|
|
||||||
timeout: 3s
|
|
||||||
retries: 30
|
|
||||||
ports:
|
|
||||||
- "0.0.0.0:3000:3000"
|
|
||||||
depends_on:
|
|
||||||
postgres:
|
|
||||||
condition: service_healthy
|
|
||||||
redis:
|
|
||||||
condition: service_healthy
|
|
||||||
minio:
|
|
||||||
condition: service_healthy
|
|
||||||
environment:
|
|
||||||
# TODO: Change this to the URL of the speckle server, as accessed from the network
|
|
||||||
CANONICAL_URL: "http://127.0.0.1:8080"
|
|
||||||
SPECKLE_AUTOMATE_URL: "http://127.0.0.1:3030"
|
|
||||||
|
|
||||||
# TODO: Change thvolumes:
|
|
||||||
REDIS_URL: "redis://redis"
|
|
||||||
|
|
||||||
S3_ENDPOINT: "http://minio:9000"
|
|
||||||
S3_ACCESS_KEY: "minioadmin"
|
|
||||||
S3_SECRET_KEY: "minioadmin"
|
|
||||||
S3_BUCKET: "speckle-server"
|
|
||||||
S3_CREATE_BUCKET: "true"
|
|
||||||
|
|
||||||
FILE_SIZE_LIMIT_MB: 100
|
|
||||||
|
|
||||||
# TODO: Change this to a unique secret for this server
|
|
||||||
SESSION_SECRET: "TODO:ReplaceWithLongString"
|
|
||||||
|
|
||||||
STRATEGY_LOCAL: "true"
|
|
||||||
DEBUG: "speckle:*"
|
|
||||||
|
|
||||||
POSTGRES_URL: "postgres"
|
|
||||||
POSTGRES_USER: "speckle"
|
|
||||||
POSTGRES_PASSWORD: "speckle"
|
|
||||||
POSTGRES_DB: "speckle"
|
|
||||||
ENABLE_MP: "false"
|
|
||||||
|
|
||||||
preview-service:
|
|
||||||
image: speckle/speckle-preview-service:latest
|
|
||||||
restart: always
|
|
||||||
depends_on:
|
|
||||||
speckle-server:
|
|
||||||
condition: service_healthy
|
|
||||||
mem_limit: "1000m"
|
|
||||||
memswap_limit: "1000m"
|
|
||||||
environment:
|
|
||||||
DEBUG: "preview-service:*"
|
|
||||||
PG_CONNECTION_STRING: "postgres://speckle:speckle@postgres/speckle"
|
|
||||||
|
|
||||||
webhook-service:
|
|
||||||
image: speckle/speckle-webhook-service:latest
|
|
||||||
restart: always
|
|
||||||
depends_on:
|
|
||||||
speckle-server:
|
|
||||||
condition: service_healthy
|
|
||||||
environment:
|
|
||||||
DEBUG: "webhook-service:*"
|
|
||||||
PG_CONNECTION_STRING: "postgres://speckle:speckle@postgres/speckle"
|
|
||||||
WAIT_HOSTS: postgres:5432
|
|
||||||
|
|
||||||
fileimport-service:
|
|
||||||
image: speckle/speckle-fileimport-service:latest
|
|
||||||
restart: always
|
|
||||||
depends_on:
|
|
||||||
speckle-server:
|
|
||||||
condition: service_healthy
|
|
||||||
environment:
|
|
||||||
DEBUG: "fileimport-service:*"
|
|
||||||
PG_CONNECTION_STRING: "postgres://speckle:speckle@postgres/speckle"
|
|
||||||
WAIT_HOSTS: postgres:5432
|
|
||||||
|
|
||||||
S3_ENDPOINT: "http://minio:9000"
|
|
||||||
S3_ACCESS_KEY: "minioadmin"
|
|
||||||
S3_SECRET_KEY: "minioadmin"
|
|
||||||
S3_BUCKET: "speckle-server"
|
|
||||||
|
|
||||||
SPECKLE_SERVER_URL: "http://speckle-server:3000"
|
|
||||||
|
|
||||||
networks:
|
|
||||||
default:
|
|
||||||
name: speckle-server
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
postgres-data:
|
|
||||||
redis-data:
|
|
||||||
minio-data:
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
import os
|
|
||||||
import random
|
|
||||||
import string
|
|
||||||
import time
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.objects import Base
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
|
|
||||||
class Sub(Base):
|
|
||||||
bar: List[str]
|
|
||||||
|
|
||||||
|
|
||||||
def random_string():
|
|
||||||
letters = string.ascii_lowercase
|
|
||||||
return "".join(random.choice(letters) for _ in range(10))
|
|
||||||
|
|
||||||
|
|
||||||
BASE_PATH = SQLiteTransport.get_base_path("Speckle")
|
|
||||||
|
|
||||||
|
|
||||||
def clean_db():
|
|
||||||
os.remove(Path(BASE_PATH, "Objects.db"))
|
|
||||||
|
|
||||||
|
|
||||||
def one_pass(clean: bool, randomize: bool, child_count: int):
|
|
||||||
foo = Base()
|
|
||||||
for i in range(child_count):
|
|
||||||
stuff = random_string() if randomize else "stuff"
|
|
||||||
foo[f"@child_{i}"] = Sub(bar=["asdf", "bar", i, stuff])
|
|
||||||
|
|
||||||
if clean:
|
|
||||||
clean_db()
|
|
||||||
transport = SQLiteTransport()
|
|
||||||
start = time.time()
|
|
||||||
hash = operations.send(base=foo, transports=[transport])
|
|
||||||
send_time = time.time() - start
|
|
||||||
|
|
||||||
receive_start = time.time()
|
|
||||||
operations.receive(hash, transport)
|
|
||||||
receive_time = time.time() - receive_start
|
|
||||||
return send_time, receive_time
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
sample_size = 4
|
|
||||||
|
|
||||||
test_permutations = [
|
|
||||||
(True, True),
|
|
||||||
(False, False),
|
|
||||||
(False, True),
|
|
||||||
(True, False),
|
|
||||||
]
|
|
||||||
for clean, randomize in test_permutations:
|
|
||||||
print(f"CLEAN: {clean}, RANDOMIZE: {randomize}")
|
|
||||||
for child_count in [10, 100, 1000, 10000]:
|
|
||||||
print(f"\tCHILD COUNT: {child_count}")
|
|
||||||
for _ in range(sample_size):
|
|
||||||
send_time, receive_time = one_pass(clean, randomize, child_count)
|
|
||||||
print(f"\t\tSend: {send_time} Receive: {receive_time}")
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
from devtools import debug
|
|
||||||
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.api.wrapper import StreamWrapper
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
stream_url = "https://latest.speckle.dev/streams/7d051a6449"
|
|
||||||
wrapper = StreamWrapper(stream_url)
|
|
||||||
|
|
||||||
transport = wrapper.get_transport()
|
|
||||||
|
|
||||||
rec = operations.receive("98396753f8bf7fe1cb60c5193e9f9d86", transport)
|
|
||||||
|
|
||||||
# hash = operations.send(base=foo, transports=[transport], use_default_cache=False)
|
|
||||||
debug(rec)
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
import random
|
|
||||||
import string
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.api.wrapper import StreamWrapper
|
|
||||||
from specklepy.objects import Base
|
|
||||||
|
|
||||||
|
|
||||||
class Sub(Base):
|
|
||||||
bar: List[str]
|
|
||||||
|
|
||||||
|
|
||||||
def random_string():
|
|
||||||
letters = string.ascii_lowercase
|
|
||||||
return "".join(random.choice(letters) for _ in range(10))
|
|
||||||
|
|
||||||
|
|
||||||
def create_object(child_count: int) -> Base:
|
|
||||||
foo = Base()
|
|
||||||
for i in range(child_count):
|
|
||||||
stuff = random_string()
|
|
||||||
foo[f"@child_{i}"] = Sub(bar=["asdf", "bar", i, stuff])
|
|
||||||
return foo
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
stream_url = "http://hyperion:3000/streams/2372b54c35"
|
|
||||||
|
|
||||||
child_count = 10
|
|
||||||
foo = create_object(child_count)
|
|
||||||
|
|
||||||
wrapper = StreamWrapper(stream_url)
|
|
||||||
transport = wrapper.get_transport()
|
|
||||||
|
|
||||||
hash = operations.send(base=foo, transports=[transport], use_default_cache=False)
|
|
||||||
|
|
||||||
rec = operations.receive(hash, transport)
|
|
||||||
print(rec)
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
from devtools import debug
|
|
||||||
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.objects.geometry import Base
|
|
||||||
from specklepy.objects.units import Units
|
|
||||||
|
|
||||||
dct = {
|
|
||||||
"id": "1234abcd",
|
|
||||||
"units": None,
|
|
||||||
"speckle_type": "Base",
|
|
||||||
"applicationId": None,
|
|
||||||
"totalChildrenCount": 0,
|
|
||||||
}
|
|
||||||
base = Base()
|
|
||||||
for prop, value in dct.items():
|
|
||||||
base.__setattr__(prop, value)
|
|
||||||
|
|
||||||
|
|
||||||
debug(base)
|
|
||||||
debug(base.units)
|
|
||||||
|
|
||||||
base.units = "m"
|
|
||||||
debug(base.units)
|
|
||||||
base.units = None
|
|
||||||
|
|
||||||
debug(base.units)
|
|
||||||
|
|
||||||
foo = operations.serialize(base)
|
|
||||||
|
|
||||||
base.units = 10
|
|
||||||
|
|
||||||
debug(base.units)
|
|
||||||
debug(foo)
|
|
||||||
|
|
||||||
base.units = Units.mm
|
|
||||||
debug(base.units)
|
|
||||||
@@ -1,10 +1,9 @@
|
|||||||
"""This is an example showcasing the usage of speckle `Base` class."""
|
"""This is an example showcasing the usage of speckle `Base` class."""
|
||||||
|
|
||||||
# the speckle.objects module exposes all speckle provided classes
|
# the speckle.objects module exposes all speckle provided classes
|
||||||
from devtools import debug
|
|
||||||
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.objects import Base
|
from specklepy.objects import Base
|
||||||
|
from specklepy.api import operations
|
||||||
|
from devtools import debug
|
||||||
|
|
||||||
|
|
||||||
class ExampleSub(Base):
|
class ExampleSub(Base):
|
||||||
@@ -51,10 +50,10 @@ if __name__ == "__main__":
|
|||||||
)
|
)
|
||||||
# support for dynamic attributes
|
# support for dynamic attributes
|
||||||
custom_sub.extra_extra = "what is this?"
|
custom_sub.extra_extra = "what is this?"
|
||||||
debug(custom_sub)
|
debug(custom_sub.json())
|
||||||
|
|
||||||
serialized = operations.serialize(custom_sub)
|
serialized = operations.serialize(custom_sub)
|
||||||
deserialized = operations.deserialize(serialized)
|
deserialized = operations.deserialize(serialized)
|
||||||
# the only difference should be between the two data is that the deserialized
|
# the only difference should be between the two data is that the deserialized
|
||||||
# instance id attribute is not None.
|
# instance id attribute is not None.
|
||||||
debug(deserialized)
|
debug(deserialized.json())
|
||||||
|
|||||||
+1
-1
@@ -9,7 +9,7 @@ def patch(tag):
|
|||||||
lines = f.readlines()
|
lines = f.readlines()
|
||||||
|
|
||||||
if "version" not in lines[2]:
|
if "version" not in lines[2]:
|
||||||
raise Exception("Invalid pyproject.toml. Could not patch version.")
|
raise Exception(f"Invalid pyproject.toml. Could not patch version.")
|
||||||
|
|
||||||
lines[2] = f'version = "{tag}"\n'
|
lines[2] = f'version = "{tag}"\n'
|
||||||
with open("pyproject.toml", "w") as file:
|
with open("pyproject.toml", "w") as file:
|
||||||
|
|||||||
Generated
+818
-1544
File diff suppressed because it is too large
Load Diff
+10
-33
@@ -1,6 +1,6 @@
|
|||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "specklepy"
|
name = "specklepy"
|
||||||
version = "2.9.1"
|
version = "2.4.0"
|
||||||
description = "The Python SDK for Speckle 2.0"
|
description = "The Python SDK for Speckle 2.0"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
authors = ["Speckle Systems <devops@speckle.systems>"]
|
authors = ["Speckle Systems <devops@speckle.systems>"]
|
||||||
@@ -8,38 +8,22 @@ license = "Apache-2.0"
|
|||||||
repository = "https://github.com/specklesystems/speckle-py"
|
repository = "https://github.com/specklesystems/speckle-py"
|
||||||
documentation = "https://speckle.guide/dev/py-examples.html"
|
documentation = "https://speckle.guide/dev/py-examples.html"
|
||||||
homepage = "https://speckle.systems/"
|
homepage = "https://speckle.systems/"
|
||||||
packages = [
|
|
||||||
{ include = "specklepy", from = "src" },
|
|
||||||
{ include = "speckle_automate", from = "src" },
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
[tool.poetry.dependencies]
|
[tool.poetry.dependencies]
|
||||||
python = ">=3.8.0, <4.0"
|
python = "^3.6.5"
|
||||||
pydantic = "^2.0"
|
pydantic = "^1.8.2"
|
||||||
appdirs = "^1.4.4"
|
appdirs = "^1.4.4"
|
||||||
gql = {extras = ["requests", "websockets"], version = "^3.3.0"}
|
gql = {version = ">=3.0.0b1", extras = ["all"], allow-prereleases = true}
|
||||||
ujson = "^5.3.0"
|
ujson = "^4.3.0"
|
||||||
Deprecated = "^1.2.13"
|
|
||||||
stringcase = "^1.2.0"
|
|
||||||
attrs = "^23.1.0"
|
|
||||||
httpx = "^0.25.0"
|
|
||||||
|
|
||||||
[tool.poetry.group.dev.dependencies]
|
[tool.poetry.dev-dependencies]
|
||||||
black = "^22.8.0"
|
black = "^20.8b1"
|
||||||
isort = "^5.7.0"
|
isort = "^5.7.0"
|
||||||
pytest = "^7.1.3"
|
pytest = "^6.2.2"
|
||||||
pytest-ordering = "^0.6"
|
pytest-ordering = "^0.6"
|
||||||
pytest-cov = "^3.0.0"
|
pytest-cov = "^3.0.0"
|
||||||
devtools = "^0.8.0"
|
|
||||||
pylint = "^2.14.4"
|
|
||||||
mypy = "^0.982"
|
|
||||||
pre-commit = "^2.20.0"
|
|
||||||
commitizen = "^2.38.0"
|
|
||||||
ruff = "^0.0.187"
|
|
||||||
types-deprecated = "^1.2.9"
|
|
||||||
types-ujson = "^5.6.0.0"
|
|
||||||
types-requests = "^2.28.11.5"
|
|
||||||
|
|
||||||
[tool.black]
|
[tool.black]
|
||||||
exclude = '''
|
exclude = '''
|
||||||
@@ -58,16 +42,9 @@ exclude = '''
|
|||||||
'''
|
'''
|
||||||
include = '\.pyi?$'
|
include = '\.pyi?$'
|
||||||
line-length = 88
|
line-length = 88
|
||||||
target-version = ["py37", "py38", "py39", "py310", "py311"]
|
target-version = ["py36", "py37", "py38"]
|
||||||
|
|
||||||
|
|
||||||
[tool.commitizen]
|
|
||||||
name = "cz_conventional_commits"
|
|
||||||
version = "2.9.2"
|
|
||||||
tag_format = "$version"
|
|
||||||
[build-system]
|
[build-system]
|
||||||
requires = ["poetry-core>=1.0.0"]
|
requires = ["poetry-core>=1.0.0"]
|
||||||
build-backend = "poetry.core.masonry.api"
|
build-backend = "poetry.core.masonry.api"
|
||||||
|
|
||||||
[tool.isort]
|
|
||||||
profile = "black"
|
|
||||||
|
|||||||
@@ -0,0 +1,147 @@
|
|||||||
|
import re
|
||||||
|
from gql.client import SyncClientSession
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from typing import Dict
|
||||||
|
|
||||||
|
from specklepy.api import resources
|
||||||
|
from specklepy.api.resources import (
|
||||||
|
branch,
|
||||||
|
commit,
|
||||||
|
stream,
|
||||||
|
object,
|
||||||
|
server,
|
||||||
|
user,
|
||||||
|
subscriptions,
|
||||||
|
)
|
||||||
|
from specklepy.api.models import ServerInfo
|
||||||
|
from gql import Client, gql
|
||||||
|
from gql.transport.requests import RequestsHTTPTransport
|
||||||
|
from gql.transport.aiohttp import AIOHTTPTransport
|
||||||
|
from gql.transport.websockets import WebsocketsTransport
|
||||||
|
|
||||||
|
|
||||||
|
class SpeckleClient:
|
||||||
|
"""
|
||||||
|
The `SpeckleClient` is your entry point for interacting with your Speckle Server's GraphQL API.
|
||||||
|
You'll need to have access to a server to use it, or you can use our public server `speckle.xyz`.
|
||||||
|
|
||||||
|
To authenticate the client, you'll need to have downloaded the [Speckle Manager](https://speckle.guide/#speckle-manager)
|
||||||
|
and added your account.
|
||||||
|
|
||||||
|
```py
|
||||||
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.api.credentials import get_default_account
|
||||||
|
|
||||||
|
# initialise the client
|
||||||
|
client = SpeckleClient(host="speckle.xyz") # or whatever your host is
|
||||||
|
# client = SpeckleClient(host="localhost:3000", use_ssl=False) or use local server
|
||||||
|
|
||||||
|
# authenticate the client with a token (account has been added in Speckle Manager)
|
||||||
|
account = get_default_account()
|
||||||
|
client.authenticate(token=account.token)
|
||||||
|
|
||||||
|
# create a new stream. this returns the stream id
|
||||||
|
new_stream_id = client.stream.create(name="a shiny new stream")
|
||||||
|
|
||||||
|
# use that stream id to get the stream from the server
|
||||||
|
new_stream = client.stream.get(id=new_stream_id)
|
||||||
|
```
|
||||||
|
"""
|
||||||
|
|
||||||
|
DEFAULT_HOST = "speckle.xyz"
|
||||||
|
USE_SSL = True
|
||||||
|
|
||||||
|
def __init__(self, host: str = DEFAULT_HOST, use_ssl: bool = USE_SSL) -> None:
|
||||||
|
ws_protocol = "ws"
|
||||||
|
http_protocol = "http"
|
||||||
|
|
||||||
|
if use_ssl:
|
||||||
|
ws_protocol = "wss"
|
||||||
|
http_protocol = "https"
|
||||||
|
|
||||||
|
# sanitise host input by removing protocol and trailing slash
|
||||||
|
host = re.sub(r"((^\w+:|^)\/\/)|(\/$)", "", host)
|
||||||
|
|
||||||
|
self.url = f"{http_protocol}://{host}"
|
||||||
|
self.graphql = self.url + "/graphql"
|
||||||
|
self.ws_url = f"{ws_protocol}://{host}/graphql"
|
||||||
|
self.me = None
|
||||||
|
|
||||||
|
self.httpclient = Client(
|
||||||
|
transport=RequestsHTTPTransport(url=self.graphql, verify=True, retries=3)
|
||||||
|
)
|
||||||
|
self.wsclient = None
|
||||||
|
|
||||||
|
self._init_resources()
|
||||||
|
|
||||||
|
# Check compatibility with the server
|
||||||
|
try:
|
||||||
|
serverInfo = self.server.get()
|
||||||
|
if not isinstance(serverInfo, ServerInfo):
|
||||||
|
raise Exception("Couldn't get ServerInfo")
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException(f"{self.url} is not a compatible Speckle Server", ex)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return (
|
||||||
|
f"SpeckleClient( server: {self.url}, authenticated: {self.me is not None} )"
|
||||||
|
)
|
||||||
|
|
||||||
|
def authenticate(self, token: str) -> None:
|
||||||
|
"""Authenticate the client using a personal access token
|
||||||
|
The token is saved in the client object and a synchronous GraphQL entrypoint is created
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
token {str} -- an api token
|
||||||
|
"""
|
||||||
|
self.me = {"token": token}
|
||||||
|
headers = {
|
||||||
|
"Authorization": f"Bearer {self.me['token']}",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
httptransport = RequestsHTTPTransport(
|
||||||
|
url=self.graphql, headers=headers, verify=True, retries=3
|
||||||
|
)
|
||||||
|
wstransport = WebsocketsTransport(
|
||||||
|
url=self.ws_url,
|
||||||
|
init_payload={"Authorization": f"Bearer {self.me['token']}"},
|
||||||
|
)
|
||||||
|
self.httpclient = Client(transport=httptransport)
|
||||||
|
self.wsclient = Client(transport=wstransport)
|
||||||
|
|
||||||
|
self._init_resources()
|
||||||
|
|
||||||
|
def execute_query(self, query: str) -> Dict:
|
||||||
|
return self.httpclient.execute(query)
|
||||||
|
|
||||||
|
def _init_resources(self) -> None:
|
||||||
|
self.stream = stream.Resource(
|
||||||
|
me=self.me, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
self.commit = commit.Resource(
|
||||||
|
me=self.me, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
self.branch = branch.Resource(
|
||||||
|
me=self.me, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
self.object = object.Resource(
|
||||||
|
me=self.me, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
self.server = server.Resource(
|
||||||
|
me=self.me, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
self.user = user.Resource(me=self.me, basepath=self.url, client=self.httpclient)
|
||||||
|
self.subscribe = subscriptions.Resource(
|
||||||
|
me=self.me,
|
||||||
|
basepath=self.ws_url,
|
||||||
|
client=self.wsclient,
|
||||||
|
)
|
||||||
|
|
||||||
|
def __getattr__(self, name):
|
||||||
|
try:
|
||||||
|
attr = getattr(resources, name)
|
||||||
|
return attr.Resource(me=self.me, basepath=self.url, client=self.httpclient)
|
||||||
|
except:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Method {name} is not supported by the SpeckleClient class"
|
||||||
|
)
|
||||||
@@ -0,0 +1,230 @@
|
|||||||
|
import os
|
||||||
|
from warnings import warn
|
||||||
|
from pydantic import BaseModel
|
||||||
|
from typing import List, Optional
|
||||||
|
from urllib.parse import urlparse, unquote
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.api.models import ServerInfo
|
||||||
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
from specklepy.transports.server.server import ServerTransport
|
||||||
|
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
||||||
|
|
||||||
|
|
||||||
|
class UserInfo(BaseModel):
|
||||||
|
name: str
|
||||||
|
email: str
|
||||||
|
company: Optional[str]
|
||||||
|
id: str
|
||||||
|
|
||||||
|
|
||||||
|
class Account(BaseModel):
|
||||||
|
isDefault: bool = None
|
||||||
|
token: str
|
||||||
|
refreshToken: str = None
|
||||||
|
serverInfo: ServerInfo
|
||||||
|
userInfo: UserInfo
|
||||||
|
id: str = None
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return f"Account(email: {self.userInfo.email}, server: {self.serverInfo.url}, isDefault: {self.isDefault})"
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
|
||||||
|
def get_local_accounts(base_path: str = None) -> List[Account]:
|
||||||
|
"""Gets all the accounts present in this environment
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
base_path {str} -- custom base path if you are not using the system default
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[Account] -- list of all local accounts or an empty list if no accounts were found
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.ACCOUNT_LIST)
|
||||||
|
account_storage = SQLiteTransport(scope="Accounts", base_path=base_path)
|
||||||
|
json_path = os.path.join(account_storage._base_path, "Accounts")
|
||||||
|
os.makedirs(json_path, exist_ok=True)
|
||||||
|
json_acct_files = [file for file in os.listdir(json_path) if file.endswith(".json")]
|
||||||
|
|
||||||
|
accounts = []
|
||||||
|
res = account_storage.get_all_objects()
|
||||||
|
if res:
|
||||||
|
accounts.extend(Account.parse_raw(r[1]) for r in res)
|
||||||
|
if json_acct_files:
|
||||||
|
try:
|
||||||
|
accounts.extend(
|
||||||
|
Account.parse_file(os.path.join(json_path, json_file))
|
||||||
|
for json_file in json_acct_files
|
||||||
|
)
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException(
|
||||||
|
"Invalid json accounts could not be read. Please fix or remove them.",
|
||||||
|
ex,
|
||||||
|
)
|
||||||
|
|
||||||
|
return accounts
|
||||||
|
|
||||||
|
|
||||||
|
def get_default_account(base_path: str = None) -> Account:
|
||||||
|
"""Gets this environment's default account if any. If there is no default, the first found will be returned and set as default.
|
||||||
|
Arguments:
|
||||||
|
base_path {str} -- custom base path if you are not using the system default
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Account -- the default account or None if no local accounts were found
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.ACCOUNT_DEFAULT)
|
||||||
|
accounts = get_local_accounts(base_path=base_path)
|
||||||
|
if not accounts:
|
||||||
|
return None
|
||||||
|
|
||||||
|
default = next((acc for acc in accounts if acc.isDefault), None)
|
||||||
|
if not default:
|
||||||
|
default = accounts[0]
|
||||||
|
default.isDefault = True
|
||||||
|
|
||||||
|
return default
|
||||||
|
|
||||||
|
|
||||||
|
class StreamWrapper:
|
||||||
|
"""
|
||||||
|
The `StreamWrapper` gives you some handy helpers to deal with urls and get authenticated clients and transports.
|
||||||
|
|
||||||
|
Construct a `StreamWrapper` with a stream, branch, commit, or object URL. The corresponding ids will be stored
|
||||||
|
in the wrapper. If you have local accounts on the machine, you can use the `get_account` and `get_client` methods
|
||||||
|
to get a local account for the server. You can also pass a token into `get_client` if you don't have a corresponding
|
||||||
|
local account for the server.
|
||||||
|
|
||||||
|
```py
|
||||||
|
from specklepy.api.credentials import StreamWrapper
|
||||||
|
|
||||||
|
# provide any stream, branch, commit, object, or globals url
|
||||||
|
wrapper = StreamWrapper("https://speckle.xyz/streams/3073b96e86/commits/604bea8cc6")
|
||||||
|
|
||||||
|
# get an authenticated SpeckleClient if you have a local account for the server
|
||||||
|
client = wrapper.get_client()
|
||||||
|
|
||||||
|
# get an authenticated ServerTransport if you have a local account for the server
|
||||||
|
transport = wrapper.get_transport()
|
||||||
|
```
|
||||||
|
"""
|
||||||
|
|
||||||
|
stream_url: str = None
|
||||||
|
use_ssl: bool = True
|
||||||
|
host: str = None
|
||||||
|
stream_id: str = None
|
||||||
|
commit_id: str = None
|
||||||
|
object_id: str = None
|
||||||
|
branch_name: str = None
|
||||||
|
client: SpeckleClient = None
|
||||||
|
account: Account = None
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"StreamWrapper( server: {self.host}, stream_id: {self.stream_id}, type: {self.type} )"
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self) -> str:
|
||||||
|
if self.object_id:
|
||||||
|
return "object"
|
||||||
|
elif self.commit_id:
|
||||||
|
return "commit"
|
||||||
|
elif self.branch_name:
|
||||||
|
return "branch"
|
||||||
|
else:
|
||||||
|
return "stream" if self.stream_id else "invalid"
|
||||||
|
|
||||||
|
def __init__(self, url: str) -> None:
|
||||||
|
metrics.track("streamwrapper")
|
||||||
|
self.stream_url = url
|
||||||
|
parsed = urlparse(url)
|
||||||
|
self.host = parsed.netloc
|
||||||
|
self.use_ssl = parsed.scheme == "https"
|
||||||
|
segments = parsed.path.strip("/").split("/", 3)
|
||||||
|
|
||||||
|
if not segments or len(segments) < 2:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Cannot parse {url} into a stream wrapper class - invalid URL provided."
|
||||||
|
)
|
||||||
|
|
||||||
|
while segments:
|
||||||
|
segment = segments.pop(0)
|
||||||
|
if segments and segment.lower() == "streams":
|
||||||
|
self.stream_id = segments.pop(0)
|
||||||
|
elif segments and segment.lower() == "commits":
|
||||||
|
self.commit_id = segments.pop(0)
|
||||||
|
elif segments and segment.lower() == "branches":
|
||||||
|
self.branch_name = unquote(segments.pop(0))
|
||||||
|
elif segments and segment.lower() == "objects":
|
||||||
|
self.object_id = segments.pop(0)
|
||||||
|
elif segment.lower() == "globals":
|
||||||
|
self.branch_name = "globals"
|
||||||
|
if segments:
|
||||||
|
self.commit_id = segments.pop(0)
|
||||||
|
else:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Cannot parse {url} into a stream wrapper class - invalid URL provided."
|
||||||
|
)
|
||||||
|
|
||||||
|
if not self.stream_id:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Cannot parse {url} into a stream wrapper class - no stream id found."
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_account(self) -> Account:
|
||||||
|
"""
|
||||||
|
Gets an account object for this server from the local accounts db (added via Speckle Manager or a json file)
|
||||||
|
"""
|
||||||
|
if self.account:
|
||||||
|
return self.account
|
||||||
|
|
||||||
|
self.account = next(
|
||||||
|
(a for a in get_local_accounts() if self.host in a.serverInfo.url),
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.account
|
||||||
|
|
||||||
|
def get_client(self, token: str = None) -> SpeckleClient:
|
||||||
|
"""
|
||||||
|
Gets an authenticated client for this server. You may provide a token if there aren't any local accounts on this
|
||||||
|
machine. If no account is found and no token is provided, an unauthenticated client is returned.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
token {str} -- optional token if no local account is available (defaults to None)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SpeckleClient -- authenticated with a corresponding local account or the provided token
|
||||||
|
"""
|
||||||
|
if self.client and token is None:
|
||||||
|
return self.client
|
||||||
|
|
||||||
|
if not self.account:
|
||||||
|
self.get_account()
|
||||||
|
|
||||||
|
if not self.client:
|
||||||
|
self.client = SpeckleClient(host=self.host, use_ssl=self.use_ssl)
|
||||||
|
|
||||||
|
if self.account is None and token is None:
|
||||||
|
warn(f"No local account found for server {self.host}", SpeckleWarning)
|
||||||
|
return self.client
|
||||||
|
|
||||||
|
self.client.authenticate(self.account.token if self.account else token)
|
||||||
|
|
||||||
|
return self.client
|
||||||
|
|
||||||
|
def get_transport(self, token: str = None) -> ServerTransport:
|
||||||
|
"""
|
||||||
|
Gets a server transport for this stream using an authenticated client. If there is no local account for this
|
||||||
|
server and the client was not authenticated with a token, this will throw an exception.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
ServerTransport -- constructed for this stream with a pre-authenticated client
|
||||||
|
"""
|
||||||
|
if not self.client or not self.client.me:
|
||||||
|
self.get_client(token)
|
||||||
|
return ServerTransport(self.stream_id, self.client)
|
||||||
@@ -0,0 +1,119 @@
|
|||||||
|
# generated by datamodel-codegen:
|
||||||
|
# filename: stream_schema.json
|
||||||
|
# timestamp: 2020-11-17T14:33:13+00:00
|
||||||
|
|
||||||
|
from datetime import datetime
|
||||||
|
from typing import Any, Dict, List, Optional
|
||||||
|
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class Collaborator(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
name: Optional[str]
|
||||||
|
role: Optional[str]
|
||||||
|
avatar: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class Commit(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
message: Optional[str]
|
||||||
|
authorName: Optional[str]
|
||||||
|
authorId: Optional[str]
|
||||||
|
authorAvatar: Optional[str]
|
||||||
|
branchName: Optional[str]
|
||||||
|
createdAt: Optional[str]
|
||||||
|
sourceApplication: Optional[str]
|
||||||
|
referencedObject: Optional[str]
|
||||||
|
totalChildrenCount: Optional[int]
|
||||||
|
parents: Optional[List[str]]
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return f"Commit( id: {self.id}, message: {self.message}, referencedObject: {self.referencedObject}, authorName: {self.authorName}, branchName: {self.branchName}, createdAt: {self.createdAt} )"
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
|
||||||
|
class Commits(BaseModel):
|
||||||
|
totalCount: Optional[int]
|
||||||
|
cursor: Optional[Any]
|
||||||
|
items: List[Commit] = []
|
||||||
|
|
||||||
|
|
||||||
|
class Object(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
speckleType: Optional[str]
|
||||||
|
applicationId: Optional[str]
|
||||||
|
totalChildrenCount: Optional[int]
|
||||||
|
createdAt: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class Branch(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
name: Optional[str]
|
||||||
|
description: Optional[str]
|
||||||
|
commits: Optional[Commits]
|
||||||
|
|
||||||
|
|
||||||
|
class Branches(BaseModel):
|
||||||
|
totalCount: Optional[int]
|
||||||
|
cursor: Optional[datetime]
|
||||||
|
items: List[Branch] = []
|
||||||
|
|
||||||
|
|
||||||
|
class Stream(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
name: Optional[str]
|
||||||
|
description: Optional[str]
|
||||||
|
isPublic: Optional[bool]
|
||||||
|
createdAt: Optional[str]
|
||||||
|
updatedAt: Optional[str]
|
||||||
|
collaborators: List[Collaborator] = []
|
||||||
|
branches: Optional[Branches]
|
||||||
|
commit: Optional[Commit]
|
||||||
|
object: Optional[Object]
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"Stream( id: {self.id}, name: {self.name}, description: {self.description}, isPublic: {self.isPublic})"
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
|
||||||
|
class Streams(BaseModel):
|
||||||
|
totalCount: Optional[int]
|
||||||
|
cursor: Optional[datetime]
|
||||||
|
items: List[Stream] = []
|
||||||
|
|
||||||
|
|
||||||
|
class User(BaseModel):
|
||||||
|
id: Optional[str]
|
||||||
|
email: Optional[str]
|
||||||
|
name: Optional[str]
|
||||||
|
bio: Optional[str]
|
||||||
|
company: Optional[str]
|
||||||
|
avatar: Optional[str]
|
||||||
|
verified: Optional[bool]
|
||||||
|
role: Optional[str]
|
||||||
|
streams: Optional[Streams]
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"User( id: {self.id}, name: {self.name}, email: {self.email}, company: {self.company} )"
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
|
||||||
|
class ServerInfo(BaseModel):
|
||||||
|
name: Optional[str]
|
||||||
|
company: Optional[str]
|
||||||
|
url: Optional[str]
|
||||||
|
description: Optional[str]
|
||||||
|
adminContact: Optional[str]
|
||||||
|
canonicalUrl: Optional[str]
|
||||||
|
roles: Optional[List[dict]]
|
||||||
|
scopes: Optional[List[dict]]
|
||||||
|
authStrategies: Optional[List[dict]]
|
||||||
|
version: Optional[str]
|
||||||
@@ -1,16 +1,15 @@
|
|||||||
from typing import List, Optional
|
from typing import List
|
||||||
|
from specklepy.logging import metrics
|
||||||
#from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
from specklepy.objects.base import Base
|
from specklepy.objects.base import Base
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
||||||
|
|
||||||
|
|
||||||
def send(
|
def send(
|
||||||
base: Base,
|
base: Base,
|
||||||
transports: Optional[List[AbstractTransport]] = None,
|
transports: List[AbstractTransport] = [],
|
||||||
use_default_cache: bool = True,
|
use_default_cache: bool = True,
|
||||||
):
|
):
|
||||||
"""Sends an object via the provided transports. Defaults to the local cache.
|
"""Sends an object via the provided transports. Defaults to the local cache.
|
||||||
@@ -18,41 +17,35 @@ def send(
|
|||||||
Arguments:
|
Arguments:
|
||||||
obj {Base} -- the object you want to send
|
obj {Base} -- the object you want to send
|
||||||
transports {list} -- where you want to send them
|
transports {list} -- where you want to send them
|
||||||
use_default_cache {bool} -- toggle for the default cache.
|
use_default_cache {bool} -- toggle for the default cache. If set to false, it will only send to the provided transports
|
||||||
If set to false, it will only send to the provided transports
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the object id of the sent object
|
str -- the object id of the sent object
|
||||||
"""
|
"""
|
||||||
|
metrics.track(metrics.SEND)
|
||||||
if not transports and not use_default_cache:
|
if not transports and not use_default_cache:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
message=(
|
message="You need to provide at least one transport: cannot send with an empty transport list and no default cache"
|
||||||
"You need to provide at least one transport: cannot send with an empty"
|
|
||||||
" transport list and no default cache"
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
if isinstance(transports, AbstractTransport):
|
|
||||||
transports = [transports]
|
|
||||||
|
|
||||||
if transports is None:
|
|
||||||
transports = []
|
|
||||||
|
|
||||||
if use_default_cache:
|
if use_default_cache:
|
||||||
transports.insert(0, SQLiteTransport())
|
transports.insert(0, SQLiteTransport())
|
||||||
|
|
||||||
serializer = BaseObjectSerializer(write_transports=transports)
|
serializer = BaseObjectSerializer(write_transports=transports)
|
||||||
|
|
||||||
obj_hash, _ = serializer.write_json(base=base)
|
for t in transports:
|
||||||
|
t.begin_write()
|
||||||
|
hash, _ = serializer.write_json(base=base)
|
||||||
|
|
||||||
return obj_hash
|
for t in transports:
|
||||||
|
t.end_write()
|
||||||
|
|
||||||
|
return hash
|
||||||
|
|
||||||
|
|
||||||
def receive(
|
def receive(
|
||||||
obj_id: str,
|
obj_id: str,
|
||||||
remote_transport: Optional[AbstractTransport] = None,
|
remote_transport: AbstractTransport = None,
|
||||||
local_transport: Optional[AbstractTransport] = None,
|
local_transport: AbstractTransport = None,
|
||||||
) -> Base:
|
) -> Base:
|
||||||
"""Receives an object from a transport.
|
"""Receives an object from a transport.
|
||||||
|
|
||||||
@@ -65,22 +58,20 @@ def receive(
|
|||||||
Returns:
|
Returns:
|
||||||
Base -- the base object
|
Base -- the base object
|
||||||
"""
|
"""
|
||||||
|
metrics.track(metrics.RECEIVE)
|
||||||
if not local_transport:
|
if not local_transport:
|
||||||
local_transport = SQLiteTransport()
|
local_transport = SQLiteTransport()
|
||||||
|
|
||||||
serializer = BaseObjectSerializer(read_transport=local_transport)
|
serializer = BaseObjectSerializer(read_transport=local_transport)
|
||||||
|
|
||||||
# try local transport first. if the parent is there, we assume all the children are there and continue with deserialization using the local transport
|
# try local transport first. if the parent is there, we assume all the children are there and continue wth deserialisation using the local transport
|
||||||
obj_string = local_transport.get_object(obj_id)
|
obj_string = local_transport.get_object(obj_id)
|
||||||
if obj_string:
|
if obj_string:
|
||||||
return serializer.read_json(obj_string=obj_string)
|
return serializer.read_json(obj_string=obj_string)
|
||||||
|
|
||||||
if not remote_transport:
|
if not remote_transport:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
message=(
|
message="Could not find the specified object using the local transport, and you didn't provide a fallback remote from which to pull it."
|
||||||
"Could not find the specified object using the local transport, and you"
|
|
||||||
" didn't provide a fallback remote from which to pull it."
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
obj_string = remote_transport.copy_object_and_children(
|
obj_string = remote_transport.copy_object_and_children(
|
||||||
@@ -92,48 +83,38 @@ def receive(
|
|||||||
|
|
||||||
def serialize(base: Base, write_transports: List[AbstractTransport] = []) -> str:
|
def serialize(base: Base, write_transports: List[AbstractTransport] = []) -> str:
|
||||||
"""
|
"""
|
||||||
Serialize a base object. If no write transports are provided,
|
Serialize a base object. If no write transports are provided, the object will be serialized
|
||||||
the object will be serialized
|
|
||||||
without detaching or chunking any of the attributes.
|
without detaching or chunking any of the attributes.
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
base {Base} -- the object to serialize
|
base {Base} -- the object to serialize
|
||||||
write_transports {List[AbstractTransport]}
|
write_transports {List[AbstractTransport]} -- optional: the transports to write to
|
||||||
-- optional: the transports to write to
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the serialized object
|
str -- the serialized object
|
||||||
"""
|
"""
|
||||||
|
metrics.track(metrics.SERIALIZE)
|
||||||
serializer = BaseObjectSerializer(write_transports=write_transports)
|
serializer = BaseObjectSerializer(write_transports=write_transports)
|
||||||
|
|
||||||
return serializer.write_json(base)[1]
|
return serializer.write_json(base)[1]
|
||||||
|
|
||||||
|
|
||||||
def deserialize(
|
def deserialize(obj_string: str, read_transport: AbstractTransport = None) -> Base:
|
||||||
obj_string: str, read_transport: Optional[AbstractTransport] = None
|
|
||||||
) -> Base:
|
|
||||||
"""
|
"""
|
||||||
Deserialize a string object into a Base object.
|
Deserialize a string object into a Base object. If the object contains referenced child objects that are not stored in the local db, a read transport needs to be provided in order to recompose the base with the children objects.
|
||||||
|
|
||||||
If the object contains referenced child objects that are not stored in the local db,
|
|
||||||
a read transport needs to be provided in order to recompose
|
|
||||||
the base with the children objects.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
obj_string {str} -- the string object to deserialize
|
obj_string {str} -- the string object to deserialize
|
||||||
read_transport {AbstractTransport}
|
read_transport {AbstractTransport} -- the transport to fetch children objects from
|
||||||
-- the transport to fetch children objects from
|
(defaults to SQLiteTransport)
|
||||||
(defaults to SQLiteTransport)
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Base -- the deserialized object
|
Base -- the deserialized object
|
||||||
"""
|
"""
|
||||||
|
metrics.track(metrics.DESERIALIZE)
|
||||||
if not read_transport:
|
if not read_transport:
|
||||||
read_transport = SQLiteTransport()
|
read_transport = SQLiteTransport()
|
||||||
|
|
||||||
serializer = BaseObjectSerializer(read_transport=read_transport)
|
serializer = BaseObjectSerializer(read_transport=read_transport)
|
||||||
|
|
||||||
return serializer.read_json(obj_string=obj_string)
|
return serializer.read_json(obj_string=obj_string)
|
||||||
|
|
||||||
|
|
||||||
__all__ = ["receive", "send", "serialize", "deserialize"]
|
|
||||||
@@ -0,0 +1,81 @@
|
|||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
from typing import Dict, List
|
||||||
|
from gql.client import Client
|
||||||
|
from gql.gql import gql
|
||||||
|
from gql.transport.exceptions import TransportQueryError
|
||||||
|
from specklepy.logging.exceptions import GraphQLException, SpeckleException
|
||||||
|
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
||||||
|
|
||||||
|
|
||||||
|
class ResourceBase(object):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
me: Dict,
|
||||||
|
basepath: str,
|
||||||
|
client: Client,
|
||||||
|
name: str,
|
||||||
|
methods: list,
|
||||||
|
) -> None:
|
||||||
|
self.me = me
|
||||||
|
self.basepath = basepath
|
||||||
|
self.client = client
|
||||||
|
self.name = name
|
||||||
|
self.methods = methods
|
||||||
|
self.schema = None
|
||||||
|
|
||||||
|
def _step_into_response(self, response: dict, return_type: str or List):
|
||||||
|
"""Step into the dict to get the relevant data"""
|
||||||
|
if return_type is None:
|
||||||
|
return response
|
||||||
|
elif isinstance(return_type, str):
|
||||||
|
return response[return_type]
|
||||||
|
elif isinstance(return_type, List):
|
||||||
|
for key in return_type:
|
||||||
|
response = response[key]
|
||||||
|
return response
|
||||||
|
|
||||||
|
def _parse_response(self, response: dict or list, schema=None):
|
||||||
|
"""Try to create a class instance from the response"""
|
||||||
|
if isinstance(response, list):
|
||||||
|
return [self._parse_response(response=r, schema=schema) for r in response]
|
||||||
|
if schema:
|
||||||
|
return schema.parse_obj(response)
|
||||||
|
elif self.schema:
|
||||||
|
try:
|
||||||
|
return self.schema.parse_obj(response)
|
||||||
|
except:
|
||||||
|
s = BaseObjectSerializer(read_transport=SQLiteTransport())
|
||||||
|
return s.recompose_base(response)
|
||||||
|
else:
|
||||||
|
return response
|
||||||
|
|
||||||
|
def make_request(
|
||||||
|
self,
|
||||||
|
query: gql,
|
||||||
|
params: Dict = None,
|
||||||
|
return_type: str or List = None,
|
||||||
|
schema=None,
|
||||||
|
parse_response: bool = True,
|
||||||
|
) -> Dict or GraphQLException:
|
||||||
|
"""Executes the GraphQL query"""
|
||||||
|
try:
|
||||||
|
response = self.client.execute(query, variable_values=params)
|
||||||
|
except Exception as e:
|
||||||
|
if isinstance(e, TransportQueryError):
|
||||||
|
return GraphQLException(
|
||||||
|
message=f"Failed to execute the GraphQL {self.name} request. Errors: {e.errors}",
|
||||||
|
errors=e.errors,
|
||||||
|
data=e.data,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
return SpeckleException(
|
||||||
|
message=f"Failed to execute the GraphQL {self.name} request. Inner exception: {e}",
|
||||||
|
exception=e,
|
||||||
|
)
|
||||||
|
|
||||||
|
response = self._step_into_response(response=response, return_type=return_type)
|
||||||
|
|
||||||
|
if parse_response:
|
||||||
|
return self._parse_response(response=response, schema=schema)
|
||||||
|
else:
|
||||||
|
return response
|
||||||
@@ -1,8 +1,12 @@
|
|||||||
import pkgutil
|
from pathlib import Path
|
||||||
import sys
|
import sys
|
||||||
|
import inspect
|
||||||
|
import pkgutil
|
||||||
from importlib import import_module
|
from importlib import import_module
|
||||||
|
|
||||||
for _, name, _ in pkgutil.iter_modules(__path__):
|
|
||||||
|
for (_, name, _) in pkgutil.iter_modules(__path__):
|
||||||
|
|
||||||
imported_module = import_module("." + name, package=__name__)
|
imported_module = import_module("." + name, package=__name__)
|
||||||
|
|
||||||
if hasattr(imported_module, "Resource"):
|
if hasattr(imported_module, "Resource"):
|
||||||
@@ -1,22 +1,20 @@
|
|||||||
from typing import Optional
|
from specklepy.api.resources import stream
|
||||||
|
from typing import List, Optional
|
||||||
from gql import gql
|
from gql import gql
|
||||||
|
from pydantic.main import BaseModel
|
||||||
from specklepy.core.api.models import Branch
|
from specklepy.api.resource import ResourceBase
|
||||||
from specklepy.core.api.resource import ResourceBase
|
from specklepy.api.models import Branch
|
||||||
|
|
||||||
NAME = "branch"
|
NAME = "branch"
|
||||||
|
METHODS = ["create"]
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for branches"""
|
"""API Access class for branches"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, me, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
account=account,
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
)
|
)
|
||||||
self.schema = Branch
|
self.schema = Branch
|
||||||
|
|
||||||
@@ -32,6 +30,7 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
id {str} -- the newly created branch's id
|
id {str} -- the newly created branch's id
|
||||||
"""
|
"""
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation BranchCreate($branch: BranchCreateInput!) {
|
mutation BranchCreate($branch: BranchCreateInput!) {
|
||||||
@@ -62,6 +61,7 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
Branch -- the fetched branch with its latest commits
|
Branch -- the fetched branch with its latest commits
|
||||||
"""
|
"""
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query BranchGet($stream_id: String!, $name: String!, $commits_limit: Int!) {
|
query BranchGet($stream_id: String!, $name: String!, $commits_limit: Int!) {
|
||||||
@@ -86,7 +86,7 @@ class Resource(ResourceBase):
|
|||||||
createdAt
|
createdAt
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
@@ -111,11 +111,7 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query BranchesGet(
|
query BranchesGet($stream_id: String!, $branches_limit: Int!, $commits_limit: Int!) {
|
||||||
$stream_id: String!,
|
|
||||||
$branches_limit: Int!,
|
|
||||||
$commits_limit: Int!
|
|
||||||
) {
|
|
||||||
stream(id: $stream_id) {
|
stream(id: $stream_id) {
|
||||||
branches(limit: $branches_limit) {
|
branches(limit: $branches_limit) {
|
||||||
items {
|
items {
|
||||||
@@ -154,11 +150,7 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def update(
|
def update(
|
||||||
self,
|
self, stream_id: str, branch_id: str, name: str = None, description: str = None
|
||||||
stream_id: str,
|
|
||||||
branch_id: str,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
description: Optional[str] = None,
|
|
||||||
):
|
):
|
||||||
"""Update a branch
|
"""Update a branch
|
||||||
|
|
||||||
@@ -169,7 +161,7 @@ class Resource(ResourceBase):
|
|||||||
description {str} -- optional: the updated branch description
|
description {str} -- optional: the updated branch description
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
bool -- True if update is successful
|
bool -- True if update is successfull
|
||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
@@ -204,6 +196,7 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if deletion is successful
|
bool -- True if deletion is successful
|
||||||
"""
|
"""
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation BranchDelete($branch: BranchDeleteInput!) {
|
mutation BranchDelete($branch: BranchDeleteInput!) {
|
||||||
@@ -1,23 +1,19 @@
|
|||||||
from typing import List, Optional, Union
|
from typing import Optional, List
|
||||||
|
|
||||||
from gql import gql
|
from gql import gql
|
||||||
|
from specklepy.api.resource import ResourceBase
|
||||||
|
from specklepy.api.models import Commit
|
||||||
|
|
||||||
from specklepy.core.api.models import Commit
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
NAME = "commit"
|
NAME = "commit"
|
||||||
|
METHODS = []
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for commits"""
|
"""API Access class for commits"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, me, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
account=account,
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
)
|
)
|
||||||
self.schema = Commit
|
self.schema = Commit
|
||||||
|
|
||||||
@@ -107,21 +103,17 @@ class Resource(ResourceBase):
|
|||||||
branch_name: str = "main",
|
branch_name: str = "main",
|
||||||
message: str = "",
|
message: str = "",
|
||||||
source_application: str = "python",
|
source_application: str = "python",
|
||||||
parents: Optional[List[str]] = None,
|
parents: List[str] = None,
|
||||||
) -> Union[str, SpeckleException]:
|
) -> str:
|
||||||
"""
|
"""
|
||||||
Creates a commit on a branch
|
Creates a commit on a branch
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the stream you want to commit to
|
stream_id {str} -- the stream you want to commit to
|
||||||
object_id {str} -- the hash of your commit object
|
object_id {str} -- the hash of your commit object
|
||||||
branch_name {str}
|
branch_name {str} -- the name of the branch to commit to (defaults to "main")
|
||||||
-- the name of the branch to commit to (defaults to "main")
|
message {str} -- optional: a message to give more information about the commit
|
||||||
message {str}
|
source_application{str} -- optional: the application from which the commit was created (defaults to "python")
|
||||||
-- optional: a message to give more information about the commit
|
|
||||||
source_application{str}
|
|
||||||
-- optional: the application from which the commit was created
|
|
||||||
(defaults to "python")
|
|
||||||
parents {List[str]} -- optional: the id of the parent commits
|
parents {List[str]} -- optional: the id of the parent commits
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
@@ -129,8 +121,7 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitCreate ($commit: CommitCreateInput!)
|
mutation CommitCreate ($commit: CommitCreateInput!){ commitCreate(commit: $commit)}
|
||||||
{ commitCreate(commit: $commit)}
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {
|
params = {
|
||||||
@@ -154,8 +145,7 @@ class Resource(ResourceBase):
|
|||||||
Update a commit
|
Update a commit
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str}
|
stream_id {str} -- the id of the stream that contains the commit you'd like to update
|
||||||
-- the id of the stream that contains the commit you'd like to update
|
|
||||||
commit_id {str} -- the id of the commit you'd like to update
|
commit_id {str} -- the id of the commit you'd like to update
|
||||||
message {str} -- the updated commit message
|
message {str} -- the updated commit message
|
||||||
|
|
||||||
@@ -164,8 +154,7 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitUpdate($commit: CommitUpdateInput!)
|
mutation CommitUpdate($commit: CommitUpdateInput!){ commitUpdate(commit: $commit)}
|
||||||
{ commitUpdate(commit: $commit)}
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {
|
params = {
|
||||||
@@ -181,8 +170,7 @@ class Resource(ResourceBase):
|
|||||||
Delete a commit
|
Delete a commit
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str}
|
stream_id {str} -- the id of the stream that contains the commit you'd like to delete
|
||||||
-- the id of the stream that contains the commit you'd like to delete
|
|
||||||
commit_id {str} -- the id of the commit you'd like to delete
|
commit_id {str} -- the id of the commit you'd like to delete
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
@@ -190,8 +178,7 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitDelete($commit: CommitDeleteInput!)
|
mutation CommitDelete($commit: CommitDeleteInput!){ commitDelete(commit: $commit)}
|
||||||
{ commitDelete(commit: $commit)}
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {"commit": {"streamId": stream_id, "id": commit_id}}
|
params = {"commit": {"streamId": stream_id, "id": commit_id}}
|
||||||
@@ -1,22 +1,19 @@
|
|||||||
from typing import Dict, List
|
from typing import Dict, List
|
||||||
|
|
||||||
from gql import gql
|
from gql import gql
|
||||||
|
from graphql.language import parser
|
||||||
from specklepy.core.api.resource import ResourceBase
|
from specklepy.api.resource import ResourceBase
|
||||||
from specklepy.objects.base import Base
|
from specklepy.objects.base import Base
|
||||||
|
|
||||||
NAME = "object"
|
NAME = "object"
|
||||||
|
METHODS = []
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for objects"""
|
"""API Access class for objects"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, me, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
account=account,
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
)
|
)
|
||||||
self.schema = Base
|
self.schema = Base
|
||||||
|
|
||||||
@@ -52,37 +49,28 @@ class Resource(ResourceBase):
|
|||||||
params = {"stream_id": stream_id, "object_id": object_id}
|
params = {"stream_id": stream_id, "object_id": object_id}
|
||||||
|
|
||||||
return self.make_request(
|
return self.make_request(
|
||||||
query=query,
|
query=query, params=params, return_type=["stream", "object", "data"]
|
||||||
params=params,
|
|
||||||
return_type=["stream", "object", "data"],
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def create(self, stream_id: str, objects: List[Dict]) -> str:
|
def create(self, stream_id: str, objects: List[Dict]) -> str:
|
||||||
"""
|
"""
|
||||||
Not advised - generally, you want to use `operations.send()`.
|
Not advised - generally, you want to use `operations.send()`.
|
||||||
|
|
||||||
Create a new object on a stream.
|
Create a new object on a stream. To send a base object, you can prepare it by running it through the
|
||||||
To send a base object, you can prepare it by running it through the
|
`BaseObjectSerializer.traverse_base()` function to get a valid (serialisable) object to send.
|
||||||
`BaseObjectSerializer.traverse_base()` function to get a valid (serialisable)
|
|
||||||
object to send.
|
|
||||||
|
|
||||||
NOTE: this does not create a commit - you can create one with
|
NOTE: this does not create a commit - you can create one with `SpeckleClient.commit.create`. Dynamic fields will be located in the 'data' dict of the received `Base` object
|
||||||
`SpeckleClient.commit.create`.
|
|
||||||
Dynamic fields will be located in the 'data' dict of the received `Base` object
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream you want to send the object to
|
stream_id {str} -- the id of the stream you want to send the object to
|
||||||
objects {List[Dict]}
|
objects {List[Dict]} -- a list of base dictionary objects (NOTE: must be json serialisable)
|
||||||
-- a list of base dictionary objects (NOTE: must be json serialisable)
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the id of the object
|
str -- the id of the object
|
||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation ObjectCreate($object_input: ObjectCreateInput!) {
|
mutation ObjectCreate($object_input: ObjectCreateInput!) { objectCreate(objectInput: $object_input) }
|
||||||
objectCreate(objectInput: $object_input)
|
|
||||||
}
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {"object_input": {"streamId": stream_id, "objects": objects}}
|
params = {"object_input": {"streamId": stream_id, "objects": objects}}
|
||||||
@@ -1,24 +1,20 @@
|
|||||||
import re
|
from typing import Dict, List
|
||||||
from typing import Any, Dict, List, Tuple
|
|
||||||
|
|
||||||
from gql import gql
|
from gql import gql
|
||||||
|
from gql.client import Client
|
||||||
|
from specklepy.api.models import ServerInfo
|
||||||
|
from specklepy.api.resource import ResourceBase
|
||||||
|
|
||||||
from specklepy.core.api.models import ServerInfo
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import GraphQLException
|
|
||||||
|
|
||||||
NAME = "server"
|
NAME = "server"
|
||||||
|
METHODS = ["get", "apps"]
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for the server"""
|
"""API Access class for the server"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, me, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
account=account,
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def get(self) -> ServerInfo:
|
def get(self) -> ServerInfo:
|
||||||
@@ -60,39 +56,6 @@ class Resource(ResourceBase):
|
|||||||
query=query, return_type="serverInfo", schema=ServerInfo
|
query=query, return_type="serverInfo", schema=ServerInfo
|
||||||
)
|
)
|
||||||
|
|
||||||
def version(self) -> Tuple[Any, ...]:
|
|
||||||
"""Get the server version
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
the server version in the format (major, minor, patch, (tag, build))
|
|
||||||
eg (2, 6, 3) for a stable build and (2, 6, 4, 'alpha', 4711) for alpha
|
|
||||||
"""
|
|
||||||
# not tracking as it will be called along with other mutations / queries as a check
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query Server {
|
|
||||||
serverInfo {
|
|
||||||
version
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
ver = self.make_request(
|
|
||||||
query=query, return_type=["serverInfo", "version"], parse_response=False
|
|
||||||
)
|
|
||||||
if isinstance(ver, Exception):
|
|
||||||
raise GraphQLException(
|
|
||||||
f"Could not get server version for {self.basepath}", [ver]
|
|
||||||
)
|
|
||||||
|
|
||||||
# pylint: disable=consider-using-generator; (list comp is faster)
|
|
||||||
return tuple(
|
|
||||||
[
|
|
||||||
int(segment) if segment.isdigit() else segment
|
|
||||||
for segment in re.split(r"\.|-", ver)
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
def apps(self) -> Dict:
|
def apps(self) -> Dict:
|
||||||
"""Get the apps registered on the server
|
"""Get the apps registered on the server
|
||||||
|
|
||||||
@@ -0,0 +1,364 @@
|
|||||||
|
from gql import gql
|
||||||
|
from typing import Dict, List, Optional
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.api.models import Stream
|
||||||
|
from specklepy.api.resource import ResourceBase
|
||||||
|
|
||||||
|
|
||||||
|
NAME = "stream"
|
||||||
|
METHODS = [
|
||||||
|
"list",
|
||||||
|
"create",
|
||||||
|
"get",
|
||||||
|
"update",
|
||||||
|
"delete",
|
||||||
|
"search",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class Resource(ResourceBase):
|
||||||
|
"""API Access class for streams"""
|
||||||
|
|
||||||
|
def __init__(self, me, basepath, client) -> None:
|
||||||
|
super().__init__(
|
||||||
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
|
)
|
||||||
|
|
||||||
|
self.schema = Stream
|
||||||
|
|
||||||
|
def get(self, id: str, branch_limit: int = 10, commit_limit: int = 10) -> Stream:
|
||||||
|
"""Get the specified stream from the server
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
id {str} -- the stream id
|
||||||
|
branch_limit {int} -- the maximum number of branches to return
|
||||||
|
commit_limit {int} -- the maximum number of commits to return
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Stream -- the retrieved stream
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_GET)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
query Stream($id: String!, $branch_limit: Int!, $commit_limit: Int!) {
|
||||||
|
stream(id: $id) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
isPublic
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
collaborators {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
branches(limit: $branch_limit) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
commits(limit: $commit_limit) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
authorName
|
||||||
|
authorId
|
||||||
|
createdAt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"id": id, "branch_limit": branch_limit, "commit_limit": commit_limit}
|
||||||
|
|
||||||
|
return self.make_request(query=query, params=params, return_type="stream")
|
||||||
|
|
||||||
|
def list(self, stream_limit: int = 10) -> List[Stream]:
|
||||||
|
"""Get a list of the user's streams
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
stream_limit {int} -- The maximum number of streams to return
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[Stream] -- A list of Stream objects
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_LIST)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
query User($stream_limit: Int!) {
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
email
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
profiles
|
||||||
|
role
|
||||||
|
streams(limit: $stream_limit) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
isPublic
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
collaborators {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"stream_limit": stream_limit}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type=["user", "streams", "items"]
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(
|
||||||
|
self,
|
||||||
|
name: str = "Anonymous Python Stream",
|
||||||
|
description: str = "No description provided",
|
||||||
|
is_public: bool = True,
|
||||||
|
) -> str:
|
||||||
|
"""Create a new stream
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
name {str} -- the name of the string
|
||||||
|
description {str} -- a short description of the stream
|
||||||
|
is_public {bool} -- whether or not the stream can be viewed by anyone with the id
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
id {str} -- the id of the newly created stream
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_CREATE)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation StreamCreate($stream: StreamCreateInput!) {
|
||||||
|
streamCreate(stream: $stream)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"stream": {"name": name, "description": description, "isPublic": is_public}
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type="streamCreate", parse_response=False
|
||||||
|
)
|
||||||
|
|
||||||
|
def update(
|
||||||
|
self, id: str, name: str = None, description: str = None, is_public: bool = None
|
||||||
|
) -> bool:
|
||||||
|
"""Update an existing stream
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
id {str} -- the id of the stream to be updated
|
||||||
|
name {str} -- the name of the string
|
||||||
|
description {str} -- a short description of the stream
|
||||||
|
is_public {bool} -- whether or not the stream can be viewed by anyone with the id
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool -- whether the stream update was successful
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_UPDATE)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation StreamUpdate($stream: StreamUpdateInput!) {
|
||||||
|
streamUpdate(stream: $stream)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"id": id,
|
||||||
|
"name": name,
|
||||||
|
"description": description,
|
||||||
|
"isPublic": is_public,
|
||||||
|
}
|
||||||
|
# remove None values so graphql doesn't cry
|
||||||
|
params = {"stream": {k: v for k, v in params.items() if v is not None}}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type="streamUpdate", parse_response=False
|
||||||
|
)
|
||||||
|
|
||||||
|
def delete(self, id: str) -> bool:
|
||||||
|
"""Delete a stream given its id
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
id {str} -- the id of the stream to delete
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool -- whether the deletion was successful
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_DELETE)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation StreamDelete($id: String!) {
|
||||||
|
streamDelete(id: $id)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"id": id}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type="streamDelete", parse_response=False
|
||||||
|
)
|
||||||
|
|
||||||
|
def search(
|
||||||
|
self,
|
||||||
|
search_query: str,
|
||||||
|
limit: int = 25,
|
||||||
|
branch_limit: int = 10,
|
||||||
|
commit_limit: int = 10,
|
||||||
|
):
|
||||||
|
"""Search for streams by name, description, or id
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
search_query {str} -- a string to search for
|
||||||
|
limit {int} -- the maximum number of results to return
|
||||||
|
branch_limit {int} -- the maximum number of branches to return
|
||||||
|
commit_limit {int} -- the maximum number of commits to return
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[Stream] -- a list of Streams that match the search query
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.STREAM_SEARCH)
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
query StreamSearch($search_query: String!,$limit: Int!, $branch_limit:Int!, $commit_limit:Int!) {
|
||||||
|
streams(query: $search_query, limit: $limit) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
isPublic
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
collaborators {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
branches(limit: $branch_limit) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
commits(limit: $commit_limit) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
authorName
|
||||||
|
authorId
|
||||||
|
createdAt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"search_query": search_query,
|
||||||
|
"limit": limit,
|
||||||
|
"branch_limit": branch_limit,
|
||||||
|
"commit_limit": commit_limit,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type=["streams", "items"]
|
||||||
|
)
|
||||||
|
|
||||||
|
def grant_permission(self, stream_id: str, user_id: str, role: str):
|
||||||
|
"""Grant permissions to a user on a given stream
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
stream_id {str} -- the id of the stream to grant permissions to
|
||||||
|
user_id {str} -- the id of the user to grant permissions for
|
||||||
|
role {str} -- the role to grant the user
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool -- True if the operation was successful
|
||||||
|
"""
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation StreamGrantPermission($permission_params: StreamGrantPermissionInput !) {
|
||||||
|
streamGrantPermission(permissionParams: $permission_params)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"permission_params": {
|
||||||
|
"streamId": stream_id,
|
||||||
|
"userId": user_id,
|
||||||
|
"role": role,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query,
|
||||||
|
params=params,
|
||||||
|
return_type="streamGrantPermission",
|
||||||
|
parse_response=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
def revoke_permission(self, stream_id: str, user_id: str):
|
||||||
|
"""Revoke permissions from a user on a given stream
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
stream_id {str} -- the id of the stream to revoke permissions from
|
||||||
|
user_id {str} -- the id of the user to revoke permissions from
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool -- True if the operation was successful
|
||||||
|
"""
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation StreamRevokePermission($permission_params: StreamRevokePermissionInput !) {
|
||||||
|
streamRevokePermission(permissionParams: $permission_params)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"permission_params": {"streamId": stream_id, "userId": user_id}}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query,
|
||||||
|
params=params,
|
||||||
|
return_type="streamRevokePermission",
|
||||||
|
parse_response=False,
|
||||||
|
)
|
||||||
+25
-38
@@ -1,14 +1,16 @@
|
|||||||
|
from typing import Callable, Dict, List, Optional, Any
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
from typing import Callable, Dict, List, Optional, Union
|
|
||||||
|
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from graphql import DocumentNode
|
from specklepy.api.resource import ResourceBase
|
||||||
|
from specklepy.api.resources.stream import Stream
|
||||||
from specklepy.core.api.resource import ResourceBase
|
from specklepy.logging.exceptions import GraphQLException, SpeckleException
|
||||||
from specklepy.core.api.resources.stream import Stream
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
NAME = "subscribe"
|
NAME = "subscribe"
|
||||||
|
METHODS = [
|
||||||
|
"stream_added",
|
||||||
|
"stream_updated",
|
||||||
|
"stream_removed",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
def check_wsclient(function):
|
def check_wsclient(function):
|
||||||
@@ -27,22 +29,17 @@ def check_wsclient(function):
|
|||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for subscriptions"""
|
"""API Access class for subscriptions"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, me, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
account=account,
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_added(self, callback: Optional[Callable] = None):
|
async def stream_added(self, callback: Callable = None):
|
||||||
"""Subscribes to new stream added event for your profile.
|
"""Subscribes to new stream added event for your profile. Use this to display an up-to-date list of streams.
|
||||||
Use this to display an up-to-date list of streams.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
callback {Callable[Stream]} -- a function that takes the updated stream
|
callback {Callable[Stream]} -- a function that takes the updated stream as an argument and executes each time a stream is added
|
||||||
as an argument and executes each time a stream is added
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Stream -- the update stream
|
Stream -- the update stream
|
||||||
@@ -57,16 +54,12 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_updated(self, id: str, callback: Optional[Callable] = None):
|
async def stream_updated(self, id: str, callback: Callable = None):
|
||||||
"""
|
"""Subscribes to stream updated event. Use this in clients/components that pertain only to this stream.
|
||||||
Subscribes to stream updated event.
|
|
||||||
Use this in clients/components that pertain only to this stream.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
id {str} -- the stream id of the stream to subscribe to
|
id {str} -- the stream id of the stream to subscribe to
|
||||||
callback {Callable[Stream]}
|
callback {Callable[Stream]} -- a function that takes the updated stream as an argument and executes each time the stream is updated
|
||||||
-- a function that takes the updated stream
|
|
||||||
as an argument and executes each time the stream is updated
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Stream -- the update stream
|
Stream -- the update stream
|
||||||
@@ -87,17 +80,11 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_removed(self, callback: Optional[Callable] = None):
|
async def stream_removed(self, callback: Callable = None):
|
||||||
"""Subscribes to stream removed event for your profile.
|
"""Subscribes to stream removed event for your profile. Use this to display an up-to-date list of streams for your profile. NOTE: If someone revokes your permissions on a stream, this subscription will be triggered with an extra value of revokedBy in the payload.
|
||||||
Use this to display an up-to-date list of streams for your profile.
|
|
||||||
NOTE: If someone revokes your permissions on a stream,
|
|
||||||
this subscription will be triggered with an extra value of revokedBy
|
|
||||||
in the payload.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
callback {Callable[Dict]}
|
callback {Callable[Dict]} -- a function that takes the returned dict as an argument and executes each time a stream is removed
|
||||||
-- a function that takes the returned dict as an argument
|
|
||||||
and executes each time a stream is removed
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
dict -- dict containing 'id' of stream removed and optionally 'revokedBy'
|
dict -- dict containing 'id' of stream removed and optionally 'revokedBy'
|
||||||
@@ -118,15 +105,15 @@ class Resource(ResourceBase):
|
|||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def subscribe(
|
async def subscribe(
|
||||||
self,
|
self,
|
||||||
query: DocumentNode,
|
query: gql,
|
||||||
params: Optional[Dict] = None,
|
params: Dict = None,
|
||||||
callback: Optional[Callable] = None,
|
callback: Callable = None,
|
||||||
return_type: Optional[Union[str, List]] = None,
|
return_type: str or List = None,
|
||||||
schema=None,
|
schema=None,
|
||||||
parse_response: bool = True,
|
parse_response: bool = True,
|
||||||
):
|
):
|
||||||
# if self.client.transport.websocket is None:
|
# if self.client.transport.websocket is None:
|
||||||
# TODO: add multiple subs to the same ws connection
|
# TODO: add multiple subs to the same ws connection
|
||||||
async with self.client as session:
|
async with self.client as session:
|
||||||
async for res in session.subscribe(query, variable_values=params):
|
async for res in session.subscribe(query, variable_values=params):
|
||||||
res = self._step_into_response(response=res, return_type=return_type)
|
res = self._step_into_response(response=res, return_type=return_type)
|
||||||
@@ -0,0 +1,120 @@
|
|||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from typing import List, Optional
|
||||||
|
from gql import gql
|
||||||
|
from pydantic.main import BaseModel
|
||||||
|
from specklepy.api.resource import ResourceBase
|
||||||
|
from specklepy.api.models import User
|
||||||
|
|
||||||
|
NAME = "user"
|
||||||
|
METHODS = ["get"]
|
||||||
|
|
||||||
|
|
||||||
|
class Resource(ResourceBase):
|
||||||
|
"""API Access class for users"""
|
||||||
|
|
||||||
|
def __init__(self, me, basepath, client) -> None:
|
||||||
|
super().__init__(
|
||||||
|
me=me, basepath=basepath, client=client, name=NAME, methods=METHODS
|
||||||
|
)
|
||||||
|
self.schema = User
|
||||||
|
|
||||||
|
def get(self, id: str = None) -> User:
|
||||||
|
"""Gets the profile of a user. If no id argument is provided, will return the current authenticated user's profile (as extracted from the authorization header).
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
id {str} -- the user id
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
User -- the retrieved user
|
||||||
|
"""
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
query User($id: String) {
|
||||||
|
user(id: $id) {
|
||||||
|
id
|
||||||
|
email
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
profiles
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"id": id}
|
||||||
|
|
||||||
|
return self.make_request(query=query, params=params, return_type="user")
|
||||||
|
|
||||||
|
def search(self, search_query: str, limit: int = 25) -> List[User]:
|
||||||
|
"""Searches for user by name or email. The search query must be at least 3 characters long
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
search_query {str} -- a string to search for
|
||||||
|
limit {int} -- the maximum number of results to return
|
||||||
|
Returns:
|
||||||
|
List[User] -- a list of User objects that match the search query
|
||||||
|
"""
|
||||||
|
if len(search_query) < 3:
|
||||||
|
return SpeckleException(
|
||||||
|
message="User search query must be at least 3 characters"
|
||||||
|
)
|
||||||
|
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
query UserSearch($search_query: String!, $limit: Int!) {
|
||||||
|
userSearch(query: $search_query, limit: $limit) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
params = {"search_query": search_query, "limit": limit}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type=["userSearch", "items"]
|
||||||
|
)
|
||||||
|
|
||||||
|
def update(
|
||||||
|
self, name: str = None, company: str = None, bio: str = None, avatar: str = None
|
||||||
|
):
|
||||||
|
"""Updates your user profile. All arguments are optional.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
name {str} -- your name
|
||||||
|
company {str} -- the company you may or may not work for
|
||||||
|
bio {str} -- tell us about yourself
|
||||||
|
avatar {str} -- a nice photo of yourself
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool -- True if your profile was updated successfully
|
||||||
|
"""
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation UserUpdate($user: UserUpdateInput!) {
|
||||||
|
userUpdate(user: $user)
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
params = {"name": name, "company": company, "bio": bio, "avatar": avatar}
|
||||||
|
|
||||||
|
params = {"user": {k: v for k, v in params.items() if v is not None}}
|
||||||
|
|
||||||
|
if not params["user"]:
|
||||||
|
return SpeckleException(
|
||||||
|
message="You must provide at least one field to update your user profile"
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type="userUpdate", parse_response=False
|
||||||
|
)
|
||||||
@@ -0,0 +1,640 @@
|
|||||||
|
|
||||||
|
|
||||||
|
scalar DateTime
|
||||||
|
|
||||||
|
scalar EmailAddress
|
||||||
|
|
||||||
|
scalar BigInt
|
||||||
|
|
||||||
|
scalar JSONObject
|
||||||
|
|
||||||
|
|
||||||
|
directive @hasScope(scope: String!) on FIELD_DEFINITION
|
||||||
|
directive @hasRole(role: String!) on FIELD_DEFINITION
|
||||||
|
|
||||||
|
type Query {
|
||||||
|
"""
|
||||||
|
Stare into the void.
|
||||||
|
"""
|
||||||
|
_: String
|
||||||
|
}
|
||||||
|
type Mutation{
|
||||||
|
"""
|
||||||
|
The void stares back.
|
||||||
|
"""
|
||||||
|
_: String
|
||||||
|
}
|
||||||
|
type Subscription{
|
||||||
|
"""
|
||||||
|
It's lonely in the void.
|
||||||
|
"""
|
||||||
|
_: String
|
||||||
|
},extend type Query {
|
||||||
|
"""
|
||||||
|
Gets a specific app from the server.
|
||||||
|
"""
|
||||||
|
app( id: String! ): ServerApp
|
||||||
|
|
||||||
|
"""
|
||||||
|
Returns all the publicly available apps on this server.
|
||||||
|
"""
|
||||||
|
apps: [ServerAppListItem]
|
||||||
|
}
|
||||||
|
|
||||||
|
type ServerApp {
|
||||||
|
id: String!
|
||||||
|
secret: String!
|
||||||
|
name: String!
|
||||||
|
description: String
|
||||||
|
termsAndConditionsLink: String
|
||||||
|
logo: String
|
||||||
|
public: Boolean
|
||||||
|
trustByDefault: Boolean
|
||||||
|
author: AppAuthor
|
||||||
|
createdAt: DateTime!
|
||||||
|
redirectUrl: String!
|
||||||
|
scopes: [Scope]!
|
||||||
|
}
|
||||||
|
|
||||||
|
type ServerAppListItem {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
description: String
|
||||||
|
termsAndConditionsLink: String
|
||||||
|
logo: String
|
||||||
|
author: AppAuthor
|
||||||
|
}
|
||||||
|
|
||||||
|
type AppAuthor {
|
||||||
|
name: String
|
||||||
|
id: String
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type User {
|
||||||
|
"""
|
||||||
|
Returns the apps you have authorized.
|
||||||
|
"""
|
||||||
|
authorizedApps: [ServerAppListItem]
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:read")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Returns the apps you have created.
|
||||||
|
"""
|
||||||
|
createdApps: [ServerAppListItem]
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:read")
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
"""
|
||||||
|
Register a new third party application.
|
||||||
|
"""
|
||||||
|
appCreate(app: AppCreateInput!): String!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:write")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Update an existing third party application. **Note: This will invalidate all existing tokens, refresh tokens and access codes and will require existing users to re-authorize it.**
|
||||||
|
"""
|
||||||
|
appUpdate(app: AppUpdateInput!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:write")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Deletes a thirty party application.
|
||||||
|
"""
|
||||||
|
appDelete(appId: String!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:write")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Revokes (de-authorizes) an application that you have previously authorized.
|
||||||
|
"""
|
||||||
|
appRevokeAccess(appId: String!): Boolean
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "apps:write")
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
input AppCreateInput {
|
||||||
|
name: String!
|
||||||
|
description: String!
|
||||||
|
termsAndConditionsLink: String
|
||||||
|
logo: String
|
||||||
|
public: Boolean
|
||||||
|
redirectUrl: String!
|
||||||
|
scopes: [String]!
|
||||||
|
}
|
||||||
|
|
||||||
|
input AppUpdateInput {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
description: String!
|
||||||
|
termsAndConditionsLink: String
|
||||||
|
logo: String
|
||||||
|
public: Boolean
|
||||||
|
redirectUrl: String!
|
||||||
|
scopes: [String]!
|
||||||
|
}
|
||||||
|
,extend type ServerInfo {
|
||||||
|
"""
|
||||||
|
The authentication strategies available on this server.
|
||||||
|
"""
|
||||||
|
authStrategies: [AuthStrategy]
|
||||||
|
}
|
||||||
|
|
||||||
|
type AuthStrategy {
|
||||||
|
id: String!,
|
||||||
|
name: String!,
|
||||||
|
icon: String!,
|
||||||
|
url: String!,
|
||||||
|
color: String
|
||||||
|
}
|
||||||
|
,extend type User{
|
||||||
|
"""
|
||||||
|
Returns a list of your personal api tokens.
|
||||||
|
"""
|
||||||
|
apiTokens: [ApiToken]
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "tokens:read")
|
||||||
|
}
|
||||||
|
|
||||||
|
type ApiToken {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
lastChars: String!
|
||||||
|
scopes: [String]!
|
||||||
|
createdAt: DateTime! #date
|
||||||
|
lifespan: BigInt!
|
||||||
|
lastUsed: String! #date
|
||||||
|
}
|
||||||
|
|
||||||
|
input ApiTokenCreateInput {
|
||||||
|
scopes: [String!]!,
|
||||||
|
name: String!,
|
||||||
|
lifespan: BigInt
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
"""
|
||||||
|
Creates an personal api token.
|
||||||
|
"""
|
||||||
|
apiTokenCreate(token: ApiTokenCreateInput!):String!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "tokens:write")
|
||||||
|
"""
|
||||||
|
Revokes (deletes) an personal api token.
|
||||||
|
"""
|
||||||
|
apiTokenRevoke(token: String!):Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "tokens:write")
|
||||||
|
}
|
||||||
|
,extend type Stream {
|
||||||
|
commits(limit: Int! = 25, cursor: String): CommitCollection
|
||||||
|
commit(id: String!): Commit
|
||||||
|
branches(limit: Int! = 25, cursor: String): BranchCollection
|
||||||
|
branch(name: String!): Branch
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type User {
|
||||||
|
commits(limit: Int! = 25, cursor: String): CommitCollectionUser
|
||||||
|
}
|
||||||
|
|
||||||
|
type Branch {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
author: User!
|
||||||
|
description: String
|
||||||
|
commits(limit: Int! = 25, cursor: String): CommitCollection
|
||||||
|
}
|
||||||
|
|
||||||
|
type Commit {
|
||||||
|
id: String!
|
||||||
|
referencedObject: String!
|
||||||
|
message: String
|
||||||
|
authorName: String
|
||||||
|
authorId: String
|
||||||
|
createdAt: DateTime
|
||||||
|
}
|
||||||
|
|
||||||
|
type CommitCollectionUserNode {
|
||||||
|
id: String!
|
||||||
|
referencedObject: String!
|
||||||
|
message: String
|
||||||
|
streamId: String
|
||||||
|
streamName: String
|
||||||
|
createdAt: DateTime
|
||||||
|
}
|
||||||
|
|
||||||
|
type BranchCollection {
|
||||||
|
totalCount: Int!
|
||||||
|
cursor: String
|
||||||
|
items: [Branch]
|
||||||
|
}
|
||||||
|
|
||||||
|
type CommitCollection {
|
||||||
|
totalCount: Int!
|
||||||
|
cursor: String
|
||||||
|
items: [Commit]
|
||||||
|
}
|
||||||
|
|
||||||
|
type CommitCollectionUser {
|
||||||
|
totalCount: Int!
|
||||||
|
cursor: String
|
||||||
|
items: [CommitCollectionUserNode]
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
branchCreate(branch: BranchCreateInput!): String!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
branchUpdate(branch: BranchUpdateInput!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
branchDelete(branch: BranchDeleteInput!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
|
||||||
|
commitCreate(commit: CommitCreateInput!): String!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
commitUpdate(commit: CommitUpdateInput!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
commitDelete(commit: CommitDeleteInput!): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Subscription {
|
||||||
|
# TODO: auth for these subscriptions
|
||||||
|
"""
|
||||||
|
Subscribe to branch created event
|
||||||
|
"""
|
||||||
|
branchCreated(streamId: String!): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
"""
|
||||||
|
Subscribe to branch updated event.
|
||||||
|
"""
|
||||||
|
branchUpdated(streamId: String!, branchId: String): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
"""
|
||||||
|
Subscribe to branch deleted event
|
||||||
|
"""
|
||||||
|
branchDeleted(streamId: String!): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Subscribe to commit created event
|
||||||
|
"""
|
||||||
|
commitCreated(streamId: String!): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
"""
|
||||||
|
Subscribe to commit updated event.
|
||||||
|
"""
|
||||||
|
commitUpdated(streamId: String!, commitId: String): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
"""
|
||||||
|
Subscribe to commit deleted event
|
||||||
|
"""
|
||||||
|
commitDeleted(streamId: String!): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
}
|
||||||
|
|
||||||
|
input BranchCreateInput {
|
||||||
|
streamId: String!
|
||||||
|
name: String!
|
||||||
|
description: String
|
||||||
|
}
|
||||||
|
|
||||||
|
input BranchUpdateInput {
|
||||||
|
streamId: String!
|
||||||
|
id: String!
|
||||||
|
name: String
|
||||||
|
description: String
|
||||||
|
}
|
||||||
|
|
||||||
|
input BranchDeleteInput {
|
||||||
|
streamId: String!
|
||||||
|
id: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
input CommitCreateInput {
|
||||||
|
streamId: String!
|
||||||
|
branchName: String!
|
||||||
|
objectId: String!
|
||||||
|
message: String
|
||||||
|
previousCommitIds: [String]
|
||||||
|
}
|
||||||
|
|
||||||
|
input CommitUpdateInput {
|
||||||
|
streamId: String!
|
||||||
|
id: String!
|
||||||
|
message: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
input CommitDeleteInput {
|
||||||
|
streamId: String!
|
||||||
|
id: String!
|
||||||
|
}
|
||||||
|
,extend type Stream {
|
||||||
|
object( id: String! ): Object
|
||||||
|
}
|
||||||
|
|
||||||
|
type Object {
|
||||||
|
id: String!
|
||||||
|
speckleType: String!
|
||||||
|
applicationId: String
|
||||||
|
createdAt: DateTime
|
||||||
|
totalChildrenCount: Int
|
||||||
|
"""
|
||||||
|
The full object, with all its props & other things. **NOTE:** If you're requesting objects for the purpose of recreating & displaying, you probably only want to request this specific field.
|
||||||
|
"""
|
||||||
|
data: JSONObject
|
||||||
|
"""
|
||||||
|
Get any objects that this object references. In the case of commits, this will give you a commit's constituent objects.
|
||||||
|
**NOTE**: Providing any of the two last arguments ( `query`, `orderBy` ) will trigger a different code branch that executes a much more expensive SQL query. It is not recommended to do so for basic clients that are interested in purely getting all the objects of a given commit.
|
||||||
|
"""
|
||||||
|
children(
|
||||||
|
limit: Int! = 100,
|
||||||
|
depth: Int! = 50,
|
||||||
|
select: [String],
|
||||||
|
cursor: String,
|
||||||
|
query: [JSONObject!],
|
||||||
|
orderBy: JSONObject ): ObjectCollection!
|
||||||
|
}
|
||||||
|
|
||||||
|
type ObjectCollection {
|
||||||
|
totalCount: Int!
|
||||||
|
cursor: String
|
||||||
|
objects: [Object]!
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
objectCreate( objectInput: ObjectCreateInput! ): [String]!
|
||||||
|
}
|
||||||
|
|
||||||
|
input ObjectCreateInput {
|
||||||
|
"""
|
||||||
|
The stream against which these objects will be created.
|
||||||
|
"""
|
||||||
|
streamId: String!
|
||||||
|
"""
|
||||||
|
The objects you want to create.
|
||||||
|
"""
|
||||||
|
objects: [JSONObject]!
|
||||||
|
},extend type Query {
|
||||||
|
serverInfo: ServerInfo!
|
||||||
|
}
|
||||||
|
|
||||||
|
"""
|
||||||
|
Information about this server.
|
||||||
|
"""
|
||||||
|
type ServerInfo {
|
||||||
|
name: String!
|
||||||
|
company: String
|
||||||
|
description: String
|
||||||
|
adminContact: String
|
||||||
|
canonicalUrl: String
|
||||||
|
termsOfService: String
|
||||||
|
roles: [Role]!
|
||||||
|
scopes: [Scope]!
|
||||||
|
}
|
||||||
|
|
||||||
|
"""
|
||||||
|
Available roles.
|
||||||
|
"""
|
||||||
|
type Role {
|
||||||
|
name: String!
|
||||||
|
description: String!
|
||||||
|
resourceTarget: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
"""
|
||||||
|
Available scopes.
|
||||||
|
"""
|
||||||
|
type Scope {
|
||||||
|
name: String!
|
||||||
|
description: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
serverInfoUpdate(info: ServerInfoUpdateInput!): Boolean
|
||||||
|
@hasRole(role: "server:admin")
|
||||||
|
@hasScope(scope: "server:setup")
|
||||||
|
}
|
||||||
|
|
||||||
|
input ServerInfoUpdateInput {
|
||||||
|
name: String!
|
||||||
|
company: String
|
||||||
|
description: String
|
||||||
|
adminContact: String
|
||||||
|
termsOfService: String
|
||||||
|
}
|
||||||
|
,extend type Query {
|
||||||
|
"""
|
||||||
|
Returns a specific stream.
|
||||||
|
"""
|
||||||
|
stream( id: String! ): Stream
|
||||||
|
|
||||||
|
"""
|
||||||
|
All the streams of the current user, pass in the `query` parameter to seach by name, description or ID.
|
||||||
|
"""
|
||||||
|
streams( query: String, limit: Int = 25, cursor: String ): StreamCollection
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
}
|
||||||
|
|
||||||
|
type Stream {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
description: String
|
||||||
|
isPublic: Boolean!
|
||||||
|
createdAt: DateTime!
|
||||||
|
updatedAt: DateTime!
|
||||||
|
collaborators: [ StreamCollaborator ]!
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type User {
|
||||||
|
"""
|
||||||
|
All the streams that a user has access to.
|
||||||
|
"""
|
||||||
|
streams( limit: Int! = 25, cursor: String ): StreamCollection
|
||||||
|
}
|
||||||
|
|
||||||
|
type StreamCollaborator {
|
||||||
|
id: String!
|
||||||
|
name: String!
|
||||||
|
role: String!
|
||||||
|
company: String
|
||||||
|
avatar: String
|
||||||
|
}
|
||||||
|
|
||||||
|
type StreamCollection {
|
||||||
|
totalCount: Int!
|
||||||
|
cursor: String
|
||||||
|
items: [ Stream ]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
"""
|
||||||
|
Creates a new stream.
|
||||||
|
"""
|
||||||
|
streamCreate( stream: StreamCreateInput! ): String
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
"""
|
||||||
|
Updates an existing stream.
|
||||||
|
"""
|
||||||
|
streamUpdate( stream: StreamUpdateInput! ): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
"""
|
||||||
|
Deletes an existing stream.
|
||||||
|
"""
|
||||||
|
streamDelete( id: String! ): Boolean!
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
"""
|
||||||
|
Grants permissions to a user on a given stream.
|
||||||
|
"""
|
||||||
|
streamGrantPermission( permissionParams: StreamGrantPermissionInput! ): Boolean
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
"""
|
||||||
|
Revokes the permissions of a user on a given stream.
|
||||||
|
"""
|
||||||
|
streamRevokePermission( permissionParams: StreamRevokePermissionInput! ): Boolean
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:write")
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Subscription {
|
||||||
|
|
||||||
|
#
|
||||||
|
# User bound subscriptions that operate on the stream collection of an user
|
||||||
|
# Example relevant view/usecase: updating the list of streams for a user.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""
|
||||||
|
Subscribes to new stream added event for your profile. Use this to display an up-to-date list of streams.
|
||||||
|
**NOTE**: If someone shares a stream with you, this subscription will be triggered with an extra value of `sharedBy` in the payload.
|
||||||
|
"""
|
||||||
|
userStreamAdded: JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "profile:read")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Subscribes to stream removed event for your profile. Use this to display an up-to-date list of streams for your profile.
|
||||||
|
**NOTE**: If someone revokes your permissions on a stream, this subscription will be triggered with an extra value of `revokedBy` in the payload.
|
||||||
|
"""
|
||||||
|
userStreamRemoved: JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "profile:read")
|
||||||
|
|
||||||
|
#
|
||||||
|
# Stream bound subscriptions that operate on the stream itself.
|
||||||
|
# Example relevant view/usecase: a single stream connector, or view, or component in a web app
|
||||||
|
#
|
||||||
|
|
||||||
|
"""
|
||||||
|
Subscribes to stream updated event. Use this in clients/components that pertain only to this stream.
|
||||||
|
"""
|
||||||
|
streamUpdated( streamId: String ): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
|
||||||
|
"""
|
||||||
|
Subscribes to stream deleted event. Use this in clients/components that pertain only to this stream.
|
||||||
|
"""
|
||||||
|
streamDeleted( streamId: String ): JSONObject
|
||||||
|
@hasRole(role: "server:user")
|
||||||
|
@hasScope(scope: "streams:read")
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
input StreamCreateInput {
|
||||||
|
name: String
|
||||||
|
description: String
|
||||||
|
isPublic: Boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
input StreamUpdateInput {
|
||||||
|
id: String!
|
||||||
|
name: String
|
||||||
|
description: String
|
||||||
|
isPublic: Boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
input StreamGrantPermissionInput {
|
||||||
|
streamId: String!,
|
||||||
|
userId: String!,
|
||||||
|
role: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
input StreamRevokePermissionInput {
|
||||||
|
streamId: String!,
|
||||||
|
userId: String!
|
||||||
|
}
|
||||||
|
,extend type Query {
|
||||||
|
"""
|
||||||
|
Gets the profile of a user. If no id argument is provided, will return the current authenticated user's profile (as extracted from the authorization header).
|
||||||
|
"""
|
||||||
|
user(id: String): User
|
||||||
|
userSearch(
|
||||||
|
query: String!
|
||||||
|
limit: Int! = 25
|
||||||
|
cursor: String
|
||||||
|
): UserSearchResultCollection
|
||||||
|
userPwdStrength(pwd: String!): JSONObject
|
||||||
|
}
|
||||||
|
|
||||||
|
"""
|
||||||
|
Base user type.
|
||||||
|
"""
|
||||||
|
type User {
|
||||||
|
id: String!
|
||||||
|
suuid: String
|
||||||
|
email: String
|
||||||
|
name: String
|
||||||
|
bio: String
|
||||||
|
company: String
|
||||||
|
avatar: String
|
||||||
|
verified: Boolean
|
||||||
|
profiles: JSONObject
|
||||||
|
role: String
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserSearchResultCollection {
|
||||||
|
cursor: String
|
||||||
|
items: [UserSearchResult]
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserSearchResult {
|
||||||
|
id: String!
|
||||||
|
name: String
|
||||||
|
bio: String
|
||||||
|
company: String
|
||||||
|
avatar: String
|
||||||
|
verified: Boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
extend type Mutation {
|
||||||
|
"""
|
||||||
|
Edits a user's profile.
|
||||||
|
"""
|
||||||
|
userUpdate(user: UserUpdateInput!): Boolean!
|
||||||
|
}
|
||||||
|
|
||||||
|
input UserUpdateInput {
|
||||||
|
name: String
|
||||||
|
company: String
|
||||||
|
bio: String
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
from typing import Any, List
|
||||||
|
|
||||||
|
|
||||||
|
class SpeckleException(Exception):
|
||||||
|
def __init__(self, message: str, exception: Exception = None) -> None:
|
||||||
|
self.message = message
|
||||||
|
self.exception = exception
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"SpeckleException: {self.message}"
|
||||||
|
|
||||||
|
|
||||||
|
class SerializationException(SpeckleException):
|
||||||
|
def __init__(self, message: str, object: Any, exception: Exception = None) -> None:
|
||||||
|
super().__init__(message=message)
|
||||||
|
self.object = object
|
||||||
|
self.unhandled_type = type(object)
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"SpeckleException: Could not serialize object of type {self.unhandled_type}"
|
||||||
|
|
||||||
|
|
||||||
|
class GraphQLException(SpeckleException):
|
||||||
|
def __init__(self, message: str, errors: List, data=None) -> None:
|
||||||
|
super().__init__(message=message)
|
||||||
|
self.errors = errors
|
||||||
|
self.data = data
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"GraphQLException: {self.message}"
|
||||||
|
|
||||||
|
|
||||||
|
class SpeckleWarning(Warning):
|
||||||
|
def __init__(self, *args: object) -> None:
|
||||||
|
super().__init__(*args)
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
import os
|
||||||
|
import queue
|
||||||
|
import logging
|
||||||
|
import requests
|
||||||
|
import threading
|
||||||
|
from requests.sessions import session
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
|
||||||
|
"""
|
||||||
|
Anonymous telemetry to help us understand how to make a better Speckle.
|
||||||
|
This really helps us to deliver a better open source project and product!
|
||||||
|
"""
|
||||||
|
TRACK = True
|
||||||
|
HOST_APP = "python"
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
METRICS_TRACKER = None
|
||||||
|
|
||||||
|
# actions
|
||||||
|
RECEIVE = "receive"
|
||||||
|
SEND = "send"
|
||||||
|
STREAM_CREATE = "stream/create"
|
||||||
|
STREAM_GET = "stream/get"
|
||||||
|
STREAM_UPDATE = "stream/update"
|
||||||
|
STREAM_DELETE = "stream/delete"
|
||||||
|
STREAM_DETAILS = "stream/details"
|
||||||
|
STREAM_LIST = "stream/list"
|
||||||
|
STREAM_VIEW = "stream/view"
|
||||||
|
STREAM_SEARCH = "stream/search"
|
||||||
|
|
||||||
|
ACCOUNT_DEFAULT = "account/default"
|
||||||
|
ACCOUNT_DETAILS = "account/details"
|
||||||
|
ACCOUNT_LIST = "account/list"
|
||||||
|
|
||||||
|
SERIALIZE = "serialization/serialize"
|
||||||
|
DESERIALIZE = "serialization/deserialize"
|
||||||
|
|
||||||
|
|
||||||
|
def disable():
|
||||||
|
global TRACK
|
||||||
|
TRACK = False
|
||||||
|
|
||||||
|
|
||||||
|
def set_host_app(host_app: str):
|
||||||
|
global HOST_APP
|
||||||
|
HOST_APP = host_app
|
||||||
|
|
||||||
|
|
||||||
|
def track(action: str):
|
||||||
|
if not TRACK:
|
||||||
|
return
|
||||||
|
try:
|
||||||
|
global METRICS_TRACKER
|
||||||
|
if not METRICS_TRACKER:
|
||||||
|
METRICS_TRACKER = MetricsTracker()
|
||||||
|
|
||||||
|
page_params = {
|
||||||
|
"rec": 1,
|
||||||
|
"idsite": METRICS_TRACKER.site_id,
|
||||||
|
"uid": METRICS_TRACKER.suuid,
|
||||||
|
"action_name": action,
|
||||||
|
"url": f"http://connectors/{HOST_APP}/{action}",
|
||||||
|
"urlref": f"http://connectors/{HOST_APP}/{action}",
|
||||||
|
"_cvar": {"1": ["hostApplication", HOST_APP]},
|
||||||
|
}
|
||||||
|
|
||||||
|
event_params = {
|
||||||
|
"rec": 1,
|
||||||
|
"idsite": METRICS_TRACKER.site_id,
|
||||||
|
"uid": MetricsTracker.suuid,
|
||||||
|
"_cvar": {"1": ["hostApplication", HOST_APP]},
|
||||||
|
"e_c": HOST_APP,
|
||||||
|
"e_a": action,
|
||||||
|
}
|
||||||
|
|
||||||
|
METRICS_TRACKER.queue.put_nowait([event_params, page_params])
|
||||||
|
except Exception as ex:
|
||||||
|
# wrapping this whole thing in a try except as we never want a failure here to annoy users!
|
||||||
|
LOG.error("Error queueing metrics request: " + str(ex))
|
||||||
|
|
||||||
|
|
||||||
|
class Singleton(type):
|
||||||
|
_instances = {}
|
||||||
|
|
||||||
|
def __call__(cls, *args, **kwargs):
|
||||||
|
if cls not in cls._instances:
|
||||||
|
cls._instances[cls] = super(Singleton, cls).__call__(*args, **kwargs)
|
||||||
|
return cls._instances[cls]
|
||||||
|
|
||||||
|
|
||||||
|
class MetricsTracker(metaclass=Singleton):
|
||||||
|
matomo_url = "https://speckle.matomo.cloud/matomo.php"
|
||||||
|
site_id = 2
|
||||||
|
host_app = "python"
|
||||||
|
suuid = None
|
||||||
|
sending_thread = None
|
||||||
|
queue = queue.Queue(1000)
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
self.sending_thread = threading.Thread(
|
||||||
|
target=self._send_tracking_requests, daemon=True
|
||||||
|
)
|
||||||
|
self.set_suuid()
|
||||||
|
self.sending_thread.start()
|
||||||
|
|
||||||
|
def set_suuid(self):
|
||||||
|
try:
|
||||||
|
file_path = os.path.join(SQLiteTransport.get_base_path("Speckle"), "suuid")
|
||||||
|
with open(file_path, "r") as file:
|
||||||
|
self.suuid = file.read()
|
||||||
|
except:
|
||||||
|
self.suuid = "unknown-suuid"
|
||||||
|
|
||||||
|
def _send_tracking_requests(self):
|
||||||
|
session = requests.Session()
|
||||||
|
while True:
|
||||||
|
params = self.queue.get()
|
||||||
|
|
||||||
|
try:
|
||||||
|
session.post(self.matomo_url, params=params[0])
|
||||||
|
session.post(self.matomo_url, params=params[1])
|
||||||
|
except Exception as ex:
|
||||||
|
LOG.error("Error sending metrics request: " + str(ex))
|
||||||
|
|
||||||
|
self.queue.task_done()
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
"""Builtin Speckle object kit."""
|
||||||
|
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
|
||||||
|
__all__ = ["Base"]
|
||||||
@@ -1,25 +1,19 @@
|
|||||||
import contextlib
|
import typing
|
||||||
from enum import Enum
|
|
||||||
from inspect import isclass
|
|
||||||
from typing import (
|
from typing import (
|
||||||
Any,
|
Any,
|
||||||
|
Callable,
|
||||||
ClassVar,
|
ClassVar,
|
||||||
Dict,
|
Dict,
|
||||||
ForwardRef,
|
|
||||||
List,
|
List,
|
||||||
Optional,
|
Optional,
|
||||||
Set,
|
Set,
|
||||||
Tuple,
|
|
||||||
Type,
|
Type,
|
||||||
Union,
|
|
||||||
get_type_hints,
|
get_type_hints,
|
||||||
)
|
)
|
||||||
from warnings import warn
|
from warnings import warn
|
||||||
|
|
||||||
from stringcase import pascalcase
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from specklepy.objects.units import get_units_from_string
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleInvalidUnitException
|
|
||||||
from specklepy.objects.units import Units, get_units_from_string
|
|
||||||
from specklepy.transports.memory import MemoryTransport
|
from specklepy.transports.memory import MemoryTransport
|
||||||
|
|
||||||
PRIMITIVES = (int, float, str, bool)
|
PRIMITIVES = (int, float, str, bool)
|
||||||
@@ -94,71 +88,23 @@ class _RegisteringBase:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
speckle_type: ClassVar[str]
|
speckle_type: ClassVar[str]
|
||||||
_speckle_type_override: ClassVar[Optional[str]] = None
|
_type_registry: ClassVar[Dict[str, "Base"]] = {}
|
||||||
_speckle_namespace: ClassVar[Optional[str]] = None
|
|
||||||
_type_registry: ClassVar[Dict[str, Type["Base"]]] = {}
|
|
||||||
_attr_types: ClassVar[Dict[str, Type]] = {}
|
_attr_types: ClassVar[Dict[str, Type]] = {}
|
||||||
# dict of chunkable props and their max chunk size
|
|
||||||
_chunkable: Dict[str, int] = {}
|
class Config:
|
||||||
_chunk_size_default: int = 1000
|
validate_assignment = True
|
||||||
_detachable: Set[str] = set() # list of defined detachable props
|
|
||||||
_serialize_ignore: Set[str] = set()
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_registered_type(cls, speckle_type: str) -> Optional[Type["Base"]]:
|
def get_registered_type(cls, speckle_type: str) -> Optional[Type["Base"]]:
|
||||||
"""Get the registered type from the protected mapping via the `speckle_type`"""
|
"""Get the registered type from the protected mapping via the `speckle_type`"""
|
||||||
for full_name in reversed(speckle_type.split(":")):
|
return cls._type_registry.get(speckle_type, None)
|
||||||
maybe_type = cls._type_registry.get(full_name, None)
|
|
||||||
if maybe_type:
|
|
||||||
return maybe_type
|
|
||||||
return None
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _determine_speckle_type(cls) -> str:
|
|
||||||
"""
|
|
||||||
This method brings the speckle_type construction in par with peckle-sharp/Core.
|
|
||||||
|
|
||||||
The implementation differs, because in Core the basis of the speckle_type if
|
|
||||||
type.FullName, which includes the dotnet namespace name too.
|
|
||||||
Copying that behavior is hard in python, where the concept of namespaces
|
|
||||||
means something entirely different.
|
|
||||||
|
|
||||||
So we enabled a speckle_type override mechanism, that enables
|
|
||||||
"""
|
|
||||||
base_name = "Base"
|
|
||||||
if cls.__name__ == base_name:
|
|
||||||
return base_name
|
|
||||||
|
|
||||||
bases = [
|
|
||||||
b._full_name()
|
|
||||||
for b in reversed(cls.mro())
|
|
||||||
if issubclass(b, Base) and b.__name__ != base_name
|
|
||||||
]
|
|
||||||
return ":".join(bases)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _full_name(cls) -> str:
|
|
||||||
base_name = "Base"
|
|
||||||
if cls.__name__ == base_name:
|
|
||||||
return base_name
|
|
||||||
|
|
||||||
if cls._speckle_type_override:
|
|
||||||
return cls._speckle_type_override
|
|
||||||
|
|
||||||
# convert the module names to PascalCase to match c# namespace naming convention
|
|
||||||
# also drop specklepy from the beginning
|
|
||||||
namespace = ".".join(
|
|
||||||
pascalcase(m)
|
|
||||||
for m in filter(lambda name: name != "specklepy", cls.__module__.split("."))
|
|
||||||
)
|
|
||||||
return f"{namespace}.{cls.__name__}"
|
|
||||||
|
|
||||||
def __init_subclass__(
|
def __init_subclass__(
|
||||||
cls,
|
cls,
|
||||||
speckle_type: Optional[str] = None,
|
speckle_type: str = None,
|
||||||
chunkable: Optional[Dict[str, int]] = None,
|
chunkable: Dict[str, int] = None,
|
||||||
detachable: Optional[Set[str]] = None,
|
detachable: Set[str] = None,
|
||||||
serialize_ignore: Optional[Set[str]] = None,
|
serialize_ignore: Set[str] = None,
|
||||||
**kwargs: Dict[str, Any],
|
**kwargs: Dict[str, Any],
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
@@ -168,15 +114,14 @@ class _RegisteringBase:
|
|||||||
initialization. This is reused to register each subclassing type into a class
|
initialization. This is reused to register each subclassing type into a class
|
||||||
level dictionary.
|
level dictionary.
|
||||||
"""
|
"""
|
||||||
cls._speckle_type_override = speckle_type
|
if speckle_type in cls._type_registry:
|
||||||
cls.speckle_type = cls._determine_speckle_type()
|
|
||||||
if cls._full_name() in cls._type_registry:
|
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"The speckle_type: {speckle_type} is already registered for type: "
|
f"The speckle_type: {speckle_type} is already registered for type: "
|
||||||
f"{cls._type_registry[cls._full_name()].__name__}. "
|
f"{cls._type_registry[speckle_type].__name__}. "
|
||||||
"Please choose a different type name."
|
f"Please choose a different type name."
|
||||||
)
|
)
|
||||||
cls._type_registry[cls._full_name()] = cls # type: ignore
|
cls.speckle_type = speckle_type or cls.__name__
|
||||||
|
cls._type_registry[cls.speckle_type] = cls # type: ignore
|
||||||
try:
|
try:
|
||||||
cls._attr_types = get_type_hints(cls)
|
cls._attr_types = get_type_hints(cls)
|
||||||
except Exception:
|
except Exception:
|
||||||
@@ -191,138 +136,16 @@ class _RegisteringBase:
|
|||||||
super().__init_subclass__(**kwargs)
|
super().__init_subclass__(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
# T = TypeVar("T")
|
|
||||||
|
|
||||||
# how i wish the code below would be correct, but we're also parsing into floats
|
|
||||||
# and converting into strings if the original type is string, but the value isn't
|
|
||||||
# def _validate_type(t: type, value: T) -> Tuple[bool, T]:
|
|
||||||
|
|
||||||
|
|
||||||
def _validate_type(t: Optional[type], value: Any) -> Tuple[bool, Any]:
|
|
||||||
# this should be reworked. Its only ok to return null for Optionals...
|
|
||||||
# if t is None and value is None:
|
|
||||||
if value is None:
|
|
||||||
return True, value
|
|
||||||
|
|
||||||
# after fixing the None t above, this should be
|
|
||||||
# if t is Any:
|
|
||||||
# if t is None:
|
|
||||||
|
|
||||||
if t is None or t is Any:
|
|
||||||
return True, value
|
|
||||||
|
|
||||||
if isclass(t) and issubclass(t, Enum):
|
|
||||||
if isinstance(value, t):
|
|
||||||
return True, value
|
|
||||||
if value in t._value2member_map_:
|
|
||||||
return True, t(value)
|
|
||||||
|
|
||||||
if getattr(t, "__module__", None) == "typing":
|
|
||||||
if isinstance(t, ForwardRef):
|
|
||||||
return True, value
|
|
||||||
|
|
||||||
origin = getattr(t, "__origin__")
|
|
||||||
# below is what in nicer for >= py38
|
|
||||||
# origin = get_origin(t)
|
|
||||||
|
|
||||||
# recursive validation for Unions on both types preferring the fist type
|
|
||||||
if origin is Union:
|
|
||||||
# below is what in nicer for >= py38
|
|
||||||
# t_1, t_2 = get_args(t)
|
|
||||||
args = t.__args__ # type: ignore
|
|
||||||
for arg_t in args:
|
|
||||||
t_success, t_value = _validate_type(arg_t, value)
|
|
||||||
if t_success:
|
|
||||||
return True, t_value
|
|
||||||
return False, value
|
|
||||||
if origin is dict:
|
|
||||||
if not isinstance(value, dict):
|
|
||||||
return False, value
|
|
||||||
if value == {}:
|
|
||||||
return True, value
|
|
||||||
if not getattr(t, "__args__", None):
|
|
||||||
return True, value
|
|
||||||
t_key, t_value = t.__args__ # type: ignore
|
|
||||||
|
|
||||||
if (
|
|
||||||
getattr(t_key, "__name__", None),
|
|
||||||
getattr(t_value, "__name__", None),
|
|
||||||
) == ("KT", "VT"):
|
|
||||||
return True, value
|
|
||||||
# we're only checking the first item, but the for loop and return after
|
|
||||||
# evaluating the first item is the fastest way
|
|
||||||
for dict_key, dict_value in value.items():
|
|
||||||
valid_key, _ = _validate_type(t_key, dict_key)
|
|
||||||
valid_value, _ = _validate_type(t_value, dict_value)
|
|
||||||
|
|
||||||
if valid_key and valid_value:
|
|
||||||
return True, value
|
|
||||||
return False, value
|
|
||||||
|
|
||||||
if origin is list:
|
|
||||||
if not isinstance(value, list):
|
|
||||||
return False, value
|
|
||||||
if value == []:
|
|
||||||
return True, value
|
|
||||||
if not hasattr(t, "__args__"):
|
|
||||||
return True, value
|
|
||||||
t_items = t.__args__[0] # type: ignore
|
|
||||||
if getattr(t_items, "__name__", None) == "T":
|
|
||||||
return True, value
|
|
||||||
first_item_valid, _ = _validate_type(t_items, value[0])
|
|
||||||
if first_item_valid:
|
|
||||||
return True, value
|
|
||||||
return False, value
|
|
||||||
|
|
||||||
if origin is tuple:
|
|
||||||
if not isinstance(value, tuple):
|
|
||||||
return False, value
|
|
||||||
if not hasattr(t, "__args__"):
|
|
||||||
return True, value
|
|
||||||
args = t.__args__ # type: ignore
|
|
||||||
if args == tuple():
|
|
||||||
return True, value
|
|
||||||
# we're not checking for empty tuple, cause tuple lengths must match
|
|
||||||
if len(args) != len(value):
|
|
||||||
return False, value
|
|
||||||
values = []
|
|
||||||
for t_item, v_item in zip(args, value):
|
|
||||||
item_valid, item_value = _validate_type(t_item, v_item)
|
|
||||||
if not item_valid:
|
|
||||||
return False, value
|
|
||||||
values.append(item_value)
|
|
||||||
return True, tuple(values)
|
|
||||||
|
|
||||||
if origin is set:
|
|
||||||
if not isinstance(value, set):
|
|
||||||
return False, value
|
|
||||||
if not hasattr(t, "__args__"):
|
|
||||||
return True, value
|
|
||||||
t_items = t.__args__[0] # type: ignore
|
|
||||||
first_item_valid, _ = _validate_type(t_items, next(iter(value)))
|
|
||||||
if first_item_valid:
|
|
||||||
return True, value
|
|
||||||
return False, value
|
|
||||||
|
|
||||||
if isinstance(value, t):
|
|
||||||
return True, value
|
|
||||||
|
|
||||||
with contextlib.suppress(ValueError, TypeError):
|
|
||||||
if t is float and value is not None:
|
|
||||||
return True, float(value)
|
|
||||||
# TODO: dafuq, i had to add this not list check
|
|
||||||
# but it would also fail for objects and other complex values
|
|
||||||
if t is str and value and not isinstance(value, list):
|
|
||||||
return True, str(value)
|
|
||||||
|
|
||||||
return False, value
|
|
||||||
|
|
||||||
|
|
||||||
class Base(_RegisteringBase):
|
class Base(_RegisteringBase):
|
||||||
id: Union[str, None] = None
|
id: Optional[str] = None
|
||||||
totalChildrenCount: Union[int, None] = None
|
totalChildrenCount: Optional[int] = None
|
||||||
applicationId: Union[str, None] = None
|
applicationId: Optional[str] = None
|
||||||
_units: Union[None, str] = None
|
_units: str = "m"
|
||||||
|
# dict of chunkable props and their max chunk size
|
||||||
|
_chunkable: Dict[str, int] = {}
|
||||||
|
_chunk_size_default: int = 1000
|
||||||
|
_detachable: Set[str] = set() # list of defined detachable props
|
||||||
|
_serialize_ignore: Set[str] = set()
|
||||||
|
|
||||||
def __init__(self, **kwargs) -> None:
|
def __init__(self, **kwargs) -> None:
|
||||||
super().__init__()
|
super().__init__()
|
||||||
@@ -390,14 +213,11 @@ class Base(_RegisteringBase):
|
|||||||
@classmethod
|
@classmethod
|
||||||
def update_forward_refs(cls) -> None:
|
def update_forward_refs(cls) -> None:
|
||||||
"""
|
"""
|
||||||
Attempts to populate the internal defined types dict for type checking
|
Attempts to populate the internal defined types dict for type checking sometime after defining the class.
|
||||||
sometime after defining the class.
|
This is already done when defining the class, but can be called again if references to undefined types were
|
||||||
This is already done when defining the class, but can be called
|
|
||||||
again if references to undefined types were
|
|
||||||
included.
|
included.
|
||||||
|
|
||||||
See `objects.geometry` for an example of how this is used with
|
See `objects.geometry` for an example of how this is used with the Brep class definitions
|
||||||
the Brep class definitions.
|
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
cls._attr_types = get_type_hints(cls)
|
cls._attr_types = get_type_hints(cls)
|
||||||
@@ -418,27 +238,51 @@ class Base(_RegisteringBase):
|
|||||||
"Invalid Name: Base member names cannot contain characters '.' or '/'",
|
"Invalid Name: Base member names cannot contain characters '.' or '/'",
|
||||||
)
|
)
|
||||||
|
|
||||||
def _type_check(self, name: str, value: Any) -> Any:
|
def _type_check(self, name: str, value: Any):
|
||||||
"""
|
"""
|
||||||
Lightweight type checking of values before setting them
|
Lightweight type checking of values before setting them
|
||||||
|
|
||||||
NOTE: Does not check subscripted types within generics as the performance hit
|
NOTE: Does not check subscripted types within generics as the performance hit of checking
|
||||||
of checking each item within a given collection isn't worth it.
|
each item within a given collection isn't worth it. Eg if you have a type Dict[str, float],
|
||||||
Eg if you have a type Dict[str, float],
|
|
||||||
we will only check if the value you're trying to set is a dict.
|
we will only check if the value you're trying to set is a dict.
|
||||||
"""
|
"""
|
||||||
types = getattr(self, "_attr_types", {})
|
types = getattr(self, "_attr_types", {})
|
||||||
t = types.get(name, None)
|
t = types.get(name, None)
|
||||||
|
|
||||||
valid, checked_value = _validate_type(t, value)
|
if t is None:
|
||||||
|
return value
|
||||||
|
|
||||||
if valid:
|
if t.__module__ == "typing":
|
||||||
return checked_value
|
origin = getattr(t, "__origin__")
|
||||||
|
t = (
|
||||||
|
tuple(getattr(sub_t, "__origin__", sub_t) for sub_t in t.__args__)
|
||||||
|
if origin is typing.Union
|
||||||
|
else origin
|
||||||
|
)
|
||||||
|
|
||||||
|
if not isinstance(t, (type, tuple)):
|
||||||
|
warn(
|
||||||
|
f"Unrecognised type '{t}' provided for attribute '{name}'. Type will not been validated."
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
if isinstance(value, t):
|
||||||
|
return value
|
||||||
|
|
||||||
|
# to be friendly, we'll parse ints and strs into floats, but not the other way around
|
||||||
|
# (to avoid unexpected rounding)
|
||||||
|
if isinstance(t, tuple):
|
||||||
|
t = t[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
if t is float:
|
||||||
|
return float(value)
|
||||||
|
if t is str and value:
|
||||||
|
return str(value)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
f"Cannot set '{self.__class__.__name__}.{name}':"
|
f"Cannot set '{self.__class__.__name__}.{name}': it expects type '{t.__name__}', but received type '{type(value).__name__}'"
|
||||||
f"it expects type '{str(t)}',"
|
|
||||||
f"but received type '{type(value).__name__}'"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def add_chunkable_attrs(self, **kwargs: int) -> None:
|
def add_chunkable_attrs(self, **kwargs: int) -> None:
|
||||||
@@ -446,8 +290,7 @@ class Base(_RegisteringBase):
|
|||||||
Mark defined attributes as chunkable for serialisation
|
Mark defined attributes as chunkable for serialisation
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
kwargs {int} -- the name of the attribute as the keyword
|
kwargs {int} -- the name of the attribute as the keyword and the chunk size as the arg
|
||||||
and the chunk size as the arg
|
|
||||||
"""
|
"""
|
||||||
chunkable = {k: v for k, v in kwargs.items() if isinstance(v, int)}
|
chunkable = {k: v for k, v in kwargs.items() if isinstance(v, int)}
|
||||||
self._chunkable = dict(self._chunkable, **chunkable)
|
self._chunkable = dict(self._chunkable, **chunkable)
|
||||||
@@ -457,25 +300,17 @@ class Base(_RegisteringBase):
|
|||||||
Mark defined attributes as detachable for serialisation
|
Mark defined attributes as detachable for serialisation
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
names {Set[str]} -- the names of the attributes to detach as a set of string
|
names {Set[str]} -- the names of the attributes to detach as a set of strings
|
||||||
"""
|
"""
|
||||||
self._detachable = self._detachable.union(names)
|
self._detachable = self._detachable.union(names)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def units(self) -> Union[str, None]:
|
def units(self):
|
||||||
return self._units
|
return self._units
|
||||||
|
|
||||||
@units.setter
|
@units.setter
|
||||||
def units(self, value: Union[str, Units, None]):
|
def units(self, value: str):
|
||||||
"""While this property accepts any string value, geometry expects units to be specific strings (see Units enum)"""
|
self._units = get_units_from_string(value)
|
||||||
if isinstance(value, str) or value is None:
|
|
||||||
self._units = value
|
|
||||||
elif isinstance(value, Units):
|
|
||||||
self._units = value.value
|
|
||||||
else:
|
|
||||||
raise SpeckleInvalidUnitException(
|
|
||||||
f"Unknown type {type(value)} received for units"
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_member_names(self) -> List[str]:
|
def get_member_names(self) -> List[str]:
|
||||||
"""Get all of the property names on this object, dynamic or not"""
|
"""Get all of the property names on this object, dynamic or not"""
|
||||||
@@ -488,7 +323,7 @@ class Base(_RegisteringBase):
|
|||||||
|
|
||||||
def get_serializable_attributes(self) -> List[str]:
|
def get_serializable_attributes(self) -> List[str]:
|
||||||
"""Get the attributes that should be serialized"""
|
"""Get the attributes that should be serialized"""
|
||||||
return sorted(list(set(self.get_member_names()) - self._serialize_ignore))
|
return list(set(self.get_member_names()) - self._serialize_ignore)
|
||||||
|
|
||||||
def get_typed_member_names(self) -> List[str]:
|
def get_typed_member_names(self) -> List[str]:
|
||||||
"""Get all of the names of the defined (typed) properties of this object"""
|
"""Get all of the names of the defined (typed) properties of this object"""
|
||||||
@@ -505,17 +340,13 @@ class Base(_RegisteringBase):
|
|||||||
|
|
||||||
def get_id(self, decompose: bool = False) -> str:
|
def get_id(self, decompose: bool = False) -> str:
|
||||||
"""
|
"""
|
||||||
Gets the id (a unique hash) of this object.
|
Gets the id (a unique hash) of this object. ⚠️ This method fully serializes the object which,
|
||||||
⚠️ This method fully serializes the object which,
|
in the case of large objects (with many sub-objects), has a tangible cost. Avoid using it!
|
||||||
in the case of large objects (with many sub-objects), has a tangible cost.
|
|
||||||
Avoid using it!
|
|
||||||
|
|
||||||
Note: the hash of a decomposed object differs from that of a
|
Note: the hash of a decomposed object differs from that of a non-decomposed object
|
||||||
non-decomposed object
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
decompose {bool} -- if True, will decompose the object in
|
decompose {bool} -- if True, will decompose the object in the process of hashing it
|
||||||
the process of hashing it
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the hash (id) of the fully serialized object
|
str -- the hash (id) of the fully serialized object
|
||||||
@@ -539,7 +370,6 @@ class Base(_RegisteringBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def _handle_object_count(self, obj: Any, parsed: List) -> int:
|
def _handle_object_count(self, obj: Any, parsed: List) -> int:
|
||||||
# pylint: disable=isinstance-second-argument-not-valid-type
|
|
||||||
count = 0
|
count = 0
|
||||||
if obj is None:
|
if obj is None:
|
||||||
return count
|
return count
|
||||||
@@ -568,8 +398,7 @@ Base.update_forward_refs()
|
|||||||
|
|
||||||
|
|
||||||
class DataChunk(Base, speckle_type="Speckle.Core.Models.DataChunk"):
|
class DataChunk(Base, speckle_type="Speckle.Core.Models.DataChunk"):
|
||||||
data: Union[List[Any], None] = None
|
data: List[Any] = None
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
super().__init__()
|
|
||||||
self.data = []
|
self.data = []
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
from enum import Enum
|
from enum import Enum
|
||||||
from typing import Any, Callable, Dict, List, Optional, Type
|
from typing import Any, Callable, List, Type
|
||||||
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.objects.base import Base
|
from specklepy.objects.base import Base
|
||||||
@@ -43,8 +43,8 @@ def curve_from_list(args: List[float]):
|
|||||||
|
|
||||||
|
|
||||||
class ObjectArray:
|
class ObjectArray:
|
||||||
def __init__(self, data: Optional[list] = None) -> None:
|
def __init__(self) -> None:
|
||||||
self.data = data or []
|
self.data = []
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_objects(cls, objects: List[Base]) -> "ObjectArray":
|
def from_objects(cls, objects: List[Base]) -> "ObjectArray":
|
||||||
@@ -60,34 +60,38 @@ class ObjectArray:
|
|||||||
"All objects in chunk should have the same speckle_type. "
|
"All objects in chunk should have the same speckle_type. "
|
||||||
f"Found {speckle_type} and {obj.speckle_type}"
|
f"Found {speckle_type} and {obj.speckle_type}"
|
||||||
)
|
)
|
||||||
data_list.encode_object(obj=obj)
|
data_list.encode_object(object=obj)
|
||||||
|
|
||||||
return data_list
|
return data_list
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def decode_data(
|
def decode_data(
|
||||||
data: List[Any], decoder: Callable[[List[Any]], Base], **kwargs: Dict[str, Any]
|
data: List[Any], decoder: Callable[[List[Any]], Base]
|
||||||
) -> List[Base]:
|
) -> List[Base]:
|
||||||
bases: List[Base] = []
|
bases = []
|
||||||
if not data:
|
if not data:
|
||||||
return bases
|
return bases
|
||||||
|
|
||||||
index = 0
|
index = 0
|
||||||
while index < len(data):
|
while index < len(data):
|
||||||
item_length = int(data[index])
|
item_length = data[index]
|
||||||
item_start = index + 1
|
item_start = index + 1
|
||||||
item_end = item_start + item_length
|
item_end = item_start + item_length
|
||||||
item_data = data[item_start:item_end]
|
item_data = data[item_start:item_end]
|
||||||
index = item_end
|
index = item_end
|
||||||
decoded_data = decoder(item_data, **kwargs)
|
# TODO: investigate what's going on w this fail
|
||||||
bases.append(decoded_data)
|
try:
|
||||||
|
decoded_data = decoder(item_data)
|
||||||
|
bases.append(decoded_data)
|
||||||
|
except ValueError:
|
||||||
|
continue
|
||||||
return bases
|
return bases
|
||||||
|
|
||||||
def decode(self, decoder: Callable[[List[Any]], Any], **kwargs: Dict[str, Any]):
|
def decode(self, decoder: Callable[[List[Any]], Any]):
|
||||||
return self.decode_data(data=self.data, decoder=decoder, **kwargs)
|
return self.decode_data(data=self.data, decoder=decoder)
|
||||||
|
|
||||||
def encode_object(self, obj: Base):
|
def encode_object(self, object: Base):
|
||||||
encoded = obj.to_list()
|
encoded = object.to_list()
|
||||||
encoded.insert(0, len(encoded))
|
encoded.insert(0, len(encoded))
|
||||||
self.data.extend(encoded)
|
self.data.extend(encoded)
|
||||||
|
|
||||||
@@ -124,7 +128,8 @@ class CurveArray(ObjectArray):
|
|||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _curve_decoder(cls, data: List[float]) -> Base:
|
def _curve_decoder(cls, data: List[float]) -> Base:
|
||||||
crv_array = cls(data)
|
crv_array = cls()
|
||||||
|
crv_array.data = data
|
||||||
return crv_array.to_curve()
|
return crv_array.to_curve()
|
||||||
|
|
||||||
def to_curves(self) -> List[Base]:
|
def to_curves(self) -> List[Base]:
|
||||||
@@ -1,7 +1,5 @@
|
|||||||
from enum import Enum
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from specklepy.objects.geometry import Point
|
from specklepy.objects.geometry import Point
|
||||||
|
from typing import List
|
||||||
|
|
||||||
from .base import Base
|
from .base import Base
|
||||||
|
|
||||||
@@ -17,27 +15,19 @@ DETACHABLE = {"detach_this", "origin", "detached_list"}
|
|||||||
|
|
||||||
|
|
||||||
class FakeGeo(Base, chunkable={"dots": 50}, detachable={"pointslist"}):
|
class FakeGeo(Base, chunkable={"dots": 50}, detachable={"pointslist"}):
|
||||||
pointslist: Optional[List[Base]] = None
|
pointslist: List[Base] = None
|
||||||
dots: Optional[List[int]] = None
|
dots: List[int] = None
|
||||||
|
|
||||||
|
|
||||||
class FakeDirection(Enum):
|
|
||||||
NORTH = 1
|
|
||||||
EAST = 2
|
|
||||||
SOUTH = 3
|
|
||||||
WEST = 4
|
|
||||||
|
|
||||||
|
|
||||||
class FakeMesh(FakeGeo, chunkable=CHUNKABLE_PROPS, detachable=DETACHABLE):
|
class FakeMesh(FakeGeo, chunkable=CHUNKABLE_PROPS, detachable=DETACHABLE):
|
||||||
vertices: Optional[List[float]] = None
|
vertices: List[float] = None
|
||||||
faces: Optional[List[int]] = None
|
faces: List[int] = None
|
||||||
colors: Optional[List[int]] = None
|
colors: List[int] = None
|
||||||
textureCoordinates: Optional[List[float]] = None
|
textureCoordinates: List[float] = None
|
||||||
cardinal_dir: Optional[FakeDirection] = None
|
test_bases: List[Base] = None
|
||||||
test_bases: Optional[List[Base]] = None
|
detach_this: Base = None
|
||||||
detach_this: Optional[Base] = None
|
detached_list: List[Base] = None
|
||||||
detached_list: Optional[List[Base]] = None
|
_origin: Point = None
|
||||||
_origin: Optional[Point] = None
|
|
||||||
|
|
||||||
# def __init__(self, **kwargs) -> None:
|
# def __init__(self, **kwargs) -> None:
|
||||||
# super(FakeMesh, self).__init__(**kwargs)
|
# super(FakeMesh, self).__init__(**kwargs)
|
||||||
@@ -0,0 +1,744 @@
|
|||||||
|
from enum import Enum
|
||||||
|
from typing import Any, List, Optional
|
||||||
|
|
||||||
|
from .base import Base
|
||||||
|
from .encoding import CurveArray, CurveTypeEncoding, ObjectArray
|
||||||
|
from .units import get_encoding_from_units, get_units_from_encoding
|
||||||
|
|
||||||
|
GEOMETRY = "Objects.Geometry."
|
||||||
|
|
||||||
|
|
||||||
|
class Interval(Base, speckle_type="Objects.Primitive.Interval"):
|
||||||
|
start: float = 0.0
|
||||||
|
end: float = 0.0
|
||||||
|
|
||||||
|
def length(self):
|
||||||
|
return abs(self.start - self.end)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Interval":
|
||||||
|
return cls(start=args[0], end=args[1])
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
return [self.start, self.end]
|
||||||
|
|
||||||
|
|
||||||
|
class Point(Base, speckle_type=GEOMETRY + "Point"):
|
||||||
|
x: float = 0.0
|
||||||
|
y: float = 0.0
|
||||||
|
z: float = 0.0
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return f"{self.__class__.__name__}(x: {self.x}, y: {self.y}, z: {self.z}, id: {self.id}, speckle_type: {self.speckle_type})"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[float]) -> "Point":
|
||||||
|
return cls(x=args[0], y=args[1], z=args[2])
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
return [self.x, self.y, self.z]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_coords(cls, x: float = 0.0, y: float = 0.0, z: float = 0.0):
|
||||||
|
pt = Point()
|
||||||
|
pt.x, pt.y, pt.z = x, y, z
|
||||||
|
return pt
|
||||||
|
|
||||||
|
|
||||||
|
class Vector(Point, speckle_type=GEOMETRY + "Vector"):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class ControlPoint(Point, speckle_type=GEOMETRY + "ControlPoint"):
|
||||||
|
weight: float = None
|
||||||
|
|
||||||
|
|
||||||
|
class Plane(Base, speckle_type=GEOMETRY + "Plane"):
|
||||||
|
origin: Point = Point()
|
||||||
|
normal: Vector = Vector()
|
||||||
|
xdir: Vector = Vector()
|
||||||
|
ydir: Vector = Vector()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Plane":
|
||||||
|
return cls(
|
||||||
|
origin=Point.from_list(args[0:3]),
|
||||||
|
normal=Vector.from_list(args[3:6]),
|
||||||
|
xdir=Vector.from_list(args[6:9]),
|
||||||
|
ydir=Vector.from_list(args[9:12]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.extend(self.origin.to_list())
|
||||||
|
encoded.extend(self.normal.to_list())
|
||||||
|
encoded.extend(self.xdir.to_list())
|
||||||
|
encoded.extend(self.ydir.to_list())
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Box(Base, speckle_type=GEOMETRY + "Box"):
|
||||||
|
basePlane: Plane = Plane()
|
||||||
|
ySize: Interval = Interval()
|
||||||
|
zSize: Interval = Interval()
|
||||||
|
xSize: Interval = Interval()
|
||||||
|
area: float = None
|
||||||
|
volume: float = None
|
||||||
|
|
||||||
|
|
||||||
|
class Line(Base, speckle_type=GEOMETRY + "Line"):
|
||||||
|
start: Point = Point()
|
||||||
|
end: Point = None
|
||||||
|
domain: Interval = None
|
||||||
|
bbox: Box = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Line":
|
||||||
|
return cls(
|
||||||
|
start=Point.from_list(args[0:3]),
|
||||||
|
end=Point.from_list(args[3:6]),
|
||||||
|
domain=Interval.from_list(args[6:9]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.extend(self.start.to_list())
|
||||||
|
encoded.extend(self.end.to_list())
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Arc(Base, speckle_type=GEOMETRY + "Arc"):
|
||||||
|
radius: float = None
|
||||||
|
startAngle: float = None
|
||||||
|
endAngle: float = None
|
||||||
|
angleRadians: float = None
|
||||||
|
plane: Plane = None
|
||||||
|
domain: Interval = None
|
||||||
|
startPoint: Point = None
|
||||||
|
midPoint: Point = None
|
||||||
|
endPoint: Point = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Arc":
|
||||||
|
return cls(
|
||||||
|
radius=args[1],
|
||||||
|
startAngle=args[2],
|
||||||
|
endAngle=args[3],
|
||||||
|
angleRadians=args[4],
|
||||||
|
domain=Interval.from_list(args[5:7]),
|
||||||
|
plane=Plane.from_list(args[7:20]),
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Arc.value)
|
||||||
|
encoded.append(self.radius)
|
||||||
|
encoded.append(self.startAngle)
|
||||||
|
encoded.append(self.endAngle)
|
||||||
|
encoded.append(self.angleRadians)
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
encoded.extend(self.plane.to_list())
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Circle(Base, speckle_type=GEOMETRY + "Circle"):
|
||||||
|
radius: float = None
|
||||||
|
plane: Plane = None
|
||||||
|
domain: Interval = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Circle":
|
||||||
|
return cls(
|
||||||
|
radius=args[1],
|
||||||
|
domain=Interval.from_list(args[2:4]),
|
||||||
|
plane=Plane.from_list(args[4:17]),
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Circle.value)
|
||||||
|
encoded.append(self.radius),
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
encoded.extend(self.plane.to_list())
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Ellipse(Base, speckle_type=GEOMETRY + "Ellipse"):
|
||||||
|
firstRadius: float = None
|
||||||
|
secondRadius: float = None
|
||||||
|
plane: Plane = None
|
||||||
|
domain: Interval = None
|
||||||
|
trimDomain: Interval = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Ellipse":
|
||||||
|
return cls(
|
||||||
|
firstRadius=args[1],
|
||||||
|
secondRadius=args[2],
|
||||||
|
domain=Interval.from_list(args[3:5]),
|
||||||
|
plane=Plane.from_list(args[5:18]),
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Ellipse.value)
|
||||||
|
encoded.append(self.firstRadius)
|
||||||
|
encoded.append(self.secondRadius)
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
encoded.extend(self.plane.to_list())
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Polyline(Base, speckle_type=GEOMETRY + "Polyline", chunkable={"value": 20000}):
|
||||||
|
value: List[float] = None
|
||||||
|
closed: bool = None
|
||||||
|
domain: Interval = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_points(cls, points: List[Point]):
|
||||||
|
polyline = cls()
|
||||||
|
polyline.units = points[0].units
|
||||||
|
polyline.value = []
|
||||||
|
for point in points:
|
||||||
|
polyline.value.extend([point.x, point.y, point.z])
|
||||||
|
return polyline
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Polyline":
|
||||||
|
point_count = args[4]
|
||||||
|
return cls(
|
||||||
|
closed=bool(args[1]),
|
||||||
|
domain=Interval.from_list(args[2:4]),
|
||||||
|
value=args[5 : 5 + point_count],
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Polyline.value)
|
||||||
|
encoded.append(int(self.closed))
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
encoded.append(len(self.value))
|
||||||
|
encoded.extend(self.value)
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
def as_points(self) -> List[Point]:
|
||||||
|
"""Converts the `value` attribute to a list of Points"""
|
||||||
|
if not self.value:
|
||||||
|
return
|
||||||
|
|
||||||
|
if len(self.value) % 3:
|
||||||
|
raise ValueError("Points array malformed: length%3 != 0.")
|
||||||
|
|
||||||
|
values = iter(self.value)
|
||||||
|
return [
|
||||||
|
Point(x=v, y=next(values), z=next(values), units=self.units) for v in values
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class Curve(
|
||||||
|
Base,
|
||||||
|
speckle_type=GEOMETRY + "Curve",
|
||||||
|
chunkable={"points": 20000, "weights": 20000, "knots": 20000},
|
||||||
|
):
|
||||||
|
degree: int = None
|
||||||
|
periodic: bool = None
|
||||||
|
rational: bool = None
|
||||||
|
points: List[float] = None
|
||||||
|
weights: List[float] = None
|
||||||
|
knots: List[float] = None
|
||||||
|
domain: Interval = None
|
||||||
|
displayValue: Polyline = None
|
||||||
|
closed: bool = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
def as_points(self) -> List[Point]:
|
||||||
|
"""Converts the `value` attribute to a list of Points"""
|
||||||
|
if not self.points:
|
||||||
|
return
|
||||||
|
|
||||||
|
if len(self.points) % 3:
|
||||||
|
raise ValueError("Points array malformed: length%3 != 0.")
|
||||||
|
|
||||||
|
values = iter(self.points)
|
||||||
|
return [
|
||||||
|
Point(x=v, y=next(values), z=next(values), units=self.units) for v in values
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Curve":
|
||||||
|
point_count = args[7]
|
||||||
|
weights_count = args[8]
|
||||||
|
knots_count = args[9]
|
||||||
|
|
||||||
|
points_start = 10
|
||||||
|
weights_start = 10 + point_count
|
||||||
|
knots_start = weights_start + weights_count
|
||||||
|
knots_end = knots_start + knots_count
|
||||||
|
|
||||||
|
return cls(
|
||||||
|
degree=args[1],
|
||||||
|
periodic=bool(args[2]),
|
||||||
|
rational=bool(args[3]),
|
||||||
|
closed=bool(args[4]),
|
||||||
|
domain=Interval.from_list(args[5:7]),
|
||||||
|
points=args[points_start:weights_start],
|
||||||
|
weights=args[weights_start:knots_start],
|
||||||
|
knots=args[knots_start:knots_end],
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Curve.value)
|
||||||
|
encoded.append(self.degree)
|
||||||
|
encoded.append(int(self.periodic))
|
||||||
|
encoded.append(int(self.rational))
|
||||||
|
encoded.append(int(self.closed))
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
encoded.append(len(self.points))
|
||||||
|
encoded.append(len(self.weights))
|
||||||
|
encoded.append(len(self.knots))
|
||||||
|
encoded.extend(self.points)
|
||||||
|
encoded.extend(self.weights)
|
||||||
|
encoded.extend(self.knots)
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Polycurve(Base, speckle_type=GEOMETRY + "Polycurve"):
|
||||||
|
segments: List[Base] = None
|
||||||
|
domain: Interval = None
|
||||||
|
closed: bool = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
length: float = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Polycurve":
|
||||||
|
curve_arrays = CurveArray()
|
||||||
|
curve_arrays.data = args[4:-1]
|
||||||
|
return cls(
|
||||||
|
closed=bool(args[1]),
|
||||||
|
domain=Interval.from_list(args[2:4]),
|
||||||
|
segments=curve_arrays.to_curves(),
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(CurveTypeEncoding.Polycurve.value)
|
||||||
|
encoded.append(int(self.closed))
|
||||||
|
encoded.extend(self.domain.to_list())
|
||||||
|
curve_array = CurveArray.from_curves(self.segments)
|
||||||
|
encoded.extend(curve_array.data)
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Extrusion(Base, speckle_type=GEOMETRY + "Extrusion"):
|
||||||
|
capped: bool = None
|
||||||
|
profile: Base = None
|
||||||
|
pathStart: Point = None
|
||||||
|
pathEnd: Point = None
|
||||||
|
pathCurve: Base = None
|
||||||
|
pathTangent: Base = None
|
||||||
|
profiles: List[Base] = None
|
||||||
|
length: float = None
|
||||||
|
area: float = None
|
||||||
|
volume: float = None
|
||||||
|
bbox: Box = None
|
||||||
|
|
||||||
|
|
||||||
|
class Mesh(
|
||||||
|
Base,
|
||||||
|
speckle_type=GEOMETRY + "Mesh",
|
||||||
|
chunkable={
|
||||||
|
"vertices": 2000,
|
||||||
|
"faces": 2000,
|
||||||
|
"colors": 2000,
|
||||||
|
"textureCoordinates": 2000,
|
||||||
|
},
|
||||||
|
):
|
||||||
|
vertices: List[float] = None
|
||||||
|
faces: List[int] = None
|
||||||
|
colors: List[int] = None
|
||||||
|
textureCoordinates: List[float] = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
volume: float = None
|
||||||
|
|
||||||
|
def transform_to(self, transform: "Transform") -> "Mesh":
|
||||||
|
mesh = Mesh(vertices=transform.apply_to_points_values(self.vertices))
|
||||||
|
for attr in set(self.get_serializable_attributes()) - {"vertices"}:
|
||||||
|
orig_val = getattr(self, attr, None)
|
||||||
|
if orig_val:
|
||||||
|
setattr(mesh, attr, orig_val)
|
||||||
|
|
||||||
|
return mesh
|
||||||
|
|
||||||
|
|
||||||
|
class Surface(Base, speckle_type=GEOMETRY + "Surface"):
|
||||||
|
degreeU: int = None
|
||||||
|
degreeV: int = None
|
||||||
|
rational: bool = None
|
||||||
|
area: float = None
|
||||||
|
pointData: List[float] = None
|
||||||
|
countU: int = None
|
||||||
|
countV: int = None
|
||||||
|
bbox: Box = None
|
||||||
|
closedU: bool = None
|
||||||
|
closedV: bool = None
|
||||||
|
domainU: Interval = None
|
||||||
|
domainV: Interval = None
|
||||||
|
knotsU: List[float] = None
|
||||||
|
knotsV: List[float] = None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "Surface":
|
||||||
|
point_count = int(args[11])
|
||||||
|
knots_u_count = int(args[12])
|
||||||
|
knots_v_count = int(args[13])
|
||||||
|
|
||||||
|
start_point_data = 14
|
||||||
|
start_knots_u = start_point_data + point_count
|
||||||
|
start_knots_v = start_knots_u + knots_u_count
|
||||||
|
|
||||||
|
return cls(
|
||||||
|
degreeU=int(args[0]),
|
||||||
|
degreeV=int(args[1]),
|
||||||
|
countU=int(args[2]),
|
||||||
|
countV=int(args[3]),
|
||||||
|
rational=bool(args[4]),
|
||||||
|
closedU=bool(args[5]),
|
||||||
|
closedV=bool(args[6]),
|
||||||
|
domainU=Interval(start=args[7], end=args[8]),
|
||||||
|
domainV=Interval(start=args[9], end=args[10]),
|
||||||
|
pointData=args[start_point_data:start_knots_u],
|
||||||
|
knotsU=args[start_knots_u:start_knots_v],
|
||||||
|
knotsV=args[start_knots_v : start_knots_v + knots_v_count],
|
||||||
|
units=get_units_from_encoding(args[-1]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(self.degreeU)
|
||||||
|
encoded.append(self.degreeV)
|
||||||
|
encoded.append(self.countU)
|
||||||
|
encoded.append(self.countV)
|
||||||
|
encoded.append(int(self.rational))
|
||||||
|
encoded.append(int(self.closedU))
|
||||||
|
encoded.append(int(self.closedV))
|
||||||
|
encoded.extend(self.domainU.to_list())
|
||||||
|
encoded.extend(self.domainV.to_list())
|
||||||
|
encoded.append(len(self.pointData))
|
||||||
|
encoded.append(len(self.knotsU))
|
||||||
|
encoded.append(len(self.knotsV))
|
||||||
|
encoded.extend(self.pointData)
|
||||||
|
encoded.extend(self.knotsU)
|
||||||
|
encoded.extend(self.knotsV)
|
||||||
|
encoded.append(get_encoding_from_units(self.units))
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class BrepFace(Base, speckle_type=GEOMETRY + "BrepFace"):
|
||||||
|
_Brep: "Brep" = None
|
||||||
|
SurfaceIndex: int = None
|
||||||
|
LoopIndices: List[int] = None
|
||||||
|
OuterLoopIndex: int = None
|
||||||
|
OrientationReversed: bool = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _outer_loop(self):
|
||||||
|
return self._Brep.Loops[self.OuterLoopIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _surface(self):
|
||||||
|
return self._Brep.Surfaces[self.SurfaceIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _loops(self):
|
||||||
|
if self.LoopIndices:
|
||||||
|
return [self._Brep.Loops[i] for i in self.LoopIndices]
|
||||||
|
|
||||||
|
|
||||||
|
class BrepEdge(Base, speckle_type=GEOMETRY + "BrepEdge"):
|
||||||
|
_Brep: "Brep" = None
|
||||||
|
Curve3dIndex: int = None
|
||||||
|
TrimIndices: List[int] = None
|
||||||
|
StartIndex: int = None
|
||||||
|
EndIndex: int = None
|
||||||
|
ProxyCurveIsReversed: bool = None
|
||||||
|
Domain: Interval = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _start_vertex(self):
|
||||||
|
return self._Brep.Vertices[self.StartIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _end_vertex(self):
|
||||||
|
return self._Brep.Vertices[self.EndIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _trims(self):
|
||||||
|
if self.TrimIndices:
|
||||||
|
return [self._Brep.Trims[i] for i in self.TrimIndices]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _curve(self):
|
||||||
|
return self._Brep.Curve3D[self.Curve3dIndex]
|
||||||
|
|
||||||
|
|
||||||
|
class BrepLoop(Base, speckle_type=GEOMETRY + "BrepLoop"):
|
||||||
|
_Brep: "Brep" = None
|
||||||
|
FaceIndex: int = None
|
||||||
|
TrimIndices: List[int] = None
|
||||||
|
Type: str = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _face(self):
|
||||||
|
return self._Brep.Faces[self.FaceIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _trims(self):
|
||||||
|
if self.TrimIndices:
|
||||||
|
return [self._Brep.Trims[i] for i in self.TrimIndices]
|
||||||
|
|
||||||
|
|
||||||
|
class BrepTrimTypeEnum(int, Enum):
|
||||||
|
Unknown = 0
|
||||||
|
Boundary = 1
|
||||||
|
Mated = 2
|
||||||
|
Seam = 3
|
||||||
|
Singular = 4
|
||||||
|
CurveOnSurface = 5
|
||||||
|
PointOnSurface = 6
|
||||||
|
Slit = 7
|
||||||
|
|
||||||
|
|
||||||
|
class BrepTrim(Base, speckle_type=GEOMETRY + "BrepTrim"):
|
||||||
|
_Brep: "Brep" = None
|
||||||
|
EdgeIndex: int = None
|
||||||
|
StartIndex: int = None
|
||||||
|
EndIndex: int = None
|
||||||
|
FaceIndex: int = None
|
||||||
|
LoopIndex: int = None
|
||||||
|
CurveIndex: int = None
|
||||||
|
IsoStatus: int = None
|
||||||
|
TrimType: str = None
|
||||||
|
IsReversed: bool = None
|
||||||
|
Domain: Interval = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _face(self):
|
||||||
|
return self._Brep.Faces[self.FaceIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _loop(self):
|
||||||
|
return self._Brep.Loops[self.LoopIndex]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _edge(self):
|
||||||
|
return self._Brep.Edges[self.EdgeIndex] if self.EdgeIndex != -1 else None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _curve_2d(self):
|
||||||
|
return self._Brep.Curve2D[self.CurveIndex]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_list(cls, args: List[Any]) -> "BrepTrim":
|
||||||
|
return cls(
|
||||||
|
EdgeIndex=args[0],
|
||||||
|
StartIndex=args[1],
|
||||||
|
EndIndex=args[2],
|
||||||
|
FaceIndex=args[3],
|
||||||
|
LoopIndex=args[4],
|
||||||
|
CurveIndex=args[5],
|
||||||
|
IsoStatus=args[6],
|
||||||
|
TrimType=BrepTrimTypeEnum(args[7]).name,
|
||||||
|
IsReversed=bool(args[8]),
|
||||||
|
)
|
||||||
|
|
||||||
|
def to_list(self) -> List[Any]:
|
||||||
|
encoded = []
|
||||||
|
encoded.append(self.EdgeIndex)
|
||||||
|
encoded.append(self.StartIndex)
|
||||||
|
encoded.append(self.EndIndex)
|
||||||
|
encoded.append(self.FaceIndex)
|
||||||
|
encoded.append(self.LoopIndex)
|
||||||
|
encoded.append(self.CurveIndex)
|
||||||
|
encoded.append(self.IsoStatus)
|
||||||
|
encoded.append(getattr(BrepTrimTypeEnum, self.TrimType).value)
|
||||||
|
encoded.append(self.IsReversed)
|
||||||
|
return encoded
|
||||||
|
|
||||||
|
|
||||||
|
class Brep(
|
||||||
|
Base,
|
||||||
|
speckle_type=GEOMETRY + "Brep",
|
||||||
|
chunkable={
|
||||||
|
"SurfacesValue": 200,
|
||||||
|
"Curve3DValues": 200,
|
||||||
|
"Curve2DValues": 200,
|
||||||
|
"VerticesValue": 5000,
|
||||||
|
"Edges": 5000,
|
||||||
|
"Loops": 5000,
|
||||||
|
"TrimsValue": 5000,
|
||||||
|
"Faces": 5000,
|
||||||
|
},
|
||||||
|
detachable={"displayValue"},
|
||||||
|
serialize_ignore={"Surfaces", "Curve3D", "Curve2D", "Vertices", "Trims"},
|
||||||
|
):
|
||||||
|
provenance: str = None
|
||||||
|
bbox: Box = None
|
||||||
|
area: float = None
|
||||||
|
volume: float = None
|
||||||
|
displayValue: Mesh = None
|
||||||
|
Surfaces: List[Surface] = None
|
||||||
|
Curve3D: List[Base] = None
|
||||||
|
Curve2D: List[Base] = None
|
||||||
|
Vertices: List[Point] = None
|
||||||
|
IsClosed: bool = None
|
||||||
|
Orientation: int = None
|
||||||
|
|
||||||
|
def _inject_self_into_children(self, children: Optional[List[Base]]) -> List[Base]:
|
||||||
|
if children is None:
|
||||||
|
return children
|
||||||
|
|
||||||
|
for child in children:
|
||||||
|
child._Brep = self
|
||||||
|
return children
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Edges(self) -> List[BrepEdge]:
|
||||||
|
return self._inject_self_into_children(self._Edges)
|
||||||
|
|
||||||
|
@Edges.setter
|
||||||
|
def Edges(self, value: List[BrepEdge]):
|
||||||
|
self._Edges = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Loops(self) -> List[BrepLoop]:
|
||||||
|
return self._inject_self_into_children(self._Loops)
|
||||||
|
|
||||||
|
@Loops.setter
|
||||||
|
def Loops(self, value: List[BrepLoop]):
|
||||||
|
self._Loops = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Faces(self) -> List[BrepFace]:
|
||||||
|
return self._inject_self_into_children(self._Faces)
|
||||||
|
|
||||||
|
@Faces.setter
|
||||||
|
def Faces(self, value: List[BrepFace]):
|
||||||
|
self._Faces = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def SurfacesValue(self) -> List[float]:
|
||||||
|
if self.Surfaces is None:
|
||||||
|
return None
|
||||||
|
return ObjectArray.from_objects(self.Surfaces).data
|
||||||
|
|
||||||
|
@SurfacesValue.setter
|
||||||
|
def SurfacesValue(self, value: List[float]):
|
||||||
|
self.Surfaces = ObjectArray.decode_data(value, Surface.from_list)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Curve3DValues(self) -> List[float]:
|
||||||
|
if self.Curve3D is None:
|
||||||
|
return None
|
||||||
|
return CurveArray.from_curves(self.Curve3D).data
|
||||||
|
|
||||||
|
@Curve3DValues.setter
|
||||||
|
def Curve3DValues(self, value: List[float]):
|
||||||
|
crv_array = CurveArray()
|
||||||
|
crv_array.data = value
|
||||||
|
self.Curve3D = crv_array.to_curves()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Curve2DValues(self) -> List[Base]:
|
||||||
|
if self.Curve2D is None:
|
||||||
|
return None
|
||||||
|
return CurveArray.from_curves(self.Curve2D).data
|
||||||
|
|
||||||
|
@Curve2DValues.setter
|
||||||
|
def Curve2DValues(self, value: List[float]):
|
||||||
|
crv_array = CurveArray()
|
||||||
|
crv_array.data = value
|
||||||
|
self.Curve2D = crv_array.to_curves()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def VerticesValue(self) -> List[Point]:
|
||||||
|
if self.Vertices is None:
|
||||||
|
return None
|
||||||
|
encoded_unit = get_encoding_from_units(self.Vertices[0].units)
|
||||||
|
values = [encoded_unit]
|
||||||
|
for vertex in self.Vertices:
|
||||||
|
values.extend(vertex.to_list())
|
||||||
|
return values
|
||||||
|
|
||||||
|
@VerticesValue.setter
|
||||||
|
def VerticesValue(self, value: List[float]):
|
||||||
|
value = value.copy()
|
||||||
|
units = get_units_from_encoding(value.pop(0))
|
||||||
|
|
||||||
|
vertices = []
|
||||||
|
|
||||||
|
for i in range(0, len(value), 3):
|
||||||
|
vertex = Point.from_list(value[i : i + 3])
|
||||||
|
vertex._units = units
|
||||||
|
vertices.append(vertex)
|
||||||
|
|
||||||
|
self.Vertices = vertices
|
||||||
|
|
||||||
|
@property
|
||||||
|
def Trims(self) -> List[BrepTrim]:
|
||||||
|
return self._inject_self_into_children(self._Trims)
|
||||||
|
|
||||||
|
@Trims.setter
|
||||||
|
def Trims(self, value: List[BrepTrim]):
|
||||||
|
self._Trims = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def TrimsValue(self) -> List[float]:
|
||||||
|
if self.Trims is None:
|
||||||
|
return None
|
||||||
|
values = []
|
||||||
|
for trim in self.Trims:
|
||||||
|
values.extend(trim.to_list())
|
||||||
|
return values
|
||||||
|
|
||||||
|
@TrimsValue.setter
|
||||||
|
def TrimsValue(self, value: List[float]):
|
||||||
|
self.Trims = [
|
||||||
|
BrepTrim.from_list(value[i : i + 9]) for i in range(0, len(value), 9)
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
BrepEdge.update_forward_refs()
|
||||||
|
BrepLoop.update_forward_refs()
|
||||||
|
BrepTrim.update_forward_refs()
|
||||||
|
BrepFace.update_forward_refs()
|
||||||
@@ -1,12 +1,8 @@
|
|||||||
from typing import Any, List, Optional
|
from typing import List
|
||||||
from deprecated import deprecated
|
|
||||||
|
|
||||||
from specklepy.objects.geometry import Point, Vector
|
from specklepy.objects.geometry import Point, Vector
|
||||||
|
|
||||||
from .base import Base
|
from .base import Base
|
||||||
|
|
||||||
OTHER = "Objects.Other."
|
OTHER = "Objects.Other."
|
||||||
OTHER_REVIT = OTHER + "Revit."
|
|
||||||
|
|
||||||
IDENTITY_TRANSFORM = [
|
IDENTITY_TRANSFORM = [
|
||||||
1.0,
|
1.0,
|
||||||
@@ -28,23 +24,8 @@ IDENTITY_TRANSFORM = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
class Material(Base, speckle_type=OTHER + "Material"):
|
|
||||||
"""Generic class for materials containing generic parameters."""
|
|
||||||
|
|
||||||
name: Optional[str] = None
|
|
||||||
|
|
||||||
|
|
||||||
class RevitMaterial(Material, speckle_type="Objects.Other.Revit." + "RevitMaterial"):
|
|
||||||
materialCategory: Optional[str] = None
|
|
||||||
materialClass: Optional[str] = None
|
|
||||||
shininess: Optional[int] = None
|
|
||||||
smoothness: Optional[int] = None
|
|
||||||
transparency: Optional[int] = None
|
|
||||||
parameters: Optional[Base] = None
|
|
||||||
|
|
||||||
|
|
||||||
class RenderMaterial(Base, speckle_type=OTHER + "RenderMaterial"):
|
class RenderMaterial(Base, speckle_type=OTHER + "RenderMaterial"):
|
||||||
name: Optional[str] = None
|
name: str = None
|
||||||
opacity: float = 1
|
opacity: float = 1
|
||||||
metalness: float = 0
|
metalness: float = 0
|
||||||
roughness: float = 1
|
roughness: float = 1
|
||||||
@@ -52,67 +33,35 @@ class RenderMaterial(Base, speckle_type=OTHER + "RenderMaterial"):
|
|||||||
emissive: int = -16777216 # black arbg
|
emissive: int = -16777216 # black arbg
|
||||||
|
|
||||||
|
|
||||||
class MaterialQuantity(Base, speckle_type=OTHER + "MaterialQuantity"):
|
|
||||||
material: Optional[Material] = None
|
|
||||||
volume: Optional[float] = None
|
|
||||||
area: Optional[float] = None
|
|
||||||
|
|
||||||
|
|
||||||
class DisplayStyle(Base, speckle_type=OTHER + "DisplayStyle"):
|
|
||||||
"""
|
|
||||||
Minimal display style class.
|
|
||||||
Developed primarily for display styles in Rhino and AutoCAD.
|
|
||||||
Rhino object attributes uses OpenNURBS definition for linetypes and lineweights.
|
|
||||||
"""
|
|
||||||
|
|
||||||
name: Optional[str] = None
|
|
||||||
color: int = -2894893 # light gray arbg
|
|
||||||
linetype: Optional[str] = None
|
|
||||||
lineweight: float = 0
|
|
||||||
|
|
||||||
|
|
||||||
class Transform(
|
class Transform(
|
||||||
Base,
|
Base,
|
||||||
speckle_type=OTHER + "Transform",
|
speckle_type=OTHER + "Transform",
|
||||||
serialize_ignore={"translation", "scaling", "is_identity", "value"},
|
serialize_ignore={"translation", "scaling", "is_identity"},
|
||||||
):
|
):
|
||||||
"""The 4x4 transformation matrix
|
"""The 4x4 transformation matrix
|
||||||
|
|
||||||
The 3x3 sub-matrix determines scaling.
|
The 3x3 sub-matrix determines scaling.
|
||||||
The 4th column defines translation,
|
The 4th column defines translation, where the last value is a divisor (usually equal to 1).
|
||||||
where the last value is a divisor (usually equal to 1).
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
_value: Optional[List[float]] = None
|
_value: List[float] = None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@deprecated(version="2.12", reason="Use matrix")
|
|
||||||
def value(self) -> List[float]:
|
def value(self) -> List[float]:
|
||||||
|
"""The transform matrix represented as a flat list of 16 floats"""
|
||||||
return self._value
|
return self._value
|
||||||
|
|
||||||
@value.setter
|
@value.setter
|
||||||
def value(self, value: List[float]) -> None:
|
def value(self, value: List[float]) -> None:
|
||||||
self.matrix = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def matrix(self) -> List[float]:
|
|
||||||
"""The transform matrix represented as a flat list of 16 floats"""
|
|
||||||
return self._value
|
|
||||||
|
|
||||||
@matrix.setter
|
|
||||||
def matrix(self, value: List[float]) -> None:
|
|
||||||
try:
|
try:
|
||||||
value = [float(x) for x in value]
|
value = [float(x) for x in value]
|
||||||
except (ValueError, TypeError) as error:
|
except (ValueError, TypeError):
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Could not create a Transform object with the requested value. Input"
|
f"Could not create a Transform object with the requested value. Input must be a 16 element list of numbers. Value provided: {value}"
|
||||||
f" must be a 16 element list of numbers. Value provided: {value}"
|
)
|
||||||
) from error
|
|
||||||
|
|
||||||
if len(value) != 16:
|
if len(value) != 16:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Could not create a Transform object: input list should be 16 floats"
|
f"Could not create a Transform object: input list should be 16 floats long, but was {len(value)} long"
|
||||||
f" long, but was {len(value)} long"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
self._value = value
|
self._value = value
|
||||||
@@ -129,7 +78,7 @@ class Transform(
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def is_identity(self) -> bool:
|
def is_identity(self) -> bool:
|
||||||
return self._value == IDENTITY_TRANSFORM
|
return self.value == IDENTITY_TRANSFORM
|
||||||
|
|
||||||
def apply_to_point(self, point: Point) -> Point:
|
def apply_to_point(self, point: Point) -> Point:
|
||||||
"""Transform a single speckle Point
|
"""Transform a single speckle Point
|
||||||
@@ -177,16 +126,14 @@ class Transform(
|
|||||||
"""Transform a list of speckle Points
|
"""Transform a list of speckle Points
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
points {List[float]}
|
points {List[float]} -- a flat list of floats representing points to transform
|
||||||
-- a flat list of floats representing points to transform
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
List[float] -- a new transformed list
|
List[float] -- a new transformed list
|
||||||
"""
|
"""
|
||||||
if len(points_value) % 3 != 0:
|
if len(points_value) % 3 != 0:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Cannot apply transform as the points list is malformed: expected"
|
"Cannot apply transform as the points list is malformed: expected length to be multiple of 3"
|
||||||
" length to be multiple of 3"
|
|
||||||
)
|
)
|
||||||
transformed = []
|
transformed = []
|
||||||
for i in range(0, len(points_value), 3):
|
for i in range(0, len(points_value), 3):
|
||||||
@@ -223,13 +170,11 @@ class Transform(
|
|||||||
][:3]
|
][:3]
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_list(cls, value: Optional[List[float]] = None) -> "Transform":
|
def from_list(cls, value: List[float] = None) -> "Transform":
|
||||||
"""Returns a Transform object from a list of 16 numbers.
|
"""Returns a Transform object from a list of 16 numbers. If no value is provided, an identity transform will be returned.
|
||||||
If no value is provided, an identity transform will be returned.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
value {List[float]} -- the matrix as a flat list of 16 numbers
|
value {List[float]} -- the matrix as a flat list of 16 numbers (defaults to the identity transform)
|
||||||
(defaults to the identity transform)
|
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Transform -- a complete transform object
|
Transform -- a complete transform object
|
||||||
@@ -242,57 +187,13 @@ class Transform(
|
|||||||
class BlockDefinition(
|
class BlockDefinition(
|
||||||
Base, speckle_type=OTHER + "BlockDefinition", detachable={"geometry"}
|
Base, speckle_type=OTHER + "BlockDefinition", detachable={"geometry"}
|
||||||
):
|
):
|
||||||
name: Optional[str] = None
|
name: str = None
|
||||||
basePoint: Optional[Point] = None
|
basePoint: Point = None
|
||||||
geometry: Optional[List[Base]] = None
|
geometry: List[Base] = None
|
||||||
|
|
||||||
|
|
||||||
class Instance(
|
|
||||||
Base, speckle_type=OTHER + "Instance", detachable={"definition"}
|
|
||||||
):
|
|
||||||
transform: Optional[Transform] = None
|
|
||||||
definition: Optional[Base] = None
|
|
||||||
|
|
||||||
|
|
||||||
class BlockInstance(
|
class BlockInstance(
|
||||||
Instance, speckle_type=OTHER + "BlockInstance", serialize_ignore={"blockDefinition"}
|
Base, speckle_type=OTHER + "BlockInstance", detachable={"blockDefinition"}
|
||||||
):
|
):
|
||||||
@property
|
blockDefinition: BlockDefinition = None
|
||||||
@deprecated(version="2.13", reason="Use definition")
|
transform: Transform = None
|
||||||
def blockDefinition(self) -> Optional[BlockDefinition]:
|
|
||||||
if isinstance(self.definition, BlockDefinition):
|
|
||||||
return self.definition
|
|
||||||
return None
|
|
||||||
|
|
||||||
@blockDefinition.setter
|
|
||||||
def blockDefinition(self, value: Optional[BlockDefinition]) -> None:
|
|
||||||
self.definition = value
|
|
||||||
|
|
||||||
class RevitInstance(Instance, speckle_type=OTHER_REVIT + "RevitInstance"):
|
|
||||||
level: Optional[Base] = None
|
|
||||||
facingFlipped: bool
|
|
||||||
handFlipped: bool
|
|
||||||
parameters: Optional[Base] = None
|
|
||||||
elementId: Optional[str]
|
|
||||||
|
|
||||||
# TODO: prob move this into a built elements module, but just trialling this for now
|
|
||||||
class RevitParameter(
|
|
||||||
Base, speckle_type="Objects.BuiltElements.Revit.Parameter"
|
|
||||||
):
|
|
||||||
name: Optional[str] = None
|
|
||||||
value: Any = None
|
|
||||||
applicationUnitType: Optional[str] = None # eg UnitType UT_Length
|
|
||||||
applicationUnit: Optional[str] = None # DisplayUnitType eg DUT_MILLIMITERS
|
|
||||||
applicationInternalName: Optional[
|
|
||||||
str
|
|
||||||
] = None # BuiltInParameterName or GUID for shared parameter
|
|
||||||
isShared: bool = False
|
|
||||||
isReadOnly: bool = False
|
|
||||||
isTypeParameter: bool = False
|
|
||||||
|
|
||||||
class Collection(
|
|
||||||
Base, speckle_type="Speckle.Core.Models.Collection", detachable={"elements"}
|
|
||||||
):
|
|
||||||
name: Optional[str] = None
|
|
||||||
collectionType: Optional[str] = None
|
|
||||||
elements: Optional[List[Base]] = None
|
|
||||||
@@ -0,0 +1,57 @@
|
|||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
UNITS = ["mm", "cm", "m", "in", "ft", "yd", "mi"]
|
||||||
|
|
||||||
|
UNITS_STRINGS = {
|
||||||
|
"mm": ["mm", "mil", "millimeters", "millimetres"],
|
||||||
|
"cm": ["cm", "centimetre", "centimeter", "centimetres", "centimeters"],
|
||||||
|
"m": ["m", "meter", "meters", "metre", "metres"],
|
||||||
|
"km": ["km", "kilometer", "kilometre", "kilometers", "kilometres"],
|
||||||
|
"in": ["in", "inch", "inches"],
|
||||||
|
"ft": ["ft", "foot", "feet"],
|
||||||
|
"yd": ["yd", "yard", "yards"],
|
||||||
|
"mi": ["mi", "mile", "miles"],
|
||||||
|
"none": ["none", "null"],
|
||||||
|
}
|
||||||
|
|
||||||
|
UNITS_ENCODINGS = {
|
||||||
|
"none": 0,
|
||||||
|
"mm": 1,
|
||||||
|
"cm": 2,
|
||||||
|
"m": 3,
|
||||||
|
"km": 4,
|
||||||
|
"in": 5,
|
||||||
|
"ft": 6,
|
||||||
|
"yd": 7,
|
||||||
|
"mi": 8,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def get_units_from_string(unit: str):
|
||||||
|
unit = str.lower(unit)
|
||||||
|
for name, alternates in UNITS_STRINGS.items():
|
||||||
|
if unit in alternates:
|
||||||
|
return name
|
||||||
|
|
||||||
|
raise SpeckleException(
|
||||||
|
message=f"Could not understand what unit {unit} is referring to. Please enter a valid unit (eg {UNITS})."
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_units_from_encoding(unit: int):
|
||||||
|
for name, encoding in UNITS_ENCODINGS.items():
|
||||||
|
if unit == encoding:
|
||||||
|
return name
|
||||||
|
|
||||||
|
raise SpeckleException(
|
||||||
|
message=f"Could not understand what unit {unit} is referring to. Please enter a valid unit encoding (eg {UNITS_ENCODINGS})."
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_encoding_from_units(unit: str):
|
||||||
|
try:
|
||||||
|
return UNITS_ENCODINGS[unit]
|
||||||
|
except KeyError:
|
||||||
|
raise SpeckleException(
|
||||||
|
message=f"No encoding exists for unit {unit}. Please enter a valid unit to encode (eg {UNITS_ENCODINGS})."
|
||||||
|
)
|
||||||
+63
-149
@@ -1,17 +1,14 @@
|
|||||||
|
import ujson
|
||||||
import hashlib
|
import hashlib
|
||||||
import re
|
import re
|
||||||
import warnings
|
|
||||||
from enum import Enum
|
|
||||||
from typing import Any, Dict, List, Optional, Tuple
|
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
from warnings import warn
|
from typing import Any, Dict, List, Tuple
|
||||||
|
|
||||||
import ujson
|
|
||||||
|
|
||||||
# import for serialization
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
from specklepy.objects.base import Base, DataChunk
|
from specklepy.objects.base import Base, DataChunk
|
||||||
|
from specklepy.logging.exceptions import SerializationException, SpeckleException
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
import specklepy.objects.geometry
|
||||||
|
import specklepy.objects.other
|
||||||
|
|
||||||
PRIMITIVES = (int, float, str, bool)
|
PRIMITIVES = (int, float, str, bool)
|
||||||
|
|
||||||
@@ -20,83 +17,35 @@ def hash_obj(obj: Any) -> str:
|
|||||||
return hashlib.sha256(ujson.dumps(obj).encode()).hexdigest()[:32]
|
return hashlib.sha256(ujson.dumps(obj).encode()).hexdigest()[:32]
|
||||||
|
|
||||||
|
|
||||||
def safe_json_loads(obj: str, obj_id=None) -> Any:
|
|
||||||
try:
|
|
||||||
return ujson.loads(obj)
|
|
||||||
except ValueError as err:
|
|
||||||
import json
|
|
||||||
|
|
||||||
warn(
|
|
||||||
f"Failed to deserialise object (id: {obj_id}). This is likely a ujson big"
|
|
||||||
f" int error - falling back to json. \nError: {err}",
|
|
||||||
SpeckleWarning,
|
|
||||||
)
|
|
||||||
return json.loads(obj)
|
|
||||||
|
|
||||||
|
|
||||||
class BaseObjectSerializer:
|
class BaseObjectSerializer:
|
||||||
read_transport: AbstractTransport
|
read_transport: AbstractTransport
|
||||||
write_transports: List[AbstractTransport]
|
write_transports: List[AbstractTransport]
|
||||||
detach_lineage: List[bool] # tracks depth and whether or not to detach
|
detach_lineage: List[bool] = [] # tracks depth and whether or not to detach
|
||||||
lineage: List[str] # keeps track of hash chain through the object tree
|
lineage: List[str] = [] # keeps track of hash chain through the object tree
|
||||||
family_tree: Dict[str, Dict[str, int]]
|
family_tree: Dict[str, Dict[str, int]] = {}
|
||||||
closure_table: Dict[str, Dict[str, int]]
|
closure_table: Dict[str, Dict[str, int]] = {}
|
||||||
deserialized: Dict[
|
|
||||||
str, Base
|
|
||||||
] # holds deserialized objects so objects with same id return the same instance
|
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self, write_transports: List[AbstractTransport] = [], read_transport=None
|
||||||
write_transports: Optional[List[AbstractTransport]] = None,
|
|
||||||
read_transport: Optional[AbstractTransport] = None,
|
|
||||||
) -> None:
|
) -> None:
|
||||||
self.write_transports = write_transports or []
|
self.write_transports = write_transports
|
||||||
self.read_transport = read_transport
|
self.read_transport = read_transport
|
||||||
self.detach_lineage = []
|
|
||||||
self.lineage = []
|
|
||||||
self.family_tree = {}
|
|
||||||
self.closure_table = {}
|
|
||||||
self.deserialized = {}
|
|
||||||
|
|
||||||
def write_json(self, base: Base):
|
def write_json(self, base: Base):
|
||||||
"""Serializes a given base object into a json string
|
self.__reset_writer()
|
||||||
Arguments:
|
self.detach_lineage = [True]
|
||||||
base {Base} -- the base object to be decomposed and serialized
|
hash, obj = self.traverse_base(base)
|
||||||
|
return hash, ujson.dumps(obj)
|
||||||
|
|
||||||
Returns:
|
def traverse_base(self, base: Base) -> Tuple[str, Dict]:
|
||||||
(str, str) -- a tuple containing the object id of the base object and
|
|
||||||
the serialized object string
|
|
||||||
"""
|
|
||||||
|
|
||||||
obj_id, obj = self.traverse_base(base)
|
|
||||||
|
|
||||||
return obj_id, ujson.dumps(obj)
|
|
||||||
|
|
||||||
def traverse_base(self, base: Base) -> Tuple[str, Dict[str, Any]]:
|
|
||||||
"""Decomposes the given base object and builds a serializable dictionary
|
"""Decomposes the given base object and builds a serializable dictionary
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
base {Base} -- the base object to be decomposed and serialized
|
base {Base} -- the base object to be decomposed and serialized
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
(str, dict) -- a tuple containing the object id of the base object and
|
(str, dict) -- a tuple containing the hash (id) of the base object and the constructed serializable dictionary
|
||||||
the constructed serializable dictionary
|
|
||||||
"""
|
"""
|
||||||
self.__reset_writer()
|
|
||||||
|
|
||||||
if self.write_transports:
|
|
||||||
for wt in self.write_transports:
|
|
||||||
wt.begin_write()
|
|
||||||
|
|
||||||
obj_id, obj = self._traverse_base(base)
|
|
||||||
|
|
||||||
if self.write_transports:
|
|
||||||
for wt in self.write_transports:
|
|
||||||
wt.end_write()
|
|
||||||
|
|
||||||
return obj_id, obj
|
|
||||||
|
|
||||||
def _traverse_base(self, base: Base) -> Tuple[str, Dict]:
|
|
||||||
if not self.detach_lineage:
|
if not self.detach_lineage:
|
||||||
self.detach_lineage = [True]
|
self.detach_lineage = [True]
|
||||||
|
|
||||||
@@ -111,8 +60,8 @@ class BaseObjectSerializer:
|
|||||||
chunkable = False
|
chunkable = False
|
||||||
detach = False
|
detach = False
|
||||||
|
|
||||||
# skip props marked to be ignored with "__" or "_"
|
# skip nulls or props marked to be ignored with "__" or "_"
|
||||||
if prop.startswith(("__", "_")):
|
if value is None or prop.startswith(("__", "_")):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# don't prepopulate id as this will mess up hashing
|
# don't prepopulate id as this will mess up hashing
|
||||||
@@ -135,22 +84,17 @@ class BaseObjectSerializer:
|
|||||||
prop.startswith("@") or prop in base._detachable or chunkable
|
prop.startswith("@") or prop in base._detachable or chunkable
|
||||||
)
|
)
|
||||||
|
|
||||||
# 1. handle None and primitives (ints, floats, strings, and bools)
|
# 1. handle primitives (ints, floats, strings, and bools)
|
||||||
if value is None or isinstance(value, PRIMITIVES):
|
if isinstance(value, PRIMITIVES):
|
||||||
object_builder[prop] = value
|
object_builder[prop] = value
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# NOTE: for dynamic props, this won't be re-serialised as an enum but as an int
|
|
||||||
if isinstance(value, Enum):
|
|
||||||
object_builder[prop] = value.value
|
|
||||||
continue
|
|
||||||
|
|
||||||
# 2. handle Base objects
|
# 2. handle Base objects
|
||||||
elif isinstance(value, Base):
|
elif isinstance(value, Base):
|
||||||
child_obj = self.traverse_value(value, detach=detach)
|
child_obj = self.traverse_value(value, detach=detach)
|
||||||
if detach and self.write_transports:
|
if detach and self.write_transports:
|
||||||
ref_id = child_obj["id"]
|
ref_hash = child_obj["id"]
|
||||||
object_builder[prop] = self.detach_helper(ref_id=ref_id)
|
object_builder[prop] = self.detach_helper(ref_hash=ref_hash)
|
||||||
else:
|
else:
|
||||||
object_builder[prop] = child_obj
|
object_builder[prop] = child_obj
|
||||||
|
|
||||||
@@ -169,8 +113,8 @@ class BaseObjectSerializer:
|
|||||||
chunk_refs = []
|
chunk_refs = []
|
||||||
for c in chunks:
|
for c in chunks:
|
||||||
self.detach_lineage.append(detach)
|
self.detach_lineage.append(detach)
|
||||||
ref_id, _ = self._traverse_base(c)
|
ref_hash, _ = self.traverse_base(c)
|
||||||
ref_obj = self.detach_helper(ref_id=ref_id)
|
ref_obj = self.detach_helper(ref_hash=ref_hash)
|
||||||
chunk_refs.append(ref_obj)
|
chunk_refs.append(ref_obj)
|
||||||
object_builder[prop] = chunk_refs
|
object_builder[prop] = chunk_refs
|
||||||
|
|
||||||
@@ -189,20 +133,20 @@ class BaseObjectSerializer:
|
|||||||
}
|
}
|
||||||
object_builder["totalChildrenCount"] = len(closure)
|
object_builder["totalChildrenCount"] = len(closure)
|
||||||
|
|
||||||
obj_id = hash_obj(object_builder)
|
hash = hash_obj(object_builder)
|
||||||
|
|
||||||
object_builder["id"] = obj_id
|
object_builder["id"] = hash
|
||||||
if closure:
|
if closure:
|
||||||
object_builder["__closure"] = self.closure_table[obj_id] = closure
|
object_builder["__closure"] = self.closure_table[hash] = closure
|
||||||
|
|
||||||
# write detached or root objects to transports
|
# write detached or root objects to transports
|
||||||
if detached and self.write_transports:
|
if detached and self.write_transports:
|
||||||
for t in self.write_transports:
|
for t in self.write_transports:
|
||||||
t.save_object(id=obj_id, serialized_object=ujson.dumps(object_builder))
|
t.save_object(id=hash, serialized_object=ujson.dumps(object_builder))
|
||||||
|
|
||||||
del self.lineage[-1]
|
del self.lineage[-1]
|
||||||
|
|
||||||
return obj_id, object_builder
|
return hash, object_builder
|
||||||
|
|
||||||
def traverse_value(self, obj: Any, detach: bool = False) -> Any:
|
def traverse_value(self, obj: Any, detach: bool = False) -> Any:
|
||||||
"""Decomposes a given object and constructs a serializable object or dictionary
|
"""Decomposes a given object and constructs a serializable object or dictionary
|
||||||
@@ -213,15 +157,9 @@ class BaseObjectSerializer:
|
|||||||
Returns:
|
Returns:
|
||||||
Any -- a serializable version of the given object
|
Any -- a serializable version of the given object
|
||||||
"""
|
"""
|
||||||
if obj is None:
|
|
||||||
return None
|
|
||||||
if isinstance(obj, PRIMITIVES):
|
if isinstance(obj, PRIMITIVES):
|
||||||
return obj
|
return obj
|
||||||
|
|
||||||
# NOTE: for dynamic props, this won't be re-serialised as an enum but as an int
|
|
||||||
if isinstance(obj, Enum):
|
|
||||||
return obj.value
|
|
||||||
|
|
||||||
elif isinstance(obj, (list, tuple, set)):
|
elif isinstance(obj, (list, tuple, set)):
|
||||||
if not detach:
|
if not detach:
|
||||||
return [self.traverse_value(o) for o in obj]
|
return [self.traverse_value(o) for o in obj]
|
||||||
@@ -230,15 +168,15 @@ class BaseObjectSerializer:
|
|||||||
for o in obj:
|
for o in obj:
|
||||||
if isinstance(o, Base):
|
if isinstance(o, Base):
|
||||||
self.detach_lineage.append(detach)
|
self.detach_lineage.append(detach)
|
||||||
ref_id, _ = self._traverse_base(o)
|
hash, _ = self.traverse_base(o)
|
||||||
detached_list.append(self.detach_helper(ref_id=ref_id))
|
detached_list.append(self.detach_helper(ref_hash=hash))
|
||||||
else:
|
else:
|
||||||
detached_list.append(self.traverse_value(o, detach))
|
detached_list.append(self.traverse_value(o, detach))
|
||||||
return detached_list
|
return detached_list
|
||||||
|
|
||||||
elif isinstance(obj, dict):
|
elif isinstance(obj, dict):
|
||||||
for k, v in obj.items():
|
for k, v in obj.items():
|
||||||
if isinstance(v, PRIMITIVES) or v is None:
|
if isinstance(v, PRIMITIVES):
|
||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
obj[k] = self.traverse_value(v)
|
obj[k] = self.traverse_value(v)
|
||||||
@@ -246,28 +184,24 @@ class BaseObjectSerializer:
|
|||||||
|
|
||||||
elif isinstance(obj, Base):
|
elif isinstance(obj, Base):
|
||||||
self.detach_lineage.append(detach)
|
self.detach_lineage.append(detach)
|
||||||
_, base_obj = self._traverse_base(obj)
|
_, base_obj = self.traverse_base(obj)
|
||||||
return base_obj
|
return base_obj
|
||||||
|
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
return obj.dict()
|
return obj.dict()
|
||||||
except Exception:
|
except:
|
||||||
warn(
|
SerializationException(
|
||||||
f"Failed to handle {type(obj)} in"
|
message=f"Failed to handle {type(obj)} in `BaseObjectSerializer.traverse_value`",
|
||||||
" `BaseObjectSerializer.traverse_value`",
|
object=obj,
|
||||||
SpeckleWarning,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return str(obj)
|
return str(obj)
|
||||||
|
|
||||||
def detach_helper(self, ref_id: str) -> Dict[str, str]:
|
def detach_helper(self, ref_hash: str) -> Dict[str, str]:
|
||||||
"""
|
"""Helper to keep track of detached objects and their depth in the family tree and create reference objects to place in the parent object
|
||||||
Helper to keep track of detached objects and their depth in the family tree
|
|
||||||
and create reference objects to place in the parent object
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
ref_id {str} -- the id of the fully traversed object
|
ref_hash {str} -- the hash of the fully traversed object
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
dict -- a reference object to be inserted into the given object's parent
|
dict -- a reference object to be inserted into the given object's parent
|
||||||
@@ -276,22 +210,19 @@ class BaseObjectSerializer:
|
|||||||
for parent in self.lineage:
|
for parent in self.lineage:
|
||||||
if parent not in self.family_tree:
|
if parent not in self.family_tree:
|
||||||
self.family_tree[parent] = {}
|
self.family_tree[parent] = {}
|
||||||
if ref_id not in self.family_tree[parent] or self.family_tree[parent][
|
if ref_hash not in self.family_tree[parent] or self.family_tree[parent][
|
||||||
ref_id
|
ref_hash
|
||||||
] > len(self.detach_lineage):
|
] > len(self.detach_lineage):
|
||||||
self.family_tree[parent][ref_id] = len(self.detach_lineage)
|
self.family_tree[parent][ref_hash] = len(self.detach_lineage)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"referencedId": ref_id,
|
"referencedId": ref_hash,
|
||||||
"speckle_type": "reference",
|
"speckle_type": "reference",
|
||||||
}
|
}
|
||||||
|
|
||||||
def __reset_writer(self) -> None:
|
def __reset_writer(self) -> None:
|
||||||
"""
|
"""Reinitializes the lineage, and other variables that get used during the json writing process"""
|
||||||
Reinitializes the lineage, and other variables that get used during the json
|
self.detach_lineage = []
|
||||||
writing process
|
|
||||||
"""
|
|
||||||
self.detach_lineage = [True]
|
|
||||||
self.lineage = []
|
self.lineage = []
|
||||||
self.family_tree = {}
|
self.family_tree = {}
|
||||||
self.closure_table = {}
|
self.closure_table = {}
|
||||||
@@ -307,9 +238,7 @@ class BaseObjectSerializer:
|
|||||||
"""
|
"""
|
||||||
if not obj_string:
|
if not obj_string:
|
||||||
return None
|
return None
|
||||||
|
obj = ujson.loads(obj_string)
|
||||||
self.deserialized = {}
|
|
||||||
obj = safe_json_loads(obj_string)
|
|
||||||
return self.recompose_base(obj=obj)
|
return self.recompose_base(obj=obj)
|
||||||
|
|
||||||
def recompose_base(self, obj: dict) -> Base:
|
def recompose_base(self, obj: dict) -> Base:
|
||||||
@@ -325,10 +254,7 @@ class BaseObjectSerializer:
|
|||||||
if not obj:
|
if not obj:
|
||||||
return
|
return
|
||||||
if isinstance(obj, str):
|
if isinstance(obj, str):
|
||||||
obj = safe_json_loads(obj)
|
obj = ujson.loads(obj)
|
||||||
|
|
||||||
if "id" in obj and obj["id"] in self.deserialized:
|
|
||||||
return self.deserialized[obj["id"]]
|
|
||||||
|
|
||||||
if "speckle_type" in obj and obj["speckle_type"] == "reference":
|
if "speckle_type" in obj and obj["speckle_type"] == "reference":
|
||||||
obj = self.get_child(obj=obj)
|
obj = self.get_child(obj=obj)
|
||||||
@@ -360,31 +286,23 @@ class BaseObjectSerializer:
|
|||||||
|
|
||||||
# 2. handle referenced child objects
|
# 2. handle referenced child objects
|
||||||
elif "referencedId" in value:
|
elif "referencedId" in value:
|
||||||
ref_id = value["referencedId"]
|
ref_hash = value["referencedId"]
|
||||||
ref_obj_str = self.read_transport.get_object(id=ref_id)
|
ref_obj_str = self.read_transport.get_object(id=ref_hash)
|
||||||
if ref_obj_str:
|
if not ref_obj_str:
|
||||||
ref_obj = safe_json_loads(ref_obj_str, ref_id)
|
raise SpeckleException(
|
||||||
base.__setattr__(prop, self.recompose_base(obj=ref_obj))
|
f"Could not find the referenced child object of id `{ref_hash}` in the given read transport: {self.read_transport.name}"
|
||||||
else:
|
|
||||||
warnings.warn(
|
|
||||||
f"Could not find the referenced child object of id `{ref_id}`"
|
|
||||||
f" in the given read transport: {self.read_transport.name}",
|
|
||||||
SpeckleWarning,
|
|
||||||
)
|
)
|
||||||
base.__setattr__(prop, self.handle_value(value))
|
ref_obj = ujson.loads(ref_obj_str)
|
||||||
|
base.__setattr__(prop, self.recompose_base(obj=ref_obj))
|
||||||
|
|
||||||
# 3. handle all other cases (base objects, lists, and dicts)
|
# 3. handle all other cases (base objects, lists, and dicts)
|
||||||
else:
|
else:
|
||||||
base.__setattr__(prop, self.handle_value(value))
|
base.__setattr__(prop, self.handle_value(value))
|
||||||
|
|
||||||
if "id" in obj:
|
|
||||||
self.deserialized[obj["id"]] = base
|
|
||||||
|
|
||||||
return base
|
return base
|
||||||
|
|
||||||
def handle_value(self, obj: Any):
|
def handle_value(self, obj: Any):
|
||||||
"""Helper for recomposing a base object by handling the dictionary
|
"""Helper for recomposing a base object by handling the dictionary representation's values
|
||||||
representation's values
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
obj {Any} -- a value from the base object dictionary
|
obj {Any} -- a value from the base object dictionary
|
||||||
@@ -426,14 +344,10 @@ class BaseObjectSerializer:
|
|||||||
return obj
|
return obj
|
||||||
|
|
||||||
def get_child(self, obj: Dict):
|
def get_child(self, obj: Dict):
|
||||||
ref_id = obj["referencedId"]
|
ref_hash = obj["referencedId"]
|
||||||
ref_obj_str = self.read_transport.get_object(id=ref_id)
|
ref_obj_str = self.read_transport.get_object(id=ref_hash)
|
||||||
if not ref_obj_str:
|
if not ref_obj_str:
|
||||||
warnings.warn(
|
raise SpeckleException(
|
||||||
f"Could not find the referenced child object of id `{ref_id}` in the"
|
f"Could not find the referenced child object of id `{ref_hash}` in the given read transport: {self.read_transport.name}"
|
||||||
f" given read transport: {self.read_transport.name}",
|
|
||||||
SpeckleWarning,
|
|
||||||
)
|
)
|
||||||
return obj
|
return ujson.loads(ref_obj_str)
|
||||||
|
|
||||||
return safe_json_loads(ref_obj_str, ref_id)
|
|
||||||
+25
-15
@@ -1,12 +1,24 @@
|
|||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
from typing import Dict, List, Optional
|
from typing import Any, Optional, List, Dict
|
||||||
|
from pydantic import BaseModel
|
||||||
|
from pydantic.main import Extra
|
||||||
|
|
||||||
|
# __________________
|
||||||
|
# | |
|
||||||
|
# | this is v wip |
|
||||||
|
# | pls be careful |
|
||||||
|
# |__________________|
|
||||||
|
# (\__/) ||
|
||||||
|
# (•ㅅ•) ||
|
||||||
|
# / づ
|
||||||
|
|
||||||
|
|
||||||
class AbstractTransport(ABC):
|
class AbstractTransport(ABC, BaseModel):
|
||||||
|
_name: str = "Abstract"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@abstractmethod
|
|
||||||
def name(self):
|
def name(self):
|
||||||
pass
|
return type(self)._name
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def begin_write(self) -> None:
|
def begin_write(self) -> None:
|
||||||
@@ -15,9 +27,7 @@ class AbstractTransport(ABC):
|
|||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def end_write(self) -> None:
|
def end_write(self) -> None:
|
||||||
"""
|
"""Optional: signals to the transport that no more items will need to be written."""
|
||||||
Optional: signals to the transport that no more items will need to be written.
|
|
||||||
"""
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
@@ -38,8 +48,7 @@ class AbstractTransport(ABC):
|
|||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
id {str} -- the hash of the object
|
id {str} -- the hash of the object
|
||||||
source_transport {AbstractTransport)
|
source_transport {AbstractTransport) -- the transport through which the object can be found
|
||||||
-- the transport through which the object can be found
|
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -51,8 +60,7 @@ class AbstractTransport(ABC):
|
|||||||
id {str} -- the hash of the object
|
id {str} -- the hash of the object
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the full string representation
|
str -- the full string representation of the object (or null if no object is found)
|
||||||
of the object (or null if no object is found)
|
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -64,8 +72,7 @@ class AbstractTransport(ABC):
|
|||||||
id_list -- List of object id to be checked
|
id_list -- List of object id to be checked
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Dict[str, bool] -- keys: input ids, values:
|
Dict[str, bool] -- keys: input ids, values: whether the transport has that object
|
||||||
whether the transport has that object
|
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -77,9 +84,12 @@ class AbstractTransport(ABC):
|
|||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
id {str} -- the id of the object you want to copy
|
id {str} -- the id of the object you want to copy
|
||||||
target_transport {AbstractTransport}
|
target_transport {AbstractTransport} -- the transport you want to copy the object to
|
||||||
-- the transport you want to copy the object to
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the string representation of the root object
|
str -- the string representation of the root object
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class Config:
|
||||||
|
extra = Extra.allow
|
||||||
|
arbitrary_types_allowed = True
|
||||||
@@ -1,18 +1,18 @@
|
|||||||
from typing import Any, Dict, List
|
import json
|
||||||
|
from typing import Any, List, Dict
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
|
||||||
|
|
||||||
class MemoryTransport(AbstractTransport):
|
class MemoryTransport(AbstractTransport):
|
||||||
def __init__(self, name="Memory") -> None:
|
_name: str = "Memory"
|
||||||
super().__init__()
|
objects: dict = {}
|
||||||
self._name = name
|
saved_object_count: int = 0
|
||||||
self.objects = {}
|
|
||||||
self.saved_object_count = 0
|
|
||||||
|
|
||||||
@property
|
def __init__(self, name=None, **data: Any) -> None:
|
||||||
def name(self) -> str:
|
super().__init__(**data)
|
||||||
return self._name
|
if name:
|
||||||
|
self._name = name
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
def __repr__(self) -> str:
|
||||||
return f"MemoryTransport(objects: {len(self.objects)})"
|
return f"MemoryTransport(objects: {len(self.objects)})"
|
||||||
@@ -28,7 +28,10 @@ class MemoryTransport(AbstractTransport):
|
|||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def get_object(self, id: str) -> str or None:
|
def get_object(self, id: str) -> str or None:
|
||||||
return self.objects[id] if id in self.objects else None
|
if id in self.objects:
|
||||||
|
return self.objects[id]
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
def has_objects(self, id_list: List[str]) -> Dict[str, bool]:
|
def has_objects(self, id_list: List[str]) -> Dict[str, bool]:
|
||||||
return {id: (id in self.objects) for id in id_list}
|
return {id: (id in self.objects) for id in id_list}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
from .server import ServerTransport
|
||||||
+10
-28
@@ -1,11 +1,11 @@
|
|||||||
import gzip
|
|
||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
import queue
|
|
||||||
import threading
|
import threading
|
||||||
|
import queue
|
||||||
|
import time
|
||||||
|
import gzip
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
@@ -18,7 +18,6 @@ class BatchSender(object):
|
|||||||
stream_id,
|
stream_id,
|
||||||
token,
|
token,
|
||||||
max_batch_size_mb=1,
|
max_batch_size_mb=1,
|
||||||
max_batch_length=20000,
|
|
||||||
batch_buffer_length=10,
|
batch_buffer_length=10,
|
||||||
thread_count=4,
|
thread_count=4,
|
||||||
):
|
):
|
||||||
@@ -27,7 +26,6 @@ class BatchSender(object):
|
|||||||
self._token = token
|
self._token = token
|
||||||
|
|
||||||
self.max_size = int(max_batch_size_mb * 1000 * 1000)
|
self.max_size = int(max_batch_size_mb * 1000 * 1000)
|
||||||
self.max_batch_length = int(max_batch_length)
|
|
||||||
self._batches = queue.Queue(batch_buffer_length)
|
self._batches = queue.Queue(batch_buffer_length)
|
||||||
self._crt_batch = []
|
self._crt_batch = []
|
||||||
self._crt_batch_size = 0
|
self._crt_batch_size = 0
|
||||||
@@ -41,11 +39,7 @@ class BatchSender(object):
|
|||||||
self._create_threads()
|
self._create_threads()
|
||||||
|
|
||||||
crt_obj_size = len(obj)
|
crt_obj_size = len(obj)
|
||||||
crt_batch_length = len(self._crt_batch)
|
if not self._crt_batch or self._crt_batch_size + crt_obj_size < self.max_size:
|
||||||
if not self._crt_batch or (
|
|
||||||
self._crt_batch_size + crt_obj_size < self.max_size
|
|
||||||
and crt_batch_length < self.max_batch_length
|
|
||||||
):
|
|
||||||
self._crt_batch.append((id, obj))
|
self._crt_batch.append((id, obj))
|
||||||
self._crt_batch_size += crt_obj_size
|
self._crt_batch_size += crt_obj_size
|
||||||
return
|
return
|
||||||
@@ -96,18 +90,12 @@ class BatchSender(object):
|
|||||||
self._exception = self._exception or ex
|
self._exception = self._exception or ex
|
||||||
LOG.error("ServerTransport sending thread error: " + str(ex))
|
LOG.error("ServerTransport sending thread error: " + str(ex))
|
||||||
|
|
||||||
def _bg_send_batch(self, session: requests.Session, batch):
|
def _bg_send_batch(self, session, batch):
|
||||||
object_ids = [obj[0] for obj in batch]
|
object_ids = [obj[0] for obj in batch]
|
||||||
response = session.post(
|
server_has_object = session.post(
|
||||||
url=f"{self.server_url}/api/diff/{self.stream_id}",
|
url=f"{self.server_url}/api/diff/{self.stream_id}",
|
||||||
data={"objects": json.dumps(object_ids)},
|
data={"objects": json.dumps(object_ids)},
|
||||||
)
|
).json()
|
||||||
if response.status_code == 403:
|
|
||||||
raise SpeckleException(
|
|
||||||
f"Invalid credentials - cannot send objects to server {self.server_url}"
|
|
||||||
)
|
|
||||||
response.raise_for_status()
|
|
||||||
server_has_object = response.json()
|
|
||||||
|
|
||||||
new_object_ids = [x for x in object_ids if not server_has_object[x]]
|
new_object_ids = [x for x in object_ids if not server_has_object[x]]
|
||||||
new_object_ids = set(new_object_ids)
|
new_object_ids = set(new_object_ids)
|
||||||
@@ -115,8 +103,7 @@ class BatchSender(object):
|
|||||||
|
|
||||||
if not new_objects:
|
if not new_objects:
|
||||||
LOG.info(
|
LOG.info(
|
||||||
f"Uploading batch of {len(batch)} objects: all objects are already in"
|
f"Uploading batch of {len(batch)} objects: all objects are already in the server"
|
||||||
" the server"
|
|
||||||
)
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
@@ -135,16 +122,11 @@ class BatchSender(object):
|
|||||||
if r.status_code != 201:
|
if r.status_code != 201:
|
||||||
LOG.warning("Upload server response: %s", r.text)
|
LOG.warning("Upload server response: %s", r.text)
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
message=(
|
message=f"Could not save the object to the server - status code {r.status_code}"
|
||||||
"Could not save the object to the server - status code"
|
|
||||||
f" {r.status_code} ({r.text[:1000]})"
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
except json.JSONDecodeError as error:
|
except json.JSONDecodeError as error:
|
||||||
return SpeckleException(
|
return SpeckleException(
|
||||||
f"Failed to send objects to {self.server_url}. Please ensure this"
|
f"Failed to send objects to {self.server_url}. Please ensure this stream ({self.stream_id}) exists on this server and that you have permission to send to it.",
|
||||||
f" stream ({self.stream_id}) exists on this server and that you have"
|
|
||||||
" permission to send to it.",
|
|
||||||
error,
|
error,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
import json
|
import json
|
||||||
from typing import Any, Dict, List, Optional
|
import time
|
||||||
from warnings import warn
|
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
from specklepy.core.api.client import SpeckleClient
|
from typing import Any, Dict, List, Type
|
||||||
from specklepy.core.api.credentials import Account, get_account_from_token
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
|
||||||
from .batch_sender import BatchSender
|
from .batch_sender import BatchSender
|
||||||
@@ -14,13 +14,12 @@ from .batch_sender import BatchSender
|
|||||||
|
|
||||||
class ServerTransport(AbstractTransport):
|
class ServerTransport(AbstractTransport):
|
||||||
"""
|
"""
|
||||||
The `ServerTransport` is the vehicle through which you transport objects to and
|
The `ServerTransport` is the vehicle through which you transport objects to and from a Speckle Server. Provide it to
|
||||||
from a Speckle Server. Provide it to `operations.send()` or `operations.receive()`.
|
`operations.send()` or `operations.receive()`.
|
||||||
|
|
||||||
The `ServerTransport` can be authenticated two different ways:
|
The `ServerTransport` can be authenticted two different ways:
|
||||||
1. by providing a `SpeckleClient`
|
1. by providing a `SpeckleClient`
|
||||||
2. by providing an `Account`
|
2. by providing a `token` and `url`
|
||||||
3. by providing a `token` and `url`
|
|
||||||
|
|
||||||
```py
|
```py
|
||||||
from specklepy.api import operations
|
from specklepy.api import operations
|
||||||
@@ -29,15 +28,14 @@ class ServerTransport(AbstractTransport):
|
|||||||
# here's the data you want to send
|
# here's the data you want to send
|
||||||
block = Block(length=2, height=4)
|
block = Block(length=2, height=4)
|
||||||
|
|
||||||
# next create the server transport - this is the vehicle through which
|
# next create the server transport - this is the vehicle through which you will send and receive
|
||||||
# you will send and receive
|
|
||||||
transport = ServerTransport(stream_id=new_stream_id, client=client)
|
transport = ServerTransport(stream_id=new_stream_id, client=client)
|
||||||
|
|
||||||
# this serialises the block and sends it to the transport
|
# this serialises the block and sends it to the transport
|
||||||
hash = operations.send(base=block, transports=[transport])
|
hash = operations.send(base=block, transports=[transport])
|
||||||
|
|
||||||
# you can now create a commit on your stream with this object
|
# you can now create a commit on your stream with this object
|
||||||
commit_id = client.commit.create(
|
commid_id = client.commit.create(
|
||||||
stream_id=new_stream_id,
|
stream_id=new_stream_id,
|
||||||
obj_id=hash,
|
obj_id=hash,
|
||||||
message="this is a block I made in speckle-py",
|
message="this is a block I made in speckle-py",
|
||||||
@@ -45,58 +43,47 @@ class ServerTransport(AbstractTransport):
|
|||||||
```
|
```
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
_name = "RemoteTransport"
|
||||||
|
url: str = None
|
||||||
|
stream_id: str = None
|
||||||
|
saved_obj_count: int = 0
|
||||||
|
session: requests.Session = None
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
client: Optional[SpeckleClient] = None,
|
client: SpeckleClient = None,
|
||||||
account: Optional[Account] = None,
|
token: str = None,
|
||||||
token: Optional[str] = None,
|
url: str = None,
|
||||||
url: Optional[str] = None,
|
**data: Any,
|
||||||
name: str = "RemoteTransport",
|
|
||||||
) -> None:
|
) -> None:
|
||||||
super().__init__()
|
super().__init__(**data)
|
||||||
if client is None and account is None and token is None and url is None:
|
# TODO: replace client with account or some other auth avenue
|
||||||
|
if client is None and token is None and url is None:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"You must provide either a client or a token and url to construct a"
|
"You must provide either a client or a token and url to construct a ServerTransport."
|
||||||
" ServerTransport."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
self._name = name
|
if client:
|
||||||
self.account = None
|
if not client.me:
|
||||||
self.saved_obj_count = 0
|
raise SpeckleException(
|
||||||
if account:
|
"The provided SpeckleClient was not authenticated."
|
||||||
self.account = account
|
|
||||||
url = account.serverInfo.url
|
|
||||||
elif client:
|
|
||||||
url = client.url
|
|
||||||
if not client.account.token:
|
|
||||||
warn(
|
|
||||||
SpeckleWarning(
|
|
||||||
"Unauthenticated Speckle Client provided to Server Transport"
|
|
||||||
f" for {self.url}. Receiving from private streams will fail."
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
else:
|
token = client.me["token"]
|
||||||
self.account = client.account
|
url = client.url
|
||||||
else:
|
|
||||||
self.account = get_account_from_token(token, url)
|
|
||||||
|
|
||||||
self.stream_id = stream_id
|
self.stream_id = stream_id
|
||||||
self.url = url
|
self.url = url
|
||||||
|
|
||||||
self._batch_sender = BatchSender(
|
self._batch_sender = BatchSender(
|
||||||
self.url, self.stream_id, self.account.token, max_batch_size_mb=1
|
self.url, self.stream_id, token, max_batch_size_mb=1
|
||||||
)
|
)
|
||||||
|
|
||||||
self.session = requests.Session()
|
self.session = requests.Session()
|
||||||
self.session.headers.update(
|
self.session.headers.update(
|
||||||
{"Authorization": f"Bearer {self.account.token}", "Accept": "text/plain"}
|
{"Authorization": f"Bearer {token}", "Accept": "text/plain"}
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self) -> str:
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
def begin_write(self) -> None:
|
def begin_write(self) -> None:
|
||||||
self.saved_obj_count = 0
|
self.saved_obj_count = 0
|
||||||
|
|
||||||
@@ -121,10 +108,8 @@ class ServerTransport(AbstractTransport):
|
|||||||
# return obj
|
# return obj
|
||||||
|
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"Getting a single object using `ServerTransport.get_object()` is not"
|
"Getting a single object using `ServerTransport.get_object()` is not implemented. To get an object from the server, please use the `SpeckleClient.object.get()` route",
|
||||||
" implemented. To get an object from the server, please use the"
|
NotImplementedError,
|
||||||
" `SpeckleClient.object.get()` route",
|
|
||||||
NotImplementedError(),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def has_objects(self, id_list: List[str]) -> Dict[str, bool]:
|
def has_objects(self, id_list: List[str]) -> Dict[str, bool]:
|
||||||
@@ -135,12 +120,12 @@ class ServerTransport(AbstractTransport):
|
|||||||
) -> str:
|
) -> str:
|
||||||
endpoint = f"{self.url}/objects/{self.stream_id}/{id}/single"
|
endpoint = f"{self.url}/objects/{self.stream_id}/{id}/single"
|
||||||
r = self.session.get(endpoint)
|
r = self.session.get(endpoint)
|
||||||
r.encoding = "utf-8"
|
if r.encoding is None:
|
||||||
|
r.encoding = "utf-8"
|
||||||
|
|
||||||
if r.status_code != 200:
|
if r.status_code != 200:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
f"Can't get object {self.stream_id}/{id}: HTTP error"
|
f"Can't get object {self.stream_id}/{id}: HTTP error {r.status_code} ({r.text[:1000]})"
|
||||||
f" {r.status_code} ({r.text[:1000]})"
|
|
||||||
)
|
)
|
||||||
root_obj_serialized = r.text
|
root_obj_serialized = r.text
|
||||||
root_obj = json.loads(root_obj_serialized)
|
root_obj = json.loads(root_obj_serialized)
|
||||||
@@ -158,17 +143,34 @@ class ServerTransport(AbstractTransport):
|
|||||||
r = self.session.post(
|
r = self.session.post(
|
||||||
endpoint, data={"objects": json.dumps(new_children_ids)}, stream=True
|
endpoint, data={"objects": json.dumps(new_children_ids)}, stream=True
|
||||||
)
|
)
|
||||||
r.encoding = "utf-8"
|
if r.encoding is None:
|
||||||
|
r.encoding = "utf-8"
|
||||||
lines = r.iter_lines(decode_unicode=True)
|
lines = r.iter_lines(decode_unicode=True)
|
||||||
|
|
||||||
# iter through returned objects saving them as we go
|
# iter through returned objects saving them as we go
|
||||||
target_transport.begin_write()
|
|
||||||
for line in lines:
|
for line in lines:
|
||||||
if line:
|
if line:
|
||||||
hash, obj = line.split("\t")
|
hash, obj = line.split("\t")
|
||||||
target_transport.save_object(hash, obj)
|
target_transport.save_object(hash, obj)
|
||||||
|
|
||||||
target_transport.save_object(id, root_obj_serialized)
|
target_transport.save_object(id, root_obj_serialized)
|
||||||
target_transport.end_write()
|
|
||||||
|
|
||||||
return root_obj_serialized
|
return root_obj_serialized
|
||||||
|
|
||||||
|
# async def stream_res(self, endpoint: str) -> str:
|
||||||
|
# data = b""
|
||||||
|
# async with aiohttp.ClientSession() as session:
|
||||||
|
# session.headers.update(
|
||||||
|
# {
|
||||||
|
# "Authorization": f"{self.session.headers['Authorization']}",
|
||||||
|
# "Accept": "text/plain",
|
||||||
|
# }
|
||||||
|
# )
|
||||||
|
# async with session.get(endpoint) as res:
|
||||||
|
# while True:
|
||||||
|
# chunk = await res.content.read(self.chunk_size)
|
||||||
|
# if not chunk:
|
||||||
|
# break
|
||||||
|
# data += chunk
|
||||||
|
|
||||||
|
# return data.decode("utf-8")
|
||||||
@@ -1,31 +1,38 @@
|
|||||||
import os
|
import os
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
import sched
|
||||||
import sqlite3
|
import sqlite3
|
||||||
|
from typing import Any, List, Dict
|
||||||
|
from appdirs import user_data_dir
|
||||||
from contextlib import closing
|
from contextlib import closing
|
||||||
from typing import Any, Dict, List, Optional, Tuple
|
|
||||||
|
|
||||||
from specklepy.core.helpers import speckle_path_provider
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
|
||||||
class SQLiteTransport(AbstractTransport):
|
class SQLiteTransport(AbstractTransport):
|
||||||
|
_name = "SQLite"
|
||||||
|
_base_path: str = None
|
||||||
|
_root_path: str = None
|
||||||
|
_is_writing: bool = False
|
||||||
|
_scheduler = sched.scheduler(time.time, time.sleep)
|
||||||
|
_polling_interval = 0.5 # seconds
|
||||||
|
__connection: sqlite3.Connection = None
|
||||||
|
app_name: str = ""
|
||||||
|
scope: str = ""
|
||||||
|
saved_obj_count: int = 0
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
base_path: Optional[str] = None,
|
base_path: str = None,
|
||||||
app_name: Optional[str] = None,
|
app_name: str = None,
|
||||||
scope: Optional[str] = None,
|
scope: str = None,
|
||||||
max_batch_size_mb: float = 10.0,
|
**data: Any,
|
||||||
name: str = "SQLite",
|
|
||||||
) -> None:
|
) -> None:
|
||||||
super().__init__()
|
super().__init__(**data)
|
||||||
self._name = name
|
|
||||||
self.app_name = app_name or "Speckle"
|
self.app_name = app_name or "Speckle"
|
||||||
self.scope = scope or "Objects"
|
self.scope = scope or "Objects"
|
||||||
self._base_path = base_path or self.get_base_path(self.app_name)
|
self._base_path = base_path or self.get_base_path(self.app_name)
|
||||||
self.max_size = int(max_batch_size_mb * 1000 * 1000)
|
|
||||||
self.saved_obj_count = 0
|
|
||||||
self._current_batch: List[Tuple[str, str]] = []
|
|
||||||
self._current_batch_size = 0
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
os.makedirs(self._base_path, exist_ok=True)
|
os.makedirs(self._base_path, exist_ok=True)
|
||||||
@@ -36,24 +43,66 @@ class SQLiteTransport(AbstractTransport):
|
|||||||
self.__initialise()
|
self.__initialise()
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
f"SQLiteTransport could not initialise {self.scope}.db at"
|
f"SQLiteTransport could not initialise {self.scope}.db at {self._base_path}. Either provide a different `base_path` or use an alternative transport.",
|
||||||
f" {self._base_path}. Either provide a different `base_path` or use an"
|
|
||||||
" alternative transport.",
|
|
||||||
ex,
|
ex,
|
||||||
)
|
)
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
def __repr__(self) -> str:
|
||||||
return f"SQLiteTransport(app: '{self.app_name}', scope: '{self.scope}')"
|
return f"SQLiteTransport(app: '{self.app_name}', scope: '{self.scope}')"
|
||||||
|
|
||||||
@property
|
# def __write_timer_elapsed(self):
|
||||||
def name(self) -> str:
|
# print("WRITE TIMER ELAPSED")
|
||||||
return self._name
|
# proc = Process(target=_run_queue, args=(self.__queue, self._root_path))
|
||||||
|
# proc.start()
|
||||||
|
# proc.join()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_base_path(app_name):
|
def get_base_path(app_name):
|
||||||
return str(
|
# from appdirs https://github.com/ActiveState/appdirs/blob/master/appdirs.py
|
||||||
speckle_path_provider.user_application_data_path().joinpath(app_name)
|
# default mac path is not the one we use (we use unix path), so using special case for this
|
||||||
)
|
system = sys.platform
|
||||||
|
if system.startswith("java"):
|
||||||
|
import platform
|
||||||
|
|
||||||
|
os_name = platform.java_ver()[3][0]
|
||||||
|
if os_name.startswith("Mac"):
|
||||||
|
system = "darwin"
|
||||||
|
|
||||||
|
if system != "darwin":
|
||||||
|
return user_data_dir(appname=app_name, appauthor=False, roaming=True)
|
||||||
|
|
||||||
|
path = os.path.expanduser("~/.config/")
|
||||||
|
return os.path.join(path, app_name)
|
||||||
|
|
||||||
|
# def __consume_queue(self):
|
||||||
|
# if self._is_writing or self.__queue.empty():
|
||||||
|
# return
|
||||||
|
# print("CONSUME QUEUE")
|
||||||
|
# self._is_writing = True
|
||||||
|
# while not self.__queue.empty():
|
||||||
|
# data = self.__queue.get()
|
||||||
|
# self.save_object(data[0], data[1])
|
||||||
|
# self._is_writing = False
|
||||||
|
|
||||||
|
# self._scheduler.enter(
|
||||||
|
# delay=self._polling_interval, priority=1, action=self.__consume_queue
|
||||||
|
# )
|
||||||
|
# self._scheduler.run(blocking=True)
|
||||||
|
|
||||||
|
# def save_object(self, id: str, serialized_object: str) -> None:
|
||||||
|
# """Adds an object to the queue and schedules it to be saved.
|
||||||
|
|
||||||
|
# Arguments:
|
||||||
|
# id {str} -- the object id
|
||||||
|
# serialized_object {str} -- the full string representation of the object
|
||||||
|
# """
|
||||||
|
# print("SAVE OBJECT")
|
||||||
|
# self.__queue.put((id, serialized_object))
|
||||||
|
|
||||||
|
# self._scheduler.enter(
|
||||||
|
# delay=self._polling_interval, priority=1, action=self.__consume_queue
|
||||||
|
# )
|
||||||
|
# self._scheduler.run(blocking=True)
|
||||||
|
|
||||||
def save_object_from_transport(
|
def save_object_from_transport(
|
||||||
self, id: str, source_transport: AbstractTransport
|
self, id: str, source_transport: AbstractTransport
|
||||||
@@ -62,50 +111,29 @@ class SQLiteTransport(AbstractTransport):
|
|||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
id {str} -- the object id
|
id {str} -- the object id
|
||||||
source_transport {AbstractTransport)
|
source_transport {AbstractTransport) -- the transport through which the object can be found
|
||||||
-- the transport through which the object can be found
|
|
||||||
"""
|
"""
|
||||||
serialized_object = source_transport.get_object(id)
|
serialized_object = source_transport.get_object(id)
|
||||||
self.save_object(id, serialized_object)
|
self.save_object(id, serialized_object)
|
||||||
|
|
||||||
def save_object(self, id: str, serialized_object: str) -> None:
|
def save_object(self, id: str, serialized_object: str) -> None:
|
||||||
"""
|
"""Directly saves an object into the database.
|
||||||
Adds an object to the current batch to be written to the db.
|
|
||||||
If the current batch is full,
|
|
||||||
the batch is written to the db and the current batch is reset.
|
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
id {str} -- the object id
|
id {str} -- the object id
|
||||||
serialized_object {str} -- the full string representation of the object
|
serialized_object {str} -- the full string representation of the object
|
||||||
"""
|
"""
|
||||||
obj_size = len(serialized_object)
|
|
||||||
if (
|
|
||||||
not self._current_batch
|
|
||||||
or self._current_batch_size + obj_size < self.max_size
|
|
||||||
):
|
|
||||||
self._current_batch.append((id, serialized_object))
|
|
||||||
self._current_batch_size += obj_size
|
|
||||||
return
|
|
||||||
|
|
||||||
self.save_current_batch()
|
|
||||||
self._current_batch = [(id, serialized_object)]
|
|
||||||
self._current_batch_size = obj_size
|
|
||||||
|
|
||||||
def save_current_batch(self) -> None:
|
|
||||||
"""Save the current batch of objects to the local db"""
|
|
||||||
self.__check_connection()
|
self.__check_connection()
|
||||||
try:
|
try:
|
||||||
with closing(self.__connection.cursor()) as c:
|
with closing(self.__connection.cursor()) as c:
|
||||||
c.executemany(
|
c.execute(
|
||||||
"INSERT OR IGNORE INTO objects(hash, content) VALUES(?,?)",
|
"INSERT OR IGNORE INTO objects(hash, content) VALUES(?,?)",
|
||||||
self._current_batch,
|
(id, serialized_object),
|
||||||
)
|
)
|
||||||
self.__connection.commit()
|
self.__connection.commit()
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"Could not save the batch of objects to the local db. Inner exception:"
|
f"Could not save the object to the local db. Inner exception: {ex}", ex
|
||||||
f" {ex}",
|
|
||||||
ex,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def get_object(self, id: str) -> str or None:
|
def get_object(self, id: str) -> str or None:
|
||||||
@@ -128,14 +156,10 @@ class SQLiteTransport(AbstractTransport):
|
|||||||
return ret
|
return ret
|
||||||
|
|
||||||
def begin_write(self):
|
def begin_write(self):
|
||||||
self._object_cache = []
|
|
||||||
self.saved_obj_count = 0
|
self.saved_obj_count = 0
|
||||||
|
|
||||||
def end_write(self):
|
def end_write(self):
|
||||||
if self._current_batch:
|
pass
|
||||||
self.save_current_batch()
|
|
||||||
self._current_batch = []
|
|
||||||
self._current_batch_size = 0
|
|
||||||
|
|
||||||
def copy_object_and_children(
|
def copy_object_and_children(
|
||||||
self, id: str, target_transport: AbstractTransport
|
self, id: str, target_transport: AbstractTransport
|
||||||
@@ -143,10 +167,7 @@ class SQLiteTransport(AbstractTransport):
|
|||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def get_all_objects(self):
|
def get_all_objects(self):
|
||||||
"""
|
"""Returns all the objects in the store. NOTE: do not use for large collections!"""
|
||||||
Returns all the objects in the store.
|
|
||||||
NOTE: do not use for large collections!
|
|
||||||
"""
|
|
||||||
self.__check_connection()
|
self.__check_connection()
|
||||||
with closing(self.__connection.cursor()) as c:
|
with closing(self.__connection.cursor()) as c:
|
||||||
rows = c.execute("SELECT * FROM objects").fetchall()
|
rows = c.execute("SELECT * FROM objects").fetchall()
|
||||||
@@ -177,4 +198,20 @@ class SQLiteTransport(AbstractTransport):
|
|||||||
self.__connection = sqlite3.connect(self._root_path)
|
self.__connection = sqlite3.connect(self._root_path)
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
self.close()
|
self.__connection.close()
|
||||||
|
|
||||||
|
|
||||||
|
# def _run_queue(queue: Queue, root_path: str):
|
||||||
|
# if queue.empty():
|
||||||
|
# return
|
||||||
|
# print("RUN QUEUE")
|
||||||
|
# conn = sqlite3.connect(root_path)
|
||||||
|
# while not queue.empty():
|
||||||
|
# data = queue.get()
|
||||||
|
# with closing(conn.cursor()) as c:
|
||||||
|
# c.execute(
|
||||||
|
# "INSERT OR IGNORE INTO objects(hash, content) VALUES(?,?)",
|
||||||
|
# (data[0], data[1]),
|
||||||
|
# )
|
||||||
|
# conn.commit()
|
||||||
|
# conn.close()
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
"""This module contains an SDK for working with Speckle Automate."""
|
|
||||||
from speckle_automate.automation_context import AutomationContext
|
|
||||||
from speckle_automate.runner import execute_automate_function, run_function
|
|
||||||
from speckle_automate.schema import (
|
|
||||||
AutomateBase,
|
|
||||||
AutomationResult,
|
|
||||||
AutomationRunData,
|
|
||||||
AutomationStatus,
|
|
||||||
ObjectResult,
|
|
||||||
ObjectResultLevel,
|
|
||||||
)
|
|
||||||
|
|
||||||
__all__ = [
|
|
||||||
"AutomationContext",
|
|
||||||
"AutomateBase",
|
|
||||||
"AutomationStatus",
|
|
||||||
"AutomationResult",
|
|
||||||
"AutomationRunData",
|
|
||||||
"ObjectResult",
|
|
||||||
"ObjectResultLevel",
|
|
||||||
"run_function",
|
|
||||||
"execute_automate_function",
|
|
||||||
]
|
|
||||||
@@ -1,306 +0,0 @@
|
|||||||
"""This module provides an abstraction layer above the Speckle Automate runtime."""
|
|
||||||
from dataclasses import dataclass, field
|
|
||||||
from pathlib import Path
|
|
||||||
import time
|
|
||||||
from typing import Optional, Union
|
|
||||||
|
|
||||||
import httpx
|
|
||||||
from gql import gql
|
|
||||||
from specklepy.api import operations
|
|
||||||
from specklepy.api.client import SpeckleClient
|
|
||||||
from specklepy.core.api.models import Branch
|
|
||||||
from specklepy.objects import Base
|
|
||||||
from specklepy.transports.memory import MemoryTransport
|
|
||||||
from specklepy.transports.server import ServerTransport
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
from speckle_automate.schema import (
|
|
||||||
AutomateBase,
|
|
||||||
AutomationResult,
|
|
||||||
AutomationRunData,
|
|
||||||
AutomationStatus,
|
|
||||||
ObjectResult,
|
|
||||||
ObjectResultLevel,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class AutomationContext:
|
|
||||||
"""A context helper class.
|
|
||||||
|
|
||||||
This class exposes methods to work with the Speckle Automate context inside
|
|
||||||
Speckle Automate functions.
|
|
||||||
|
|
||||||
An instance of AutomationContext is injected into every run of a function.
|
|
||||||
"""
|
|
||||||
|
|
||||||
automation_run_data: AutomationRunData
|
|
||||||
speckle_client: SpeckleClient
|
|
||||||
_server_transport: ServerTransport
|
|
||||||
_speckle_token: str
|
|
||||||
|
|
||||||
#: keep a memory transponrt at hand, to speed up things if needed
|
|
||||||
_memory_transport: MemoryTransport = field(default_factory=MemoryTransport)
|
|
||||||
|
|
||||||
#: added for performance measuring
|
|
||||||
_init_time: float = field(default_factory=time.perf_counter)
|
|
||||||
_automation_result: AutomationResult = field(default_factory=AutomationResult)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def initialize(
|
|
||||||
cls, automation_run_data: Union[str, AutomationRunData], speckle_token: str
|
|
||||||
) -> "AutomationContext":
|
|
||||||
"""Bootstrap the AutomateSDK from raw data.
|
|
||||||
|
|
||||||
Todo:
|
|
||||||
----
|
|
||||||
* bootstrap a structlog logger instance
|
|
||||||
* expose a logger, that ppl can use instead of print
|
|
||||||
"""
|
|
||||||
# parse the json value if its not an initialized project data instance
|
|
||||||
automation_run_data = (
|
|
||||||
automation_run_data
|
|
||||||
if isinstance(automation_run_data, AutomationRunData)
|
|
||||||
else AutomationRunData.model_validate_json(automation_run_data)
|
|
||||||
)
|
|
||||||
speckle_client = SpeckleClient(
|
|
||||||
automation_run_data.speckle_server_url,
|
|
||||||
automation_run_data.speckle_server_url.startswith("https"),
|
|
||||||
)
|
|
||||||
speckle_client.authenticate_with_token(speckle_token)
|
|
||||||
if not speckle_client.account:
|
|
||||||
msg = (
|
|
||||||
f"Could not autenticate to {automation_run_data.speckle_server_url}",
|
|
||||||
"with the provided token",
|
|
||||||
)
|
|
||||||
raise ValueError(msg)
|
|
||||||
server_transport = ServerTransport(
|
|
||||||
automation_run_data.project_id, speckle_client
|
|
||||||
)
|
|
||||||
return cls(automation_run_data, speckle_client, server_transport, speckle_token)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def run_status(self) -> AutomationStatus:
|
|
||||||
"""Get the status of the automation run."""
|
|
||||||
return self._automation_result.run_status
|
|
||||||
|
|
||||||
def elapsed(self) -> float:
|
|
||||||
"""Return the elapsed time in seconds since the initialization time."""
|
|
||||||
return time.perf_counter() - self._init_time
|
|
||||||
|
|
||||||
def receive_version(self) -> Base:
|
|
||||||
"""Receive the Speckle project version that triggered this automation run."""
|
|
||||||
commit = self.speckle_client.commit.get(
|
|
||||||
self.automation_run_data.project_id, self.automation_run_data.version_id
|
|
||||||
)
|
|
||||||
if not commit.referencedObject:
|
|
||||||
raise ValueError("The commit has no referencedObject, cannot receive it.")
|
|
||||||
base = operations.receive(
|
|
||||||
commit.referencedObject, self._server_transport, self._memory_transport
|
|
||||||
)
|
|
||||||
print(
|
|
||||||
f"It took {self.elapsed():2f} seconds to receive",
|
|
||||||
f" the speckle version {self.automation_run_data.version_id}",
|
|
||||||
)
|
|
||||||
return base
|
|
||||||
|
|
||||||
def create_new_version_in_project(
|
|
||||||
self, root_object: Base, model_id: str, version_message: str = ""
|
|
||||||
) -> None:
|
|
||||||
"""Save a base model to a new version on the project.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
root_object (Base): The Speckle base object for the new version.
|
|
||||||
model_id (str): For now please use a `branchName`!
|
|
||||||
version_message (str): The message for the new version.
|
|
||||||
"""
|
|
||||||
if model_id == self.automation_run_data.model_id:
|
|
||||||
raise ValueError(
|
|
||||||
f"The target model id: {model_id} cannot match the model id"
|
|
||||||
f" that triggered this automation: {self.automation_run_data.model_id}"
|
|
||||||
)
|
|
||||||
|
|
||||||
branch = self._get_model(model_id)
|
|
||||||
if not branch.name:
|
|
||||||
raise ValueError(f"The model {model_id} has no name.")
|
|
||||||
|
|
||||||
root_object_id = operations.send(
|
|
||||||
root_object,
|
|
||||||
[self._server_transport, self._memory_transport],
|
|
||||||
use_default_cache=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
version_id = self.speckle_client.commit.create(
|
|
||||||
stream_id=self.automation_run_data.project_id,
|
|
||||||
object_id=root_object_id,
|
|
||||||
branch_name=branch.name,
|
|
||||||
message=version_message,
|
|
||||||
source_application="SpeckleAutomate",
|
|
||||||
)
|
|
||||||
|
|
||||||
if isinstance(version_id, SpeckleException):
|
|
||||||
raise version_id
|
|
||||||
|
|
||||||
self._automation_result.result_versions.append(version_id)
|
|
||||||
|
|
||||||
def _get_model(self, model_id: str) -> Branch:
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query ProjectModel($projectId: String!, $modelId: String!){
|
|
||||||
project(id: $projectId) {
|
|
||||||
model(id: $modelId) {
|
|
||||||
name
|
|
||||||
id
|
|
||||||
description
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"projectId": self.automation_run_data.project_id, "modelId": model_id}
|
|
||||||
response = self.speckle_client.httpclient.execute(query, params)
|
|
||||||
return Branch.model_validate(response["project"]["model"])
|
|
||||||
|
|
||||||
def report_run_status(self) -> None:
|
|
||||||
"""Report the current run status to the project of this automation."""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation ReportFunctionRunStatus(
|
|
||||||
$automationId: String!,
|
|
||||||
$automationRevisionId: String!,
|
|
||||||
$automationRunId: String!,
|
|
||||||
$versionId: String!,
|
|
||||||
$functionId: String!,
|
|
||||||
$runStatus: AutomationRunStatus!
|
|
||||||
$elapsed: Float!
|
|
||||||
$resultVersionIds: [String!]!
|
|
||||||
$statusMessage: String
|
|
||||||
$objectResults: JSONObject
|
|
||||||
){
|
|
||||||
automationMutations {
|
|
||||||
functionRunStatusReport(input: {
|
|
||||||
automationId: $automationId
|
|
||||||
automationRevisionId: $automationRevisionId
|
|
||||||
automationRunId: $automationRunId
|
|
||||||
versionId: $versionId
|
|
||||||
functionRuns: [
|
|
||||||
{
|
|
||||||
functionId: $functionId
|
|
||||||
status: $runStatus,
|
|
||||||
elapsed: $elapsed,
|
|
||||||
resultVersionIds: $resultVersionIds,
|
|
||||||
statusMessage: $statusMessage
|
|
||||||
results: $objectResults
|
|
||||||
}]
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
if self.run_status in [AutomationStatus.SUCCEEDED, AutomationStatus.FAILED]:
|
|
||||||
object_results = {
|
|
||||||
"version": "1.0.0",
|
|
||||||
"values": {
|
|
||||||
"speckleObjects": self._automation_result.model_dump(by_alias=True)[
|
|
||||||
"objectResults"
|
|
||||||
],
|
|
||||||
"blobs": self._automation_result.blobs,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
object_results = None
|
|
||||||
params = {
|
|
||||||
"automationId": self.automation_run_data.automation_id,
|
|
||||||
"automationRevisionId": self.automation_run_data.automation_revision_id,
|
|
||||||
"automationRunId": self.automation_run_data.automation_run_id,
|
|
||||||
"versionId": self.automation_run_data.version_id,
|
|
||||||
"functionId": self.automation_run_data.function_id,
|
|
||||||
"runStatus": self.run_status.value,
|
|
||||||
"statusMessage": self._automation_result.status_message,
|
|
||||||
"elapsed": self.elapsed(),
|
|
||||||
"resultVersionIds": self._automation_result.result_versions,
|
|
||||||
"objectResults": object_results,
|
|
||||||
}
|
|
||||||
self.speckle_client.httpclient.execute(query, params)
|
|
||||||
|
|
||||||
def store_file_result(self, file_path: Union[Path, str]) -> None:
|
|
||||||
"""Save a file attached to the project of this automation."""
|
|
||||||
path_obj = (
|
|
||||||
Path(file_path).resolve() if isinstance(file_path, str) else file_path
|
|
||||||
)
|
|
||||||
|
|
||||||
class UploadResult(AutomateBase):
|
|
||||||
blob_id: str
|
|
||||||
file_name: str
|
|
||||||
upload_status: int
|
|
||||||
|
|
||||||
class BlobUploadResponse(AutomateBase):
|
|
||||||
upload_results: list[UploadResult]
|
|
||||||
|
|
||||||
if not path_obj.exists():
|
|
||||||
raise ValueError("The given file path doesn't exist")
|
|
||||||
files = {path_obj.name: open(str(path_obj), "rb")}
|
|
||||||
|
|
||||||
url = (
|
|
||||||
f"{self.automation_run_data.speckle_server_url}/api/stream/"
|
|
||||||
f"{self.automation_run_data.project_id}/blob"
|
|
||||||
)
|
|
||||||
data = (
|
|
||||||
httpx.post(
|
|
||||||
url,
|
|
||||||
files=files,
|
|
||||||
headers={"authorization": f"Bearer {self._speckle_token}"},
|
|
||||||
)
|
|
||||||
.raise_for_status()
|
|
||||||
.json()
|
|
||||||
)
|
|
||||||
|
|
||||||
upload_response = BlobUploadResponse.model_validate(data)
|
|
||||||
|
|
||||||
if len(upload_response.upload_results) != 1:
|
|
||||||
raise ValueError("Expecting one upload result.")
|
|
||||||
|
|
||||||
for upload_result in upload_response.upload_results:
|
|
||||||
self._automation_result.blobs.append(upload_result.blob_id)
|
|
||||||
|
|
||||||
def mark_run_failed(self, status_message: str) -> None:
|
|
||||||
"""Mark the current run a failure."""
|
|
||||||
self._mark_run(AutomationStatus.FAILED, status_message)
|
|
||||||
|
|
||||||
def mark_run_success(self, status_message: Optional[str]) -> None:
|
|
||||||
"""Mark the current run a success with an optional message."""
|
|
||||||
self._mark_run(AutomationStatus.SUCCEEDED, status_message)
|
|
||||||
|
|
||||||
def _mark_run(
|
|
||||||
self, status: AutomationStatus, status_message: Optional[str]
|
|
||||||
) -> None:
|
|
||||||
duration = self.elapsed()
|
|
||||||
self._automation_result.status_message = status_message
|
|
||||||
self._automation_result.run_status = status
|
|
||||||
self._automation_result.elapsed = duration
|
|
||||||
|
|
||||||
msg = f"Automation run {status.value} after {duration:2f} seconds."
|
|
||||||
print("\n".join([msg, status_message]) if status_message else msg)
|
|
||||||
|
|
||||||
def add_object_error(self, object_id: str, error_cause: str) -> None:
|
|
||||||
"""Add an error to a given Speckle object."""
|
|
||||||
self._add_object_result(object_id, ObjectResultLevel.ERROR, error_cause)
|
|
||||||
|
|
||||||
def add_object_warning(self, object_id: str, warning: str) -> None:
|
|
||||||
"""Add a warning to a given Speckle object."""
|
|
||||||
self._add_object_result(object_id, ObjectResultLevel.WARNING, warning)
|
|
||||||
|
|
||||||
def add_object_info(self, object_id: str, info: str) -> None:
|
|
||||||
"""Add an info message to a given Speckle object."""
|
|
||||||
self._add_object_result(object_id, ObjectResultLevel.INFO, info)
|
|
||||||
|
|
||||||
def _add_object_result(
|
|
||||||
self, object_id: str, level: ObjectResultLevel, status_message: str
|
|
||||||
) -> None:
|
|
||||||
print(
|
|
||||||
f"Object {object_id} was marked with {level.value.upper()}",
|
|
||||||
f" cause: {status_message}",
|
|
||||||
)
|
|
||||||
self._automation_result.object_results[object_id].append(
|
|
||||||
ObjectResult(level=level, status_message=status_message)
|
|
||||||
)
|
|
||||||
@@ -1,155 +0,0 @@
|
|||||||
"""Function execution module.
|
|
||||||
|
|
||||||
Provides mechanisms to execute any function,
|
|
||||||
that conforms to the AutomateFunction "interface"
|
|
||||||
"""
|
|
||||||
import json
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import traceback
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Callable, Optional, TypeVar, Union, overload
|
|
||||||
|
|
||||||
from speckle_automate.automation_context import AutomationContext
|
|
||||||
from speckle_automate.schema import AutomateBase, AutomationRunData, AutomationStatus
|
|
||||||
|
|
||||||
T = TypeVar("T", bound=AutomateBase)
|
|
||||||
|
|
||||||
AutomateFunction = Callable[[AutomationContext, T], None]
|
|
||||||
AutomateFunctionWithoutInputs = Callable[[AutomationContext], None]
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
|
||||||
def execute_automate_function(
|
|
||||||
automate_function: AutomateFunction[T],
|
|
||||||
input_schema: type[T],
|
|
||||||
) -> None:
|
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
|
||||||
def execute_automate_function(automate_function: AutomateFunctionWithoutInputs) -> None:
|
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
def execute_automate_function(
|
|
||||||
automate_function: Union[AutomateFunction[T], AutomateFunctionWithoutInputs],
|
|
||||||
input_schema: Optional[type[T]] = None,
|
|
||||||
):
|
|
||||||
"""Runs the provided automate function with the input schema."""
|
|
||||||
# first arg is the python file name, we do not need that
|
|
||||||
args = sys.argv[1:]
|
|
||||||
|
|
||||||
if len(args) < 2:
|
|
||||||
raise ValueError("too few arguments specified need minimum 2")
|
|
||||||
|
|
||||||
if len(args) > 4:
|
|
||||||
raise ValueError("too many arguments specified, max supported is 4")
|
|
||||||
|
|
||||||
# we rely on a command name convention to decide what to do.
|
|
||||||
# this is here, so that the function authors do not see any of this
|
|
||||||
command = args[0]
|
|
||||||
|
|
||||||
if command == "generate_schema":
|
|
||||||
path = Path(args[1])
|
|
||||||
schema = json.dumps(
|
|
||||||
input_schema.model_json_schema(by_alias=True) if input_schema else {}
|
|
||||||
)
|
|
||||||
path.write_text(schema)
|
|
||||||
|
|
||||||
elif command == "run":
|
|
||||||
automation_run_data = args[1]
|
|
||||||
function_inputs = args[2]
|
|
||||||
|
|
||||||
speckle_token = os.environ.get("SPECKLE_TOKEN", None)
|
|
||||||
if not speckle_token and len(args) != 4:
|
|
||||||
raise ValueError("Cannot get speckle token from arguments or environment")
|
|
||||||
|
|
||||||
speckle_token = speckle_token if speckle_token else args[3]
|
|
||||||
|
|
||||||
inputs = (
|
|
||||||
input_schema.model_validate_json(function_inputs)
|
|
||||||
if input_schema
|
|
||||||
else input_schema
|
|
||||||
)
|
|
||||||
|
|
||||||
if inputs:
|
|
||||||
automation_context = run_function(
|
|
||||||
automate_function, # type: ignore
|
|
||||||
automation_run_data,
|
|
||||||
speckle_token,
|
|
||||||
inputs,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
automation_context = run_function(
|
|
||||||
automate_function, # type: ignore
|
|
||||||
automation_run_data,
|
|
||||||
speckle_token,
|
|
||||||
)
|
|
||||||
|
|
||||||
exit_code = (
|
|
||||||
0 if automation_context.run_status == AutomationStatus.SUCCEEDED else 1
|
|
||||||
)
|
|
||||||
exit(exit_code)
|
|
||||||
|
|
||||||
else:
|
|
||||||
raise NotImplementedError(f"Command: '{command}' is not supported.")
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
|
||||||
def run_function(
|
|
||||||
automate_function: AutomateFunction[T],
|
|
||||||
automation_run_data: Union[AutomationRunData, str],
|
|
||||||
speckle_token: str,
|
|
||||||
inputs: T,
|
|
||||||
) -> AutomationContext:
|
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
|
||||||
def run_function(
|
|
||||||
automate_function: AutomateFunctionWithoutInputs,
|
|
||||||
automation_run_data: Union[AutomationRunData, str],
|
|
||||||
speckle_token: str,
|
|
||||||
) -> AutomationContext:
|
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
def run_function(
|
|
||||||
automate_function: Union[AutomateFunction[T], AutomateFunctionWithoutInputs],
|
|
||||||
automation_run_data: Union[AutomationRunData, str],
|
|
||||||
speckle_token: str,
|
|
||||||
inputs: Optional[T] = None,
|
|
||||||
) -> AutomationContext:
|
|
||||||
"""Run the provided function with the automate sdk context."""
|
|
||||||
automation_context = AutomationContext.initialize(
|
|
||||||
automation_run_data, speckle_token
|
|
||||||
)
|
|
||||||
automation_context.report_run_status()
|
|
||||||
|
|
||||||
try:
|
|
||||||
# avoiding complex type gymnastics here on the internals.
|
|
||||||
# the external type overloads make this correct
|
|
||||||
if inputs:
|
|
||||||
automate_function(automation_context, inputs) # type: ignore
|
|
||||||
else:
|
|
||||||
automate_function(automation_context) # type: ignore
|
|
||||||
|
|
||||||
# the function author forgot to mark the function success
|
|
||||||
if automation_context.run_status not in [
|
|
||||||
AutomationStatus.FAILED,
|
|
||||||
AutomationStatus.SUCCEEDED,
|
|
||||||
]:
|
|
||||||
automation_context.mark_run_success(
|
|
||||||
"WARNING: Automate assumed a success status,"
|
|
||||||
" but it was not marked as so by the function."
|
|
||||||
)
|
|
||||||
except Exception:
|
|
||||||
trace = traceback.format_exc()
|
|
||||||
print(trace)
|
|
||||||
automation_context.mark_run_failed(
|
|
||||||
"Function error. Check the automation run logs for details."
|
|
||||||
)
|
|
||||||
finally:
|
|
||||||
automation_context.report_run_status()
|
|
||||||
return automation_context
|
|
||||||
@@ -1,73 +0,0 @@
|
|||||||
""""""
|
|
||||||
from collections import defaultdict
|
|
||||||
from enum import Enum
|
|
||||||
from typing import Optional, List, Dict
|
|
||||||
|
|
||||||
from pydantic import BaseModel, ConfigDict, Field
|
|
||||||
from stringcase import camelcase
|
|
||||||
|
|
||||||
|
|
||||||
class AutomateBase(BaseModel):
|
|
||||||
"""Use this class as a base model for automate related DTO."""
|
|
||||||
|
|
||||||
model_config = ConfigDict(alias_generator=camelcase, populate_by_name=True)
|
|
||||||
|
|
||||||
|
|
||||||
class AutomationRunData(BaseModel):
|
|
||||||
"""Values of the project / model that triggered the run of this function."""
|
|
||||||
|
|
||||||
project_id: str
|
|
||||||
model_id: str
|
|
||||||
branch_name: str
|
|
||||||
version_id: str
|
|
||||||
speckle_server_url: str
|
|
||||||
|
|
||||||
automation_id: str
|
|
||||||
automation_revision_id: str
|
|
||||||
automation_run_id: str
|
|
||||||
|
|
||||||
function_id: str
|
|
||||||
function_release: str
|
|
||||||
|
|
||||||
model_config = ConfigDict(
|
|
||||||
alias_generator=camelcase, populate_by_name=True, protected_namespaces=()
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class AutomationStatus(str, Enum):
|
|
||||||
"""Set the status of the automation."""
|
|
||||||
|
|
||||||
INITIALIZING = "INITIALIZING"
|
|
||||||
RUNNING = "RUNNING"
|
|
||||||
FAILED = "FAILED"
|
|
||||||
SUCCEEDED = "SUCCEEDED"
|
|
||||||
|
|
||||||
|
|
||||||
class ObjectResultLevel(str, Enum):
|
|
||||||
"""Possible status message levels for object reports."""
|
|
||||||
|
|
||||||
INFO = "INFO"
|
|
||||||
WARNING = "WARNING"
|
|
||||||
ERROR = "ERROR"
|
|
||||||
|
|
||||||
|
|
||||||
class ObjectResult(AutomateBase):
|
|
||||||
"""An object level result."""
|
|
||||||
|
|
||||||
level: ObjectResultLevel
|
|
||||||
status_message: str
|
|
||||||
|
|
||||||
|
|
||||||
class AutomationResult(AutomateBase):
|
|
||||||
"""Schema accepted by the Speckle server as a result for an automation run."""
|
|
||||||
|
|
||||||
elapsed: float = 0
|
|
||||||
result_view: Optional[str] = None
|
|
||||||
result_versions: List[str] = Field(default_factory=list)
|
|
||||||
blobs: List[str] = Field(default_factory=list)
|
|
||||||
run_status: AutomationStatus = AutomationStatus.RUNNING
|
|
||||||
status_message: Optional[str] = None
|
|
||||||
|
|
||||||
object_results: Dict[str, List[ObjectResult]] = Field(
|
|
||||||
default_factory=lambda: defaultdict(list) # typing: ignore
|
|
||||||
)
|
|
||||||
@@ -1,159 +0,0 @@
|
|||||||
import re
|
|
||||||
from typing import Dict
|
|
||||||
from warnings import warn
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import Client
|
|
||||||
from gql.transport.exceptions import TransportServerError
|
|
||||||
from gql.transport.requests import RequestsHTTPTransport
|
|
||||||
from gql.transport.websockets import WebsocketsTransport
|
|
||||||
|
|
||||||
from specklepy.api import resources
|
|
||||||
from specklepy.api.credentials import Account, get_account_from_token
|
|
||||||
from specklepy.api.resources import (
|
|
||||||
user,
|
|
||||||
active_user,
|
|
||||||
branch,
|
|
||||||
commit,
|
|
||||||
object,
|
|
||||||
other_user,
|
|
||||||
server,
|
|
||||||
stream,
|
|
||||||
subscriptions,
|
|
||||||
)
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
|
|
||||||
from specklepy.core.api.client import SpeckleClient as CoreSpeckleClient
|
|
||||||
|
|
||||||
|
|
||||||
class SpeckleClient(CoreSpeckleClient):
|
|
||||||
"""
|
|
||||||
The `SpeckleClient` is your entry point for interacting with
|
|
||||||
your Speckle Server's GraphQL API.
|
|
||||||
You'll need to have access to a server to use it,
|
|
||||||
or you can use our public server `speckle.xyz`.
|
|
||||||
|
|
||||||
To authenticate the client, you'll need to have downloaded
|
|
||||||
the [Speckle Manager](https://speckle.guide/#speckle-manager)
|
|
||||||
and added your account.
|
|
||||||
|
|
||||||
```py
|
|
||||||
from specklepy.api.client import SpeckleClient
|
|
||||||
from specklepy.api.credentials import get_default_account
|
|
||||||
|
|
||||||
# initialise the client
|
|
||||||
client = SpeckleClient(host="speckle.xyz") # or whatever your host is
|
|
||||||
# client = SpeckleClient(host="localhost:3000", use_ssl=False) or use local server
|
|
||||||
|
|
||||||
# authenticate the client with an account (account has been added in Speckle Manager)
|
|
||||||
account = get_default_account()
|
|
||||||
client.authenticate_with_account(account)
|
|
||||||
|
|
||||||
# create a new stream. this returns the stream id
|
|
||||||
new_stream_id = client.stream.create(name="a shiny new stream")
|
|
||||||
|
|
||||||
# use that stream id to get the stream from the server
|
|
||||||
new_stream = client.stream.get(id=new_stream_id)
|
|
||||||
```
|
|
||||||
"""
|
|
||||||
|
|
||||||
DEFAULT_HOST = "speckle.xyz"
|
|
||||||
USE_SSL = True
|
|
||||||
|
|
||||||
def __init__(self, host: str = DEFAULT_HOST, use_ssl: bool = USE_SSL) -> None:
|
|
||||||
super().__init__(
|
|
||||||
host=host,
|
|
||||||
use_ssl=use_ssl,
|
|
||||||
)
|
|
||||||
self.account = Account()
|
|
||||||
|
|
||||||
def _init_resources(self) -> None:
|
|
||||||
self.server = server.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
server_version = None
|
|
||||||
try:
|
|
||||||
server_version = self.server.version()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
self.user = user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.other_user = other_user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.active_user = active_user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.stream = stream.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.commit = commit.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.branch = branch.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.object = object.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.subscribe = subscriptions.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.ws_url,
|
|
||||||
client=self.wsclient,
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(
|
|
||||||
version="2.6.0",
|
|
||||||
reason=(
|
|
||||||
"Renamed: please use `authenticate_with_account` or"
|
|
||||||
" `authenticate_with_token` instead."
|
|
||||||
),
|
|
||||||
)
|
|
||||||
def authenticate(self, token: str) -> None:
|
|
||||||
"""Authenticate the client using a personal access token
|
|
||||||
The token is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str} -- an api token
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Client Authenticate_deprecated"})
|
|
||||||
return super().authenticate(token)
|
|
||||||
|
|
||||||
def authenticate_with_token(self, token: str) -> None:
|
|
||||||
"""
|
|
||||||
Authenticate the client using a personal access token.
|
|
||||||
The token is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str} -- an api token
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Client Authenticate With Token"})
|
|
||||||
return super().authenticate_with_token(token)
|
|
||||||
|
|
||||||
def authenticate_with_account(self, account: Account) -> None:
|
|
||||||
"""Authenticate the client using an Account object
|
|
||||||
The account is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
account {Account} -- the account object which can be found with
|
|
||||||
`get_default_account` or `get_local_accounts`
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Client Authenticate With Account"})
|
|
||||||
return super().authenticate_with_account(account)
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
import os
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from pydantic import BaseModel, Field # pylint: disable=no-name-in-module
|
|
||||||
|
|
||||||
from specklepy.api.models import ServerInfo
|
|
||||||
from specklepy.core.helpers import speckle_path_provider
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
# following imports seem to be unnecessary, but they need to stay
|
|
||||||
# to not break the scripts using these functions as non-core
|
|
||||||
from specklepy.core.api.credentials import (Account, UserInfo,
|
|
||||||
StreamWrapper, # deprecated
|
|
||||||
get_local_accounts as core_get_local_accounts,
|
|
||||||
get_account_from_token as core_get_account_from_token)
|
|
||||||
|
|
||||||
|
|
||||||
def get_local_accounts(base_path: Optional[str] = None) -> List[Account]:
|
|
||||||
"""Gets all the accounts present in this environment
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Account] -- list of all local accounts or an empty list if
|
|
||||||
no accounts were found
|
|
||||||
"""
|
|
||||||
accounts = core_get_local_accounts(base_path)
|
|
||||||
|
|
||||||
metrics.track(
|
|
||||||
metrics.SDK,
|
|
||||||
next(
|
|
||||||
(acc for acc in accounts if acc.isDefault),
|
|
||||||
accounts[0] if accounts else None,
|
|
||||||
),
|
|
||||||
{"name": "Get Local Accounts"}
|
|
||||||
)
|
|
||||||
|
|
||||||
return accounts
|
|
||||||
|
|
||||||
def get_default_account(base_path: Optional[str] = None) -> Optional[Account]:
|
|
||||||
"""
|
|
||||||
Gets this environment's default account if any. If there is no default,
|
|
||||||
the first found will be returned and set as default.
|
|
||||||
Arguments:
|
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Account -- the default account or None if no local accounts were found
|
|
||||||
"""
|
|
||||||
accounts = core_get_local_accounts(base_path=base_path)
|
|
||||||
if not accounts:
|
|
||||||
return None
|
|
||||||
|
|
||||||
default = next((acc for acc in accounts if acc.isDefault), None)
|
|
||||||
if not default:
|
|
||||||
default = accounts[0]
|
|
||||||
default.isDefault = True
|
|
||||||
metrics.initialise_tracker(default)
|
|
||||||
|
|
||||||
return default
|
|
||||||
|
|
||||||
def get_account_from_token(token: str, server_url: str = None) -> Account:
|
|
||||||
"""Gets the local account for the token if it exists
|
|
||||||
Arguments:
|
|
||||||
token {str} -- the api token
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Account -- the local account with this token or a shell account containing
|
|
||||||
just the token and url if no local account is found
|
|
||||||
"""
|
|
||||||
account = core_get_account_from_token(token, server_url)
|
|
||||||
|
|
||||||
metrics.track( metrics.SDK, account, {"name": "Get Account From Token"} )
|
|
||||||
return account
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
from dataclasses import dataclass
|
|
||||||
from enum import Enum
|
|
||||||
from unicodedata import name
|
|
||||||
|
|
||||||
# following imports seem to be unnecessary, but they need to stay
|
|
||||||
# to not break the scripts using these functions as non-core
|
|
||||||
from specklepy.core.api.host_applications import (HostApplication, HostAppVersion,
|
|
||||||
get_host_app_from_string,
|
|
||||||
_app_name_host_app_mapping,
|
|
||||||
RHINO,GRASSHOPPER,REVIT,DYNAMO,UNITY,GSA,
|
|
||||||
CIVIL,AUTOCAD,MICROSTATION,OPENROADS,
|
|
||||||
OPENRAIL,OPENBUILDINGS,ETABS,SAP2000,CSIBRIDGE,
|
|
||||||
SAFE,TEKLASTRUCTURES,DXF,EXCEL,UNREAL,POWERBI,
|
|
||||||
BLENDER,QGIS,ARCGIS,SKETCHUP,ARCHICAD,TOPSOLID,
|
|
||||||
PYTHON,NET,OTHER)
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
print(HostAppVersion.v)
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
from datetime import datetime
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from pydantic import BaseModel, Field
|
|
||||||
|
|
||||||
# following imports seem to be unnecessary, but they need to stay
|
|
||||||
# to not break the scripts using these functions as non-core
|
|
||||||
from specklepy.core.api.models import (Collaborator, Commit,
|
|
||||||
Commits, Object, Branch, Branches,
|
|
||||||
Stream, Streams, User, LimitedUser,
|
|
||||||
PendingStreamCollaborator, Activity,
|
|
||||||
ActivityCollection, ServerInfo)
|
|
||||||
@@ -1,100 +0,0 @@
|
|||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
from specklepy.objects.base import Base
|
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
from specklepy.core.api.operations import (send as core_send,
|
|
||||||
receive as _untracked_receive,
|
|
||||||
serialize as core_serialize,
|
|
||||||
deserialize as core_deserialize)
|
|
||||||
|
|
||||||
|
|
||||||
def send(
|
|
||||||
base: Base,
|
|
||||||
transports: Optional[List[AbstractTransport]] = None,
|
|
||||||
use_default_cache: bool = True,
|
|
||||||
):
|
|
||||||
"""Sends an object via the provided transports. Defaults to the local cache.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
obj {Base} -- the object you want to send
|
|
||||||
transports {list} -- where you want to send them
|
|
||||||
use_default_cache {bool} -- toggle for the default cache.
|
|
||||||
If set to false, it will only send to the provided transports
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the object id of the sent object
|
|
||||||
"""
|
|
||||||
if transports is None:
|
|
||||||
metrics.track(metrics.SEND)
|
|
||||||
else:
|
|
||||||
metrics.track(metrics.SEND, getattr(transports[0], "account", None))
|
|
||||||
|
|
||||||
return core_send(base, transports, use_default_cache)
|
|
||||||
|
|
||||||
|
|
||||||
def receive(
|
|
||||||
obj_id: str,
|
|
||||||
remote_transport: Optional[AbstractTransport] = None,
|
|
||||||
local_transport: Optional[AbstractTransport] = None,
|
|
||||||
) -> Base:
|
|
||||||
"""Receives an object from a transport.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
obj_id {str} -- the id of the object to receive
|
|
||||||
remote_transport {Transport} -- the transport to receive from
|
|
||||||
local_transport {Transport} -- the local cache to check for existing objects
|
|
||||||
(defaults to `SQLiteTransport`)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Base -- the base object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.RECEIVE, getattr(remote_transport, "account", None))
|
|
||||||
return _untracked_receive(obj_id, remote_transport, local_transport)
|
|
||||||
|
|
||||||
|
|
||||||
def serialize(base: Base, write_transports: List[AbstractTransport] = []) -> str:
|
|
||||||
"""
|
|
||||||
Serialize a base object. If no write transports are provided,
|
|
||||||
the object will be serialized
|
|
||||||
without detaching or chunking any of the attributes.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
base {Base} -- the object to serialize
|
|
||||||
write_transports {List[AbstractTransport]}
|
|
||||||
-- optional: the transports to write to
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the serialized object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Serialize"})
|
|
||||||
return core_serialize(base, write_transports)
|
|
||||||
|
|
||||||
def deserialize(
|
|
||||||
obj_string: str, read_transport: Optional[AbstractTransport] = None
|
|
||||||
) -> Base:
|
|
||||||
"""
|
|
||||||
Deserialize a string object into a Base object.
|
|
||||||
|
|
||||||
If the object contains referenced child objects that are not stored in the local db,
|
|
||||||
a read transport needs to be provided in order to recompose
|
|
||||||
the base with the children objects.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
obj_string {str} -- the string object to deserialize
|
|
||||||
read_transport {AbstractTransport}
|
|
||||||
-- the transport to fetch children objects from
|
|
||||||
(defaults to SQLiteTransport)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Base -- the deserialized object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Deserialize"})
|
|
||||||
return core_deserialize(obj_string, read_transport)
|
|
||||||
|
|
||||||
|
|
||||||
__all__ = ["receive", "send", "serialize", "deserialize"]
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
from threading import Lock
|
|
||||||
from typing import Any, Dict, List, Optional, Tuple, Type, Union
|
|
||||||
|
|
||||||
from gql.client import Client
|
|
||||||
from gql.transport.exceptions import TransportQueryError
|
|
||||||
from graphql import DocumentNode
|
|
||||||
|
|
||||||
from specklepy.api.credentials import Account
|
|
||||||
from specklepy.logging.exceptions import (
|
|
||||||
GraphQLException,
|
|
||||||
SpeckleException,
|
|
||||||
UnsupportedException,
|
|
||||||
)
|
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
# following imports seem to be unnecessary, but they need to stay
|
|
||||||
# to not break the scripts using these functions as non-core
|
|
||||||
from specklepy.core.api.resource import ResourceBase as CoreResourceBase
|
|
||||||
|
|
||||||
|
|
||||||
class ResourceBase(CoreResourceBase):
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
account: Account,
|
|
||||||
basepath: str,
|
|
||||||
client: Client,
|
|
||||||
name: str,
|
|
||||||
server_version: Optional[Tuple[Any, ...]] = None,
|
|
||||||
) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account = account,
|
|
||||||
basepath = basepath,
|
|
||||||
client = client,
|
|
||||||
name = name,
|
|
||||||
server_version = server_version
|
|
||||||
)
|
|
||||||
|
|
||||||
@@ -1,120 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import ActivityCollection, PendingStreamCollaborator, User
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.active_user import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for users"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = User
|
|
||||||
|
|
||||||
def get(self) -> User:
|
|
||||||
"""Gets the profile of a user. If no id argument is provided,
|
|
||||||
will return the current authenticated user's profile
|
|
||||||
(as extracted from the authorization header).
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
User -- the retrieved user
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "User Active Get"})
|
|
||||||
return super().get()
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
company: Optional[str] = None,
|
|
||||||
bio: Optional[str] = None,
|
|
||||||
avatar: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Updates your user profile. All arguments are optional.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- your name
|
|
||||||
company {str} -- the company you may or may not work for
|
|
||||||
bio {str} -- tell us about yourself
|
|
||||||
avatar {str} -- a nice photo of yourself
|
|
||||||
|
|
||||||
Returns @deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT):
|
|
||||||
bool -- True if your profile was updated successfully
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Active Update"})
|
|
||||||
return super().update(name, company, bio, avatar)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
If no id argument is provided, will return the current authenticated user's
|
|
||||||
activity (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz as they will be
|
|
||||||
converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime} -- latest cutoff for activity
|
|
||||||
(ie: return all activity _before_ this time)
|
|
||||||
after {datetime} -- oldest cutoff for activity
|
|
||||||
(ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Active Activity"})
|
|
||||||
return super().activity(limit, action_type, before, after, cursor)
|
|
||||||
|
|
||||||
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the active user's pending stream invites
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the current user
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Active Invites All Get"})
|
|
||||||
return super().get_all_pending_invites()
|
|
||||||
|
|
||||||
def get_pending_invite(
|
|
||||||
self, stream_id: str, token: Optional[str] = None
|
|
||||||
) -> Optional[PendingStreamCollaborator]:
|
|
||||||
"""Get a particular pending invite for the active user on a given stream.
|
|
||||||
If no invite_id is provided, any valid invite will be returned.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to look for invites on
|
|
||||||
token {str} -- the token of the invite to look for (optional)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PendingStreamCollaborator
|
|
||||||
-- the invite for the given stream (or None if it isn't found)
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Active Invite Get"})
|
|
||||||
return super().get_pending_invite(stream_id, token)
|
|
||||||
@@ -1,98 +0,0 @@
|
|||||||
from typing import Optional
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import Branch
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.branch import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for branches"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
)
|
|
||||||
self.schema = Branch
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self, stream_id: str, name: str, description: str = "No description provided"
|
|
||||||
) -> str:
|
|
||||||
"""Create a new branch on this stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- the name of the new branch
|
|
||||||
description {str} -- a short description of the branch
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
id {str} -- the newly created branch's id
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Create"})
|
|
||||||
return super().create(stream_id, name, description)
|
|
||||||
|
|
||||||
def get(self, stream_id: str, name: str, commits_limit: int = 10):
|
|
||||||
"""Get a branch by name from a stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to get the branch from
|
|
||||||
name {str} -- the name of the branch to get
|
|
||||||
commits_limit {int} -- maximum number of commits to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Branch -- the fetched branch with its latest commits
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Get"})
|
|
||||||
return super().get(stream_id, name, commits_limit)
|
|
||||||
|
|
||||||
def list(self, stream_id: str, branches_limit: int = 10, commits_limit: int = 10):
|
|
||||||
"""Get a list of branches from a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to get the branches from
|
|
||||||
branches_limit {int} -- maximum number of branches to get
|
|
||||||
commits_limit {int} -- maximum number of commits to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Branch] -- the branches on the stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch List"})
|
|
||||||
return super().list(stream_id, branches_limit, commits_limit)
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
branch_id: str,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
description: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Update a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream containing the branch to update
|
|
||||||
branch_id {str} -- the id of the branch to update
|
|
||||||
name {str} -- optional: the updated branch name
|
|
||||||
description {str} -- optional: the updated branch description
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if update is successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Update"})
|
|
||||||
return super().update(stream_id, branch_id, name, description)
|
|
||||||
|
|
||||||
def delete(self, stream_id: str, branch_id: str):
|
|
||||||
"""Delete a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream containing the branch to delete
|
|
||||||
branch_id {str} -- the branch to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if deletion is successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Delete"})
|
|
||||||
return super().delete(stream_id, branch_id)
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import Commit
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.commit import Resource as CoreResource
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for commits"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
)
|
|
||||||
self.schema = Commit
|
|
||||||
|
|
||||||
def get(self, stream_id: str, commit_id: str) -> Commit:
|
|
||||||
"""
|
|
||||||
Gets a commit given a stream and the commit id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream where we can find the commit
|
|
||||||
commit_id {str} -- the id of the commit you want to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Commit -- the retrieved commit object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Get"})
|
|
||||||
return super().get(stream_id, commit_id)
|
|
||||||
|
|
||||||
def list(self, stream_id: str, limit: int = 10) -> List[Commit]:
|
|
||||||
"""
|
|
||||||
Get a list of commits on a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream where the commits are
|
|
||||||
limit {int} -- the maximum number of commits to fetch (default = 10)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Commit] -- a list of the most recent commit objects
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit List"})
|
|
||||||
return super().list(stream_id, limit)
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
object_id: str,
|
|
||||||
branch_name: str = "main",
|
|
||||||
message: str = "",
|
|
||||||
source_application: str = "python",
|
|
||||||
parents: Optional[List[str]] = None,
|
|
||||||
) -> Union[str, SpeckleException]:
|
|
||||||
"""
|
|
||||||
Creates a commit on a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream you want to commit to
|
|
||||||
object_id {str} -- the hash of your commit object
|
|
||||||
branch_name {str}
|
|
||||||
-- the name of the branch to commit to (defaults to "main")
|
|
||||||
message {str}
|
|
||||||
-- optional: a message to give more information about the commit
|
|
||||||
source_application{str}
|
|
||||||
-- optional: the application from which the commit was created
|
|
||||||
(defaults to "python")
|
|
||||||
parents {List[str]} -- optional: the id of the parent commits
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the id of the created commit
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Create"})
|
|
||||||
return super().create(
|
|
||||||
stream_id, object_id, branch_name, message, source_application, parents
|
|
||||||
)
|
|
||||||
|
|
||||||
def update(self, stream_id: str, commit_id: str, message: str) -> bool:
|
|
||||||
"""
|
|
||||||
Update a commit
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream that contains the commit you'd like to update
|
|
||||||
commit_id {str} -- the id of the commit you'd like to update
|
|
||||||
message {str} -- the updated commit message
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation succeeded
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Update"})
|
|
||||||
return super().update(stream_id, commit_id, message)
|
|
||||||
|
|
||||||
def delete(self, stream_id: str, commit_id: str) -> bool:
|
|
||||||
"""
|
|
||||||
Delete a commit
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream that contains the commit you'd like to delete
|
|
||||||
commit_id {str} -- the id of the commit you'd like to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation succeeded
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Delete"})
|
|
||||||
return super().delete(stream_id, commit_id)
|
|
||||||
|
|
||||||
def received(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
commit_id: str,
|
|
||||||
source_application: str = "python",
|
|
||||||
message: Optional[str] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""
|
|
||||||
Mark a commit object a received by the source application.
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Received"})
|
|
||||||
return super().received(stream_id, commit_id, source_application, message)
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
from typing import Dict, List
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.objects.base import Base
|
|
||||||
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.object import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for objects"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
)
|
|
||||||
self.schema = Base
|
|
||||||
|
|
||||||
def get(self, stream_id: str, object_id: str) -> Base:
|
|
||||||
"""
|
|
||||||
Get a stream object
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream for the object
|
|
||||||
object_id {str} -- the hash of the object you want to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Base -- the returned Base object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Object Get"})
|
|
||||||
return super().get(stream_id, object_id)
|
|
||||||
|
|
||||||
def create(self, stream_id: str, objects: List[Dict]) -> str:
|
|
||||||
"""
|
|
||||||
Not advised - generally, you want to use `operations.send()`.
|
|
||||||
|
|
||||||
Create a new object on a stream.
|
|
||||||
To send a base object, you can prepare it by running it through the
|
|
||||||
`BaseObjectSerializer.traverse_base()` function to get a valid (serialisable)
|
|
||||||
object to send.
|
|
||||||
|
|
||||||
NOTE: this does not create a commit - you can create one with
|
|
||||||
`SpeckleClient.commit.create`.
|
|
||||||
Dynamic fields will be located in the 'data' dict of the received `Base` object
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream you want to send the object to
|
|
||||||
objects {List[Dict]}
|
|
||||||
-- a list of base dictionary objects (NOTE: must be json serialisable)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the id of the object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Object Create"})
|
|
||||||
return super().create(stream_id, objects)
|
|
||||||
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import ActivityCollection, LimitedUser
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.other_user import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for other users, that are not the currently active user."""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = LimitedUser
|
|
||||||
|
|
||||||
def get(self, id: str) -> LimitedUser:
|
|
||||||
"""
|
|
||||||
Gets the profile of another user.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
LimitedUser -- the retrieved profile of another user
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Other User Get"})
|
|
||||||
return super().get(id)
|
|
||||||
|
|
||||||
def search(
|
|
||||||
self, search_query: str, limit: int = 25
|
|
||||||
) -> Union[List[LimitedUser], SpeckleException]:
|
|
||||||
"""Searches for user by name or email. The search query must be at least
|
|
||||||
3 characters long
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
Returns:
|
|
||||||
List[LimitedUser] -- a list of User objects that match the search query
|
|
||||||
"""
|
|
||||||
if len(search_query) < 3:
|
|
||||||
return SpeckleException(
|
|
||||||
message="User search query must be at least 3 characters"
|
|
||||||
)
|
|
||||||
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Other User Search"})
|
|
||||||
return super().search(search_query, limit)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
user_id: str,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
) -> ActivityCollection:
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of
|
|
||||||
any tz as they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime} -- latest cutoff for activity
|
|
||||||
(ie: return all activity _before_ this time)
|
|
||||||
after {datetime} -- oldest cutoff for activity
|
|
||||||
(ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Other User Activity"})
|
|
||||||
return super().activity(user_id, limit, action_type, before, after, cursor)
|
|
||||||
|
|
||||||
@@ -1,76 +0,0 @@
|
|||||||
import re
|
|
||||||
from typing import Any, Dict, List, Tuple
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import ServerInfo
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import GraphQLException
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.server import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for the server"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
)
|
|
||||||
|
|
||||||
def get(self) -> ServerInfo:
|
|
||||||
"""Get the server info
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
dict -- the server info in dictionary form
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Server Get"})
|
|
||||||
return super().get()
|
|
||||||
|
|
||||||
def version(self) -> Tuple[Any, ...]:
|
|
||||||
"""Get the server version
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
the server version in the format (major, minor, patch, (tag, build))
|
|
||||||
eg (2, 6, 3) for a stable build and (2, 6, 4, 'alpha', 4711) for alpha
|
|
||||||
"""
|
|
||||||
# not tracking as it will be called along with other mutations / queries as a check
|
|
||||||
return super().version()
|
|
||||||
|
|
||||||
def apps(self) -> Dict:
|
|
||||||
"""Get the apps registered on the server
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
dict -- a dictionary of apps registered on the server
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Server Apps"})
|
|
||||||
return super().apps()
|
|
||||||
|
|
||||||
def create_token(self, name: str, scopes: List[str], lifespan: int) -> str:
|
|
||||||
"""Create a personal API token
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
scopes {List[str]} -- the scopes to grant with this token
|
|
||||||
name {str} -- a name for your new token
|
|
||||||
lifespan {int} -- duration before the token expires
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the new API token. note: this is the only time you'll see the token!
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Server Create Token"})
|
|
||||||
return super().create_token(name, scopes, lifespan)
|
|
||||||
|
|
||||||
def revoke_token(self, token: str) -> bool:
|
|
||||||
"""Revokes (deletes) a personal API token
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str} -- the token to revoke (delete)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the token was successfully deleted
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Server Revoke Token"})
|
|
||||||
return super().revoke_token(token)
|
|
||||||
@@ -1,304 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import ActivityCollection, PendingStreamCollaborator, Stream
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, UnsupportedException
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.stream import Resource as CoreResource
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for streams"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.schema = Stream
|
|
||||||
|
|
||||||
def get(self, id: str, branch_limit: int = 10, commit_limit: int = 10) -> Stream:
|
|
||||||
"""Get the specified stream from the server
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the stream id
|
|
||||||
branch_limit {int} -- the maximum number of branches to return
|
|
||||||
commit_limit {int} -- the maximum number of commits to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the retrieved stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Get"})
|
|
||||||
return super().get(id, branch_limit, commit_limit)
|
|
||||||
|
|
||||||
def list(self, stream_limit: int = 10) -> List[Stream]:
|
|
||||||
"""Get a list of the user's streams
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_limit {int} -- The maximum number of streams to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Stream] -- A list of Stream objects
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream List"})
|
|
||||||
return super().list(stream_limit)
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self,
|
|
||||||
name: str = "Anonymous Python Stream",
|
|
||||||
description: str = "No description provided",
|
|
||||||
is_public: bool = True,
|
|
||||||
) -> str:
|
|
||||||
"""Create a new stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- the name of the string
|
|
||||||
description {str} -- a short description of the stream
|
|
||||||
is_public {bool}
|
|
||||||
-- whether or not the stream can be viewed by anyone with the id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
id {str} -- the id of the newly created stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Create"})
|
|
||||||
return super().create(name, description, is_public)
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
id: str,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
description: Optional[str] = None,
|
|
||||||
is_public: Optional[bool] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""Update an existing stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the id of the stream to be updated
|
|
||||||
name {str} -- the name of the string
|
|
||||||
description {str} -- a short description of the stream
|
|
||||||
is_public {bool}
|
|
||||||
-- whether or not the stream can be viewed by anyone with the id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- whether the stream update was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Update"})
|
|
||||||
return super().update(id, name, description, is_public)
|
|
||||||
|
|
||||||
def delete(self, id: str) -> bool:
|
|
||||||
"""Delete a stream given its id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the id of the stream to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- whether the deletion was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Delete"})
|
|
||||||
return super().delete(id)
|
|
||||||
|
|
||||||
def search(
|
|
||||||
self,
|
|
||||||
search_query: str,
|
|
||||||
limit: int = 25,
|
|
||||||
branch_limit: int = 10,
|
|
||||||
commit_limit: int = 10,
|
|
||||||
):
|
|
||||||
"""Search for streams by name, description, or id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
branch_limit {int} -- the maximum number of branches to return
|
|
||||||
commit_limit {int} -- the maximum number of commits to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Stream] -- a list of Streams that match the search query
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Search"})
|
|
||||||
return super().search(search_query, limit, branch_limit, commit_limit)
|
|
||||||
|
|
||||||
def favorite(self, stream_id: str, favorited: bool = True):
|
|
||||||
"""Favorite or unfavorite the given stream.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to favorite / unfavorite
|
|
||||||
favorited {bool}
|
|
||||||
-- whether to favorite (True) or unfavorite (False) the stream
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the stream with its `id`, `name`, and `favoritedDate`
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Favorite"})
|
|
||||||
return super().favorite(stream_id, favorited)
|
|
||||||
|
|
||||||
def get_all_pending_invites(
|
|
||||||
self, stream_id: str
|
|
||||||
) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the pending invites on a stream.
|
|
||||||
You must be a `stream:owner` to query this.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream id from which to get the pending invites
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the specified stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Invite Get"})
|
|
||||||
return super().get_all_pending_invites(stream_id)
|
|
||||||
|
|
||||||
def invite(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
email: Optional[str] = None,
|
|
||||||
user_id: Optional[str] = None,
|
|
||||||
role: str = "stream:contributor", # should default be reviewer?
|
|
||||||
message: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Invite someone to a stream using either their email or user id
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to invite the user to
|
|
||||||
email {str} -- the email of the user to invite (use this OR `user_id`)
|
|
||||||
user_id {str} -- the id of the user to invite (use this OR `email`)
|
|
||||||
role {str}
|
|
||||||
-- the role to assign to the user (defaults to `stream:contributor`)
|
|
||||||
message {str}
|
|
||||||
-- a message to send along with this invite to the specified user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Invite Create"})
|
|
||||||
return super().invite(stream_id, email, user_id, role, message)
|
|
||||||
|
|
||||||
def invite_batch(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
emails: Optional[List[str]] = None,
|
|
||||||
user_ids: Optional[List[None]] = None,
|
|
||||||
message: Optional[str] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""Invite a batch of users to a specified stream.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to invite the user to
|
|
||||||
emails {List[str]}
|
|
||||||
-- the email of the user to invite (use this and/or `user_ids`)
|
|
||||||
user_id {List[str]}
|
|
||||||
-- the id of the user to invite (use this and/or `emails`)
|
|
||||||
message {str}
|
|
||||||
-- a message to send along with this invite to the specified user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Invite Batch Create"})
|
|
||||||
return super().invite_batch(stream_id, emails, user_ids, message)
|
|
||||||
|
|
||||||
def invite_cancel(self, stream_id: str, invite_id: str) -> bool:
|
|
||||||
"""Cancel an existing stream invite
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream invite
|
|
||||||
invite_id {str} -- the id of the invite to use
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- true if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Invite Cancel"})
|
|
||||||
return super().invite_cancel(stream_id, invite_id)
|
|
||||||
|
|
||||||
def invite_use(self, stream_id: str, token: str, accept: bool = True) -> bool:
|
|
||||||
"""Accept or decline a stream invite
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream for which the user has a pending invite
|
|
||||||
token {str} -- the token of the invite to use
|
|
||||||
accept {bool} -- whether or not to accept the invite (defaults to True)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- true if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Invite Use"})
|
|
||||||
return super().invite_use(stream_id, token, accept)
|
|
||||||
|
|
||||||
def update_permission(self, stream_id: str, user_id: str, role: str):
|
|
||||||
"""Updates permissions for a user on a given stream
|
|
||||||
|
|
||||||
Valid for Speckle Server >=2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to grant permissions to
|
|
||||||
user_id {str} -- the id of the user to grant permissions for
|
|
||||||
role {str} -- the role to grant the user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Permission Update", "role": role})
|
|
||||||
return super().update_permission(stream_id, user_id, role)
|
|
||||||
|
|
||||||
def revoke_permission(self, stream_id: str, user_id: str):
|
|
||||||
"""Revoke permissions from a user on a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to revoke permissions from
|
|
||||||
user_id {str} -- the id of the user to revoke permissions from
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Permission Revoke"})
|
|
||||||
return super().revoke_permission(stream_id, user_id)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
limit: int = 20,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz
|
|
||||||
as they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
stream_id {str} -- the id of the stream to get activity from
|
|
||||||
action_type {str}
|
|
||||||
-- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime}
|
|
||||||
-- latest cutoff for activity (ie: return all activity _before_ this time)
|
|
||||||
after {datetime}
|
|
||||||
-- oldest cutoff for activity (ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Stream Activity"})
|
|
||||||
return super().activity(stream_id, action_type, limit, before, after, cursor)
|
|
||||||
|
|
||||||
@@ -1,109 +0,0 @@
|
|||||||
from functools import wraps
|
|
||||||
from typing import Callable, Dict, List, Optional, Union
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
from graphql import DocumentNode
|
|
||||||
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.api.resources.stream import Stream
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.core.api.resources.subscriptions import Resource as CoreResource
|
|
||||||
|
|
||||||
def check_wsclient(function):
|
|
||||||
@wraps(function)
|
|
||||||
async def check_wsclient_wrapper(self, *args, **kwargs):
|
|
||||||
if self.client is None:
|
|
||||||
raise SpeckleException(
|
|
||||||
"You must authenticate before you can subscribe to events"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return await function(self, *args, **kwargs)
|
|
||||||
|
|
||||||
return check_wsclient_wrapper
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for subscriptions"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
)
|
|
||||||
|
|
||||||
@check_wsclient
|
|
||||||
async def stream_added(self, callback: Optional[Callable] = None):
|
|
||||||
"""Subscribes to new stream added event for your profile.
|
|
||||||
Use this to display an up-to-date list of streams.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
callback {Callable[Stream]} -- a function that takes the updated stream
|
|
||||||
as an argument and executes each time a stream is added
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the update stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Subscription Stream Added"})
|
|
||||||
return super().stream_added(callback)
|
|
||||||
|
|
||||||
@check_wsclient
|
|
||||||
async def stream_updated(self, id: str, callback: Optional[Callable] = None):
|
|
||||||
"""
|
|
||||||
Subscribes to stream updated event.
|
|
||||||
Use this in clients/components that pertain only to this stream.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the stream id of the stream to subscribe to
|
|
||||||
callback {Callable[Stream]}
|
|
||||||
-- a function that takes the updated stream
|
|
||||||
as an argument and executes each time the stream is updated
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the update stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Subscription Stream Updated"})
|
|
||||||
return super().stream_updated(id, callback)
|
|
||||||
|
|
||||||
@check_wsclient
|
|
||||||
async def stream_removed(self, callback: Optional[Callable] = None):
|
|
||||||
"""Subscribes to stream removed event for your profile.
|
|
||||||
Use this to display an up-to-date list of streams for your profile.
|
|
||||||
NOTE: If someone revokes your permissions on a stream,
|
|
||||||
this subscription will be triggered with an extra value of revokedBy
|
|
||||||
in the payload.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
callback {Callable[Dict]}
|
|
||||||
-- a function that takes the returned dict as an argument
|
|
||||||
and executes each time a stream is removed
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
dict -- dict containing 'id' of stream removed and optionally 'revokedBy'
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Subscription Stream Removed"})
|
|
||||||
return super().stream_removed(callback)
|
|
||||||
|
|
||||||
@check_wsclient
|
|
||||||
async def subscribe(
|
|
||||||
self,
|
|
||||||
query: DocumentNode,
|
|
||||||
params: Optional[Dict] = None,
|
|
||||||
callback: Optional[Callable] = None,
|
|
||||||
return_type: Optional[Union[str, List]] = None,
|
|
||||||
schema=None,
|
|
||||||
parse_response: bool = True,
|
|
||||||
):
|
|
||||||
# if self.client.transport.websocket is None:
|
|
||||||
# TODO: add multiple subs to the same ws connection
|
|
||||||
async with self.client as session:
|
|
||||||
async for res in session.subscribe(query, variable_values=params):
|
|
||||||
res = self._step_into_response(response=res, return_type=return_type)
|
|
||||||
if parse_response:
|
|
||||||
res = self._parse_response(response=res, schema=schema)
|
|
||||||
if callback is not None:
|
|
||||||
callback(res)
|
|
||||||
else:
|
|
||||||
return res
|
|
||||||
@@ -1,158 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.api.models import ActivityCollection, PendingStreamCollaborator, User
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.core.api.resources.user import Resource as CoreResource
|
|
||||||
|
|
||||||
DEPRECATION_VERSION = "2.9.0"
|
|
||||||
DEPRECATION_TEXT = (
|
|
||||||
"The user resource is deprecated, please use the active_user or other_user"
|
|
||||||
" resources"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
|
||||||
"""API Access class for users"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = User
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get(self, id: Optional[str] = None) -> User:
|
|
||||||
"""
|
|
||||||
Gets the profile of a user.
|
|
||||||
If no id argument is provided, will return the current authenticated
|
|
||||||
user's profile (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
User -- the retrieved user
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Get_deprecated"})
|
|
||||||
return super().get(id)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def search(
|
|
||||||
self, search_query: str, limit: int = 25
|
|
||||||
) -> Union[List[User], SpeckleException]:
|
|
||||||
"""
|
|
||||||
Searches for user by name or email.
|
|
||||||
The search query must be at least 3 characters long
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
Returns:
|
|
||||||
List[User] -- a list of User objects that match the search query
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Search_deprecated"})
|
|
||||||
return super().search(search_query, limit)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
company: Optional[str] = None,
|
|
||||||
bio: Optional[str] = None,
|
|
||||||
avatar: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Updates your user profile. All arguments are optional.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- your name
|
|
||||||
company {str} -- the company you may or may not work for
|
|
||||||
bio {str} -- tell us about yourself
|
|
||||||
avatar {str} -- a nice photo of yourself
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if your profile was updated successfully
|
|
||||||
"""
|
|
||||||
#metrics.track(metrics.USER, self.account, {"name": "update"})
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Update_deprecated"})
|
|
||||||
return super().update(name, company, bio, avatar)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
user_id: Optional[str] = None,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
If no id argument is provided, will return the current authenticated
|
|
||||||
user's activity (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz as
|
|
||||||
they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime}
|
|
||||||
-- latest cutoff for activity (ie: return all activity _before_ this time)
|
|
||||||
after {datetime}
|
|
||||||
-- oldest cutoff for activity (ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Activity_deprecated"})
|
|
||||||
return super().activity(user_id, limit, action_type, before, after, cursor)
|
|
||||||
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the active user's pending stream invites
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the current user
|
|
||||||
"""
|
|
||||||
#metrics.track(metrics.INVITE, self.account, {"name": "get"})
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User GetAllInvites_deprecated"})
|
|
||||||
return super().get_all_pending_invites()
|
|
||||||
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get_pending_invite(
|
|
||||||
self, stream_id: str, token: Optional[str] = None
|
|
||||||
) -> Optional[PendingStreamCollaborator]:
|
|
||||||
"""Get a particular pending invite for the active user on a given stream.
|
|
||||||
If no invite_id is provided, any valid invite will be returned.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to look for invites on
|
|
||||||
token {str} -- the token of the invite to look for (optional)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PendingStreamCollaborator
|
|
||||||
-- the invite for the given stream (or None if it isn't found)
|
|
||||||
"""
|
|
||||||
#metrics.track(metrics.INVITE, self.account, {"name": "get"})
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User GetInvite_deprecated"})
|
|
||||||
return super().get_pending_invite(stream_id, token)
|
|
||||||
|
|
||||||
@@ -1,94 +0,0 @@
|
|||||||
from urllib.parse import unquote, urlparse
|
|
||||||
from warnings import warn
|
|
||||||
|
|
||||||
from specklepy.api.client import SpeckleClient
|
|
||||||
from specklepy.api.credentials import (
|
|
||||||
Account,
|
|
||||||
get_account_from_token,
|
|
||||||
get_local_accounts,
|
|
||||||
)
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
from specklepy.transports.server.server import ServerTransport
|
|
||||||
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.core.api.wrapper import StreamWrapper as CoreStreamWrapper
|
|
||||||
|
|
||||||
class StreamWrapper(CoreStreamWrapper):
|
|
||||||
"""
|
|
||||||
The `StreamWrapper` gives you some handy helpers to deal with urls and
|
|
||||||
get authenticated clients and transports.
|
|
||||||
|
|
||||||
Construct a `StreamWrapper` with a stream, branch, commit, or object URL.
|
|
||||||
The corresponding ids will be stored
|
|
||||||
in the wrapper. If you have local accounts on the machine,
|
|
||||||
you can use the `get_account` and `get_client` methods
|
|
||||||
to get a local account for the server. You can also pass a token into `get_client`
|
|
||||||
if you don't have a corresponding
|
|
||||||
local account for the server.
|
|
||||||
|
|
||||||
```py
|
|
||||||
from specklepy.api.wrapper import StreamWrapper
|
|
||||||
|
|
||||||
# provide any stream, branch, commit, object, or globals url
|
|
||||||
wrapper = StreamWrapper("https://speckle.xyz/streams/3073b96e86/commits/604bea8cc6")
|
|
||||||
|
|
||||||
# get an authenticated SpeckleClient if you have a local account for the server
|
|
||||||
client = wrapper.get_client()
|
|
||||||
|
|
||||||
# get an authenticated ServerTransport if you have a local account for the server
|
|
||||||
transport = wrapper.get_transport()
|
|
||||||
```
|
|
||||||
"""
|
|
||||||
|
|
||||||
stream_url: str = None
|
|
||||||
use_ssl: bool = True
|
|
||||||
host: str = None
|
|
||||||
stream_id: str = None
|
|
||||||
commit_id: str = None
|
|
||||||
object_id: str = None
|
|
||||||
branch_name: str = None
|
|
||||||
_client: SpeckleClient = None
|
|
||||||
_account: Account = None
|
|
||||||
|
|
||||||
def __init__(self, url: str) -> None:
|
|
||||||
super().__init__(url = url)
|
|
||||||
|
|
||||||
def get_account(self, token: str = None) -> Account:
|
|
||||||
"""
|
|
||||||
Gets an account object for this server from the local accounts db
|
|
||||||
(added via Speckle Manager or a json file)
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Stream Wrapper Get Account"})
|
|
||||||
return super().get_account(token)
|
|
||||||
|
|
||||||
def get_client(self, token: str = None) -> SpeckleClient:
|
|
||||||
"""
|
|
||||||
Gets an authenticated client for this server.
|
|
||||||
You may provide a token if there aren't any local accounts on this
|
|
||||||
machine. If no account is found and no token is provided,
|
|
||||||
an unauthenticated client is returned.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str}
|
|
||||||
-- optional token if no local account is available (defaults to None)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
SpeckleClient
|
|
||||||
-- authenticated with a corresponding local account or the provided token
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Stream Wrapper Get Client"})
|
|
||||||
return super().get_client(token)
|
|
||||||
|
|
||||||
def get_transport(self, token: str = None) -> ServerTransport:
|
|
||||||
"""
|
|
||||||
Gets a server transport for this stream using an authenticated client.
|
|
||||||
If there is no local account for this
|
|
||||||
server and the client was not authenticated with a token,
|
|
||||||
this will throw an exception.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
ServerTransport -- constructed for this stream
|
|
||||||
with a pre-authenticated client
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Stream Wrapper Get Transport"})
|
|
||||||
return super().get_transport(token)
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
"""
|
|
||||||
This is the Core SDK module of `specklepy`.
|
|
||||||
|
|
||||||
This module should be kept in sync with the functionalities of our other SDKs especially
|
|
||||||
C# Core https://github.com/specklesystems/speckle-sharp/tree/main/Core/Core
|
|
||||||
"""
|
|
||||||
@@ -1,242 +0,0 @@
|
|||||||
import re
|
|
||||||
from typing import Dict
|
|
||||||
from warnings import warn
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import Client
|
|
||||||
from gql.transport.exceptions import TransportServerError
|
|
||||||
from gql.transport.requests import RequestsHTTPTransport
|
|
||||||
from gql.transport.websockets import WebsocketsTransport
|
|
||||||
|
|
||||||
from specklepy.core.api import resources
|
|
||||||
from specklepy.core.api.credentials import Account, get_account_from_token
|
|
||||||
from specklepy.core.api.resources import (
|
|
||||||
user,
|
|
||||||
active_user,
|
|
||||||
branch,
|
|
||||||
commit,
|
|
||||||
object,
|
|
||||||
other_user,
|
|
||||||
server,
|
|
||||||
stream,
|
|
||||||
subscriptions,
|
|
||||||
)
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
|
|
||||||
|
|
||||||
class SpeckleClient:
|
|
||||||
"""
|
|
||||||
The `SpeckleClient` is your entry point for interacting with
|
|
||||||
your Speckle Server's GraphQL API.
|
|
||||||
You'll need to have access to a server to use it,
|
|
||||||
or you can use our public server `speckle.xyz`.
|
|
||||||
|
|
||||||
To authenticate the client, you'll need to have downloaded
|
|
||||||
the [Speckle Manager](https://speckle.guide/#speckle-manager)
|
|
||||||
and added your account.
|
|
||||||
|
|
||||||
```py
|
|
||||||
from specklepy.api.client import SpeckleClient
|
|
||||||
from specklepy.api.credentials import get_default_account
|
|
||||||
|
|
||||||
# initialise the client
|
|
||||||
client = SpeckleClient(host="speckle.xyz") # or whatever your host is
|
|
||||||
# client = SpeckleClient(host="localhost:3000", use_ssl=False) or use local server
|
|
||||||
|
|
||||||
# authenticate the client with an account (account has been added in Speckle Manager)
|
|
||||||
account = get_default_account()
|
|
||||||
client.authenticate_with_account(account)
|
|
||||||
|
|
||||||
# create a new stream. this returns the stream id
|
|
||||||
new_stream_id = client.stream.create(name="a shiny new stream")
|
|
||||||
|
|
||||||
# use that stream id to get the stream from the server
|
|
||||||
new_stream = client.stream.get(id=new_stream_id)
|
|
||||||
```
|
|
||||||
"""
|
|
||||||
|
|
||||||
DEFAULT_HOST = "speckle.xyz"
|
|
||||||
USE_SSL = True
|
|
||||||
|
|
||||||
def __init__(self, host: str = DEFAULT_HOST, use_ssl: bool = USE_SSL) -> None:
|
|
||||||
ws_protocol = "ws"
|
|
||||||
http_protocol = "http"
|
|
||||||
|
|
||||||
if use_ssl:
|
|
||||||
ws_protocol = "wss"
|
|
||||||
http_protocol = "https"
|
|
||||||
|
|
||||||
# sanitise host input by removing protocol and trailing slash
|
|
||||||
host = re.sub(r"((^\w+:|^)\/\/)|(\/$)", "", host)
|
|
||||||
|
|
||||||
self.url = f"{http_protocol}://{host}"
|
|
||||||
self.graphql = f"{self.url}/graphql"
|
|
||||||
self.ws_url = f"{ws_protocol}://{host}/graphql"
|
|
||||||
self.account = Account()
|
|
||||||
|
|
||||||
self.httpclient = Client(
|
|
||||||
transport=RequestsHTTPTransport(url=self.graphql, verify=True, retries=3)
|
|
||||||
)
|
|
||||||
self.wsclient = None
|
|
||||||
|
|
||||||
self._init_resources()
|
|
||||||
|
|
||||||
# ? Check compatibility with the server - i think we can skip this at this point? save a request
|
|
||||||
# try:
|
|
||||||
# server_info = self.server.get()
|
|
||||||
# if isinstance(server_info, Exception):
|
|
||||||
# raise server_info
|
|
||||||
# if not isinstance(server_info, ServerInfo):
|
|
||||||
# raise Exception("Couldn't get ServerInfo")
|
|
||||||
# except Exception as ex:
|
|
||||||
# raise SpeckleException(
|
|
||||||
# f"{self.url} is not a compatible Speckle Server", ex
|
|
||||||
# ) from ex
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return (
|
|
||||||
f"SpeckleClient( server: {self.url}, authenticated:"
|
|
||||||
f" {self.account.token is not None} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(
|
|
||||||
version="2.6.0",
|
|
||||||
reason=(
|
|
||||||
"Renamed: please use `authenticate_with_account` or"
|
|
||||||
" `authenticate_with_token` instead."
|
|
||||||
),
|
|
||||||
)
|
|
||||||
def authenticate(self, token: str) -> None:
|
|
||||||
"""Authenticate the client using a personal access token
|
|
||||||
The token is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str} -- an api token
|
|
||||||
"""
|
|
||||||
self.authenticate_with_token(token)
|
|
||||||
self._set_up_client()
|
|
||||||
|
|
||||||
def authenticate_with_token(self, token: str) -> None:
|
|
||||||
"""
|
|
||||||
Authenticate the client using a personal access token.
|
|
||||||
The token is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str} -- an api token
|
|
||||||
"""
|
|
||||||
self.account = get_account_from_token(token, self.url)
|
|
||||||
self._set_up_client()
|
|
||||||
|
|
||||||
def authenticate_with_account(self, account: Account) -> None:
|
|
||||||
"""Authenticate the client using an Account object
|
|
||||||
The account is saved in the client object and a synchronous GraphQL
|
|
||||||
entrypoint is created
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
account {Account} -- the account object which can be found with
|
|
||||||
`get_default_account` or `get_local_accounts`
|
|
||||||
"""
|
|
||||||
self.account = account
|
|
||||||
self._set_up_client()
|
|
||||||
|
|
||||||
def _set_up_client(self) -> None:
|
|
||||||
headers = {
|
|
||||||
"Authorization": f"Bearer {self.account.token}",
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
"apollographql-client-name": metrics.HOST_APP,
|
|
||||||
"apollographql-client-version": metrics.HOST_APP_VERSION,
|
|
||||||
}
|
|
||||||
httptransport = RequestsHTTPTransport(
|
|
||||||
url=self.graphql, headers=headers, verify=True, retries=3
|
|
||||||
)
|
|
||||||
wstransport = WebsocketsTransport(
|
|
||||||
url=self.ws_url,
|
|
||||||
init_payload={"Authorization": f"Bearer {self.account.token}"},
|
|
||||||
)
|
|
||||||
self.httpclient = Client(transport=httptransport)
|
|
||||||
self.wsclient = Client(transport=wstransport)
|
|
||||||
|
|
||||||
self._init_resources()
|
|
||||||
|
|
||||||
try:
|
|
||||||
user_or_error = self.active_user.get()
|
|
||||||
if isinstance(user_or_error, SpeckleException):
|
|
||||||
if isinstance(user_or_error.exception, TransportServerError):
|
|
||||||
raise user_or_error.exception
|
|
||||||
else:
|
|
||||||
raise user_or_error
|
|
||||||
except TransportServerError as ex:
|
|
||||||
if ex.code == 403:
|
|
||||||
warn(
|
|
||||||
SpeckleWarning(
|
|
||||||
"Possibly invalid token - could not authenticate Speckle Client"
|
|
||||||
f" for server {self.url}"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
raise ex
|
|
||||||
|
|
||||||
def execute_query(self, query: str) -> Dict:
|
|
||||||
return self.httpclient.execute(query)
|
|
||||||
|
|
||||||
def _init_resources(self) -> None:
|
|
||||||
self.server = server.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
server_version = None
|
|
||||||
try:
|
|
||||||
server_version = self.server.version()
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
self.user = user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.other_user = other_user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.active_user = active_user.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.stream = stream.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.url,
|
|
||||||
client=self.httpclient,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.commit = commit.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.branch = branch.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.object = object.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
self.subscribe = subscriptions.Resource(
|
|
||||||
account=self.account,
|
|
||||||
basepath=self.ws_url,
|
|
||||||
client=self.wsclient,
|
|
||||||
)
|
|
||||||
|
|
||||||
def __getattr__(self, name):
|
|
||||||
try:
|
|
||||||
attr = getattr(resources, name)
|
|
||||||
return attr.Resource(
|
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
|
||||||
)
|
|
||||||
except AttributeError:
|
|
||||||
raise SpeckleException(
|
|
||||||
f"Method {name} is not supported by the SpeckleClient class"
|
|
||||||
)
|
|
||||||
@@ -1,154 +0,0 @@
|
|||||||
import os
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from pydantic import BaseModel, Field # pylint: disable=no-name-in-module
|
|
||||||
|
|
||||||
from specklepy.core.api.models import ServerInfo
|
|
||||||
from specklepy.core.helpers import speckle_path_provider
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
|
|
||||||
class UserInfo(BaseModel):
|
|
||||||
name: Optional[str] = None
|
|
||||||
email: Optional[str] = None
|
|
||||||
company: Optional[str] = None
|
|
||||||
id: Optional[str] = None
|
|
||||||
|
|
||||||
|
|
||||||
class Account(BaseModel):
|
|
||||||
isDefault: bool = False
|
|
||||||
token: Optional[str] = None
|
|
||||||
refreshToken: Optional[str] = None
|
|
||||||
serverInfo: ServerInfo = Field(default_factory=ServerInfo)
|
|
||||||
userInfo: UserInfo = Field(default_factory=UserInfo)
|
|
||||||
id: Optional[str] = None
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return (
|
|
||||||
f"Account(email: {self.userInfo.email}, server: {self.serverInfo.url},"
|
|
||||||
f" isDefault: {self.isDefault})"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_token(cls, token: str, server_url: str = None):
|
|
||||||
acct = cls(token=token)
|
|
||||||
acct.serverInfo.url = server_url
|
|
||||||
return acct
|
|
||||||
|
|
||||||
|
|
||||||
def get_local_accounts(base_path: Optional[str] = None) -> List[Account]:
|
|
||||||
"""Gets all the accounts present in this environment
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Account] -- list of all local accounts or an empty list if
|
|
||||||
no accounts were found
|
|
||||||
"""
|
|
||||||
accounts: List[Account] = []
|
|
||||||
try:
|
|
||||||
account_storage = SQLiteTransport(scope="Accounts", base_path=base_path)
|
|
||||||
res = account_storage.get_all_objects()
|
|
||||||
account_storage.close()
|
|
||||||
if res:
|
|
||||||
accounts.extend(Account.model_validate_json(r[1]) for r in res)
|
|
||||||
except SpeckleException:
|
|
||||||
# cannot open SQLiteTransport, probably because of the lack
|
|
||||||
# of disk write permissions
|
|
||||||
pass
|
|
||||||
|
|
||||||
json_acct_files = []
|
|
||||||
json_path = str(speckle_path_provider.accounts_folder_path())
|
|
||||||
try:
|
|
||||||
os.makedirs(json_path, exist_ok=True)
|
|
||||||
json_acct_files.extend(
|
|
||||||
file for file in os.listdir(json_path) if file.endswith(".json")
|
|
||||||
)
|
|
||||||
|
|
||||||
except Exception:
|
|
||||||
# cannot find or get the json account paths
|
|
||||||
pass
|
|
||||||
|
|
||||||
if json_acct_files:
|
|
||||||
try:
|
|
||||||
accounts.extend(
|
|
||||||
Account.model_validate_json(Path(json_path, json_file).read_text())
|
|
||||||
# Account.parse_file(os.path.join(json_path, json_file))
|
|
||||||
for json_file in json_acct_files
|
|
||||||
)
|
|
||||||
except Exception as ex:
|
|
||||||
raise SpeckleException(
|
|
||||||
"Invalid json accounts could not be read. Please fix or remove them.",
|
|
||||||
ex,
|
|
||||||
) from ex
|
|
||||||
|
|
||||||
return accounts
|
|
||||||
|
|
||||||
|
|
||||||
def get_default_account(base_path: Optional[str] = None) -> Optional[Account]:
|
|
||||||
"""
|
|
||||||
Gets this environment's default account if any. If there is no default,
|
|
||||||
the first found will be returned and set as default.
|
|
||||||
Arguments:
|
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Account -- the default account or None if no local accounts were found
|
|
||||||
"""
|
|
||||||
accounts = get_local_accounts(base_path=base_path)
|
|
||||||
if not accounts:
|
|
||||||
return None
|
|
||||||
|
|
||||||
default = next((acc for acc in accounts if acc.isDefault), None)
|
|
||||||
if not default:
|
|
||||||
default = accounts[0]
|
|
||||||
default.isDefault = True
|
|
||||||
#metrics.initialise_tracker(default)
|
|
||||||
|
|
||||||
return default
|
|
||||||
|
|
||||||
|
|
||||||
def get_account_from_token(token: str, server_url: str = None) -> Account:
|
|
||||||
"""Gets the local account for the token if it exists
|
|
||||||
Arguments:
|
|
||||||
token {str} -- the api token
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Account -- the local account with this token or a shell account containing
|
|
||||||
just the token and url if no local account is found
|
|
||||||
"""
|
|
||||||
accounts = get_local_accounts()
|
|
||||||
if not accounts:
|
|
||||||
return Account.from_token(token, server_url)
|
|
||||||
|
|
||||||
acct = next((acc for acc in accounts if acc.token == token), None)
|
|
||||||
if acct:
|
|
||||||
return acct
|
|
||||||
|
|
||||||
if server_url:
|
|
||||||
url = server_url.lower()
|
|
||||||
acct = next(
|
|
||||||
(acc for acc in accounts if url in acc.serverInfo.url.lower()), None
|
|
||||||
)
|
|
||||||
if acct:
|
|
||||||
return acct
|
|
||||||
|
|
||||||
return Account.from_token(token, server_url)
|
|
||||||
|
|
||||||
|
|
||||||
class StreamWrapper:
|
|
||||||
def __init__(self, url: str = None) -> None:
|
|
||||||
raise SpeckleException(
|
|
||||||
message=(
|
|
||||||
"The StreamWrapper has moved as of v2.6.0! Please import from"
|
|
||||||
" specklepy.api.wrapper"
|
|
||||||
),
|
|
||||||
exception=DeprecationWarning(),
|
|
||||||
)
|
|
||||||
@@ -1,116 +0,0 @@
|
|||||||
from dataclasses import dataclass
|
|
||||||
from enum import Enum
|
|
||||||
from unicodedata import name
|
|
||||||
|
|
||||||
|
|
||||||
class HostAppVersion(Enum):
|
|
||||||
v = "v"
|
|
||||||
v6 = "v6"
|
|
||||||
v7 = "v7"
|
|
||||||
v2019 = "v2019"
|
|
||||||
v2020 = "v2020"
|
|
||||||
v2021 = "v2021"
|
|
||||||
v2022 = "v2022"
|
|
||||||
v2023 = "v2023"
|
|
||||||
v2024 = "v2024"
|
|
||||||
v2025 = "v2025"
|
|
||||||
vSandbox = "vSandbox"
|
|
||||||
vRevit = "vRevit"
|
|
||||||
vRevit2021 = "vRevit2021"
|
|
||||||
vRevit2022 = "vRevit2022"
|
|
||||||
vRevit2023 = "vRevit2023"
|
|
||||||
vRevit2024 = "vRevit2024"
|
|
||||||
vRevit2025 = "vRevit2025"
|
|
||||||
v25 = "v25"
|
|
||||||
v26 = "v26"
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return self.value
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.value
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class HostApplication:
|
|
||||||
name: str
|
|
||||||
slug: str
|
|
||||||
|
|
||||||
def get_version(self, version: HostAppVersion) -> str:
|
|
||||||
return f"{name.replace(' ', '')}{str(version).strip('v')}"
|
|
||||||
|
|
||||||
|
|
||||||
RHINO = HostApplication("Rhino", "rhino")
|
|
||||||
GRASSHOPPER = HostApplication("Grasshopper", "grasshopper")
|
|
||||||
REVIT = HostApplication("Revit", "revit")
|
|
||||||
DYNAMO = HostApplication("Dynamo", "dynamo")
|
|
||||||
UNITY = HostApplication("Unity", "unity")
|
|
||||||
GSA = HostApplication("GSA", "gsa")
|
|
||||||
CIVIL = HostApplication("Civil 3D", "civil3d")
|
|
||||||
AUTOCAD = HostApplication("AutoCAD", "autocad")
|
|
||||||
MICROSTATION = HostApplication("MicroStation", "microstation")
|
|
||||||
OPENROADS = HostApplication("OpenRoads", "openroads")
|
|
||||||
OPENRAIL = HostApplication("OpenRail", "openrail")
|
|
||||||
OPENBUILDINGS = HostApplication("OpenBuildings", "openbuildings")
|
|
||||||
ETABS = HostApplication("ETABS", "etabs")
|
|
||||||
SAP2000 = HostApplication("SAP2000", "sap2000")
|
|
||||||
CSIBRIDGE = HostApplication("CSIBridge", "csibridge")
|
|
||||||
SAFE = HostApplication("SAFE", "safe")
|
|
||||||
TEKLASTRUCTURES = HostApplication("Tekla Structures", "teklastructures")
|
|
||||||
DXF = HostApplication("DXF Converter", "dxf")
|
|
||||||
EXCEL = HostApplication("Excel", "excel")
|
|
||||||
UNREAL = HostApplication("Unreal", "unreal")
|
|
||||||
POWERBI = HostApplication("Power BI", "powerbi")
|
|
||||||
BLENDER = HostApplication("Blender", "blender")
|
|
||||||
QGIS = HostApplication("QGIS", "qgis")
|
|
||||||
ARCGIS = HostApplication("ArcGIS", "arcgis")
|
|
||||||
SKETCHUP = HostApplication("SketchUp", "sketchup")
|
|
||||||
ARCHICAD = HostApplication("Archicad", "archicad")
|
|
||||||
TOPSOLID = HostApplication("TopSolid", "topsolid")
|
|
||||||
PYTHON = HostApplication("Python", "python")
|
|
||||||
NET = HostApplication(".NET", "net")
|
|
||||||
OTHER = HostApplication("Other", "other")
|
|
||||||
|
|
||||||
_app_name_host_app_mapping = {
|
|
||||||
"dynamo": DYNAMO,
|
|
||||||
"revit": REVIT,
|
|
||||||
"autocad": AUTOCAD,
|
|
||||||
"civil": CIVIL,
|
|
||||||
"rhino": RHINO,
|
|
||||||
"grasshopper": GRASSHOPPER,
|
|
||||||
"unity": UNITY,
|
|
||||||
"gsa": GSA,
|
|
||||||
"microstation": MICROSTATION,
|
|
||||||
"openroads": OPENROADS,
|
|
||||||
"openrail": OPENRAIL,
|
|
||||||
"openbuildings": OPENBUILDINGS,
|
|
||||||
"etabs": ETABS,
|
|
||||||
"sap": SAP2000,
|
|
||||||
"csibridge": CSIBRIDGE,
|
|
||||||
"safe": SAFE,
|
|
||||||
"teklastructures": TEKLASTRUCTURES,
|
|
||||||
"dxf": DXF,
|
|
||||||
"excel": EXCEL,
|
|
||||||
"unreal": UNREAL,
|
|
||||||
"powerbi": POWERBI,
|
|
||||||
"blender": BLENDER,
|
|
||||||
"qgis": QGIS,
|
|
||||||
"arcgis": ARCGIS,
|
|
||||||
"sketchup": SKETCHUP,
|
|
||||||
"archicad": ARCHICAD,
|
|
||||||
"topsolid": TOPSOLID,
|
|
||||||
"python": PYTHON,
|
|
||||||
"net": NET,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def get_host_app_from_string(app_name: str) -> HostApplication:
|
|
||||||
app_name = app_name.lower().replace(" ", "")
|
|
||||||
for partial_app_name, host_app in _app_name_host_app_mapping.items():
|
|
||||||
if partial_app_name in app_name:
|
|
||||||
return host_app
|
|
||||||
return HostApplication(app_name, app_name)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
print(HostAppVersion.v)
|
|
||||||
@@ -1,198 +0,0 @@
|
|||||||
from datetime import datetime
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from pydantic import BaseModel, Field
|
|
||||||
|
|
||||||
|
|
||||||
class Collaborator(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
name: Optional[str] = None
|
|
||||||
role: Optional[str] = None
|
|
||||||
avatar: Optional[str] = None
|
|
||||||
|
|
||||||
|
|
||||||
class Commit(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
message: Optional[str] = None
|
|
||||||
authorName: Optional[str] = None
|
|
||||||
authorId: Optional[str] = None
|
|
||||||
authorAvatar: Optional[str] = None
|
|
||||||
branchName: Optional[str] = None
|
|
||||||
createdAt: Optional[datetime] = None
|
|
||||||
sourceApplication: Optional[str] = None
|
|
||||||
referencedObject: Optional[str] = None
|
|
||||||
totalChildrenCount: Optional[int] = None
|
|
||||||
parents: Optional[List[str]] = None
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return (
|
|
||||||
f"Commit( id: {self.id}, message: {self.message}, referencedObject:"
|
|
||||||
f" {self.referencedObject}, authorName: {self.authorName}, branchName:"
|
|
||||||
f" {self.branchName}, createdAt: {self.createdAt} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class Commits(BaseModel):
|
|
||||||
totalCount: Optional[int] = None
|
|
||||||
cursor: Optional[datetime] = None
|
|
||||||
items: List[Commit] = []
|
|
||||||
|
|
||||||
|
|
||||||
class Object(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
speckleType: Optional[str] = None
|
|
||||||
applicationId: Optional[str] = None
|
|
||||||
totalChildrenCount: Optional[int] = None
|
|
||||||
createdAt: Optional[datetime] = None
|
|
||||||
|
|
||||||
|
|
||||||
class Branch(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
name: Optional[str] = None
|
|
||||||
description: Optional[str] = None
|
|
||||||
commits: Optional[Commits] = None
|
|
||||||
|
|
||||||
|
|
||||||
class Branches(BaseModel):
|
|
||||||
totalCount: Optional[int] = None
|
|
||||||
cursor: Optional[datetime] = None
|
|
||||||
items: List[Branch] = []
|
|
||||||
|
|
||||||
|
|
||||||
class Stream(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
name: Optional[str] = None
|
|
||||||
role: Optional[str] = None
|
|
||||||
isPublic: Optional[bool] = None
|
|
||||||
description: Optional[str] = None
|
|
||||||
createdAt: Optional[datetime] = None
|
|
||||||
updatedAt: Optional[datetime] = None
|
|
||||||
collaborators: List[Collaborator] = Field(default_factory=list)
|
|
||||||
branches: Optional[Branches] = None
|
|
||||||
commit: Optional[Commit] = None
|
|
||||||
object: Optional[Object] = None
|
|
||||||
commentCount: Optional[int] = None
|
|
||||||
favoritedDate: Optional[datetime] = None
|
|
||||||
favoritesCount: Optional[int] = None
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return (
|
|
||||||
f"Stream( id: {self.id}, name: {self.name}, description:"
|
|
||||||
f" {self.description}, isPublic: {self.isPublic})"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class Streams(BaseModel):
|
|
||||||
totalCount: Optional[int] = None
|
|
||||||
cursor: Optional[datetime] = None
|
|
||||||
items: List[Stream] = []
|
|
||||||
|
|
||||||
|
|
||||||
class User(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
email: Optional[str] = None
|
|
||||||
name: Optional[str] = None
|
|
||||||
bio: Optional[str] = None
|
|
||||||
company: Optional[str] = None
|
|
||||||
avatar: Optional[str] = None
|
|
||||||
verified: Optional[bool] = None
|
|
||||||
role: Optional[str] = None
|
|
||||||
streams: Optional[Streams] = None
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return (
|
|
||||||
f"User( id: {self.id}, name: {self.name}, email: {self.email}, company:"
|
|
||||||
f" {self.company} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class LimitedUser(BaseModel):
|
|
||||||
"""Limited user type, for showing public info about a user to another user."""
|
|
||||||
|
|
||||||
id: str
|
|
||||||
name: Optional[str] = None
|
|
||||||
bio: Optional[str] = None
|
|
||||||
company: Optional[str] = None
|
|
||||||
avatar: Optional[str] = None
|
|
||||||
verified: Optional[bool] = None
|
|
||||||
role: Optional[str] = None
|
|
||||||
|
|
||||||
|
|
||||||
class PendingStreamCollaborator(BaseModel):
|
|
||||||
id: Optional[str] = None
|
|
||||||
inviteId: Optional[str] = None
|
|
||||||
streamId: Optional[str] = None
|
|
||||||
streamName: Optional[str] = None
|
|
||||||
title: Optional[str] = None
|
|
||||||
role: Optional[str] = None
|
|
||||||
invitedBy: Optional[User] = None
|
|
||||||
user: Optional[User] = None
|
|
||||||
token: Optional[str] = None
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return (
|
|
||||||
f"PendingStreamCollaborator( inviteId: {self.inviteId}, streamId:"
|
|
||||||
f" {self.streamId}, role: {self.role}, title: {self.title}, invitedBy:"
|
|
||||||
f" {self.user.name if self.user else None})"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class Activity(BaseModel):
|
|
||||||
actionType: Optional[str] = None
|
|
||||||
info: Optional[dict] = None
|
|
||||||
userId: Optional[str] = None
|
|
||||||
streamId: Optional[str] = None
|
|
||||||
resourceId: Optional[str] = None
|
|
||||||
resourceType: Optional[str] = None
|
|
||||||
message: Optional[str] = None
|
|
||||||
time: Optional[datetime] = None
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return (
|
|
||||||
f"Activity( streamId: {self.streamId}, actionType: {self.actionType},"
|
|
||||||
f" message: {self.message}, userId: {self.userId} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class ActivityCollection(BaseModel):
|
|
||||||
totalCount: Optional[int] = None
|
|
||||||
items: Optional[List[Activity]] = None
|
|
||||||
cursor: Optional[datetime] = None
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return (
|
|
||||||
f"ActivityCollection( totalCount: {self.totalCount}, items:"
|
|
||||||
f" {len(self.items) if self.items else 0}, cursor:"
|
|
||||||
f" {self.cursor.isoformat() if self.cursor else None} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class ServerInfo(BaseModel):
|
|
||||||
name: Optional[str] = None
|
|
||||||
company: Optional[str] = None
|
|
||||||
url: Optional[str] = None
|
|
||||||
description: Optional[str] = None
|
|
||||||
adminContact: Optional[str] = None
|
|
||||||
canonicalUrl: Optional[str] = None
|
|
||||||
roles: Optional[List[dict]] = None
|
|
||||||
scopes: Optional[List[dict]] = None
|
|
||||||
authStrategies: Optional[List[dict]] = None
|
|
||||||
version: Optional[str] = None
|
|
||||||
@@ -1,131 +0,0 @@
|
|||||||
from threading import Lock
|
|
||||||
from typing import Any, Dict, List, Optional, Tuple, Type, Union
|
|
||||||
|
|
||||||
from gql.client import Client
|
|
||||||
from gql.transport.exceptions import TransportQueryError
|
|
||||||
from graphql import DocumentNode
|
|
||||||
|
|
||||||
from specklepy.core.api.credentials import Account
|
|
||||||
from specklepy.logging.exceptions import (
|
|
||||||
GraphQLException,
|
|
||||||
SpeckleException,
|
|
||||||
UnsupportedException,
|
|
||||||
)
|
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
|
|
||||||
|
|
||||||
class ResourceBase(object):
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
account: Account,
|
|
||||||
basepath: str,
|
|
||||||
client: Client,
|
|
||||||
name: str,
|
|
||||||
server_version: Optional[Tuple[Any, ...]] = None,
|
|
||||||
) -> None:
|
|
||||||
self.account = account
|
|
||||||
self.basepath = basepath
|
|
||||||
self.client = client
|
|
||||||
self.name = name
|
|
||||||
self.server_version = server_version
|
|
||||||
self.schema: Optional[Type] = None
|
|
||||||
self.__lock = Lock()
|
|
||||||
|
|
||||||
def _step_into_response(self, response: dict, return_type: Union[str, List, None]):
|
|
||||||
"""Step into the dict to get the relevant data"""
|
|
||||||
if return_type is None:
|
|
||||||
return response
|
|
||||||
if isinstance(return_type, str):
|
|
||||||
return response[return_type]
|
|
||||||
if isinstance(return_type, List):
|
|
||||||
for key in return_type:
|
|
||||||
response = response[key]
|
|
||||||
return response
|
|
||||||
|
|
||||||
def _parse_response(self, response: Union[dict, list, None], schema=None):
|
|
||||||
"""Try to create a class instance from the response"""
|
|
||||||
if response is None:
|
|
||||||
return None
|
|
||||||
if isinstance(response, list):
|
|
||||||
return [self._parse_response(response=r, schema=schema) for r in response]
|
|
||||||
if schema:
|
|
||||||
return schema.model_validate(response)
|
|
||||||
elif self.schema:
|
|
||||||
try:
|
|
||||||
return self.schema.model_validate(response)
|
|
||||||
except Exception:
|
|
||||||
s = BaseObjectSerializer(read_transport=SQLiteTransport())
|
|
||||||
return s.recompose_base(response)
|
|
||||||
else:
|
|
||||||
return response
|
|
||||||
|
|
||||||
def make_request(
|
|
||||||
self,
|
|
||||||
query: DocumentNode,
|
|
||||||
params: Optional[Dict] = None,
|
|
||||||
return_type: Union[str, List, None] = None,
|
|
||||||
schema=None,
|
|
||||||
parse_response: bool = True,
|
|
||||||
) -> Any:
|
|
||||||
"""Executes the GraphQL query"""
|
|
||||||
try:
|
|
||||||
with self.__lock:
|
|
||||||
response = self.client.execute(query, variable_values=params)
|
|
||||||
except Exception as ex:
|
|
||||||
if isinstance(ex, TransportQueryError):
|
|
||||||
return GraphQLException(
|
|
||||||
message=(
|
|
||||||
f"Failed to execute the GraphQL {self.name} request. Errors:"
|
|
||||||
f" {ex.errors}"
|
|
||||||
),
|
|
||||||
errors=ex.errors,
|
|
||||||
data=ex.data,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return SpeckleException(
|
|
||||||
message=(
|
|
||||||
f"Failed to execute the GraphQL {self.name} request. Inner"
|
|
||||||
f" exception: {ex}"
|
|
||||||
),
|
|
||||||
exception=ex,
|
|
||||||
)
|
|
||||||
|
|
||||||
response = self._step_into_response(response=response, return_type=return_type)
|
|
||||||
|
|
||||||
if parse_response:
|
|
||||||
return self._parse_response(response=response, schema=schema)
|
|
||||||
else:
|
|
||||||
return response
|
|
||||||
|
|
||||||
def _check_server_version_at_least(
|
|
||||||
self, target_version: Tuple[Any, ...], unsupported_message: Optional[str] = None
|
|
||||||
):
|
|
||||||
"""Use this check to guard against making unsupported requests on older servers.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
target_version {tuple}
|
|
||||||
the minimum server version in the format (major, minor, patch, (tag, build))
|
|
||||||
eg (2, 6, 3) for a stable build and (2, 6, 4, 'alpha', 4711) for alpha
|
|
||||||
"""
|
|
||||||
if not unsupported_message:
|
|
||||||
unsupported_message = (
|
|
||||||
"The client method used is not supported on Speckle Server versions"
|
|
||||||
f" prior to v{'.'.join(target_version)}"
|
|
||||||
)
|
|
||||||
# if version is dev, it should be supported... (or not)
|
|
||||||
if self.server_version == ("dev",):
|
|
||||||
return
|
|
||||||
if self.server_version and self.server_version < target_version:
|
|
||||||
raise UnsupportedException(unsupported_message)
|
|
||||||
|
|
||||||
def _check_invites_supported(self):
|
|
||||||
"""Invites are only supported for Speckle Server >= 2.6.4.
|
|
||||||
Use this check to guard against making unsupported requests on older servers.
|
|
||||||
"""
|
|
||||||
self._check_server_version_at_least(
|
|
||||||
(2, 6, 4),
|
|
||||||
"Stream invites are only supported as of Speckle Server v2.6.4. Please"
|
|
||||||
" update your Speckle Server to use this method or use the"
|
|
||||||
" `grant_permission` flow instead.",
|
|
||||||
)
|
|
||||||
@@ -1,264 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.core.api.models import ActivityCollection, PendingStreamCollaborator, User
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
NAME = "active_user"
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
|
||||||
"""API Access class for users"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = User
|
|
||||||
|
|
||||||
def get(self) -> User:
|
|
||||||
"""Gets the profile of a user. If no id argument is provided,
|
|
||||||
will return the current authenticated user's profile
|
|
||||||
(as extracted from the authorization header).
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
User -- the retrieved user
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query User {
|
|
||||||
activeUser {
|
|
||||||
id
|
|
||||||
email
|
|
||||||
name
|
|
||||||
bio
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
verified
|
|
||||||
profiles
|
|
||||||
role
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {}
|
|
||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="activeUser")
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
company: Optional[str] = None,
|
|
||||||
bio: Optional[str] = None,
|
|
||||||
avatar: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Updates your user profile. All arguments are optional.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- your name
|
|
||||||
company {str} -- the company you may or may not work for
|
|
||||||
bio {str} -- tell us about yourself
|
|
||||||
avatar {str} -- a nice photo of yourself
|
|
||||||
|
|
||||||
Returns @deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT):
|
|
||||||
bool -- True if your profile was updated successfully
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation UserUpdate($user: UserUpdateInput!) {
|
|
||||||
userUpdate(user: $user)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"name": name, "company": company, "bio": bio, "avatar": avatar}
|
|
||||||
|
|
||||||
params = {"user": {k: v for k, v in params.items() if v is not None}}
|
|
||||||
|
|
||||||
if not params["user"]:
|
|
||||||
return SpeckleException(
|
|
||||||
message=(
|
|
||||||
"You must provide at least one field to update your user profile"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type="userUpdate", parse_response=False
|
|
||||||
)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
If no id argument is provided, will return the current authenticated user's
|
|
||||||
activity (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz as they will be
|
|
||||||
converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime} -- latest cutoff for activity
|
|
||||||
(ie: return all activity _before_ this time)
|
|
||||||
after {datetime} -- oldest cutoff for activity
|
|
||||||
(ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query UserActivity(
|
|
||||||
$action_type: String,
|
|
||||||
$before:DateTime,
|
|
||||||
$after: DateTime,
|
|
||||||
$cursor: DateTime,
|
|
||||||
$limit: Int
|
|
||||||
){
|
|
||||||
activeUser {
|
|
||||||
activity(
|
|
||||||
actionType: $action_type,
|
|
||||||
before: $before,
|
|
||||||
after: $after,
|
|
||||||
cursor: $cursor,
|
|
||||||
limit: $limit
|
|
||||||
) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
actionType
|
|
||||||
info
|
|
||||||
userId
|
|
||||||
streamId
|
|
||||||
resourceId
|
|
||||||
resourceType
|
|
||||||
message
|
|
||||||
time
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"limit": limit,
|
|
||||||
"action_type": action_type,
|
|
||||||
"before": before.astimezone(timezone.utc).isoformat() if before else before,
|
|
||||||
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
|
||||||
"cursor": cursor.astimezone(timezone.utc).isoformat() if cursor else cursor,
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type=["activeUser", "activity"],
|
|
||||||
schema=ActivityCollection,
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the active user's pending stream invites
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the current user
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamInvites {
|
|
||||||
streamInvites{
|
|
||||||
id
|
|
||||||
token
|
|
||||||
inviteId
|
|
||||||
streamId
|
|
||||||
streamName
|
|
||||||
title
|
|
||||||
role
|
|
||||||
invitedBy {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
return_type="streamInvites",
|
|
||||||
schema=PendingStreamCollaborator,
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_pending_invite(
|
|
||||||
self, stream_id: str, token: Optional[str] = None
|
|
||||||
) -> Optional[PendingStreamCollaborator]:
|
|
||||||
"""Get a particular pending invite for the active user on a given stream.
|
|
||||||
If no invite_id is provided, any valid invite will be returned.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to look for invites on
|
|
||||||
token {str} -- the token of the invite to look for (optional)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PendingStreamCollaborator
|
|
||||||
-- the invite for the given stream (or None if it isn't found)
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamInvite($streamId: String!, $token: String) {
|
|
||||||
streamInvite(streamId: $streamId, token: $token) {
|
|
||||||
id
|
|
||||||
token
|
|
||||||
streamId
|
|
||||||
streamName
|
|
||||||
title
|
|
||||||
role
|
|
||||||
invitedBy {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"streamId": stream_id}
|
|
||||||
if token:
|
|
||||||
params["token"] = token
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInvite",
|
|
||||||
schema=PendingStreamCollaborator,
|
|
||||||
)
|
|
||||||
@@ -1,172 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.core.api.models import ActivityCollection, LimitedUser
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
NAME = "other_user"
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
|
||||||
"""API Access class for other users, that are not the currently active user."""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = LimitedUser
|
|
||||||
|
|
||||||
def get(self, id: str) -> LimitedUser:
|
|
||||||
"""
|
|
||||||
Gets the profile of another user.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
LimitedUser -- the retrieved profile of another user
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query OtherUser($id: String!) {
|
|
||||||
otherUser(id: $id) {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
bio
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
verified
|
|
||||||
role
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"id": id}
|
|
||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="otherUser")
|
|
||||||
|
|
||||||
def search(
|
|
||||||
self, search_query: str, limit: int = 25
|
|
||||||
) -> Union[List[LimitedUser], SpeckleException]:
|
|
||||||
"""Searches for user by name or email. The search query must be at least
|
|
||||||
3 characters long
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
Returns:
|
|
||||||
List[LimitedUser] -- a list of User objects that match the search query
|
|
||||||
"""
|
|
||||||
if len(search_query) < 3:
|
|
||||||
return SpeckleException(
|
|
||||||
message="User search query must be at least 3 characters"
|
|
||||||
)
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query UserSearch($search_query: String!, $limit: Int!) {
|
|
||||||
userSearch(query: $search_query, limit: $limit) {
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
bio
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
verified
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"search_query": search_query, "limit": limit}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type=["userSearch", "items"]
|
|
||||||
)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
user_id: str,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
) -> ActivityCollection:
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of
|
|
||||||
any tz as they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime} -- latest cutoff for activity
|
|
||||||
(ie: return all activity _before_ this time)
|
|
||||||
after {datetime} -- oldest cutoff for activity
|
|
||||||
(ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query UserActivity(
|
|
||||||
$user_id: String!,
|
|
||||||
$action_type: String,
|
|
||||||
$before:DateTime,
|
|
||||||
$after: DateTime,
|
|
||||||
$cursor: DateTime,
|
|
||||||
$limit: Int
|
|
||||||
){
|
|
||||||
otherUser(id: $user_id) {
|
|
||||||
activity(
|
|
||||||
actionType: $action_type,
|
|
||||||
before: $before,
|
|
||||||
after: $after,
|
|
||||||
cursor: $cursor,
|
|
||||||
limit: $limit
|
|
||||||
) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
actionType
|
|
||||||
info
|
|
||||||
userId
|
|
||||||
streamId
|
|
||||||
resourceId
|
|
||||||
resourceType
|
|
||||||
message
|
|
||||||
time
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"user_id": user_id,
|
|
||||||
"limit": limit,
|
|
||||||
"action_type": action_type,
|
|
||||||
"before": before.astimezone(timezone.utc).isoformat() if before else before,
|
|
||||||
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
|
||||||
"cursor": cursor.astimezone(timezone.utc).isoformat() if cursor else cursor,
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type=["otherUser", "activity"],
|
|
||||||
schema=ActivityCollection,
|
|
||||||
)
|
|
||||||
@@ -1,751 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.core.api.models import ActivityCollection, PendingStreamCollaborator, Stream
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, UnsupportedException
|
|
||||||
|
|
||||||
NAME = "stream"
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
|
||||||
"""API Access class for streams"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.schema = Stream
|
|
||||||
|
|
||||||
def get(self, id: str, branch_limit: int = 10, commit_limit: int = 10) -> Stream:
|
|
||||||
"""Get the specified stream from the server
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the stream id
|
|
||||||
branch_limit {int} -- the maximum number of branches to return
|
|
||||||
commit_limit {int} -- the maximum number of commits to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the retrieved stream
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query Stream($id: String!, $branch_limit: Int!, $commit_limit: Int!) {
|
|
||||||
stream(id: $id) {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
description
|
|
||||||
isPublic
|
|
||||||
createdAt
|
|
||||||
updatedAt
|
|
||||||
commentCount
|
|
||||||
favoritesCount
|
|
||||||
collaborators {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
branches(limit: $branch_limit) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
description
|
|
||||||
commits(limit: $commit_limit) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
message
|
|
||||||
authorId
|
|
||||||
createdAt
|
|
||||||
authorName
|
|
||||||
referencedObject
|
|
||||||
sourceApplication
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"id": id, "branch_limit": branch_limit, "commit_limit": commit_limit}
|
|
||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="stream")
|
|
||||||
|
|
||||||
def list(self, stream_limit: int = 10) -> List[Stream]:
|
|
||||||
"""Get a list of the user's streams
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_limit {int} -- The maximum number of streams to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Stream] -- A list of Stream objects
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query User($stream_limit: Int!) {
|
|
||||||
user {
|
|
||||||
id
|
|
||||||
bio
|
|
||||||
name
|
|
||||||
email
|
|
||||||
avatar
|
|
||||||
company
|
|
||||||
verified
|
|
||||||
profiles
|
|
||||||
role
|
|
||||||
streams(limit: $stream_limit) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
isPublic
|
|
||||||
createdAt
|
|
||||||
updatedAt
|
|
||||||
description
|
|
||||||
commentCount
|
|
||||||
favoritesCount
|
|
||||||
collaborators {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"stream_limit": stream_limit}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type=["user", "streams", "items"]
|
|
||||||
)
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self,
|
|
||||||
name: str = "Anonymous Python Stream",
|
|
||||||
description: str = "No description provided",
|
|
||||||
is_public: bool = True,
|
|
||||||
) -> str:
|
|
||||||
"""Create a new stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- the name of the string
|
|
||||||
description {str} -- a short description of the stream
|
|
||||||
is_public {bool}
|
|
||||||
-- whether or not the stream can be viewed by anyone with the id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
id {str} -- the id of the newly created stream
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamCreate($stream: StreamCreateInput!) {
|
|
||||||
streamCreate(stream: $stream)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"stream": {"name": name, "description": description, "isPublic": is_public}
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type="streamCreate", parse_response=False
|
|
||||||
)
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
id: str,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
description: Optional[str] = None,
|
|
||||||
is_public: Optional[bool] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""Update an existing stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the id of the stream to be updated
|
|
||||||
name {str} -- the name of the string
|
|
||||||
description {str} -- a short description of the stream
|
|
||||||
is_public {bool}
|
|
||||||
-- whether or not the stream can be viewed by anyone with the id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- whether the stream update was successful
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamUpdate($stream: StreamUpdateInput!) {
|
|
||||||
streamUpdate(stream: $stream)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"id": id,
|
|
||||||
"name": name,
|
|
||||||
"description": description,
|
|
||||||
"isPublic": is_public,
|
|
||||||
}
|
|
||||||
# remove None values so graphql doesn't cry
|
|
||||||
params = {"stream": {k: v for k, v in params.items() if v is not None}}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type="streamUpdate", parse_response=False
|
|
||||||
)
|
|
||||||
|
|
||||||
def delete(self, id: str) -> bool:
|
|
||||||
"""Delete a stream given its id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the id of the stream to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- whether the deletion was successful
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamDelete($id: String!) {
|
|
||||||
streamDelete(id: $id)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"id": id}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type="streamDelete", parse_response=False
|
|
||||||
)
|
|
||||||
|
|
||||||
def search(
|
|
||||||
self,
|
|
||||||
search_query: str,
|
|
||||||
limit: int = 25,
|
|
||||||
branch_limit: int = 10,
|
|
||||||
commit_limit: int = 10,
|
|
||||||
):
|
|
||||||
"""Search for streams by name, description, or id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
branch_limit {int} -- the maximum number of branches to return
|
|
||||||
commit_limit {int} -- the maximum number of commits to return
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Stream] -- a list of Streams that match the search query
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamSearch(
|
|
||||||
$search_query: String!,
|
|
||||||
$limit: Int!,
|
|
||||||
$branch_limit:Int!,
|
|
||||||
$commit_limit:Int!
|
|
||||||
) {
|
|
||||||
streams(query: $search_query, limit: $limit) {
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
description
|
|
||||||
isPublic
|
|
||||||
createdAt
|
|
||||||
updatedAt
|
|
||||||
collaborators {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
role
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
branches(limit: $branch_limit) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
description
|
|
||||||
commits(limit: $commit_limit) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
referencedObject
|
|
||||||
message
|
|
||||||
authorName
|
|
||||||
authorId
|
|
||||||
createdAt
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"search_query": search_query,
|
|
||||||
"limit": limit,
|
|
||||||
"branch_limit": branch_limit,
|
|
||||||
"commit_limit": commit_limit,
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type=["streams", "items"]
|
|
||||||
)
|
|
||||||
|
|
||||||
def favorite(self, stream_id: str, favorited: bool = True):
|
|
||||||
"""Favorite or unfavorite the given stream.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to favorite / unfavorite
|
|
||||||
favorited {bool}
|
|
||||||
-- whether to favorite (True) or unfavorite (False) the stream
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Stream -- the stream with its `id`, `name`, and `favoritedDate`
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamFavorite($stream_id: String!, $favorited: Boolean!) {
|
|
||||||
streamFavorite(streamId: $stream_id, favorited: $favorited) {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
favoritedDate
|
|
||||||
favoritesCount
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"stream_id": stream_id,
|
|
||||||
"favorited": favorited,
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type=["streamFavorite"]
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_all_pending_invites(
|
|
||||||
self, stream_id: str
|
|
||||||
) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the pending invites on a stream.
|
|
||||||
You must be a `stream:owner` to query this.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream id from which to get the pending invites
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the specified stream
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamInvites($streamId: String!) {
|
|
||||||
stream(id: $streamId){
|
|
||||||
pendingCollaborators {
|
|
||||||
id
|
|
||||||
token
|
|
||||||
inviteId
|
|
||||||
streamId
|
|
||||||
streamName
|
|
||||||
title
|
|
||||||
role
|
|
||||||
invitedBy{
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
user {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"streamId": stream_id}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type=["stream", "pendingCollaborators"],
|
|
||||||
schema=PendingStreamCollaborator,
|
|
||||||
)
|
|
||||||
|
|
||||||
def invite(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
email: Optional[str] = None,
|
|
||||||
user_id: Optional[str] = None,
|
|
||||||
role: str = "stream:contributor", # should default be reviewer?
|
|
||||||
message: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Invite someone to a stream using either their email or user id
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to invite the user to
|
|
||||||
email {str} -- the email of the user to invite (use this OR `user_id`)
|
|
||||||
user_id {str} -- the id of the user to invite (use this OR `email`)
|
|
||||||
role {str}
|
|
||||||
-- the role to assign to the user (defaults to `stream:contributor`)
|
|
||||||
message {str}
|
|
||||||
-- a message to send along with this invite to the specified user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
if email is None and user_id is None:
|
|
||||||
raise SpeckleException(
|
|
||||||
"You must provide either an email or a user id to use the"
|
|
||||||
" `stream.invite` method"
|
|
||||||
)
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamInviteCreate($input: StreamInviteCreateInput!) {
|
|
||||||
streamInviteCreate(input: $input)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"email": email,
|
|
||||||
"userId": user_id,
|
|
||||||
"streamId": stream_id,
|
|
||||||
"message": message,
|
|
||||||
"role": role,
|
|
||||||
}
|
|
||||||
params = {"input": {k: v for k, v in params.items() if v is not None}}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInviteCreate",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def invite_batch(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
emails: Optional[List[str]] = None,
|
|
||||||
user_ids: Optional[List[None]] = None,
|
|
||||||
message: Optional[str] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""Invite a batch of users to a specified stream.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to invite the user to
|
|
||||||
emails {List[str]}
|
|
||||||
-- the email of the user to invite (use this and/or `user_ids`)
|
|
||||||
user_id {List[str]}
|
|
||||||
-- the id of the user to invite (use this and/or `emails`)
|
|
||||||
message {str}
|
|
||||||
-- a message to send along with this invite to the specified user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
if emails is None and user_ids is None:
|
|
||||||
raise SpeckleException(
|
|
||||||
"You must provide either an email or a user id to use the"
|
|
||||||
" `stream.invite` method"
|
|
||||||
)
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamInviteBatchCreate($input: [StreamInviteCreateInput!]!) {
|
|
||||||
streamInviteBatchCreate(input: $input)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
email_invites = [
|
|
||||||
{"streamId": stream_id, "message": message, "email": email}
|
|
||||||
for email in (emails if emails is not None else [])
|
|
||||||
if email is not None
|
|
||||||
]
|
|
||||||
|
|
||||||
user_invites = [
|
|
||||||
{"streamId": stream_id, "message": message, "userId": user_id}
|
|
||||||
for user_id in (user_ids if user_ids is not None else [])
|
|
||||||
if user_id is not None
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
params = {"input": [*email_invites, *user_invites]}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInviteBatchCreate",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def invite_cancel(self, stream_id: str, invite_id: str) -> bool:
|
|
||||||
"""Cancel an existing stream invite
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream invite
|
|
||||||
invite_id {str} -- the id of the invite to use
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- true if the operation was successful
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamInviteCancel($streamId: String!, $inviteId: String!) {
|
|
||||||
streamInviteCancel(streamId: $streamId, inviteId: $inviteId)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"streamId": stream_id, "inviteId": invite_id}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInviteCancel",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def invite_use(self, stream_id: str, token: str, accept: bool = True) -> bool:
|
|
||||||
"""Accept or decline a stream invite
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream for which the user has a pending invite
|
|
||||||
token {str} -- the token of the invite to use
|
|
||||||
accept {bool} -- whether or not to accept the invite (defaults to True)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- true if the operation was successful
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamInviteUse(
|
|
||||||
$accept: Boolean!,
|
|
||||||
$streamId: String!,
|
|
||||||
$token: String!
|
|
||||||
) {
|
|
||||||
streamInviteUse(accept: $accept, streamId: $streamId, token: $token)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"streamId": stream_id, "token": token, "accept": accept}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInviteUse",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_permission(self, stream_id: str, user_id: str, role: str):
|
|
||||||
"""Updates permissions for a user on a given stream
|
|
||||||
|
|
||||||
Valid for Speckle Server >=2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to grant permissions to
|
|
||||||
user_id {str} -- the id of the user to grant permissions for
|
|
||||||
role {str} -- the role to grant the user
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
if self.server_version and (
|
|
||||||
self.server_version != ("dev",) and self.server_version < (2, 6, 4)
|
|
||||||
):
|
|
||||||
raise UnsupportedException(
|
|
||||||
"Server mutation `update_permission` is only supported as of Speckle"
|
|
||||||
" Server v2.6.4. Please update your Speckle Server to use this method"
|
|
||||||
" or use the `grant_permission` method instead."
|
|
||||||
)
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamUpdatePermission(
|
|
||||||
$permission_params: StreamUpdatePermissionInput!
|
|
||||||
) {
|
|
||||||
streamUpdatePermission(permissionParams: $permission_params)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"permission_params": {
|
|
||||||
"streamId": stream_id,
|
|
||||||
"userId": user_id,
|
|
||||||
"role": role,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamUpdatePermission",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def revoke_permission(self, stream_id: str, user_id: str):
|
|
||||||
"""Revoke permissions from a user on a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to revoke permissions from
|
|
||||||
user_id {str} -- the id of the user to revoke permissions from
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation was successful
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation StreamRevokePermission(
|
|
||||||
$permission_params: StreamRevokePermissionInput!
|
|
||||||
) {
|
|
||||||
streamRevokePermission(permissionParams: $permission_params)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"permission_params": {"streamId": stream_id, "userId": user_id}}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamRevokePermission",
|
|
||||||
parse_response=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
limit: int = 20,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz
|
|
||||||
as they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
stream_id {str} -- the id of the stream to get activity from
|
|
||||||
action_type {str}
|
|
||||||
-- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime}
|
|
||||||
-- latest cutoff for activity (ie: return all activity _before_ this time)
|
|
||||||
after {datetime}
|
|
||||||
-- oldest cutoff for activity (ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamActivity(
|
|
||||||
$stream_id: String!,
|
|
||||||
$action_type: String,
|
|
||||||
$before:DateTime,
|
|
||||||
$after: DateTime,
|
|
||||||
$cursor: DateTime,
|
|
||||||
$limit: Int
|
|
||||||
){
|
|
||||||
stream(id: $stream_id) {
|
|
||||||
activity(
|
|
||||||
actionType: $action_type,
|
|
||||||
before: $before,
|
|
||||||
after: $after,
|
|
||||||
cursor: $cursor,
|
|
||||||
limit: $limit
|
|
||||||
) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
actionType
|
|
||||||
info
|
|
||||||
userId
|
|
||||||
streamId
|
|
||||||
resourceId
|
|
||||||
resourceType
|
|
||||||
message
|
|
||||||
time
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
params = {
|
|
||||||
"stream_id": stream_id,
|
|
||||||
"limit": limit,
|
|
||||||
"action_type": action_type,
|
|
||||||
"before": before.astimezone(timezone.utc).isoformat()
|
|
||||||
if before
|
|
||||||
else before,
|
|
||||||
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
|
||||||
"cursor": cursor.astimezone(timezone.utc).isoformat()
|
|
||||||
if cursor
|
|
||||||
else cursor,
|
|
||||||
}
|
|
||||||
except AttributeError as e:
|
|
||||||
raise SpeckleException(
|
|
||||||
"Could not get stream activity - `before`, `after`, and `cursor` must"
|
|
||||||
" be in `datetime` format if provided",
|
|
||||||
ValueError(),
|
|
||||||
) from e
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type=["stream", "activity"],
|
|
||||||
schema=ActivityCollection,
|
|
||||||
)
|
|
||||||
@@ -1,322 +0,0 @@
|
|||||||
from datetime import datetime, timezone
|
|
||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from deprecated import deprecated
|
|
||||||
from gql import gql
|
|
||||||
|
|
||||||
from specklepy.core.api.models import ActivityCollection, PendingStreamCollaborator, User
|
|
||||||
from specklepy.core.api.resource import ResourceBase
|
|
||||||
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
NAME = "user"
|
|
||||||
|
|
||||||
DEPRECATION_VERSION = "2.9.0"
|
|
||||||
DEPRECATION_TEXT = (
|
|
||||||
"The user resource is deprecated, please use the active_user or other_user"
|
|
||||||
" resources"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
|
||||||
"""API Access class for users"""
|
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
|
||||||
super().__init__(
|
|
||||||
account=account,
|
|
||||||
basepath=basepath,
|
|
||||||
client=client,
|
|
||||||
name=NAME,
|
|
||||||
server_version=server_version,
|
|
||||||
)
|
|
||||||
self.schema = User
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get(self, id: Optional[str] = None) -> User:
|
|
||||||
"""
|
|
||||||
Gets the profile of a user.
|
|
||||||
If no id argument is provided, will return the current authenticated
|
|
||||||
user's profile (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
id {str} -- the user id
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
User -- the retrieved user
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query User($id: String) {
|
|
||||||
user(id: $id) {
|
|
||||||
id
|
|
||||||
email
|
|
||||||
name
|
|
||||||
bio
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
verified
|
|
||||||
profiles
|
|
||||||
role
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"id": id}
|
|
||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="user")
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def search(
|
|
||||||
self, search_query: str, limit: int = 25
|
|
||||||
) -> Union[List[User], SpeckleException]:
|
|
||||||
"""
|
|
||||||
Searches for user by name or email.
|
|
||||||
The search query must be at least 3 characters long
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
search_query {str} -- a string to search for
|
|
||||||
limit {int} -- the maximum number of results to return
|
|
||||||
Returns:
|
|
||||||
List[User] -- a list of User objects that match the search query
|
|
||||||
"""
|
|
||||||
if len(search_query) < 3:
|
|
||||||
return SpeckleException(
|
|
||||||
message="User search query must be at least 3 characters"
|
|
||||||
)
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query UserSearch($search_query: String!, $limit: Int!) {
|
|
||||||
userSearch(query: $search_query, limit: $limit) {
|
|
||||||
items {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
bio
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
verified
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"search_query": search_query, "limit": limit}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type=["userSearch", "items"]
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
company: Optional[str] = None,
|
|
||||||
bio: Optional[str] = None,
|
|
||||||
avatar: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Updates your user profile. All arguments are optional.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- your name
|
|
||||||
company {str} -- the company you may or may not work for
|
|
||||||
bio {str} -- tell us about yourself
|
|
||||||
avatar {str} -- a nice photo of yourself
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if your profile was updated successfully
|
|
||||||
"""
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
mutation UserUpdate($user: UserUpdateInput!) {
|
|
||||||
userUpdate(user: $user)
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
params = {"name": name, "company": company, "bio": bio, "avatar": avatar}
|
|
||||||
|
|
||||||
params = {"user": {k: v for k, v in params.items() if v is not None}}
|
|
||||||
|
|
||||||
if not params["user"]:
|
|
||||||
return SpeckleException(
|
|
||||||
message=(
|
|
||||||
"You must provide at least one field to update your user profile"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query, params=params, return_type="userUpdate", parse_response=False
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
user_id: Optional[str] = None,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Get the activity from a given stream in an Activity collection.
|
|
||||||
Step into the activity `items` for the list of activity.
|
|
||||||
If no id argument is provided, will return the current authenticated
|
|
||||||
user's activity (as extracted from the authorization header).
|
|
||||||
|
|
||||||
Note: all timestamps arguments should be `datetime` of any tz as
|
|
||||||
they will be converted to UTC ISO format strings
|
|
||||||
|
|
||||||
user_id {str} -- the id of the user to get the activity from
|
|
||||||
action_type {str} -- filter results to a single action type
|
|
||||||
(eg: `commit_create` or `commit_receive`)
|
|
||||||
limit {int} -- max number of Activity items to return
|
|
||||||
before {datetime}
|
|
||||||
-- latest cutoff for activity (ie: return all activity _before_ this time)
|
|
||||||
after {datetime}
|
|
||||||
-- oldest cutoff for activity (ie: return all activity _after_ this time)
|
|
||||||
cursor {datetime} -- timestamp cursor for pagination
|
|
||||||
"""
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query UserActivity(
|
|
||||||
$user_id: String,
|
|
||||||
$action_type: String,
|
|
||||||
$before:DateTime,
|
|
||||||
$after: DateTime,
|
|
||||||
$cursor: DateTime,
|
|
||||||
$limit: Int
|
|
||||||
){
|
|
||||||
user(id: $user_id) {
|
|
||||||
activity(
|
|
||||||
actionType: $action_type,
|
|
||||||
before: $before,
|
|
||||||
after: $after,
|
|
||||||
cursor: $cursor,
|
|
||||||
limit: $limit
|
|
||||||
) {
|
|
||||||
totalCount
|
|
||||||
cursor
|
|
||||||
items {
|
|
||||||
actionType
|
|
||||||
info
|
|
||||||
userId
|
|
||||||
streamId
|
|
||||||
resourceId
|
|
||||||
resourceType
|
|
||||||
message
|
|
||||||
time
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {
|
|
||||||
"user_id": user_id,
|
|
||||||
"limit": limit,
|
|
||||||
"action_type": action_type,
|
|
||||||
"before": before.astimezone(timezone.utc).isoformat() if before else before,
|
|
||||||
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
|
||||||
"cursor": cursor.astimezone(timezone.utc).isoformat() if cursor else cursor,
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type=["user", "activity"],
|
|
||||||
schema=ActivityCollection,
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
|
||||||
"""Get all of the active user's pending stream invites
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[PendingStreamCollaborator]
|
|
||||||
-- a list of pending invites for the current user
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamInvites {
|
|
||||||
streamInvites{
|
|
||||||
id
|
|
||||||
token
|
|
||||||
inviteId
|
|
||||||
streamId
|
|
||||||
streamName
|
|
||||||
title
|
|
||||||
role
|
|
||||||
invitedBy {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
return_type="streamInvites",
|
|
||||||
schema=PendingStreamCollaborator,
|
|
||||||
)
|
|
||||||
|
|
||||||
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
|
||||||
def get_pending_invite(
|
|
||||||
self, stream_id: str, token: Optional[str] = None
|
|
||||||
) -> Optional[PendingStreamCollaborator]:
|
|
||||||
"""Get a particular pending invite for the active user on a given stream.
|
|
||||||
If no invite_id is provided, any valid invite will be returned.
|
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to look for invites on
|
|
||||||
token {str} -- the token of the invite to look for (optional)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
PendingStreamCollaborator
|
|
||||||
-- the invite for the given stream (or None if it isn't found)
|
|
||||||
"""
|
|
||||||
self._check_invites_supported()
|
|
||||||
|
|
||||||
query = gql(
|
|
||||||
"""
|
|
||||||
query StreamInvite($streamId: String!, $token: String) {
|
|
||||||
streamInvite(streamId: $streamId, token: $token) {
|
|
||||||
id
|
|
||||||
token
|
|
||||||
streamId
|
|
||||||
streamName
|
|
||||||
title
|
|
||||||
role
|
|
||||||
invitedBy {
|
|
||||||
id
|
|
||||||
name
|
|
||||||
company
|
|
||||||
avatar
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
|
|
||||||
params = {"streamId": stream_id}
|
|
||||||
if token:
|
|
||||||
params["token"] = token
|
|
||||||
|
|
||||||
return self.make_request(
|
|
||||||
query=query,
|
|
||||||
params=params,
|
|
||||||
return_type="streamInvite",
|
|
||||||
schema=PendingStreamCollaborator,
|
|
||||||
)
|
|
||||||
@@ -1,186 +0,0 @@
|
|||||||
from urllib.parse import unquote, urlparse
|
|
||||||
from warnings import warn
|
|
||||||
|
|
||||||
from specklepy.core.api.client import SpeckleClient
|
|
||||||
from specklepy.core.api.credentials import (
|
|
||||||
Account,
|
|
||||||
get_account_from_token,
|
|
||||||
get_local_accounts,
|
|
||||||
)
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
from specklepy.transports.server.server import ServerTransport
|
|
||||||
|
|
||||||
|
|
||||||
class StreamWrapper:
|
|
||||||
"""
|
|
||||||
The `StreamWrapper` gives you some handy helpers to deal with urls and
|
|
||||||
get authenticated clients and transports.
|
|
||||||
|
|
||||||
Construct a `StreamWrapper` with a stream, branch, commit, or object URL.
|
|
||||||
The corresponding ids will be stored
|
|
||||||
in the wrapper. If you have local accounts on the machine,
|
|
||||||
you can use the `get_account` and `get_client` methods
|
|
||||||
to get a local account for the server. You can also pass a token into `get_client`
|
|
||||||
if you don't have a corresponding
|
|
||||||
local account for the server.
|
|
||||||
|
|
||||||
```py
|
|
||||||
from specklepy.api.wrapper import StreamWrapper
|
|
||||||
|
|
||||||
# provide any stream, branch, commit, object, or globals url
|
|
||||||
wrapper = StreamWrapper("https://speckle.xyz/streams/3073b96e86/commits/604bea8cc6")
|
|
||||||
|
|
||||||
# get an authenticated SpeckleClient if you have a local account for the server
|
|
||||||
client = wrapper.get_client()
|
|
||||||
|
|
||||||
# get an authenticated ServerTransport if you have a local account for the server
|
|
||||||
transport = wrapper.get_transport()
|
|
||||||
```
|
|
||||||
"""
|
|
||||||
|
|
||||||
stream_url: str = None
|
|
||||||
use_ssl: bool = True
|
|
||||||
host: str = None
|
|
||||||
stream_id: str = None
|
|
||||||
commit_id: str = None
|
|
||||||
object_id: str = None
|
|
||||||
branch_name: str = None
|
|
||||||
_client: SpeckleClient = None
|
|
||||||
_account: Account = None
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return (
|
|
||||||
f"StreamWrapper( server: {self.host}, stream_id: {self.stream_id}, type:"
|
|
||||||
f" {self.type} )"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def type(self) -> str:
|
|
||||||
if self.object_id:
|
|
||||||
return "object"
|
|
||||||
elif self.commit_id:
|
|
||||||
return "commit"
|
|
||||||
elif self.branch_name:
|
|
||||||
return "branch"
|
|
||||||
else:
|
|
||||||
return "stream" if self.stream_id else "invalid"
|
|
||||||
|
|
||||||
def __init__(self, url: str) -> None:
|
|
||||||
self.stream_url = url
|
|
||||||
parsed = urlparse(url)
|
|
||||||
self.host = parsed.netloc
|
|
||||||
self.use_ssl = parsed.scheme == "https"
|
|
||||||
segments = parsed.path.strip("/").split("/", 3)
|
|
||||||
|
|
||||||
if not segments or len(segments) < 2:
|
|
||||||
raise SpeckleException(
|
|
||||||
f"Cannot parse {url} into a stream wrapper class - invalid URL"
|
|
||||||
" provided."
|
|
||||||
)
|
|
||||||
|
|
||||||
while segments:
|
|
||||||
segment = segments.pop(0)
|
|
||||||
if segments and segment.lower() == "streams":
|
|
||||||
self.stream_id = segments.pop(0)
|
|
||||||
elif segments and segment.lower() == "commits":
|
|
||||||
self.commit_id = segments.pop(0)
|
|
||||||
elif segments and segment.lower() == "branches":
|
|
||||||
self.branch_name = unquote(segments.pop(0))
|
|
||||||
elif segments and segment.lower() == "objects":
|
|
||||||
self.object_id = segments.pop(0)
|
|
||||||
elif segment.lower() == "globals":
|
|
||||||
self.branch_name = "globals"
|
|
||||||
if segments:
|
|
||||||
self.commit_id = segments.pop(0)
|
|
||||||
else:
|
|
||||||
raise SpeckleException(
|
|
||||||
f"Cannot parse {url} into a stream wrapper class - invalid URL"
|
|
||||||
" provided."
|
|
||||||
)
|
|
||||||
|
|
||||||
if not self.stream_id:
|
|
||||||
raise SpeckleException(
|
|
||||||
f"Cannot parse {url} into a stream wrapper class - no stream id found."
|
|
||||||
)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def server_url(self):
|
|
||||||
return f"{'https' if self.use_ssl else 'http'}://{self.host}"
|
|
||||||
|
|
||||||
def get_account(self, token: str = None) -> Account:
|
|
||||||
"""
|
|
||||||
Gets an account object for this server from the local accounts db
|
|
||||||
(added via Speckle Manager or a json file)
|
|
||||||
"""
|
|
||||||
if self._account and self._account.token:
|
|
||||||
return self._account
|
|
||||||
|
|
||||||
self._account = next(
|
|
||||||
(
|
|
||||||
a
|
|
||||||
for a in get_local_accounts()
|
|
||||||
if self.host == urlparse(a.serverInfo.url).netloc
|
|
||||||
),
|
|
||||||
None,
|
|
||||||
)
|
|
||||||
|
|
||||||
if not self._account:
|
|
||||||
self._account = get_account_from_token(token, self.server_url)
|
|
||||||
|
|
||||||
if self._client:
|
|
||||||
self._client.authenticate_with_account(self._account)
|
|
||||||
|
|
||||||
return self._account
|
|
||||||
|
|
||||||
def get_client(self, token: str = None) -> SpeckleClient:
|
|
||||||
"""
|
|
||||||
Gets an authenticated client for this server.
|
|
||||||
You may provide a token if there aren't any local accounts on this
|
|
||||||
machine. If no account is found and no token is provided,
|
|
||||||
an unauthenticated client is returned.
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
token {str}
|
|
||||||
-- optional token if no local account is available (defaults to None)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
SpeckleClient
|
|
||||||
-- authenticated with a corresponding local account or the provided token
|
|
||||||
"""
|
|
||||||
if self._client and token is None:
|
|
||||||
return self._client
|
|
||||||
|
|
||||||
if not self._account or not self._account.token:
|
|
||||||
self.get_account(token)
|
|
||||||
|
|
||||||
if not self._client:
|
|
||||||
self._client = SpeckleClient(host=self.host, use_ssl=self.use_ssl)
|
|
||||||
|
|
||||||
if self._account.token is None and token is None:
|
|
||||||
warn(f"No local account found for server {self.host}", SpeckleWarning)
|
|
||||||
return self._client
|
|
||||||
|
|
||||||
if self._account.token:
|
|
||||||
self._client.authenticate_with_account(self._account)
|
|
||||||
else:
|
|
||||||
self._client.authenticate_with_token(token)
|
|
||||||
|
|
||||||
return self._client
|
|
||||||
|
|
||||||
def get_transport(self, token: str = None) -> ServerTransport:
|
|
||||||
"""
|
|
||||||
Gets a server transport for this stream using an authenticated client.
|
|
||||||
If there is no local account for this
|
|
||||||
server and the client was not authenticated with a token,
|
|
||||||
this will throw an exception.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
ServerTransport -- constructed for this stream
|
|
||||||
with a pre-authenticated client
|
|
||||||
"""
|
|
||||||
if not self._account or not self._account.token:
|
|
||||||
self.get_account(token)
|
|
||||||
return ServerTransport(self.stream_id, account=self._account)
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
"""Common helpers module for Core."""
|
|
||||||
@@ -1,127 +0,0 @@
|
|||||||
"""
|
|
||||||
Provides uniform and consistent path helpers for `specklepy`
|
|
||||||
"""
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
|
||||||
|
|
||||||
_user_data_env_var = "SPECKLE_USERDATA_PATH"
|
|
||||||
|
|
||||||
|
|
||||||
def _path() -> Optional[Path]:
|
|
||||||
"""Read the user data path override setting."""
|
|
||||||
path_override = os.environ.get(_user_data_env_var)
|
|
||||||
if path_override:
|
|
||||||
return Path(path_override)
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
_application_name = "Speckle"
|
|
||||||
|
|
||||||
|
|
||||||
def override_application_name(application_name: str) -> None:
|
|
||||||
"""Override the global Speckle application name."""
|
|
||||||
global _application_name
|
|
||||||
_application_name = application_name
|
|
||||||
|
|
||||||
|
|
||||||
def override_application_data_path(path: Optional[str]) -> None:
|
|
||||||
"""
|
|
||||||
Override the global Speckle application data path.
|
|
||||||
|
|
||||||
If the value of path is `None` the environment variable gets deleted.
|
|
||||||
"""
|
|
||||||
if path:
|
|
||||||
os.environ[_user_data_env_var] = path
|
|
||||||
else:
|
|
||||||
os.environ.pop(_user_data_env_var, None)
|
|
||||||
|
|
||||||
|
|
||||||
_blob_folder_name = "Blobs"
|
|
||||||
|
|
||||||
|
|
||||||
def override_blob_storage_folder(blob_folder_name: str) -> None:
|
|
||||||
"""Override the global Blob storage folder name."""
|
|
||||||
global _blob_folder_name
|
|
||||||
_blob_folder_name = blob_folder_name
|
|
||||||
|
|
||||||
|
|
||||||
_accounts_folder_name = "Accounts"
|
|
||||||
|
|
||||||
|
|
||||||
def override_accounts_folder_name(accounts_folder_name: str) -> None:
|
|
||||||
"""Override the global Accounts folder name."""
|
|
||||||
global _accounts_folder_name
|
|
||||||
_accounts_folder_name = accounts_folder_name
|
|
||||||
|
|
||||||
|
|
||||||
_objects_folder_name = "Objects"
|
|
||||||
|
|
||||||
|
|
||||||
def override_objects_folder_name(objects_folder_name: str) -> None:
|
|
||||||
"""Override global Objects folder name."""
|
|
||||||
global _objects_folder_name
|
|
||||||
_objects_folder_name = objects_folder_name
|
|
||||||
|
|
||||||
|
|
||||||
def _ensure_folder_exists(base_path: Path, folder_name: str) -> Path:
|
|
||||||
path = base_path.joinpath(folder_name)
|
|
||||||
path.mkdir(exist_ok=True, parents=True)
|
|
||||||
return path
|
|
||||||
|
|
||||||
|
|
||||||
def user_application_data_path() -> Path:
|
|
||||||
"""Get the platform specific user configuration folder path"""
|
|
||||||
path_override = _path()
|
|
||||||
if path_override:
|
|
||||||
return path_override
|
|
||||||
|
|
||||||
try:
|
|
||||||
if sys.platform.startswith("win"):
|
|
||||||
app_data_path = os.getenv("APPDATA")
|
|
||||||
if not app_data_path:
|
|
||||||
raise SpeckleException(
|
|
||||||
message="Cannot get appdata path from environment."
|
|
||||||
)
|
|
||||||
return Path(app_data_path)
|
|
||||||
else:
|
|
||||||
# try getting the standard XDG_DATA_HOME value
|
|
||||||
# as that is used as an override
|
|
||||||
app_data_path = os.getenv("XDG_DATA_HOME")
|
|
||||||
if app_data_path:
|
|
||||||
return Path(app_data_path)
|
|
||||||
else:
|
|
||||||
return _ensure_folder_exists(Path.home(), ".config")
|
|
||||||
except Exception as ex:
|
|
||||||
raise SpeckleException(
|
|
||||||
message="Failed to initialize user application data path.", exception=ex
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def user_speckle_folder_path() -> Path:
|
|
||||||
"""Get the folder where the user's Speckle data should be stored."""
|
|
||||||
return _ensure_folder_exists(user_application_data_path(), _application_name)
|
|
||||||
|
|
||||||
|
|
||||||
def user_speckle_connector_installation_path(host_application: str) -> Path:
|
|
||||||
"""
|
|
||||||
Gets a connector specific installation folder.
|
|
||||||
|
|
||||||
In this folder we can put our connector installation and all python packages.
|
|
||||||
"""
|
|
||||||
return _ensure_folder_exists(
|
|
||||||
_ensure_folder_exists(user_speckle_folder_path(), "connector_installations"),
|
|
||||||
host_application,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def accounts_folder_path() -> Path:
|
|
||||||
"""Get the folder where the Speckle accounts data should be stored."""
|
|
||||||
return _ensure_folder_exists(user_speckle_folder_path(), _accounts_folder_name)
|
|
||||||
|
|
||||||
|
|
||||||
def blob_storage_path(path: Optional[Path] = None) -> Path:
|
|
||||||
return _ensure_folder_exists(path or user_speckle_folder_path(), _blob_folder_name)
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
from typing import Any, List, Optional
|
|
||||||
|
|
||||||
|
|
||||||
class SpeckleException(Exception):
|
|
||||||
def __init__(self, message: str, exception: Exception = None) -> None:
|
|
||||||
super().__init__()
|
|
||||||
self.message = message
|
|
||||||
self.exception = exception
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return f"SpeckleException: {self.message}"
|
|
||||||
|
|
||||||
|
|
||||||
class SpeckleInvalidUnitException(SpeckleException):
|
|
||||||
def __init__(self, invalid_unit: Any) -> None:
|
|
||||||
super().__init__(
|
|
||||||
message=(
|
|
||||||
"Invalid units: expected type str but received"
|
|
||||||
f" {type(invalid_unit)} ({invalid_unit})."
|
|
||||||
),
|
|
||||||
exception=None,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class SerializationException(SpeckleException):
|
|
||||||
def __init__(self, message: str, obj: Any, exception: Exception = None) -> None:
|
|
||||||
super().__init__(message=message, exception=exception)
|
|
||||||
self.obj = obj
|
|
||||||
self.unhandled_type = type(obj)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return (
|
|
||||||
"SpeckleException: Could not serialize object of type"
|
|
||||||
f" {self.unhandled_type}"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class GraphQLException(SpeckleException):
|
|
||||||
def __init__(
|
|
||||||
self, message: str, errors: Optional[List[Any]] = None, data=None
|
|
||||||
) -> None:
|
|
||||||
super().__init__(message=message)
|
|
||||||
self.errors = errors
|
|
||||||
self.data = data
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return f"GraphQLException: {self.message}"
|
|
||||||
|
|
||||||
|
|
||||||
class UnsupportedException(SpeckleException):
|
|
||||||
def __init__(self, message: str) -> None:
|
|
||||||
super().__init__(message=message)
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return f"UnsupportedException: {self.message}"
|
|
||||||
|
|
||||||
|
|
||||||
class SpeckleWarning(Warning):
|
|
||||||
def __init__(self, *args: object) -> None:
|
|
||||||
super().__init__(*args)
|
|
||||||
@@ -1,158 +0,0 @@
|
|||||||
import contextlib
|
|
||||||
import getpass
|
|
||||||
import hashlib
|
|
||||||
import logging
|
|
||||||
import platform
|
|
||||||
import queue
|
|
||||||
import sys
|
|
||||||
import threading
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
import requests
|
|
||||||
|
|
||||||
"""
|
|
||||||
Anonymous telemetry to help us understand how to make a better Speckle.
|
|
||||||
This really helps us to deliver a better open source project and product!
|
|
||||||
"""
|
|
||||||
TRACK = True
|
|
||||||
HOST_APP = "python"
|
|
||||||
HOST_APP_VERSION = f"python {'.'.join(map(str, sys.version_info[:2]))}"
|
|
||||||
PLATFORMS = {"win32": "Windows", "cygwin": "Windows", "darwin": "Mac OS X"}
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
METRICS_TRACKER = None
|
|
||||||
|
|
||||||
# actions
|
|
||||||
SDK = "SDK Action"
|
|
||||||
CONNECTOR = "Connector Action"
|
|
||||||
RECEIVE = "Receive"
|
|
||||||
SEND = "Send"
|
|
||||||
|
|
||||||
# not in use since 2.15
|
|
||||||
ACCOUNTS = "Get Local Accounts"
|
|
||||||
BRANCH = "Branch Action"
|
|
||||||
CLIENT = "Speckle Client"
|
|
||||||
COMMIT = "Commit Action"
|
|
||||||
DESERIALIZE = "serialization/deserialize"
|
|
||||||
INVITE = "Invite Action"
|
|
||||||
OTHER_USER = "Other User Action"
|
|
||||||
PERMISSION = "Permission Action"
|
|
||||||
SERIALIZE = "serialization/serialize"
|
|
||||||
SERVER = "Server Action"
|
|
||||||
STREAM = "Stream Action"
|
|
||||||
STREAM_WRAPPER = "Stream Wrapper"
|
|
||||||
USER = "User Action"
|
|
||||||
|
|
||||||
|
|
||||||
def disable():
|
|
||||||
global TRACK
|
|
||||||
TRACK = False
|
|
||||||
|
|
||||||
|
|
||||||
def enable():
|
|
||||||
global TRACK
|
|
||||||
TRACK = True
|
|
||||||
|
|
||||||
|
|
||||||
def set_host_app(host_app: str, host_app_version: Optional[str] = None):
|
|
||||||
global HOST_APP, HOST_APP_VERSION
|
|
||||||
HOST_APP = host_app
|
|
||||||
HOST_APP_VERSION = host_app_version or HOST_APP_VERSION
|
|
||||||
|
|
||||||
|
|
||||||
def track(
|
|
||||||
action: str,
|
|
||||||
account=None,
|
|
||||||
custom_props: Optional[dict] = None,
|
|
||||||
):
|
|
||||||
if not TRACK:
|
|
||||||
return
|
|
||||||
try:
|
|
||||||
initialise_tracker(account)
|
|
||||||
event_params = {
|
|
||||||
"event": action,
|
|
||||||
"properties": {
|
|
||||||
"distinct_id": METRICS_TRACKER.last_user,
|
|
||||||
"server_id": METRICS_TRACKER.last_server,
|
|
||||||
"token": METRICS_TRACKER.analytics_token,
|
|
||||||
"hostApp": HOST_APP,
|
|
||||||
"hostAppVersion": HOST_APP_VERSION,
|
|
||||||
"$os": METRICS_TRACKER.platform,
|
|
||||||
"type": "action",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
if custom_props:
|
|
||||||
event_params["properties"].update(custom_props)
|
|
||||||
|
|
||||||
METRICS_TRACKER.queue.put_nowait(event_params)
|
|
||||||
except Exception as ex:
|
|
||||||
# wrapping this whole thing in a try except as we never want a failure here to annoy users!
|
|
||||||
LOG.debug(f"Error queueing metrics request: {str(ex)}")
|
|
||||||
|
|
||||||
|
|
||||||
def initialise_tracker(account=None):
|
|
||||||
global METRICS_TRACKER
|
|
||||||
if not METRICS_TRACKER:
|
|
||||||
METRICS_TRACKER = MetricsTracker()
|
|
||||||
|
|
||||||
if account and account.userInfo.email:
|
|
||||||
METRICS_TRACKER.set_last_user(account.userInfo.email)
|
|
||||||
if account and account.serverInfo.url:
|
|
||||||
METRICS_TRACKER.set_last_server(account.serverInfo.url)
|
|
||||||
|
|
||||||
|
|
||||||
class Singleton(type):
|
|
||||||
_instances = {}
|
|
||||||
|
|
||||||
def __call__(cls, *args, **kwargs):
|
|
||||||
if cls not in cls._instances:
|
|
||||||
cls._instances[cls] = super(Singleton, cls).__call__(*args, **kwargs)
|
|
||||||
return cls._instances[cls]
|
|
||||||
|
|
||||||
|
|
||||||
class MetricsTracker(metaclass=Singleton):
|
|
||||||
analytics_url = "https://analytics.speckle.systems/track?ip=1"
|
|
||||||
analytics_token = "acd87c5a50b56df91a795e999812a3a4"
|
|
||||||
last_user = ""
|
|
||||||
last_server = None
|
|
||||||
platform = None
|
|
||||||
sending_thread = None
|
|
||||||
queue = queue.Queue(1000)
|
|
||||||
|
|
||||||
def __init__(self) -> None:
|
|
||||||
self.sending_thread = threading.Thread(
|
|
||||||
target=self._send_tracking_requests, daemon=True
|
|
||||||
)
|
|
||||||
self.platform = PLATFORMS.get(sys.platform, "linux")
|
|
||||||
self.sending_thread.start()
|
|
||||||
with contextlib.suppress(Exception):
|
|
||||||
node, user = platform.node(), getpass.getuser()
|
|
||||||
if node and user:
|
|
||||||
self.last_user = f"@{self.hash(f'{node}-{user}')}"
|
|
||||||
|
|
||||||
def set_last_user(self, email: str):
|
|
||||||
if not email:
|
|
||||||
return
|
|
||||||
self.last_user = f"@{self.hash(email)}"
|
|
||||||
|
|
||||||
def set_last_server(self, server: str):
|
|
||||||
if not server:
|
|
||||||
return
|
|
||||||
self.last_server = self.hash(server)
|
|
||||||
|
|
||||||
def hash(self, value: str):
|
|
||||||
inputList = value.lower().split("://")
|
|
||||||
input = inputList[len(inputList)-1].split("/")[0].split('?')[0]
|
|
||||||
return hashlib.md5(input.encode("utf-8")).hexdigest().upper()
|
|
||||||
|
|
||||||
def _send_tracking_requests(self):
|
|
||||||
session = requests.Session()
|
|
||||||
while True:
|
|
||||||
event_params = [self.queue.get()]
|
|
||||||
|
|
||||||
try:
|
|
||||||
session.post(self.analytics_url, json=event_params)
|
|
||||||
except Exception as ex:
|
|
||||||
LOG.debug(f"Error sending metrics request: {str(ex)}")
|
|
||||||
|
|
||||||
self.queue.task_done()
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
from typing import Optional
|
|
||||||
from specklepy.objects import Base
|
|
||||||
|
|
||||||
|
|
||||||
class CRS(Base, speckle_type="Objects.GIS.CRS"):
|
|
||||||
"""A Coordinate Reference System stored in wkt format"""
|
|
||||||
|
|
||||||
name: Optional[str] = None
|
|
||||||
authority_id: Optional[str] = None
|
|
||||||
wkt: Optional[str] = None
|
|
||||||
units_native: Optional[str] = None
|
|
||||||
offset_x: Optional[float] = None
|
|
||||||
offset_y: Optional[float] = None
|
|
||||||
rotation: Optional[float] = None
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
"""Builtin Speckle object kit."""
|
|
||||||
|
|
||||||
from specklepy.objects.GIS.layers import (
|
|
||||||
VectorLayer,
|
|
||||||
RasterLayer,
|
|
||||||
)
|
|
||||||
|
|
||||||
from specklepy.objects.GIS.geometry import (
|
|
||||||
GisPolygonGeometry,
|
|
||||||
GisPolygonElement,
|
|
||||||
GisLineElement,
|
|
||||||
GisPointElement,
|
|
||||||
GisRasterElement,
|
|
||||||
)
|
|
||||||
|
|
||||||
from specklepy.objects.GIS.CRS import (
|
|
||||||
CRS,
|
|
||||||
)
|
|
||||||
|
|
||||||
__all__ = ["VectorLayer", "RasterLayer",
|
|
||||||
"GisPolygonGeometry", "GisPolygonElement", "GisLineElement", "GisPointElement", "GisRasterElement",
|
|
||||||
"CRS"]
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
|
|
||||||
from typing import Optional, Union, List
|
|
||||||
from specklepy.objects.geometry import Point, Line, Polyline, Circle, Arc, Polycurve, Mesh
|
|
||||||
from specklepy.objects import Base
|
|
||||||
from deprecated import deprecated
|
|
||||||
|
|
||||||
class GisPolygonGeometry(Base, speckle_type="Objects.GIS.PolygonGeometry", detachable={"displayValue"}):
|
|
||||||
"""GIS Polygon Geometry"""
|
|
||||||
|
|
||||||
boundary: Optional[Union[Polyline, Arc, Line, Circle, Polycurve]] = None
|
|
||||||
voids: Optional[List[Union[Polyline, Arc, Line, Circle, Polycurve]] ] = None
|
|
||||||
displayValue: Optional[List[Mesh]] = None
|
|
||||||
|
|
||||||
class GisPolygonElement(Base, speckle_type="Objects.GIS.PolygonElement"):
|
|
||||||
"""GIS Polygon element"""
|
|
||||||
|
|
||||||
geometry: Optional[List[GisPolygonGeometry]] = None
|
|
||||||
attributes: Optional[Base] = None
|
|
||||||
|
|
||||||
class GisLineElement(Base, speckle_type="Objects.GIS.LineElement"):
|
|
||||||
"""GIS Polyline element"""
|
|
||||||
|
|
||||||
geometry: Optional[List[Union[Polyline, Arc, Line, Circle, Polycurve]]] = None,
|
|
||||||
attributes: Optional[Base] = None,
|
|
||||||
|
|
||||||
class GisPointElement(Base, speckle_type="Objects.GIS.PointElement"):
|
|
||||||
"""GIS Point element"""
|
|
||||||
|
|
||||||
geometry: Optional[List[Point]] = None,
|
|
||||||
attributes: Optional[Base] = None,
|
|
||||||
|
|
||||||
class GisRasterElement(Base, speckle_type="Objects.GIS.RasterElement", detachable={"displayValue"}):
|
|
||||||
"""GIS Raster element"""
|
|
||||||
|
|
||||||
band_count: Optional[int] = None
|
|
||||||
band_names: Optional[List[str]] = None
|
|
||||||
x_origin: Optional[float] = None
|
|
||||||
y_origin: Optional[float] = None
|
|
||||||
x_size: Optional[int] = None
|
|
||||||
y_size: Optional[int] = None
|
|
||||||
x_resolution: Optional[float] = None
|
|
||||||
y_resolution: Optional[float] = None
|
|
||||||
noDataValue: Optional[List[float]] = None
|
|
||||||
displayValue: Optional[List[Mesh]] = None
|
|
||||||
|
|
||||||
class GisTopography(GisRasterElement, speckle_type="Objects.GIS.GisTopography", detachable={"displayValue"}):
|
|
||||||
"""GIS Raster element with 3d Topography representation"""
|
|
||||||
|
|
||||||
class GisNonGeometryElement(Base, speckle_type="Objects.GIS.NonGeometryElement"):
|
|
||||||
"""GIS Table feature"""
|
|
||||||
|
|
||||||
attributes: Optional[Base] = None
|
|
||||||
|
|
||||||
@@ -1,138 +0,0 @@
|
|||||||
from typing import Any, Dict, List, Union, Optional
|
|
||||||
from specklepy.objects.base import Base
|
|
||||||
from specklepy.objects.other import Collection
|
|
||||||
|
|
||||||
from specklepy.objects.GIS.CRS import CRS
|
|
||||||
from deprecated import deprecated
|
|
||||||
|
|
||||||
@deprecated(version="2.15", reason="Use VectorLayer or RasterLayer instead")
|
|
||||||
class Layer(Base, detachable={"features"}):
|
|
||||||
"""A GIS Layer"""
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
name:str=None,
|
|
||||||
crs:CRS=None,
|
|
||||||
units: str = "m",
|
|
||||||
features: Optional[List[Base]] = None,
|
|
||||||
layerType: str = "None",
|
|
||||||
geomType: str = "None",
|
|
||||||
renderer: Optional[dict[str, Any]] = None,
|
|
||||||
**kwargs
|
|
||||||
) -> None:
|
|
||||||
super().__init__(**kwargs)
|
|
||||||
self.name = name
|
|
||||||
self.crs = crs
|
|
||||||
self.units = units
|
|
||||||
self.type = layerType
|
|
||||||
self.features = features or []
|
|
||||||
self.geomType = geomType
|
|
||||||
self.renderer = renderer or {}
|
|
||||||
|
|
||||||
@deprecated(version="2.16", reason="Use VectorLayer or RasterLayer instead")
|
|
||||||
class VectorLayer(
|
|
||||||
Collection,
|
|
||||||
detachable={"elements"},
|
|
||||||
speckle_type="VectorLayer",
|
|
||||||
serialize_ignore={"features"}):
|
|
||||||
|
|
||||||
"""GIS Vector Layer"""
|
|
||||||
name: Optional[str]=None
|
|
||||||
crs: Optional[Union[CRS, Base]]=None
|
|
||||||
units: Optional[str] = None
|
|
||||||
elements: Optional[List[Base]] = None
|
|
||||||
attributes: Optional[Base] = None
|
|
||||||
geomType: Optional[str] = "None"
|
|
||||||
renderer: Optional[Dict[str, Any]] = None
|
|
||||||
collectionType = "VectorLayer"
|
|
||||||
|
|
||||||
@property
|
|
||||||
@deprecated(version="2.14", reason="Use elements")
|
|
||||||
def features(self) -> Optional[List[Base]]:
|
|
||||||
return self.elements
|
|
||||||
|
|
||||||
@features.setter
|
|
||||||
def features(self, value: Optional[List[Base]]) -> None:
|
|
||||||
self.elements = value
|
|
||||||
|
|
||||||
@deprecated(version="2.16", reason="Use VectorLayer or RasterLayer instead")
|
|
||||||
class RasterLayer(
|
|
||||||
Collection,
|
|
||||||
detachable={"elements"},
|
|
||||||
speckle_type="RasterLayer",
|
|
||||||
serialize_ignore={"features"}):
|
|
||||||
|
|
||||||
"""GIS Raster Layer"""
|
|
||||||
|
|
||||||
name: Optional[str] = None
|
|
||||||
crs: Optional[Union[CRS, Base]]=None
|
|
||||||
units: Optional[str] = None
|
|
||||||
rasterCrs: Optional[Union[CRS, Base]]=None
|
|
||||||
elements: Optional[List[Base]] = None
|
|
||||||
geomType: Optional[str] = "None"
|
|
||||||
renderer: Optional[Dict[str, Any]] = None
|
|
||||||
collectionType = "RasterLayer"
|
|
||||||
|
|
||||||
|
|
||||||
@property
|
|
||||||
@deprecated(version="2.14", reason="Use elements")
|
|
||||||
def features(self) -> Optional[List[Base]]:
|
|
||||||
return self.elements
|
|
||||||
|
|
||||||
@features.setter
|
|
||||||
def features(self, value: Optional[List[Base]]) -> None:
|
|
||||||
self.elements = value
|
|
||||||
|
|
||||||
|
|
||||||
class VectorLayer(
|
|
||||||
Collection,
|
|
||||||
detachable={"elements"},
|
|
||||||
speckle_type="Objects.GIS.VectorLayer",
|
|
||||||
serialize_ignore={"features"}):
|
|
||||||
|
|
||||||
"""GIS Vector Layer"""
|
|
||||||
|
|
||||||
name: Optional[str]=None
|
|
||||||
crs: Optional[Union[CRS, Base]]=None
|
|
||||||
units: Optional[str] = None
|
|
||||||
elements: Optional[List[Base]] = None
|
|
||||||
attributes: Optional[Base] = None
|
|
||||||
geomType: Optional[str] = "None"
|
|
||||||
renderer: Optional[Dict[str, Any]] = None
|
|
||||||
collectionType = "VectorLayer"
|
|
||||||
|
|
||||||
@property
|
|
||||||
@deprecated(version="2.14", reason="Use elements")
|
|
||||||
def features(self) -> Optional[List[Base]]:
|
|
||||||
return self.elements
|
|
||||||
|
|
||||||
@features.setter
|
|
||||||
def features(self, value: Optional[List[Base]]) -> None:
|
|
||||||
self.elements = value
|
|
||||||
|
|
||||||
class RasterLayer(
|
|
||||||
Collection,
|
|
||||||
detachable={"elements"},
|
|
||||||
speckle_type="Objects.GIS.RasterLayer",
|
|
||||||
serialize_ignore={"features"}):
|
|
||||||
|
|
||||||
"""GIS Raster Layer"""
|
|
||||||
|
|
||||||
name: Optional[str] = None
|
|
||||||
crs: Optional[Union[CRS, Base]]=None
|
|
||||||
units: Optional[str] = None
|
|
||||||
rasterCrs: Optional[Union[CRS, Base]]=None
|
|
||||||
elements: Optional[List[Base]] = None
|
|
||||||
geomType: Optional[str] = "None"
|
|
||||||
renderer: Optional[Dict[str, Any]] = None
|
|
||||||
collectionType = "RasterLayer"
|
|
||||||
|
|
||||||
|
|
||||||
@property
|
|
||||||
@deprecated(version="2.14", reason="Use elements")
|
|
||||||
def features(self) -> Optional[List[Base]]:
|
|
||||||
return self.elements
|
|
||||||
|
|
||||||
@features.setter
|
|
||||||
def features(self, value: Optional[List[Base]]) -> None:
|
|
||||||
self.elements = value
|
|
||||||
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user