Compare commits
432 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 89f4c75cc9 | |||
| d20f7ea82a | |||
| 2e44a891e8 | |||
| edce76491f | |||
| ecbd0eab09 | |||
| b17423b282 | |||
| 166b0f5e87 | |||
| cac34120a9 | |||
| 55c4c68cf3 | |||
| be850d5ea9 | |||
| c9a5badac1 | |||
| 118fa07e37 | |||
| d71b616e2b | |||
| 35750f12c5 | |||
| 5730cdcb43 | |||
| 82b6dbbe78 | |||
| 883be4b27b | |||
| 37e2711a76 | |||
| 8dcc67fb31 | |||
| ed84820995 | |||
| 5c3dcb7bc0 | |||
| 92732e3c76 | |||
| 903951547d | |||
| 82c3dc9ffb | |||
| a0e10aae99 | |||
| bbea2a0d76 | |||
| a05ac3479b | |||
| 0bd972945e | |||
| f200544065 | |||
| 68ce9823ae | |||
| a920352407 | |||
| 24bfb6718e | |||
| e63f4b8636 | |||
| 47c6bd89af | |||
| bd38dfacc7 | |||
| 281483f0fc | |||
| 932838de8f | |||
| a0b39e4c64 | |||
| 759cd0ef58 | |||
| 46c18bbe6b | |||
| 82d39e66fe | |||
| 10f7499182 | |||
| 170d2f0450 | |||
| 040a4e2553 | |||
| e978e4f632 | |||
| eae60160a1 | |||
| c78a780e85 | |||
| 1b45f50697 | |||
| be8fae3b1c | |||
| ab41d3cbe0 | |||
| f843bb0c89 | |||
| b7933e0088 | |||
| 7e09d4f4ce | |||
| bb62109332 | |||
| 3642731f37 | |||
| 3bd849c815 | |||
| 2acf4c41c7 | |||
| 6b6ff80bf2 | |||
| 0f1f00db00 | |||
| 280927b720 | |||
| 6096cd25f6 | |||
| cc004c8e6b | |||
| a10b2594d3 | |||
| 976a52bdc8 | |||
| 09ca501a74 | |||
| 225d4f02d4 | |||
| f1b51848cf | |||
| 08fb3f6cd7 | |||
| fe7909c913 | |||
| a00e16929d | |||
| 44d1ef9f93 | |||
| 404dbd1d1e | |||
| 537a504121 | |||
| 6c03dc82c8 | |||
| 780126528d | |||
| fe03d96ae2 | |||
| 078a6c8da8 | |||
| 905377dea1 | |||
| 62c5114cb3 | |||
| 43a5302a90 | |||
| addaa996ea | |||
| 3b5421a5bc | |||
| 88e8c86fa6 | |||
| d6843b9971 | |||
| 302a9f7f30 | |||
| ede9591c6a | |||
| c5b339d891 | |||
| 2e35fb9e5c | |||
| e6b822b0e3 | |||
| 239bc4b5b9 | |||
| 4eea15ddc1 | |||
| 204aa7466e | |||
| 24019e99f3 | |||
| 64492fafa5 | |||
| 3a8d634989 | |||
| f27650af3a | |||
| 6469b6f757 | |||
| b28db0881c | |||
| b0b442de23 | |||
| 32d2fe8ead | |||
| 9fd40eac23 | |||
| b22ba1f1f1 | |||
| 5e20fe7bf1 | |||
| 6da5da23c4 | |||
| 1b59f0b026 | |||
| 78123936d2 | |||
| dbc1aefed3 | |||
| e726345b0c | |||
| e074dbcced | |||
| 62e342b2cb | |||
| 804dd37639 | |||
| 64b61f54f5 | |||
| 58789ab234 | |||
| 2696fb74ba | |||
| 57e176af91 | |||
| 437483641c | |||
| 1e971b57c3 | |||
| f04be12ec8 | |||
| 51242928ca | |||
| 77b3be9145 | |||
| cc5abdf9cb | |||
| 4eca5144a8 | |||
| 8589663049 | |||
| 956f72dd6a | |||
| a2daa68c1c | |||
| d60feb73a2 | |||
| a0ca10ad20 | |||
| f6118f3336 | |||
| c7cd2f3e91 | |||
| b374bfefd0 | |||
| d716db251f | |||
| 6d7e7c5c4b | |||
| 7dcd9288ca | |||
| 7d99f48def | |||
| 4332a8faef | |||
| a1aee8b3fa | |||
| deb8ad50c5 | |||
| 558b25b1d1 | |||
| 4db0fa69fa | |||
| 1eca211c96 | |||
| f65173581a | |||
| 223c776c63 | |||
| ccccc53f59 | |||
| ae6fc85ab4 | |||
| 7ad0785c62 | |||
| 76e4ec1535 | |||
| 4e96aade51 | |||
| 7ca00b7b77 | |||
| bddf9c0c1c | |||
| bf3ab7da4c | |||
| 4dc148181e | |||
| 357859288d | |||
| 1ce61bdda8 | |||
| 42737c4ed2 | |||
| 62ee1a4b0a | |||
| d21373873c | |||
| e3716f6206 | |||
| f6917b0761 | |||
| 04764b17eb | |||
| dbe3d759f6 | |||
| f6ff484e66 | |||
| bd000395af | |||
| 10f49579fd | |||
| 1693465dfc | |||
| c3a7ead8f5 | |||
| d151a8d0ae | |||
| c0dd88cbdb | |||
| 71d3589e72 | |||
| 5bde1bc2d6 | |||
| 75e6f0229a | |||
| 5d7e71f357 | |||
| 6c223b6fb3 | |||
| e6131a7956 | |||
| 45b50e4f26 | |||
| d9b92490ec | |||
| 37c09fa56c | |||
| cbae4d300d | |||
| 2742c12e31 | |||
| 6dd0813089 | |||
| a1831b57db | |||
| 1ff3245531 | |||
| 3b4723a186 | |||
| efe9551c5e | |||
| 23a5087fbc | |||
| 52c8e37a5b | |||
| 6a6b3d4c3d | |||
| 8f32aa014e | |||
| 11c6221972 | |||
| 262be44423 | |||
| fd3d97cf5a | |||
| 9dba99ad26 | |||
| 2810598336 | |||
| f918582ed2 | |||
| 9181440c62 | |||
| 62912d4428 | |||
| 67cf41d721 | |||
| 4ad3761478 | |||
| 6e8e08ae94 | |||
| 6e7c36223f | |||
| b1f979a10a | |||
| d1ebd84cca | |||
| fe92e49c59 | |||
| fbbd6c0dd7 | |||
| 8ffe219111 | |||
| e4d087db3a | |||
| 2e8943e961 | |||
| f254defc6b | |||
| 541e3d961f | |||
| b02f183533 | |||
| 589198f5f1 | |||
| 948a56a07f | |||
| 3eed9a60fa | |||
| c169c4eeda | |||
| 32b5ef88a1 | |||
| 3a979318ad | |||
| 1e6321c7f1 | |||
| b5fb684864 | |||
| 65048cd01b | |||
| 9d2fd5bc42 | |||
| bd35fb59c3 | |||
| 4931c95d7c | |||
| 52d53db661 | |||
| 23ee28f851 | |||
| 791190a38c | |||
| 3c7feb0bec | |||
| 2b583fd822 | |||
| 8244e3ecc7 | |||
| 5ac9d80cbc | |||
| 5e2fbaa7c2 | |||
| 703ceaf369 | |||
| a5096c41ca | |||
| 972339454d | |||
| 34c11d5931 | |||
| 854ce9f77f | |||
| 7f926cf547 | |||
| 5e8b54e3b7 | |||
| 8bd46e4e64 | |||
| 91edd4f85b | |||
| 0cb6c7f682 | |||
| 125a4bbeed | |||
| 76c4074aed | |||
| 16164a57da | |||
| 3a225fa935 | |||
| 102850b894 | |||
| 5ac85c541b | |||
| cca7b18119 | |||
| 8a34b95128 | |||
| 46d7abbaee | |||
| 67e95caf5f | |||
| 04532ed645 | |||
| 7df175d9bb | |||
| 3912fa8860 | |||
| 34de2928ae | |||
| 8a91260887 | |||
| 88eea00787 | |||
| c57d57c009 | |||
| 708e3329e3 | |||
| f0e68845c0 | |||
| 434a4376b3 | |||
| d701bedcc7 | |||
| 6238150bd5 | |||
| 3e41e8cd8e | |||
| 3962126b54 | |||
| c99c25e848 | |||
| 1ef9b91e82 | |||
| c0cfe1471a | |||
| da838280c3 | |||
| 681872e5ff | |||
| e11c41e0f8 | |||
| ec651a9237 | |||
| ece957fb0f | |||
| 5338d8ac0f | |||
| e36ea70e8a | |||
| edf2afaa89 | |||
| e0b1b272c0 | |||
| 682e82057e | |||
| 473e5cfddb | |||
| 03cd989165 | |||
| 284d841a1e | |||
| 668fc5131f | |||
| 64926bd41d | |||
| 1c9b186ea5 | |||
| ed9f1ad818 | |||
| b83c30a1c9 | |||
| c079342a55 | |||
| 6aa29d9b30 | |||
| f456e4ddbb | |||
| dcd13224d0 | |||
| 06952a5991 | |||
| 6049049813 | |||
| 75b7d30bd6 | |||
| 19e26318fc | |||
| 798dc7ff6a | |||
| 0e4cff5904 | |||
| d1502c9072 | |||
| 869629e2a3 | |||
| 48b98294fb | |||
| 8205180e5d | |||
| a2fd21f541 | |||
| 08ac76cf09 | |||
| fbf19420fa | |||
| 44336addaf | |||
| 43c9a9cace | |||
| 99e9f773b8 | |||
| 189a5847cf | |||
| eb86b4881a | |||
| 64fca5f280 | |||
| 784e9c1326 | |||
| c7f5e0718b | |||
| d2685c5cf5 | |||
| 1b83e5a84b | |||
| 77e09b9780 | |||
| 402f750200 | |||
| a43e7471a4 | |||
| a4ed7ebb08 | |||
| e7eb7c67a9 | |||
| d547cdaac0 | |||
| 6f35c1bd20 | |||
| 420c73f484 | |||
| c2859475cc | |||
| 56e8d65e2b | |||
| 7885a6be8d | |||
| b19b85c9d1 | |||
| db4b2b7f87 | |||
| 77916995bc | |||
| 3dd56dc38e | |||
| ae42bec1c3 | |||
| ea7baf8eb5 | |||
| 6a9f4bf89b | |||
| 8352bb5c9a | |||
| fc34b876fd | |||
| 183993cfc5 | |||
| 9be3b4b93d | |||
| 0b14660115 | |||
| 68c4c682a0 | |||
| 4f93ddcaf3 | |||
| e842f651b9 | |||
| 7e1bec1aba | |||
| 1fb9a4f5fe | |||
| 1668c80bed | |||
| ac6ba87c68 | |||
| 3db8565f57 | |||
| a32822f4e3 | |||
| 40956927c8 | |||
| 4628f111ba | |||
| 9c952b432d | |||
| f075988e4b | |||
| 65c829404a | |||
| 85e7a72524 | |||
| 0533aa0139 | |||
| 04b733a241 | |||
| 01036c0f2e | |||
| 4d8ca534fe | |||
| e941efd95c | |||
| be9defbfa9 | |||
| dd54c69554 | |||
| 93c1ec9556 | |||
| 659c57e840 | |||
| 1cdd4ffc9c | |||
| 5ae022d2ed | |||
| 31ca59cea8 | |||
| c91f673dba | |||
| e2c8ef1b3d | |||
| b6b0a5a3a0 | |||
| f36d63a2cf | |||
| afb9065fb9 | |||
| fcc33f8989 | |||
| 2cf9b64221 | |||
| 990cf4eb2f | |||
| b25f2ab4bc | |||
| a8786c126d | |||
| a35f8936ca | |||
| d965cc0988 | |||
| 59a6950eed | |||
| 6804282fac | |||
| a3a22a43d5 | |||
| 08aaa41b6c | |||
| 07a3213ee9 | |||
| 7a46176803 | |||
| b75501addd | |||
| 15636dbe62 | |||
| 9a2061d900 | |||
| 8dc51fb936 | |||
| e805b8ac6e | |||
| 5a1d624979 | |||
| c8808b07b3 | |||
| 1a9b847c44 | |||
| f01fcb8e66 | |||
| 67499ab20c | |||
| ebb703e68d | |||
| ef5d41da5d | |||
| 851dd9c482 | |||
| 78074cf691 | |||
| be0daa419d | |||
| a7d1b9ce30 | |||
| 05756a8e9e | |||
| b50e658333 | |||
| 88248353ab | |||
| aec94f8f7f | |||
| e6b1604bc3 | |||
| de29b93b8b | |||
| 10aa8b59b6 | |||
| b86faa6a14 | |||
| 7430611c52 | |||
| ddd52f4af9 | |||
| 35bc6b0350 | |||
| 9585d46c4e | |||
| 1900fece8b | |||
| 344b7de557 | |||
| fd09e97a53 | |||
| 459bd0901f | |||
| ae7c4bc14d | |||
| 41f1823aae | |||
| 625bd5cd84 | |||
| 8812985c67 | |||
| c838835a65 | |||
| 361ba6bfcd | |||
| 8078a4b596 | |||
| 08b106464f | |||
| 0cfe5db674 | |||
| d9dbca2c68 | |||
| ab5b55871b | |||
| 2f87956154 | |||
| 6fc4ab1539 | |||
| 7f432e768d | |||
| d0eb364b3e | |||
| 936b2d8b5a | |||
| 5a9aec80bd | |||
| 6616526279 | |||
| 6e00de58b9 | |||
| 0ec404bbec | |||
| 506aaf68ca |
+65
-36
@@ -1,53 +1,76 @@
|
|||||||
version: 2.1
|
version: 2.1
|
||||||
|
|
||||||
orbs:
|
orbs:
|
||||||
python: circleci/python@2.0.3
|
codecov: codecov/codecov@3.3.0
|
||||||
codecov: codecov/codecov@3.2.2
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
test:
|
pre-commit:
|
||||||
|
parameters:
|
||||||
|
config_file:
|
||||||
|
default: ./.pre-commit-config.yaml
|
||||||
|
description: Optional, path to pre-commit config file.
|
||||||
|
type: string
|
||||||
|
cache_prefix:
|
||||||
|
default: ''
|
||||||
|
description: |
|
||||||
|
Optional cache prefix to be used on CircleCI. Can be used for cache busting or to ensure multiple jobs use different caches.
|
||||||
|
type: string
|
||||||
docker:
|
docker:
|
||||||
- image: "cimg/python:<<parameters.tag>>"
|
- image: speckle/pre-commit-runner:latest
|
||||||
- image: "cimg/node:16.15"
|
resource_class: medium
|
||||||
- image: "cimg/redis:6.2"
|
steps:
|
||||||
- image: "cimg/postgres:14.2"
|
- checkout
|
||||||
environment:
|
- restore_cache:
|
||||||
POSTGRES_DB: speckle2_test
|
keys:
|
||||||
POSTGRES_PASSWORD: speckle
|
- cache-pre-commit-<<parameters.cache_prefix>>-{{ checksum "<<parameters.config_file>>" }}
|
||||||
POSTGRES_USER: speckle
|
- run:
|
||||||
- image: "speckle/speckle-server"
|
name: Install pre-commit hooks
|
||||||
command: ["bash", "-c", "/wait && node bin/www"]
|
command: pre-commit install-hooks --config <<parameters.config_file>>
|
||||||
environment:
|
- save_cache:
|
||||||
POSTGRES_URL: "localhost"
|
key: cache-pre-commit-<<parameters.cache_prefix>>-{{ checksum "<<parameters.config_file>>" }}
|
||||||
POSTGRES_USER: "speckle"
|
paths:
|
||||||
POSTGRES_PASSWORD: "speckle"
|
- ~/.cache/pre-commit
|
||||||
POSTGRES_DB: "speckle2_test"
|
- run:
|
||||||
REDIS_URL: "redis://localhost"
|
name: Run pre-commit
|
||||||
SESSION_SECRET: "keyboard cat"
|
command: pre-commit run --all-files
|
||||||
STRATEGY_LOCAL: "true"
|
- run:
|
||||||
CANONICAL_URL: "http://localhost:3000"
|
command: git --no-pager diff
|
||||||
WAIT_HOSTS: localhost:5432, localhost:6379
|
name: git diff
|
||||||
DISABLE_FILE_UPLOADS: "true"
|
when: on_fail
|
||||||
|
|
||||||
|
test:
|
||||||
|
machine:
|
||||||
|
image: ubuntu-2204:2023.02.1
|
||||||
|
docker_layer_caching: false
|
||||||
|
resource_class: medium
|
||||||
parameters:
|
parameters:
|
||||||
tag:
|
tag:
|
||||||
default: "3.8"
|
default: "3.11"
|
||||||
type: string
|
type: string
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- run: python --version
|
|
||||||
- run:
|
- run:
|
||||||
command: python -m pip install --upgrade pip
|
name: Install python
|
||||||
name: upgrade pip
|
command: |
|
||||||
- python/install-packages:
|
pyenv install -s << parameters.tag >>
|
||||||
pkg-manager: poetry
|
pyenv global << parameters.tag >>
|
||||||
- run: poetry run pytest --cov --cov-report xml:reports/coverage.xml --junitxml=reports/test-results.xml
|
- run:
|
||||||
|
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:
|
||||||
@@ -57,20 +80,26 @@ jobs:
|
|||||||
- 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 specklesystems -p $PYPI_PASSWORD
|
- run: poetry publish -u __token__ -p $PYPI_TOKEN
|
||||||
|
|
||||||
workflows:
|
workflows:
|
||||||
main:
|
main:
|
||||||
jobs:
|
jobs:
|
||||||
|
- pre-commit:
|
||||||
|
filters:
|
||||||
|
tags:
|
||||||
|
only: /.*/
|
||||||
- test:
|
- test:
|
||||||
matrix:
|
matrix:
|
||||||
parameters:
|
parameters:
|
||||||
tag: ["3.7", "3.8", "3.9", "3.10"]
|
tag: ["3.11"]
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
- deploy:
|
- deploy:
|
||||||
|
context: pypi
|
||||||
requires:
|
requires:
|
||||||
|
- pre-commit
|
||||||
- test
|
- test
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
|
|||||||
@@ -22,6 +22,6 @@ RUN if [ "${NODE_VERSION}" != "none" ]; then su vscode -c "umask 0002 && . /usr/
|
|||||||
|
|
||||||
USER vscode
|
USER vscode
|
||||||
|
|
||||||
RUN curl -sSL https://raw.githubusercontent.com/python-poetry/poetry/master/get-poetry.py | python -
|
RUN curl -sSL https://install.python-poetry.org | python3 -
|
||||||
|
|
||||||
ENV PATH=$PATH:$HOME/.poetry/env
|
ENV PATH=$PATH:$HOME/.poetry/env
|
||||||
|
|||||||
@@ -52,4 +52,4 @@
|
|||||||
"postCreateCommand": "poetry config virtualenvs.create false && poetry install",
|
"postCreateCommand": "poetry config virtualenvs.create false && poetry install",
|
||||||
// Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
// Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
||||||
"remoteUser": "vscode"
|
"remoteUser": "vscode"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -41,4 +41,4 @@ services:
|
|||||||
command: /bin/sh -c "while sleep 1000; do :; done"
|
command: /bin/sh -c "while sleep 1000; do :; done"
|
||||||
network_mode: host
|
network_mode: host
|
||||||
# networks:
|
# networks:
|
||||||
# default:
|
# default:
|
||||||
|
|||||||
+1
-1
@@ -1,3 +1,3 @@
|
|||||||
* text=auto eol=lf
|
* text=auto eol=lf
|
||||||
*.{cmd,[cC][mM][dD]} text eol=crlf
|
*.{cmd,[cC][mM][dD]} text eol=crlf
|
||||||
*.{bat,[bB][aA][tT]} text eol=crlf
|
*.{bat,[bB][aA][tT]} text eol=crlf
|
||||||
|
|||||||
@@ -1,77 +0,0 @@
|
|||||||
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 }}
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
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
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
repos:
|
||||||
|
- repo: https://github.com/charliermarsh/ruff-pre-commit
|
||||||
|
hooks:
|
||||||
|
- id: ruff
|
||||||
|
rev: v0.8.2
|
||||||
|
|
||||||
|
- repo: https://github.com/commitizen-tools/commitizen
|
||||||
|
hooks:
|
||||||
|
- id: commitizen
|
||||||
|
- id: commitizen-branch
|
||||||
|
stages:
|
||||||
|
- push
|
||||||
|
rev: v3.13.0
|
||||||
|
|
||||||
|
- repo: https://github.com/pycqa/isort
|
||||||
|
rev: 5.13.2
|
||||||
|
hooks:
|
||||||
|
- id: isort
|
||||||
|
|
||||||
|
- repo: https://github.com/psf/black
|
||||||
|
rev: 24.10.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.5.0
|
||||||
|
hooks:
|
||||||
|
- id: end-of-file-fixer
|
||||||
|
- id: trailing-whitespace
|
||||||
Vendored
+5
-6
@@ -4,10 +4,9 @@
|
|||||||
// 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": "debugpy",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"program": "${file}",
|
"program": "${file}",
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
@@ -15,12 +14,12 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Pytest",
|
"name": "Pytest",
|
||||||
"type": "python",
|
"type": "debugpy",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"program": "poetry",
|
"module": "pytest",
|
||||||
"args": ["run", "pytest"],
|
"args": [],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"justMyCode": true
|
"justMyCode": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,52 +2,22 @@
|
|||||||
<img src="https://user-images.githubusercontent.com/2679513/131189167-18ea5fe1-c578-47f6-9785-3748178e4312.png" width="150px"/><br/>
|
<img src="https://user-images.githubusercontent.com/2679513/131189167-18ea5fe1-c578-47f6-9785-3748178e4312.png" width="150px"/><br/>
|
||||||
Speckle | specklepy 🐍
|
Speckle | specklepy 🐍
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
|
<p align="center"><a href="https://twitter.com/SpeckleSystems"><img src="https://img.shields.io/twitter/follow/SpeckleSystems?style=social" alt="Twitter Follow"></a> <a href="https://speckle.community"><img src="https://img.shields.io/discourse/users?server=https%3A%2F%2Fspeckle.community&style=flat-square&logo=discourse&logoColor=white" alt="Community forum users"></a> <a href="https://speckle.systems"><img src="https://img.shields.io/badge/https://-speckle.systems-royalblue?style=flat-square" alt="website"></a> <a href="https://speckle.guide/dev/"><img src="https://img.shields.io/badge/docs-speckle.guide-orange?style=flat-square&logo=read-the-docs&logoColor=white" alt="docs"></a></p>
|
||||||
|
|
||||||
|
> Speckle is the first AEC data hub that connects with your favorite AEC tools. Speckle exists to overcome the challenges of working in a fragmented industry where communication, creative workflows, and the exchange of data are often hindered by siloed software and processes. It is here to make the industry better.
|
||||||
|
|
||||||
<h3 align="center">
|
<h3 align="center">
|
||||||
The Python SDK
|
The Python SDK
|
||||||
</h3>
|
</h3>
|
||||||
<p align="center"><b>Speckle</b> is the data infrastructure for the AEC industry.</p><br/>
|
|
||||||
|
|
||||||
<p align="center"><a href="https://twitter.com/SpeckleSystems"><img src="https://img.shields.io/twitter/follow/SpeckleSystems?style=social" alt="Twitter Follow"></a> <a href="https://speckle.community"><img src="https://img.shields.io/discourse/users?server=https%3A%2F%2Fspeckle.community&style=flat-square&logo=discourse&logoColor=white" alt="Community forum users"></a> <a href="https://speckle.systems"><img src="https://img.shields.io/badge/https://-speckle.systems-royalblue?style=flat-square" alt="website"></a> <a href="https://speckle.guide/dev/"><img src="https://img.shields.io/badge/docs-speckle.guide-orange?style=flat-square&logo=read-the-docs&logoColor=white" alt="docs"></a></p>
|
|
||||||
<p align="center"><a href="https://github.com/specklesystems/specklepy/"><img src="https://circleci.com/gh/specklesystems/specklepy.svg?style=svg&circle-token=76eabd350ea243575cbb258b746ed3f471f7ac29" alt="Speckle-Next"></a><a href="https://codecov.io/gh/specklesystems/specklepy">
|
|
||||||
<img src="https://codecov.io/gh/specklesystems/specklepy/branch/main/graph/badge.svg?token=8KQFL5N0YF"/>
|
|
||||||
</a> </p>
|
|
||||||
|
|
||||||
# About Speckle
|
|
||||||
|
|
||||||
What is Speckle? Check our 
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
- **Object-based:** say goodbye to files! Speckle is the first object based platform for the AEC industry
|
|
||||||
- **Version control:** Speckle is the Git & Hub for geometry and BIM data
|
|
||||||
- **Collaboration:** share your designs collaborate with others
|
|
||||||
- **3D Viewer:** see your CAD and BIM models online, share and embed them anywhere
|
|
||||||
- **Interoperability:** get your CAD and BIM models into other software without exporting or importing
|
|
||||||
- **Real time:** get real time updates and notifications and changes
|
|
||||||
- **GraphQL API:** get what you need anywhere you want it
|
|
||||||
- **Webhooks:** the base for a automation and next-gen pipelines
|
|
||||||
- **Built for developers:** we are building Speckle with developers in mind and got tools for every stack
|
|
||||||
- **Built for the AEC industry:** Speckle connectors are plugins for the most common software used in the industry such as Revit, Rhino, Grasshopper, AutoCAD, Civil 3D, Excel, Unreal Engine, Unity, QGIS, Blender and more!
|
|
||||||
|
|
||||||
### Try Speckle now!
|
|
||||||
|
|
||||||
Give Speckle a try in no time by:
|
|
||||||
|
|
||||||
- [](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
|
|
||||||
|
|
||||||
### Resources
|
|
||||||
|
|
||||||
- [](https://speckle.community) for help, feature requests or just to hang with other speckle enthusiasts, check out our community forum!
|
|
||||||
- [](https://speckle.systems) our tutorials portal is full of resources to get you started using Speckle
|
|
||||||
- [](https://speckle.guide/dev/) reference on almost any end-user and developer functionality
|
|
||||||
|
|
||||||
|
<p align="center"><a href="https://codecov.io/gh/specklesystems/specklepy"><img src="https://codecov.io/gh/specklesystems/specklepy/branch/main/graph/badge.svg?token=8KQFL5N0YF" alt="Codecov"></a></p>
|
||||||
|
|
||||||
# Repo structure
|
# Repo structure
|
||||||
|
|
||||||
## 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,6 +35,12 @@ 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:
|
||||||
@@ -82,7 +58,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
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,120 @@
|
|||||||
|
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:RELEASE.2023-10-25T06-33-25Z"
|
||||||
|
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-2:latest
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- "0.0.0.0:8080:8080"
|
||||||
|
|
||||||
|
speckle-server:
|
||||||
|
image: speckle/speckle-server:latest
|
||||||
|
restart: always
|
||||||
|
healthcheck:
|
||||||
|
test:
|
||||||
|
- CMD
|
||||||
|
- /nodejs/bin/node
|
||||||
|
- -e
|
||||||
|
- "try { require('node:http').request({headers: {'Content-Type': 'application/json'}, port:3000, hostname:'127.0.0.1', path:'/readiness', method: 'GET', timeout: 2000 }, (res) => { body = ''; res.on('data', (chunk) => {body += chunk;}); res.on('end', () => {process.exit(res.statusCode != 200 || body.toLowerCase().includes('error'));}); }).end(); } catch { process.exit(1); }"
|
||||||
|
interval: 10s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
start_period: 90s
|
||||||
|
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
|
||||||
|
MAX_PROJECT_MODELS_PER_PAGE: 500
|
||||||
|
|
||||||
|
# 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"
|
||||||
|
|
||||||
|
networks:
|
||||||
|
default:
|
||||||
|
name: speckle-server
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
postgres-data:
|
||||||
|
redis-data:
|
||||||
|
minio-data:
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
from typing import List
|
import os
|
||||||
from specklepy.objects import Base
|
import random
|
||||||
from specklepy.api import operations
|
import string
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
import time
|
import time
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
import os
|
from typing import List
|
||||||
import string
|
|
||||||
import random
|
from specklepy.api import operations
|
||||||
|
from specklepy.objects import Base
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
|
||||||
|
|
||||||
class Sub(Base):
|
class Sub(Base):
|
||||||
@@ -26,7 +27,6 @@ def clean_db():
|
|||||||
|
|
||||||
|
|
||||||
def one_pass(clean: bool, randomize: bool, child_count: int):
|
def one_pass(clean: bool, randomize: bool, child_count: int):
|
||||||
|
|
||||||
foo = Base()
|
foo = Base()
|
||||||
for i in range(child_count):
|
for i in range(child_count):
|
||||||
stuff = random_string() if randomize else "stuff"
|
stuff = random_string() if randomize else "stuff"
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
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)
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
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)
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
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,11 @@
|
|||||||
"""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 specklepy.objects import Base
|
|
||||||
from specklepy.api import operations
|
|
||||||
from devtools import debug
|
from devtools import debug
|
||||||
|
|
||||||
|
from specklepy.api import operations
|
||||||
|
from specklepy.objects import Base
|
||||||
|
|
||||||
|
|
||||||
class ExampleSub(Base):
|
class ExampleSub(Base):
|
||||||
"""
|
"""
|
||||||
|
|||||||
+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(f"Invalid pyproject.toml. Could not patch version.")
|
raise Exception("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
+1731
-898
File diff suppressed because it is too large
Load Diff
+34
-12
@@ -1,6 +1,6 @@
|
|||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "specklepy"
|
name = "specklepy"
|
||||||
version = "2.4.0"
|
version = "2.17.14"
|
||||||
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,25 +8,41 @@ 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.7.2, <4.0"
|
python = ">=3.10.0, <4.0"
|
||||||
pydantic = "^1.8.2"
|
pydantic = "^2.5"
|
||||||
appdirs = "^1.4.4"
|
appdirs = "^1.4.4"
|
||||||
gql = {extras = ["requests", "websockets"], version = "^3.3.0"}
|
gql = { extras = ["requests", "websockets"], version = "^3.3.0" }
|
||||||
ujson = "^5.3.0"
|
ujson = "^5.3.0"
|
||||||
Deprecated = "^1.2.13"
|
Deprecated = "^1.2.13"
|
||||||
|
stringcase = "^1.2.0"
|
||||||
|
attrs = "^23.1.0"
|
||||||
|
httpx = "^0.25.0"
|
||||||
|
pydantic-settings = "^2.6.1"
|
||||||
|
|
||||||
[tool.poetry.dev-dependencies]
|
[tool.poetry.group.dev.dependencies]
|
||||||
black = "^20.8b1"
|
black = "24.10.0"
|
||||||
isort = "^5.7.0"
|
isort = "^5.13.2"
|
||||||
pytest = "^6.2.2"
|
pytest = "^7.1.3"
|
||||||
|
pytest-asyncio = "^0.23.0"
|
||||||
pytest-ordering = "^0.6"
|
pytest-ordering = "^0.6"
|
||||||
pytest-cov = "^3.0.0"
|
pytest-cov = "^3.0.0"
|
||||||
devtools = "^0.8.0"
|
devtools = "^0.8.0"
|
||||||
pylint = "^2.14.4"
|
pylint = "^3.3.2"
|
||||||
|
pydantic-settings = "^2.3.0"
|
||||||
|
mypy = "^0.982"
|
||||||
|
pre-commit = "^2.20.0"
|
||||||
|
commitizen = "^3.13.0"
|
||||||
|
ruff = "^0.8.2"
|
||||||
|
types-deprecated = "^1.2.9"
|
||||||
|
types-ujson = "^5.6.0.0"
|
||||||
|
types-requests = "^2.28.11.5"
|
||||||
|
|
||||||
[tool.black]
|
[tool.black]
|
||||||
exclude = '''
|
exclude = '''
|
||||||
@@ -45,9 +61,15 @@ exclude = '''
|
|||||||
'''
|
'''
|
||||||
include = '\.pyi?$'
|
include = '\.pyi?$'
|
||||||
line-length = 88
|
line-length = 88
|
||||||
target-version = ["py37", "py38", "py39", "py310"]
|
target-version = ["py39", "py310", "py311", "py312", "py313"]
|
||||||
|
|
||||||
|
|
||||||
|
[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"
|
||||||
|
|||||||
@@ -1,170 +0,0 @@
|
|||||||
# generated by datamodel-codegen:
|
|
||||||
# filename: stream_schema.json
|
|
||||||
# timestamp: 2020-11-17T14:33:13+00:00
|
|
||||||
|
|
||||||
from datetime import datetime
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
|
|
||||||
from pydantic import BaseModel # pylint: disable=no-name-in-module
|
|
||||||
|
|
||||||
|
|
||||||
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[datetime]
|
|
||||||
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[datetime]
|
|
||||||
items: List[Commit] = []
|
|
||||||
|
|
||||||
|
|
||||||
class Object(BaseModel):
|
|
||||||
id: Optional[str]
|
|
||||||
speckleType: Optional[str]
|
|
||||||
applicationId: Optional[str]
|
|
||||||
totalChildrenCount: Optional[int]
|
|
||||||
createdAt: Optional[datetime]
|
|
||||||
|
|
||||||
|
|
||||||
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]
|
|
||||||
role: Optional[str]
|
|
||||||
isPublic: Optional[bool]
|
|
||||||
description: Optional[str]
|
|
||||||
createdAt: Optional[datetime]
|
|
||||||
updatedAt: Optional[datetime]
|
|
||||||
collaborators: List[Collaborator] = []
|
|
||||||
branches: Optional[Branches]
|
|
||||||
commit: Optional[Commit]
|
|
||||||
object: Optional[Object]
|
|
||||||
commentCount: Optional[int]
|
|
||||||
favoritedDate: Optional[datetime]
|
|
||||||
favoritesCount: Optional[int]
|
|
||||||
|
|
||||||
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 PendingStreamCollaborator(BaseModel):
|
|
||||||
id: Optional[str]
|
|
||||||
inviteId: Optional[str]
|
|
||||||
streamId: Optional[str]
|
|
||||||
streamName: Optional[str]
|
|
||||||
title: Optional[str]
|
|
||||||
role: Optional[str]
|
|
||||||
invitedBy: Optional[User]
|
|
||||||
user: Optional[User]
|
|
||||||
token: Optional[str]
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return f"PendingStreamCollaborator( inviteId: {self.inviteId}, streamId: {self.streamId}, role: {self.role}, title: {self.title}, invitedBy: {self.user.name if self.user else None})"
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class Activity(BaseModel):
|
|
||||||
actionType: Optional[str]
|
|
||||||
info: Optional[dict]
|
|
||||||
userId: Optional[str]
|
|
||||||
streamId: Optional[str]
|
|
||||||
resourceId: Optional[str]
|
|
||||||
resourceType: Optional[str]
|
|
||||||
message: Optional[str]
|
|
||||||
time: Optional[datetime]
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return f"Activity( streamId: {self.streamId}, actionType: {self.actionType}, message: {self.message}, userId: {self.userId} )"
|
|
||||||
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return self.__repr__()
|
|
||||||
|
|
||||||
|
|
||||||
class ActivityCollection(BaseModel):
|
|
||||||
totalCount: Optional[int]
|
|
||||||
items: Optional[List[Activity]]
|
|
||||||
cursor: Optional[datetime]
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return f"ActivityCollection( totalCount: {self.totalCount}, items: {len(self.items) if self.items else 0}, cursor: {self.cursor.isoformat() if self.cursor else None} )"
|
|
||||||
|
|
||||||
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,12 +0,0 @@
|
|||||||
from pathlib import Path
|
|
||||||
import sys
|
|
||||||
import pkgutil
|
|
||||||
from importlib import import_module
|
|
||||||
|
|
||||||
|
|
||||||
for (_, name, _) in pkgutil.iter_modules(__path__):
|
|
||||||
|
|
||||||
imported_module = import_module("." + name, package=__name__)
|
|
||||||
|
|
||||||
if hasattr(imported_module, "Resource"):
|
|
||||||
setattr(sys.modules[__name__], name, imported_module)
|
|
||||||
@@ -1,640 +0,0 @@
|
|||||||
|
|
||||||
|
|
||||||
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 search 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
|
|
||||||
}
|
|
||||||
@@ -1,166 +0,0 @@
|
|||||||
from warnings import warn
|
|
||||||
from urllib.parse import urlparse, unquote
|
|
||||||
from specklepy.api.credentials import (
|
|
||||||
Account,
|
|
||||||
get_account_from_token,
|
|
||||||
get_local_accounts,
|
|
||||||
)
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.api.client import SpeckleClient
|
|
||||||
from specklepy.transports.server.server import ServerTransport
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
|
|
||||||
|
|
||||||
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: {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)
|
|
||||||
metrics.track(metrics.STREAM_WRAPPER, self.get_account())
|
|
||||||
|
|
||||||
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 in a.serverInfo.url),
|
|
||||||
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,5 +0,0 @@
|
|||||||
"""Builtin Speckle object kit."""
|
|
||||||
|
|
||||||
from specklepy.objects.base import Base
|
|
||||||
|
|
||||||
__all__ = ["Base"]
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
"""Builtin Speckle object kit."""
|
|
||||||
|
|
||||||
from specklepy.objects.structural.analysis import *
|
|
||||||
from specklepy.objects.structural.properties import *
|
|
||||||
from specklepy.objects.structural.material import *
|
|
||||||
from specklepy.objects.structural.geometry import *
|
|
||||||
from specklepy.objects.structural.loading import *
|
|
||||||
from specklepy.objects.structural.axis import Axis
|
|
||||||
|
|
||||||
__all__ = [
|
|
||||||
"Element1D",
|
|
||||||
"Element2D",
|
|
||||||
"Element3D",
|
|
||||||
"Axis",
|
|
||||||
"Node",
|
|
||||||
"Restraint",
|
|
||||||
"Load",
|
|
||||||
"LoadBeam",
|
|
||||||
"LoadCase",
|
|
||||||
"LoadCombinations",
|
|
||||||
"LoadFace",
|
|
||||||
"LoadGravity",
|
|
||||||
"LoadNode",
|
|
||||||
"Model",
|
|
||||||
"ModelInfo",
|
|
||||||
"ModelSettings",
|
|
||||||
"ModelUnits",
|
|
||||||
"Concrete",
|
|
||||||
"Material",
|
|
||||||
"Steel",
|
|
||||||
"Timber",
|
|
||||||
"Property",
|
|
||||||
"Property1D",
|
|
||||||
"Property2D",
|
|
||||||
"Property3D",
|
|
||||||
"PropertyDamper",
|
|
||||||
"PropertyMass",
|
|
||||||
"PropertySpring",
|
|
||||||
"SectionProfile",
|
|
||||||
]
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
from typing import List
|
|
||||||
|
|
||||||
from ..base import Base
|
|
||||||
from ..geometry import *
|
|
||||||
from .properties import *
|
|
||||||
|
|
||||||
STRUCTURAL_ANALYSIS = "Objects.Structural.Analysis."
|
|
||||||
|
|
||||||
|
|
||||||
class ModelUnits(Base, speckle_type=STRUCTURAL_ANALYSIS + "ModelUnits"):
|
|
||||||
length: str = None
|
|
||||||
sections: str = None
|
|
||||||
displacements: str = None
|
|
||||||
stress: str = None
|
|
||||||
force: str = None
|
|
||||||
mass: str = None
|
|
||||||
time: str = None
|
|
||||||
temperature: str = None
|
|
||||||
velocity: str = None
|
|
||||||
acceleration: str = None
|
|
||||||
energy: str = None
|
|
||||||
angle: str = None
|
|
||||||
strain: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class ModelSettings(Base, speckle_type=STRUCTURAL_ANALYSIS + "ModelSettings"):
|
|
||||||
modelUnits: ModelUnits = None
|
|
||||||
steelCode: str = None
|
|
||||||
concreteCode: str = None
|
|
||||||
coincidenceTolerance: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ModelInfo(Base, speckle_type=STRUCTURAL_ANALYSIS + "ModelInfo"):
|
|
||||||
name: str = None
|
|
||||||
description: str = None
|
|
||||||
projectNumber: str = None
|
|
||||||
projectName: str = None
|
|
||||||
settings: ModelSettings = None
|
|
||||||
initials: str = None
|
|
||||||
application: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class Model(Base, speckle_type=STRUCTURAL_ANALYSIS + "Model"):
|
|
||||||
specs: ModelInfo = None
|
|
||||||
nodes: List = None
|
|
||||||
elements: List = None
|
|
||||||
loads: List = None
|
|
||||||
restraints: List = None
|
|
||||||
properties: List = None
|
|
||||||
materials: List = None
|
|
||||||
layerDescription: str = None
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
from ..base import Base
|
|
||||||
from ..geometry import Plane
|
|
||||||
|
|
||||||
|
|
||||||
class Axis(Base, speckle_type="Objects.Structural.Geometry.Axis"):
|
|
||||||
name: str = None
|
|
||||||
axisType: str = None
|
|
||||||
plane: Plane = None
|
|
||||||
@@ -1,108 +0,0 @@
|
|||||||
from enum import Enum
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from ..base import Base
|
|
||||||
from ..geometry import *
|
|
||||||
from .properties import *
|
|
||||||
from .axis import Axis
|
|
||||||
|
|
||||||
STRUCTURAL_GEOMETRY = "Objects.Structural.Geometry"
|
|
||||||
|
|
||||||
|
|
||||||
class ElementType1D(int, Enum):
|
|
||||||
Beam = 0
|
|
||||||
Brace = 1
|
|
||||||
Bar = 2
|
|
||||||
Column = 3
|
|
||||||
Rod = 4
|
|
||||||
Spring = 5
|
|
||||||
Tie = 6
|
|
||||||
Strut = 7
|
|
||||||
Link = 8
|
|
||||||
Damper = 9
|
|
||||||
Cable = 10
|
|
||||||
Spacer = 11
|
|
||||||
Other = 12
|
|
||||||
Null = 13
|
|
||||||
|
|
||||||
|
|
||||||
class ElementType2D(int, Enum):
|
|
||||||
Quad4 = 0
|
|
||||||
Quad8 = 1
|
|
||||||
Triangle3 = 2
|
|
||||||
Triangle6 = 3
|
|
||||||
|
|
||||||
|
|
||||||
class ElementType3D(int, Enum):
|
|
||||||
Brick8 = 0
|
|
||||||
Wedge6 = 1
|
|
||||||
Pyramid5 = 2
|
|
||||||
Tetra4 = 3
|
|
||||||
|
|
||||||
|
|
||||||
class Restraint(Base, speckle_type=STRUCTURAL_GEOMETRY + ".Restraint"):
|
|
||||||
code: str = None
|
|
||||||
stiffnessX: float = 0.0
|
|
||||||
stiffnessY: float = 0.0
|
|
||||||
stiffnessZ: float = 0.0
|
|
||||||
stiffnessXX: float = 0.0
|
|
||||||
stiffnessYY: float = 0.0
|
|
||||||
stiffnessZZ: float = 0.0
|
|
||||||
units: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class Node(Base, speckle_type=STRUCTURAL_GEOMETRY + ".Node"):
|
|
||||||
name: str = None
|
|
||||||
basePoint: Point = None
|
|
||||||
constraintAxis: Axis = None
|
|
||||||
restraint: Restraint = None
|
|
||||||
springProperty: PropertySpring = None
|
|
||||||
massProperty: PropertyMass = None
|
|
||||||
damperProperty: PropertyDamper = None
|
|
||||||
units: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class Element1D(Base, speckle_type=STRUCTURAL_GEOMETRY + ".Element1D"):
|
|
||||||
name: str = None
|
|
||||||
baseLine: Line = None
|
|
||||||
property: Property1D = None
|
|
||||||
type: ElementType1D = None
|
|
||||||
end1Releases: Restraint = None
|
|
||||||
end2Releases: Restraint = None
|
|
||||||
end1Offset: Vector = None
|
|
||||||
end2Offset: Vector = None
|
|
||||||
orientationNode: Node = None
|
|
||||||
orinetationAngle: float = 0.0
|
|
||||||
localAxis: Plane = None
|
|
||||||
parent: Base = None
|
|
||||||
end1Node: Node = Node
|
|
||||||
end2Node: Node = Node
|
|
||||||
topology: List = None
|
|
||||||
displayMesh: Mesh = None
|
|
||||||
units: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class Element2D(Base, speckle_type=STRUCTURAL_GEOMETRY + ".Element2D"):
|
|
||||||
name: str = None
|
|
||||||
property: Property2D = None
|
|
||||||
type: ElementType2D = None
|
|
||||||
offset: float = 0.0
|
|
||||||
orientationAngle: float = 0.0
|
|
||||||
parent: Base = None
|
|
||||||
topology: List = None
|
|
||||||
displayMesh: Mesh = None
|
|
||||||
units: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class Element3D(Base, speckle_type=STRUCTURAL_GEOMETRY + ".Element3D"):
|
|
||||||
name: str = None
|
|
||||||
baseMesh: Mesh = None
|
|
||||||
property: Property3D = None
|
|
||||||
type: ElementType3D = None
|
|
||||||
orientationAngle: float = 0.0
|
|
||||||
parent: Base = None
|
|
||||||
topology: List
|
|
||||||
units: str = None
|
|
||||||
|
|
||||||
|
|
||||||
# class Storey needs ependency on built elements first
|
|
||||||
@@ -1,174 +0,0 @@
|
|||||||
from typing import List
|
|
||||||
|
|
||||||
from ..base import Base
|
|
||||||
from ..geometry import *
|
|
||||||
from .loading import *
|
|
||||||
from .geometry import *
|
|
||||||
from .analysis import Model
|
|
||||||
|
|
||||||
STRUCTURAL_RESULTS = "Objects.Structural.Results."
|
|
||||||
|
|
||||||
|
|
||||||
class Result(Base, speckle_type=STRUCTURAL_RESULTS + "Result"):
|
|
||||||
resultCase: Base = None
|
|
||||||
permutation: str = None
|
|
||||||
description: str = None
|
|
||||||
|
|
||||||
|
|
||||||
class ResultSet1D(Result, speckle_type=STRUCTURAL_RESULTS + "ResultSet1D"):
|
|
||||||
results1D: List
|
|
||||||
|
|
||||||
|
|
||||||
class Result1D(Result, speckle_type=STRUCTURAL_RESULTS + "Result1D"):
|
|
||||||
element: Element1D = None
|
|
||||||
position: float = 0.0
|
|
||||||
dispX: float = 0.0
|
|
||||||
dispY: float = 0.0
|
|
||||||
dispZ: float = 0.0
|
|
||||||
rotXX: float = 0.0
|
|
||||||
rotYY: float = 0.0
|
|
||||||
rotZZ: float = 0.0
|
|
||||||
forceX: float = 0.0
|
|
||||||
forceY: float = 0.0
|
|
||||||
forceZ: float = 0.0
|
|
||||||
momentXX: float = 0.0
|
|
||||||
momentYY: float = 0.0
|
|
||||||
momentZZ: float = 0.0
|
|
||||||
axialStress: float = 0.0
|
|
||||||
shearStressY: float = 0.0
|
|
||||||
shearStressZ: float = 0.0
|
|
||||||
bendingStressYPos: float = 0.0
|
|
||||||
bendingStressYNeg: float = 0.0
|
|
||||||
bendingStressZPos: float = 0.0
|
|
||||||
bendingStressZNeg: float = 0.0
|
|
||||||
combinedStressMax: float = 0.0
|
|
||||||
combinedStressMin: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ResultSet2D(Result, speckle_type=STRUCTURAL_RESULTS + "ResultSet2D"):
|
|
||||||
results2D: List
|
|
||||||
|
|
||||||
|
|
||||||
class Result2D(Result, speckle_type=STRUCTURAL_RESULTS + "Result2D"):
|
|
||||||
element: Element2D = None
|
|
||||||
position: List
|
|
||||||
dispX: float = 0.0
|
|
||||||
dispY: float = 0.0
|
|
||||||
dispZ: float = 0.0
|
|
||||||
forceXX: float = 0.0
|
|
||||||
forceYY: float = 0.0
|
|
||||||
forceXY: float = 0.0
|
|
||||||
momentXX: float = 0.0
|
|
||||||
momentYY: float = 0.0
|
|
||||||
momentXY: float = 0.0
|
|
||||||
shearX: float = 0.0
|
|
||||||
shearY: float = 0.0
|
|
||||||
stressTopXX: float = 0.0
|
|
||||||
stressTopYY: float = 0.0
|
|
||||||
stressTopZZ: float = 0.0
|
|
||||||
stressTopXY: float = 0.0
|
|
||||||
stressTopYZ: float = 0.0
|
|
||||||
stressTopZX: float = 0.0
|
|
||||||
stressMidXX: float = 0.0
|
|
||||||
stressMidYY: float = 0.0
|
|
||||||
stressMidZZ: float = 0.0
|
|
||||||
stressMidXY: float = 0.0
|
|
||||||
stressMidYZ: float = 0.0
|
|
||||||
stressMidZX: float = 0.0
|
|
||||||
stressBotXX: float = 0.0
|
|
||||||
stressBotYY: float = 0.0
|
|
||||||
stressBotZZ: float = 0.0
|
|
||||||
stressBotXY: float = 0.0
|
|
||||||
stressBotYZ: float = 0.0
|
|
||||||
stressBotZX: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ResultSet3D(Result, speckle_type=STRUCTURAL_RESULTS + "ResultSet3D"):
|
|
||||||
results3D: List
|
|
||||||
|
|
||||||
|
|
||||||
class Result3D(Result, speckle_type=STRUCTURAL_RESULTS + "Result3D"):
|
|
||||||
element: Element3D = None
|
|
||||||
position: List
|
|
||||||
dispX: float = 0.0
|
|
||||||
dispY: float = 0.0
|
|
||||||
dispZ: float = 0.0
|
|
||||||
stressXX: float = 0.0
|
|
||||||
stressYY: float = 0.0
|
|
||||||
stressZZ: float = 0.0
|
|
||||||
stressXY: float = 0.0
|
|
||||||
stressYZ: float = 0.0
|
|
||||||
stressZX: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ResultGlobal(Result, speckle_type=STRUCTURAL_RESULTS + "ResultGlobal"):
|
|
||||||
model: Model = None
|
|
||||||
loadX: float = 0.0
|
|
||||||
loadY: float = 0.0
|
|
||||||
loadZ: float = 0.0
|
|
||||||
loadXX: float = 0.0
|
|
||||||
loadYY: float = 0.0
|
|
||||||
loadZZ: float = 0.0
|
|
||||||
reactionX: float = 0.0
|
|
||||||
reactionY: float = 0.0
|
|
||||||
reactionZ: float = 0.0
|
|
||||||
reactionXX: float = 0.0
|
|
||||||
reactionYY: float = 0.0
|
|
||||||
reactionZZ: float = 0.0
|
|
||||||
mode: float = 0.0
|
|
||||||
frequency: float = 0.0
|
|
||||||
loadFactor: float = 0.0
|
|
||||||
modalStiffness: float = 0.0
|
|
||||||
modalGeoStiffness: float = 0.0
|
|
||||||
effMassX: float = 0.0
|
|
||||||
effMassY: float = 0.0
|
|
||||||
effMassZ: float = 0.0
|
|
||||||
effMassXX: float = 0.0
|
|
||||||
effMassYY: float = 0.0
|
|
||||||
effMassZZ: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ResultSetNode(Result, speckle_type=STRUCTURAL_RESULTS + "ResultSetNode"):
|
|
||||||
resultsNode: List
|
|
||||||
|
|
||||||
|
|
||||||
class ResultNode(Result, speckle_type=STRUCTURAL_RESULTS + " ResultNode"):
|
|
||||||
node: Node = None
|
|
||||||
dispX: float = 0.0
|
|
||||||
dispY: float = 0.0
|
|
||||||
dispZ: float = 0.0
|
|
||||||
rotXX: float = 0.0
|
|
||||||
rotYY: float = 0.0
|
|
||||||
rotZZ: float = 0.0
|
|
||||||
reactionX: float = 0.0
|
|
||||||
reactionY: float = 0.0
|
|
||||||
reactionZ: float = 0.0
|
|
||||||
reactionXX: float = 0.0
|
|
||||||
reactionYY: float = 0.0
|
|
||||||
reactionZZ: float = 0.0
|
|
||||||
constraintX: float = 0.0
|
|
||||||
constraintY: float = 0.0
|
|
||||||
constraintZ: float = 0.0
|
|
||||||
constraintXX: float = 0.0
|
|
||||||
constraintYY: float = 0.0
|
|
||||||
constraintZZ: float = 0.0
|
|
||||||
velX: float = 0.0
|
|
||||||
velY: float = 0.0
|
|
||||||
velZ: float = 0.0
|
|
||||||
velXX: float = 0.0
|
|
||||||
velYY: float = 0.0
|
|
||||||
velZZ: float = 0.0
|
|
||||||
accX: float = 0.0
|
|
||||||
accY: float = 0.0
|
|
||||||
accZ: float = 0.0
|
|
||||||
accXX: float = 0.0
|
|
||||||
accYY: float = 0.0
|
|
||||||
accZZ: float = 0.0
|
|
||||||
|
|
||||||
|
|
||||||
class ResultSetAll(Base, speckle_type=None):
|
|
||||||
resultSet1D: ResultSet1D = None
|
|
||||||
resultSet2D: ResultSet2D = None
|
|
||||||
resultSet3D: ResultSet3D = None
|
|
||||||
resultsGlobal: ResultGlobal = None
|
|
||||||
resultsNode: ResultSetNode = None
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
from warnings import warn
|
|
||||||
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
|
||||||
|
|
||||||
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,
|
|
||||||
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):
|
|
||||||
if not isinstance(unit, str):
|
|
||||||
warn(
|
|
||||||
f"Invalid units: expected type str but received {type(unit)} ({unit}). Skipping - no units will be set.",
|
|
||||||
SpeckleWarning,
|
|
||||||
)
|
|
||||||
return
|
|
||||||
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 as e:
|
|
||||||
raise SpeckleException(message=f"No encoding exists for unit {unit}. Please enter a valid unit to encode (eg {UNITS_ENCODINGS}).") from e
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
from .server import ServerTransport
|
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
"""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,
|
||||||
|
ObjectResultLevel,
|
||||||
|
ResultCase,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"AutomationContext",
|
||||||
|
"AutomateBase",
|
||||||
|
"AutomationStatus",
|
||||||
|
"AutomationResult",
|
||||||
|
"AutomationRunData",
|
||||||
|
"ResultCase",
|
||||||
|
"ObjectResultLevel",
|
||||||
|
"run_function",
|
||||||
|
"execute_automate_function",
|
||||||
|
]
|
||||||
@@ -0,0 +1,427 @@
|
|||||||
|
"""This module provides an abstraction layer above the Speckle Automate runtime."""
|
||||||
|
|
||||||
|
import time
|
||||||
|
from dataclasses import dataclass, field
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import Any, Dict, List, Optional, Tuple, Union
|
||||||
|
|
||||||
|
import httpx
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from speckle_automate.schema import (
|
||||||
|
AutomateBase,
|
||||||
|
AutomationResult,
|
||||||
|
AutomationRunData,
|
||||||
|
AutomationStatus,
|
||||||
|
ObjectResultLevel,
|
||||||
|
ResultCase,
|
||||||
|
)
|
||||||
|
from specklepy.api import operations
|
||||||
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.core.api.models import Branch
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.transports.memory import MemoryTransport
|
||||||
|
from specklepy.transports.server import ServerTransport
|
||||||
|
|
||||||
|
|
||||||
|
@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
|
||||||
|
|
||||||
|
@property
|
||||||
|
def status_message(self) -> Optional[str]:
|
||||||
|
"""Get the current status message."""
|
||||||
|
return self._automation_result.status_message
|
||||||
|
|
||||||
|
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."""
|
||||||
|
# TODO: this is a quick hack to keep implementation consistency. Move to proper receive many versions
|
||||||
|
version_id = self.automation_run_data.triggers[0].payload.version_id
|
||||||
|
commit = self.speckle_client.commit.get(
|
||||||
|
self.automation_run_data.project_id, 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 {version_id}",
|
||||||
|
)
|
||||||
|
return base
|
||||||
|
|
||||||
|
def create_new_version_in_project(
|
||||||
|
self, root_object: Base, model_name: str, version_message: str = ""
|
||||||
|
) -> Tuple[str, str]:
|
||||||
|
"""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.
|
||||||
|
"""
|
||||||
|
|
||||||
|
branch = self.speckle_client.branch.get(
|
||||||
|
self.automation_run_data.project_id, model_name, 1
|
||||||
|
)
|
||||||
|
if isinstance(branch, Branch):
|
||||||
|
if not branch.id:
|
||||||
|
raise ValueError("Cannot use the branch without its id")
|
||||||
|
matching_trigger = [
|
||||||
|
t
|
||||||
|
for t in self.automation_run_data.triggers
|
||||||
|
if t.payload.model_id == branch.id
|
||||||
|
]
|
||||||
|
if matching_trigger:
|
||||||
|
raise ValueError(
|
||||||
|
f"The target model: {model_name} cannot match the model"
|
||||||
|
f" that triggered this automation:"
|
||||||
|
f" {matching_trigger[0].payload.model_id}"
|
||||||
|
)
|
||||||
|
model_id = branch.id
|
||||||
|
|
||||||
|
else:
|
||||||
|
# we just check if it exists
|
||||||
|
branch_create = self.speckle_client.branch.create(
|
||||||
|
self.automation_run_data.project_id,
|
||||||
|
model_name,
|
||||||
|
)
|
||||||
|
if isinstance(branch_create, Exception):
|
||||||
|
raise branch_create
|
||||||
|
model_id = branch_create
|
||||||
|
|
||||||
|
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=model_name,
|
||||||
|
message=version_message,
|
||||||
|
source_application="SpeckleAutomate",
|
||||||
|
)
|
||||||
|
|
||||||
|
if isinstance(version_id, SpeckleException):
|
||||||
|
raise version_id
|
||||||
|
|
||||||
|
self._automation_result.result_versions.append(version_id)
|
||||||
|
return model_id, version_id
|
||||||
|
|
||||||
|
@property
|
||||||
|
def context_view(self) -> Optional[str]:
|
||||||
|
return self._automation_result.result_view
|
||||||
|
|
||||||
|
def set_context_view(
|
||||||
|
self,
|
||||||
|
# f"{model_id}@{version_id} or {model_id} "
|
||||||
|
resource_ids: Optional[List[str]] = None,
|
||||||
|
include_source_model_version: bool = True,
|
||||||
|
) -> None:
|
||||||
|
link_resources = (
|
||||||
|
[
|
||||||
|
f"{t.payload.model_id}@{t.payload.version_id}"
|
||||||
|
for t in self.automation_run_data.triggers
|
||||||
|
]
|
||||||
|
if include_source_model_version
|
||||||
|
else []
|
||||||
|
)
|
||||||
|
if resource_ids:
|
||||||
|
link_resources.extend(resource_ids)
|
||||||
|
if not link_resources:
|
||||||
|
raise Exception(
|
||||||
|
"We do not have enough resource ids to compose a context view"
|
||||||
|
)
|
||||||
|
self._automation_result.result_view = (
|
||||||
|
f"/projects/{self.automation_run_data.project_id}"
|
||||||
|
f"/models/{','.join(link_resources)}"
|
||||||
|
)
|
||||||
|
|
||||||
|
def report_run_status(self) -> None:
|
||||||
|
"""Report the current run status to the project of this automation."""
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation AutomateFunctionRunStatusReport(
|
||||||
|
$projectId: String!
|
||||||
|
$functionRunId: String!
|
||||||
|
$status: AutomateRunStatus!
|
||||||
|
$statusMessage: String
|
||||||
|
$results: JSONObject
|
||||||
|
$contextView: String
|
||||||
|
){
|
||||||
|
automateFunctionRunStatusReport(input: {
|
||||||
|
projectId: $projectId
|
||||||
|
functionRunId: $functionRunId
|
||||||
|
status: $status
|
||||||
|
statusMessage: $statusMessage
|
||||||
|
contextView: $contextView
|
||||||
|
results: $results
|
||||||
|
})
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
if self.run_status in [AutomationStatus.SUCCEEDED, AutomationStatus.FAILED]:
|
||||||
|
object_results = {
|
||||||
|
"version": 1,
|
||||||
|
"values": {
|
||||||
|
"objectResults": self._automation_result.model_dump(by_alias=True)[
|
||||||
|
"objectResults"
|
||||||
|
],
|
||||||
|
"blobIds": self._automation_result.blobs,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
object_results = None
|
||||||
|
|
||||||
|
params = {
|
||||||
|
"projectId": self.automation_run_data.project_id,
|
||||||
|
"functionRunId": self.automation_run_data.function_run_id,
|
||||||
|
"status": self.run_status.value,
|
||||||
|
"statusMessage": self._automation_result.status_message,
|
||||||
|
"results": object_results,
|
||||||
|
"contextView": self._automation_result.result_view,
|
||||||
|
}
|
||||||
|
print(f"Reporting run status with content: {params}")
|
||||||
|
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.")
|
||||||
|
|
||||||
|
self._automation_result.blobs.extend(
|
||||||
|
[upload_result.blob_id for upload_result in upload_response.upload_results]
|
||||||
|
)
|
||||||
|
|
||||||
|
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_exception(self, status_message: str) -> None:
|
||||||
|
"""Mark the current run a failure."""
|
||||||
|
self._mark_run(AutomationStatus.EXCEPTION, 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 attach_error_to_objects(
|
||||||
|
self,
|
||||||
|
category: str,
|
||||||
|
object_ids: Union[str, List[str]],
|
||||||
|
message: Optional[str] = None,
|
||||||
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> None:
|
||||||
|
"""Add a new error case to the run results.
|
||||||
|
|
||||||
|
If the error cause has already created an error case,
|
||||||
|
the error will be extended with a new case refering to the causing objects.
|
||||||
|
Args:
|
||||||
|
error_tag (str): A short tag for the error type.
|
||||||
|
causing_object_ids (str[]): A list of object_id-s that are causing the error
|
||||||
|
error_messagge (Optional[str]): Optional error message.
|
||||||
|
metadata: User provided metadata key value pairs
|
||||||
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
|
"""
|
||||||
|
self.attach_result_to_objects(
|
||||||
|
ObjectResultLevel.ERROR,
|
||||||
|
category,
|
||||||
|
object_ids,
|
||||||
|
message,
|
||||||
|
metadata,
|
||||||
|
visual_overrides,
|
||||||
|
)
|
||||||
|
|
||||||
|
def attach_warning_to_objects(
|
||||||
|
self,
|
||||||
|
category: str,
|
||||||
|
object_ids: Union[str, List[str]],
|
||||||
|
message: Optional[str] = None,
|
||||||
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> None:
|
||||||
|
"""Add a new warning case to the run results."""
|
||||||
|
self.attach_result_to_objects(
|
||||||
|
ObjectResultLevel.WARNING,
|
||||||
|
category,
|
||||||
|
object_ids,
|
||||||
|
message,
|
||||||
|
metadata,
|
||||||
|
visual_overrides,
|
||||||
|
)
|
||||||
|
|
||||||
|
def attach_success_to_objects(
|
||||||
|
self,
|
||||||
|
category: str,
|
||||||
|
object_ids: Union[str, List[str]],
|
||||||
|
message: Optional[str] = None,
|
||||||
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> None:
|
||||||
|
"""Add a new success case to the run results."""
|
||||||
|
self.attach_result_to_objects(
|
||||||
|
ObjectResultLevel.SUCCESS,
|
||||||
|
category,
|
||||||
|
object_ids,
|
||||||
|
message,
|
||||||
|
metadata,
|
||||||
|
visual_overrides,
|
||||||
|
)
|
||||||
|
|
||||||
|
def attach_info_to_objects(
|
||||||
|
self,
|
||||||
|
category: str,
|
||||||
|
object_ids: Union[str, List[str]],
|
||||||
|
message: Optional[str] = None,
|
||||||
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> None:
|
||||||
|
"""Add a new info case to the run results."""
|
||||||
|
self.attach_result_to_objects(
|
||||||
|
ObjectResultLevel.INFO,
|
||||||
|
category,
|
||||||
|
object_ids,
|
||||||
|
message,
|
||||||
|
metadata,
|
||||||
|
visual_overrides,
|
||||||
|
)
|
||||||
|
|
||||||
|
def attach_result_to_objects(
|
||||||
|
self,
|
||||||
|
level: ObjectResultLevel,
|
||||||
|
category: str,
|
||||||
|
object_ids: Union[str, List[str]],
|
||||||
|
message: Optional[str] = None,
|
||||||
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> None:
|
||||||
|
if isinstance(object_ids, list):
|
||||||
|
if len(object_ids) < 1:
|
||||||
|
raise ValueError(
|
||||||
|
f"Need atleast one object_id to report a(n) {level.value.upper()}"
|
||||||
|
)
|
||||||
|
id_list = object_ids
|
||||||
|
else:
|
||||||
|
id_list = [object_ids]
|
||||||
|
print(
|
||||||
|
f"Created new {level.value.upper()}"
|
||||||
|
f" category: {category} caused by: {message}"
|
||||||
|
)
|
||||||
|
self._automation_result.object_results.append(
|
||||||
|
ResultCase(
|
||||||
|
category=category,
|
||||||
|
level=level,
|
||||||
|
object_ids=id_list,
|
||||||
|
message=message,
|
||||||
|
metadata=metadata,
|
||||||
|
visual_overrides=visual_overrides,
|
||||||
|
)
|
||||||
|
)
|
||||||
@@ -0,0 +1,155 @@
|
|||||||
|
"""Some useful helpers for working with automation data."""
|
||||||
|
|
||||||
|
import secrets
|
||||||
|
import string
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
from gql import gql
|
||||||
|
from pydantic import Field
|
||||||
|
from pydantic_settings import BaseSettings, SettingsConfigDict
|
||||||
|
|
||||||
|
from speckle_automate.schema import AutomationRunData, TestAutomationRunData
|
||||||
|
from specklepy.api.client import SpeckleClient
|
||||||
|
|
||||||
|
|
||||||
|
class TestAutomationEnvironment(BaseSettings):
|
||||||
|
"""Get known environment variables from local `.env` file"""
|
||||||
|
|
||||||
|
model_config = SettingsConfigDict(
|
||||||
|
env_file=".env",
|
||||||
|
env_file_encoding="utf-8",
|
||||||
|
env_prefix="speckle_",
|
||||||
|
extra="ignore",
|
||||||
|
)
|
||||||
|
|
||||||
|
token: str = Field()
|
||||||
|
server_url: str = Field()
|
||||||
|
project_id: str = Field()
|
||||||
|
automation_id: str = Field()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture()
|
||||||
|
def test_automation_environment() -> TestAutomationEnvironment:
|
||||||
|
return TestAutomationEnvironment()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture()
|
||||||
|
def test_automation_token(
|
||||||
|
test_automation_environment: TestAutomationEnvironment,
|
||||||
|
) -> str:
|
||||||
|
"""Provide a speckle token for the test suite."""
|
||||||
|
|
||||||
|
return test_automation_environment.token
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture()
|
||||||
|
def speckle_client(
|
||||||
|
test_automation_environment: TestAutomationEnvironment,
|
||||||
|
) -> SpeckleClient:
|
||||||
|
"""Initialize a SpeckleClient for testing."""
|
||||||
|
speckle_client = SpeckleClient(
|
||||||
|
test_automation_environment.server_url,
|
||||||
|
test_automation_environment.server_url.startswith("https"),
|
||||||
|
)
|
||||||
|
speckle_client.authenticate_with_token(test_automation_environment.token)
|
||||||
|
return speckle_client
|
||||||
|
|
||||||
|
|
||||||
|
def create_test_automation_run(
|
||||||
|
speckle_client: SpeckleClient, project_id: str, test_automation_id: str
|
||||||
|
) -> TestAutomationRunData:
|
||||||
|
"""Create test run to report local test results to"""
|
||||||
|
|
||||||
|
query = gql(
|
||||||
|
"""
|
||||||
|
mutation CreateTestRun(
|
||||||
|
$projectId: ID!,
|
||||||
|
$automationId: ID!
|
||||||
|
) {
|
||||||
|
projectMutations {
|
||||||
|
automationMutations(projectId: $projectId) {
|
||||||
|
createTestAutomationRun(automationId: $automationId) {
|
||||||
|
automationRunId
|
||||||
|
functionRunId
|
||||||
|
triggers {
|
||||||
|
payload {
|
||||||
|
modelId
|
||||||
|
versionId
|
||||||
|
}
|
||||||
|
triggerType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"automationId": test_automation_id, "projectId": project_id}
|
||||||
|
|
||||||
|
result = speckle_client.httpclient.execute(query, params)
|
||||||
|
|
||||||
|
print(result)
|
||||||
|
|
||||||
|
return (
|
||||||
|
result.get("projectMutations")
|
||||||
|
.get("automationMutations")
|
||||||
|
.get("createTestAutomationRun")
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture()
|
||||||
|
def test_automation_run(
|
||||||
|
speckle_client: SpeckleClient,
|
||||||
|
test_automation_environment: TestAutomationEnvironment,
|
||||||
|
) -> TestAutomationRunData:
|
||||||
|
return create_test_automation_run(
|
||||||
|
speckle_client,
|
||||||
|
test_automation_environment.project_id,
|
||||||
|
test_automation_environment.automation_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def create_test_automation_run_data(
|
||||||
|
speckle_client: SpeckleClient,
|
||||||
|
test_automation_environment: TestAutomationEnvironment,
|
||||||
|
) -> AutomationRunData:
|
||||||
|
"""Create automation run data for a new run for a given test automation"""
|
||||||
|
|
||||||
|
test_automation_run_data = create_test_automation_run(
|
||||||
|
speckle_client,
|
||||||
|
test_automation_environment.project_id,
|
||||||
|
test_automation_environment.automation_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
return AutomationRunData(
|
||||||
|
project_id=test_automation_environment.project_id,
|
||||||
|
speckle_server_url=test_automation_environment.server_url,
|
||||||
|
automation_id=test_automation_environment.automation_id,
|
||||||
|
automation_run_id=test_automation_run_data["automationRunId"],
|
||||||
|
function_run_id=test_automation_run_data["functionRunId"],
|
||||||
|
triggers=test_automation_run_data["triggers"],
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture()
|
||||||
|
def test_automation_run_data(
|
||||||
|
speckle_client: SpeckleClient,
|
||||||
|
test_automation_environment: TestAutomationEnvironment,
|
||||||
|
) -> AutomationRunData:
|
||||||
|
return create_test_automation_run_data(speckle_client, test_automation_environment)
|
||||||
|
|
||||||
|
|
||||||
|
def crypto_random_string(length: int) -> str:
|
||||||
|
"""Generate a semi crypto random string of a given length."""
|
||||||
|
alphabet = string.ascii_letters + string.digits
|
||||||
|
return "".join(secrets.choice(alphabet) for _ in range(length)).lower()
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"test_automation_environment",
|
||||||
|
"test_automation_token",
|
||||||
|
"speckle_client",
|
||||||
|
"test_automation_run",
|
||||||
|
"test_automation_run_data",
|
||||||
|
]
|
||||||
@@ -0,0 +1,193 @@
|
|||||||
|
"""Function execution module.
|
||||||
|
|
||||||
|
Provides mechanisms to execute any function,
|
||||||
|
that conforms to the AutomateFunction "interface"
|
||||||
|
"""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
import traceback
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import Callable, Optional, Tuple, TypeVar, Union, overload
|
||||||
|
|
||||||
|
from pydantic import create_model
|
||||||
|
from pydantic.json_schema import GenerateJsonSchema
|
||||||
|
|
||||||
|
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]
|
||||||
|
|
||||||
|
|
||||||
|
def _read_input_data(inputs_location: str) -> str:
|
||||||
|
input_path = Path(inputs_location)
|
||||||
|
if not input_path.exists():
|
||||||
|
raise ValueError(f"Cannot find the function inputs file at {input_path}")
|
||||||
|
|
||||||
|
return input_path.read_text()
|
||||||
|
|
||||||
|
|
||||||
|
def _parse_input_data(
|
||||||
|
input_location: str, input_schema: Optional[type[T]]
|
||||||
|
) -> Tuple[AutomationRunData, Optional[T], str]:
|
||||||
|
input_json_string = _read_input_data(input_location)
|
||||||
|
|
||||||
|
class FunctionRunData(AutomateBase):
|
||||||
|
speckle_token: str
|
||||||
|
automation_run_data: AutomationRunData
|
||||||
|
function_inputs: None = None
|
||||||
|
|
||||||
|
parser_model = FunctionRunData
|
||||||
|
|
||||||
|
if input_schema:
|
||||||
|
parser_model = create_model(
|
||||||
|
"FunctionRunDataWithInputs",
|
||||||
|
function_inputs=(input_schema, ...),
|
||||||
|
__base__=FunctionRunData,
|
||||||
|
)
|
||||||
|
|
||||||
|
input_data = parser_model.model_validate_json(input_json_string)
|
||||||
|
return (
|
||||||
|
input_data.automation_run_data,
|
||||||
|
input_data.function_inputs,
|
||||||
|
input_data.speckle_token,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def execute_automate_function(
|
||||||
|
automate_function: AutomateFunction[T],
|
||||||
|
input_schema: type[T],
|
||||||
|
) -> None: ...
|
||||||
|
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def execute_automate_function(
|
||||||
|
automate_function: AutomateFunctionWithoutInputs,
|
||||||
|
) -> None: ...
|
||||||
|
|
||||||
|
|
||||||
|
class AutomateGenerateJsonSchema(GenerateJsonSchema):
|
||||||
|
def generate(self, schema, mode="validation"):
|
||||||
|
json_schema = super().generate(schema, mode=mode)
|
||||||
|
json_schema["$schema"] = self.schema_dialect
|
||||||
|
return json_schema
|
||||||
|
|
||||||
|
|
||||||
|
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("Incorrect number of arguments specified need 2")
|
||||||
|
|
||||||
|
# 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, argument = args
|
||||||
|
|
||||||
|
if command == "generate_schema":
|
||||||
|
path = Path(argument)
|
||||||
|
schema = json.dumps(
|
||||||
|
input_schema.model_json_schema(
|
||||||
|
by_alias=True, schema_generator=AutomateGenerateJsonSchema
|
||||||
|
)
|
||||||
|
if input_schema
|
||||||
|
else {}
|
||||||
|
)
|
||||||
|
path.write_text(schema)
|
||||||
|
|
||||||
|
elif command == "run":
|
||||||
|
automation_run_data, function_inputs, speckle_token = _parse_input_data(
|
||||||
|
argument, input_schema
|
||||||
|
)
|
||||||
|
|
||||||
|
automation_context = AutomationContext.initialize(
|
||||||
|
automation_run_data, speckle_token
|
||||||
|
)
|
||||||
|
|
||||||
|
if function_inputs:
|
||||||
|
automation_context = run_function(
|
||||||
|
automation_context,
|
||||||
|
automate_function, # type: ignore
|
||||||
|
function_inputs, # type: ignore
|
||||||
|
)
|
||||||
|
|
||||||
|
else:
|
||||||
|
automation_context = AutomationContext.initialize(
|
||||||
|
automation_run_data, speckle_token
|
||||||
|
)
|
||||||
|
automation_context = run_function(
|
||||||
|
automation_context,
|
||||||
|
automate_function, # type: ignore
|
||||||
|
)
|
||||||
|
|
||||||
|
# if we've gotten this far, the execution should technically be completed as expected
|
||||||
|
# thus exiting with 0 is the schemantically correct thing to do
|
||||||
|
exit_code = (
|
||||||
|
1 if automation_context.run_status == AutomationStatus.EXCEPTION else 0
|
||||||
|
)
|
||||||
|
exit(exit_code)
|
||||||
|
|
||||||
|
else:
|
||||||
|
raise NotImplementedError(f"Command: '{command}' is not supported.")
|
||||||
|
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def run_function(
|
||||||
|
automation_context: AutomationContext,
|
||||||
|
automate_function: AutomateFunction[T],
|
||||||
|
inputs: T,
|
||||||
|
) -> AutomationContext: ...
|
||||||
|
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def run_function(
|
||||||
|
automation_context: AutomationContext,
|
||||||
|
automate_function: AutomateFunctionWithoutInputs,
|
||||||
|
) -> AutomationContext: ...
|
||||||
|
|
||||||
|
|
||||||
|
def run_function(
|
||||||
|
automation_context: AutomationContext,
|
||||||
|
automate_function: Union[AutomateFunction[T], AutomateFunctionWithoutInputs],
|
||||||
|
inputs: Optional[T] = None,
|
||||||
|
) -> AutomationContext:
|
||||||
|
"""Run the provided function with the automate sdk context."""
|
||||||
|
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,
|
||||||
|
AutomationStatus.EXCEPTION,
|
||||||
|
]:
|
||||||
|
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_exception(
|
||||||
|
"Function error. Check the automation run logs for details."
|
||||||
|
)
|
||||||
|
finally:
|
||||||
|
if not automation_context.context_view:
|
||||||
|
automation_context.set_context_view()
|
||||||
|
automation_context.report_run_status()
|
||||||
|
return automation_context
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
""""""
|
||||||
|
|
||||||
|
from enum import Enum
|
||||||
|
from typing import Any, Dict, List, Literal, Optional
|
||||||
|
|
||||||
|
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 VersionCreationTriggerPayload(AutomateBase):
|
||||||
|
"""Represents the version creation trigger payload."""
|
||||||
|
|
||||||
|
model_id: str
|
||||||
|
version_id: str
|
||||||
|
|
||||||
|
|
||||||
|
class VersionCreationTrigger(AutomateBase):
|
||||||
|
"""Represents a single version creation trigger for the automation run."""
|
||||||
|
|
||||||
|
trigger_type: Literal["versionCreation"]
|
||||||
|
payload: VersionCreationTriggerPayload
|
||||||
|
|
||||||
|
|
||||||
|
class AutomationRunData(BaseModel):
|
||||||
|
"""Values of the project / model that triggered the run of this function."""
|
||||||
|
|
||||||
|
project_id: str
|
||||||
|
speckle_server_url: str
|
||||||
|
automation_id: str
|
||||||
|
automation_run_id: str
|
||||||
|
function_run_id: str
|
||||||
|
|
||||||
|
triggers: List[VersionCreationTrigger]
|
||||||
|
|
||||||
|
model_config = ConfigDict(
|
||||||
|
alias_generator=camelcase, populate_by_name=True, protected_namespaces=()
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class TestAutomationRunData(BaseModel):
|
||||||
|
"""Values of the run created in the test automation for local test results."""
|
||||||
|
|
||||||
|
automation_run_id: str
|
||||||
|
function_run_id: str
|
||||||
|
|
||||||
|
triggers: List[VersionCreationTrigger]
|
||||||
|
|
||||||
|
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"
|
||||||
|
EXCEPTION = "EXCEPTION"
|
||||||
|
|
||||||
|
|
||||||
|
class ObjectResultLevel(str, Enum):
|
||||||
|
"""Possible status message levels for object reports."""
|
||||||
|
|
||||||
|
SUCCESS = "SUCCESS"
|
||||||
|
INFO = "INFO"
|
||||||
|
WARNING = "WARNING"
|
||||||
|
ERROR = "ERROR"
|
||||||
|
|
||||||
|
|
||||||
|
class ResultCase(AutomateBase):
|
||||||
|
"""A result case."""
|
||||||
|
|
||||||
|
category: str
|
||||||
|
level: ObjectResultLevel
|
||||||
|
object_ids: List[str]
|
||||||
|
message: Optional[str]
|
||||||
|
metadata: Optional[Dict[str, Any]]
|
||||||
|
visual_overrides: Optional[Dict[str, Any]]
|
||||||
|
|
||||||
|
|
||||||
|
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: list[ResultCase] = Field(default_factory=list)
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
# from specklepy import objects
|
||||||
|
|
||||||
|
# __all__ = ["objects"]
|
||||||
@@ -0,0 +1,198 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.credentials import Account
|
||||||
|
from specklepy.api.resources import (
|
||||||
|
ActiveUserResource,
|
||||||
|
ModelResource,
|
||||||
|
OtherUserResource,
|
||||||
|
ProjectInviteResource,
|
||||||
|
ProjectResource,
|
||||||
|
ServerResource,
|
||||||
|
SubscriptionResource,
|
||||||
|
VersionResource,
|
||||||
|
branch,
|
||||||
|
commit,
|
||||||
|
object,
|
||||||
|
stream,
|
||||||
|
subscriptions,
|
||||||
|
user,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.client import SpeckleClient as CoreSpeckleClient
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
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 `app.speckle.systems`.
|
||||||
|
|
||||||
|
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="app.speckle.systems") # 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 = "app.speckle.systems"
|
||||||
|
USE_SSL = True
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
host: str = DEFAULT_HOST,
|
||||||
|
use_ssl: bool = USE_SSL,
|
||||||
|
verify_certificate: bool = True,
|
||||||
|
) -> None:
|
||||||
|
super().__init__(
|
||||||
|
host=host,
|
||||||
|
use_ssl=use_ssl,
|
||||||
|
verify_certificate=verify_certificate,
|
||||||
|
)
|
||||||
|
self.account = Account()
|
||||||
|
|
||||||
|
def _init_resources(self) -> None:
|
||||||
|
self.server = ServerResource(
|
||||||
|
account=self.account, basepath=self.url, client=self.httpclient
|
||||||
|
)
|
||||||
|
|
||||||
|
server_version = None
|
||||||
|
try:
|
||||||
|
server_version = self.server.version()
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.other_user = OtherUserResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.active_user = ActiveUserResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.project = ProjectResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.project_invite = ProjectInviteResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.model = ModelResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.version = VersionResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.subscription = SubscriptionResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.ws_url,
|
||||||
|
client=self.wsclient,
|
||||||
|
# todo: why doesn't this take a server version
|
||||||
|
)
|
||||||
|
# Deprecated Resources
|
||||||
|
self.user = 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)
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
# 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 StreamWrapper # noqa: F401
|
||||||
|
from specklepy.core.api.credentials import Account, UserInfo # noqa: F401
|
||||||
|
from specklepy.core.api.credentials import (
|
||||||
|
get_account_from_token as core_get_account_from_token,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.credentials import get_local_accounts as core_get_local_accounts
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
from specklepy.core.api.host_applications import (
|
||||||
|
ARCGIS,
|
||||||
|
ARCHICAD,
|
||||||
|
AUTOCAD,
|
||||||
|
BLENDER,
|
||||||
|
CIVIL,
|
||||||
|
CSIBRIDGE,
|
||||||
|
DXF,
|
||||||
|
DYNAMO,
|
||||||
|
ETABS,
|
||||||
|
EXCEL,
|
||||||
|
GRASSHOPPER,
|
||||||
|
GSA,
|
||||||
|
MICROSTATION,
|
||||||
|
NET,
|
||||||
|
OPENBUILDINGS,
|
||||||
|
OPENRAIL,
|
||||||
|
OPENROADS,
|
||||||
|
OTHER,
|
||||||
|
POWERBI,
|
||||||
|
PYTHON,
|
||||||
|
QGIS,
|
||||||
|
REVIT,
|
||||||
|
RHINO,
|
||||||
|
SAFE,
|
||||||
|
SAP2000,
|
||||||
|
SKETCHUP,
|
||||||
|
TEKLASTRUCTURES,
|
||||||
|
TOPSOLID,
|
||||||
|
UNITY,
|
||||||
|
UNREAL,
|
||||||
|
HostApplication,
|
||||||
|
HostAppVersion,
|
||||||
|
_app_name_host_app_mapping,
|
||||||
|
get_host_app_from_string,
|
||||||
|
)
|
||||||
|
|
||||||
|
# re-exporting stuff from the moved api module
|
||||||
|
__all__ = [
|
||||||
|
"ARCGIS",
|
||||||
|
"ARCHICAD",
|
||||||
|
"AUTOCAD",
|
||||||
|
"BLENDER",
|
||||||
|
"CIVIL",
|
||||||
|
"CSIBRIDGE",
|
||||||
|
"DXF",
|
||||||
|
"DYNAMO",
|
||||||
|
"ETABS",
|
||||||
|
"EXCEL",
|
||||||
|
"GRASSHOPPER",
|
||||||
|
"GSA",
|
||||||
|
"MICROSTATION",
|
||||||
|
"NET",
|
||||||
|
"OPENBUILDINGS",
|
||||||
|
"OPENRAIL",
|
||||||
|
"OPENROADS",
|
||||||
|
"OTHER",
|
||||||
|
"POWERBI",
|
||||||
|
"PYTHON",
|
||||||
|
"QGIS",
|
||||||
|
"REVIT",
|
||||||
|
"RHINO",
|
||||||
|
"SAFE",
|
||||||
|
"SAP2000",
|
||||||
|
"SKETCHUP",
|
||||||
|
"TEKLASTRUCTURES",
|
||||||
|
"TOPSOLID",
|
||||||
|
"UNITY",
|
||||||
|
"UNREAL",
|
||||||
|
"HostApplication",
|
||||||
|
"HostAppVersion",
|
||||||
|
"_app_name_host_app_mapping",
|
||||||
|
"get_host_app_from_string",
|
||||||
|
]
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
# 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 (
|
||||||
|
Activity,
|
||||||
|
ActivityCollection,
|
||||||
|
Branch,
|
||||||
|
Branches,
|
||||||
|
Collaborator,
|
||||||
|
Commit,
|
||||||
|
Commits,
|
||||||
|
LimitedUser,
|
||||||
|
Object,
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
ServerInfo,
|
||||||
|
Stream,
|
||||||
|
Streams,
|
||||||
|
User,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"Activity",
|
||||||
|
"ActivityCollection",
|
||||||
|
"Branch",
|
||||||
|
"Branches",
|
||||||
|
"Collaborator",
|
||||||
|
"Commit",
|
||||||
|
"Commits",
|
||||||
|
"LimitedUser",
|
||||||
|
"Object",
|
||||||
|
"PendingStreamCollaborator",
|
||||||
|
"ServerInfo",
|
||||||
|
"Stream",
|
||||||
|
"Streams",
|
||||||
|
"User",
|
||||||
|
]
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.operations import deserialize as core_deserialize
|
||||||
|
from specklepy.core.api.operations import receive as _untracked_receive
|
||||||
|
from specklepy.core.api.operations import send as core_send
|
||||||
|
from specklepy.core.api.operations import serialize as core_serialize
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
|
||||||
|
|
||||||
|
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"]
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
from typing import Any, Optional, Tuple
|
||||||
|
|
||||||
|
from gql.client import Client
|
||||||
|
|
||||||
|
from specklepy.api.credentials import Account
|
||||||
|
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,
|
||||||
|
)
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
from specklepy.api.resources.current.active_user_resource import ActiveUserResource
|
||||||
|
from specklepy.api.resources.current.model_resource import ModelResource
|
||||||
|
from specklepy.api.resources.current.other_user_resource import OtherUserResource
|
||||||
|
from specklepy.api.resources.current.project_invite_resource import (
|
||||||
|
ProjectInviteResource,
|
||||||
|
)
|
||||||
|
from specklepy.api.resources.current.project_resource import ProjectResource
|
||||||
|
from specklepy.api.resources.current.server_resource import ServerResource
|
||||||
|
from specklepy.api.resources.current.subscription_resource import SubscriptionResource
|
||||||
|
from specklepy.api.resources.current.version_resource import VersionResource
|
||||||
|
from specklepy.api.resources.deprecated import (
|
||||||
|
active_user,
|
||||||
|
branch,
|
||||||
|
commit,
|
||||||
|
object,
|
||||||
|
other_user,
|
||||||
|
server,
|
||||||
|
stream,
|
||||||
|
subscriptions,
|
||||||
|
user,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"ActiveUserResource",
|
||||||
|
"ModelResource",
|
||||||
|
"OtherUserResource",
|
||||||
|
"ProjectInviteResource",
|
||||||
|
"ProjectResource",
|
||||||
|
"ServerResource",
|
||||||
|
"SubscriptionResource",
|
||||||
|
"VersionResource",
|
||||||
|
"active_user",
|
||||||
|
"branch",
|
||||||
|
"commit",
|
||||||
|
"object",
|
||||||
|
"other_user",
|
||||||
|
"server",
|
||||||
|
"stream",
|
||||||
|
"subscriptions",
|
||||||
|
"user",
|
||||||
|
]
|
||||||
@@ -0,0 +1,144 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import List, Optional, overload
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.user_inputs import UserProjectsFilter, UserUpdateInput
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
Project,
|
||||||
|
ResourceCollection,
|
||||||
|
User,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources import ActiveUserResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class ActiveUserResource(CoreResource):
|
||||||
|
"""API Access class for users. This class provides methods to get and update
|
||||||
|
the user profile, fetch user activity, and manage pending stream invitations."""
|
||||||
|
|
||||||
|
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) -> Optional[User]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Active User Get"})
|
||||||
|
return super().get()
|
||||||
|
|
||||||
|
@deprecated("Use UserUpdateInput overload", version=FE1_DEPRECATION_VERSION)
|
||||||
|
@overload
|
||||||
|
def update(
|
||||||
|
self,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
company: Optional[str] = None,
|
||||||
|
bio: Optional[str] = None,
|
||||||
|
avatar: Optional[str] = None,
|
||||||
|
) -> User: ...
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def update(self, *, input: UserUpdateInput) -> User: ...
|
||||||
|
|
||||||
|
def update(
|
||||||
|
self,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
company: Optional[str] = None,
|
||||||
|
bio: Optional[str] = None,
|
||||||
|
avatar: Optional[str] = None,
|
||||||
|
*,
|
||||||
|
input: Optional[UserUpdateInput] = None,
|
||||||
|
) -> User:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Active User Update"})
|
||||||
|
if isinstance(input, UserUpdateInput):
|
||||||
|
return super()._update(input=input)
|
||||||
|
else:
|
||||||
|
return super()._update(
|
||||||
|
input=UserUpdateInput(
|
||||||
|
name=name,
|
||||||
|
company=company,
|
||||||
|
bio=bio,
|
||||||
|
avatar=avatar,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_projects(
|
||||||
|
self,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[UserProjectsFilter] = None,
|
||||||
|
) -> ResourceCollection[Project]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Active User Get Projects"})
|
||||||
|
return super().get_projects(limit=limit, cursor=cursor, filter=filter)
|
||||||
|
|
||||||
|
def get_project_invites(self) -> List[PendingStreamCollaborator]:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Active User Get Project Invites"}
|
||||||
|
)
|
||||||
|
return super().get_project_invites()
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
def activity(
|
||||||
|
self,
|
||||||
|
limit: int = 20,
|
||||||
|
action_type: Optional[str] = None,
|
||||||
|
before: Optional[datetime] = None,
|
||||||
|
after: Optional[datetime] = None,
|
||||||
|
cursor: Optional[datetime] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Fetches collection the current authenticated user's activity
|
||||||
|
as filtered by given parameters
|
||||||
|
|
||||||
|
Note: all timestamps arguments should be `datetime` of any tz as they will be
|
||||||
|
converted to UTC ISO format strings
|
||||||
|
|
||||||
|
Args:
|
||||||
|
limit (int): The maximum number of activity items to return.
|
||||||
|
action_type (Optional[str]): Filter results to a single action type.
|
||||||
|
before (Optional[datetime]): Latest cutoff for activity to include.
|
||||||
|
after (Optional[datetime]): Oldest cutoff for an activity to include.
|
||||||
|
cursor (Optional[datetime]): Timestamp cursor for pagination.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Activity collection, filtered according to the provided parameters.
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "User Active Activity"})
|
||||||
|
return super().activity(limit, action_type, before, after, cursor)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
||||||
|
"""Fetches all of the current user's pending stream invitations.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[PendingStreamCollaborator]: A list of pending stream invitations.
|
||||||
|
"""
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "User Active Invites All Get"}
|
||||||
|
)
|
||||||
|
return super().get_all_pending_invites()
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
def get_pending_invite(
|
||||||
|
self, stream_id: str, token: Optional[str] = None
|
||||||
|
) -> Optional[PendingStreamCollaborator]:
|
||||||
|
"""Fetches a specific pending invite for the current user on a given stream.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
stream_id (str): The ID of the stream to look for invites on.
|
||||||
|
token (Optional[str]): The token of the invite to look for (optional).
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Optional[PendingStreamCollaborator]: The invite for the given stream, or None if not found.
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "User Active Invite Get"})
|
||||||
|
return super().get_pending_invite(stream_id, token)
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.model_inputs import (
|
||||||
|
CreateModelInput,
|
||||||
|
DeleteModelInput,
|
||||||
|
ModelVersionsFilter,
|
||||||
|
UpdateModelInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.inputs.project_inputs import ProjectModelsFilter
|
||||||
|
from specklepy.core.api.models import Model, ModelWithVersions, ResourceCollection
|
||||||
|
from specklepy.core.api.resources import ModelResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class ModelResource(CoreResource):
|
||||||
|
"""API Access class for models"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, model_id: str, project_id: str) -> Model:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Get"})
|
||||||
|
return super().get(model_id, project_id)
|
||||||
|
|
||||||
|
def get_with_versions(
|
||||||
|
self,
|
||||||
|
model_id: str,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
versions_limit: int = 25,
|
||||||
|
versions_cursor: Optional[str] = None,
|
||||||
|
versions_filter: Optional[ModelVersionsFilter] = None,
|
||||||
|
) -> ModelWithVersions:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Get With Versions"})
|
||||||
|
return super().get_with_versions(
|
||||||
|
model_id,
|
||||||
|
project_id,
|
||||||
|
versions_limit=versions_limit,
|
||||||
|
versions_cursor=versions_cursor,
|
||||||
|
versions_filter=versions_filter,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_models(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
models_limit: int = 25,
|
||||||
|
models_cursor: Optional[str] = None,
|
||||||
|
models_filter: Optional[ProjectModelsFilter] = None,
|
||||||
|
) -> ResourceCollection[Model]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Get Models"})
|
||||||
|
return super().get_models(
|
||||||
|
project_id,
|
||||||
|
models_limit=models_limit,
|
||||||
|
models_cursor=models_cursor,
|
||||||
|
models_filter=models_filter,
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(self, input: CreateModelInput) -> Model:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Create"})
|
||||||
|
return super().create(input)
|
||||||
|
|
||||||
|
def delete(self, input: DeleteModelInput) -> bool:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Delete"})
|
||||||
|
return super().delete(input)
|
||||||
|
|
||||||
|
def update(self, input: UpdateModelInput) -> Model:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Model Update"})
|
||||||
|
return super().update(input)
|
||||||
@@ -0,0 +1,104 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ActivityCollection,
|
||||||
|
LimitedUser,
|
||||||
|
UserSearchResultCollection,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources import OtherUserResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
|
||||||
|
class OtherUserResource(CoreResource):
|
||||||
|
"""
|
||||||
|
Provides API access to other users' profiles and activities on the platform.
|
||||||
|
This class enables fetching limited information about users, searching for users by name or email,
|
||||||
|
and accessing user activity logs with appropriate privacy and access control measures in place.
|
||||||
|
"""
|
||||||
|
|
||||||
|
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) -> Optional[LimitedUser]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Other User Get"})
|
||||||
|
return super().get(id)
|
||||||
|
|
||||||
|
def user_search(
|
||||||
|
self,
|
||||||
|
query: str,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
archived: bool = False,
|
||||||
|
emailOnly: bool = False,
|
||||||
|
) -> UserSearchResultCollection:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Other User Search"})
|
||||||
|
return super().user_search(
|
||||||
|
query, limit=limit, cursor=cursor, archived=archived, emailOnly=emailOnly
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason="Use user_search instead", version=FE1_DEPRECATION_VERSION)
|
||||||
|
def search(
|
||||||
|
self, search_query: str, limit: int = 25
|
||||||
|
) -> Union[List[LimitedUser], SpeckleException]:
|
||||||
|
"""
|
||||||
|
Searches for users by name or email.
|
||||||
|
The search requires a minimum query length of 3 characters.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
search_query (str): The search string.
|
||||||
|
limit (int): Maximum number of search results to return.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Union[List[LimitedUser], SpeckleException]: A list of users matching the search
|
||||||
|
query or an exception if the query is too short.
|
||||||
|
"""
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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:
|
||||||
|
"""
|
||||||
|
Retrieves a collection of activities for a specified user, with optional filters for activity type,
|
||||||
|
time frame, and pagination.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
user_id (str): The ID of the user whose activities are being requested.
|
||||||
|
limit (int): The maximum number of activity items to return.
|
||||||
|
action_type (Optional[str]): A specific type of activity to filter.
|
||||||
|
before (Optional[datetime]): Latest timestamp to include activities before.
|
||||||
|
after (Optional[datetime]): Earliest timestamp to include activities after.
|
||||||
|
cursor (Optional[datetime]): Timestamp for pagination cursor.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
ActivityCollection: A collection of user activities filtered according to specified criteria.
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Other User Activity"})
|
||||||
|
return super().activity(user_id, limit, action_type, before, after, cursor)
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
from typing import Any, Optional, Tuple
|
||||||
|
|
||||||
|
from gql import Client
|
||||||
|
|
||||||
|
from specklepy.core.api.credentials import Account
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectInviteCreateInput,
|
||||||
|
ProjectInviteUseInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import PendingStreamCollaborator, ProjectWithTeam
|
||||||
|
from specklepy.core.api.resources import ProjectInviteResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectInviteResource(CoreResource):
|
||||||
|
"""API Access class for project invites"""
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
account: Account,
|
||||||
|
basepath: str,
|
||||||
|
client: Client,
|
||||||
|
server_version: Optional[Tuple[Any, ...]],
|
||||||
|
) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(
|
||||||
|
self, project_id: str, input: ProjectInviteCreateInput
|
||||||
|
) -> ProjectWithTeam:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Invite Create"})
|
||||||
|
return super().create(project_id, input)
|
||||||
|
|
||||||
|
def use(self, input: ProjectInviteUseInput) -> bool:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Invite Use"})
|
||||||
|
return super().use(input)
|
||||||
|
|
||||||
|
def get(
|
||||||
|
self, project_id: str, token: Optional[str]
|
||||||
|
) -> Optional[PendingStreamCollaborator]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Invite Get"})
|
||||||
|
return super().get(project_id, token)
|
||||||
|
|
||||||
|
def cancel(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
invite_id: str,
|
||||||
|
) -> ProjectWithTeam:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Invite Cancel"})
|
||||||
|
return super().cancel(project_id, invite_id)
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectCreateInput,
|
||||||
|
ProjectModelsFilter,
|
||||||
|
ProjectUpdateInput,
|
||||||
|
ProjectUpdateRoleInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import Project, ProjectWithModels, ProjectWithTeam
|
||||||
|
from specklepy.core.api.resources import ProjectResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectResource(CoreResource):
|
||||||
|
"""API Access class for projects"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, project_id: str) -> Project:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Get "})
|
||||||
|
return super().get(project_id)
|
||||||
|
|
||||||
|
def get_with_models(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
models_limit: int = 25,
|
||||||
|
models_cursor: Optional[str] = None,
|
||||||
|
models_filter: Optional[ProjectModelsFilter] = None,
|
||||||
|
) -> ProjectWithModels:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Get With Models"})
|
||||||
|
return super().get_with_models(
|
||||||
|
project_id,
|
||||||
|
models_limit=models_limit,
|
||||||
|
models_cursor=models_cursor,
|
||||||
|
models_filter=models_filter,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_with_team(self, project_id: str) -> ProjectWithTeam:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Get With Team"})
|
||||||
|
return super().get_with_team(project_id)
|
||||||
|
|
||||||
|
def create(self, input: ProjectCreateInput) -> Project:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Create"})
|
||||||
|
return super().create(input)
|
||||||
|
|
||||||
|
def update(self, input: ProjectUpdateInput) -> Project:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Update"})
|
||||||
|
return super().update(input)
|
||||||
|
|
||||||
|
def delete(self, project_id: str) -> bool:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Delete"})
|
||||||
|
return super().delete(project_id)
|
||||||
|
|
||||||
|
def update_role(self, input: ProjectUpdateRoleInput) -> ProjectWithTeam:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Project Update Role"})
|
||||||
|
return super().update_role(input)
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
from typing import Any, Dict, List, Tuple
|
||||||
|
|
||||||
|
from specklepy.api.models import ServerInfo
|
||||||
|
from specklepy.core.api.resources import ServerResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class ServerResource(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)
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
from typing import Callable, Optional, Sequence
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
from typing_extensions import TypeVar
|
||||||
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ProjectModelsUpdatedMessage,
|
||||||
|
ProjectUpdatedMessage,
|
||||||
|
ProjectVersionsUpdatedMessage,
|
||||||
|
UserProjectsUpdatedMessage,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources import SubscriptionResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
TEventArgs = TypeVar("TEventArgs", bound=BaseModel)
|
||||||
|
|
||||||
|
|
||||||
|
class SubscriptionResource(CoreResource):
|
||||||
|
def __init__(self, account, basepath, client) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def user_projects_updated(
|
||||||
|
self, callback: Callable[[UserProjectsUpdatedMessage], None]
|
||||||
|
) -> None:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Subscription Project Models Updated"}
|
||||||
|
)
|
||||||
|
return await super().user_projects_updated(callback)
|
||||||
|
|
||||||
|
async def project_models_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectModelsUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
*,
|
||||||
|
model_ids: Optional[Sequence[str]] = None,
|
||||||
|
) -> None:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Subscription Project Models Updated"}
|
||||||
|
)
|
||||||
|
return await super().project_models_updated(callback, id, model_ids=model_ids)
|
||||||
|
|
||||||
|
async def project_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
) -> None:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Subscription Project Updated"}
|
||||||
|
)
|
||||||
|
return await super().project_updated(callback, id)
|
||||||
|
|
||||||
|
async def project_versions_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectVersionsUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
) -> None:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Subscription Project Versions Updated"}
|
||||||
|
)
|
||||||
|
return await super().project_versions_updated(callback, id)
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.model_inputs import ModelVersionsFilter
|
||||||
|
from specklepy.core.api.inputs.version_inputs import (
|
||||||
|
CreateVersionInput,
|
||||||
|
DeleteVersionsInput,
|
||||||
|
MarkReceivedVersionInput,
|
||||||
|
MoveVersionsInput,
|
||||||
|
UpdateVersionInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import ResourceCollection, Version
|
||||||
|
from specklepy.core.api.resources import VersionResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class VersionResource(CoreResource):
|
||||||
|
"""API Access class for model versions"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, version_id: str, project_id: str) -> Version:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Get"})
|
||||||
|
return super().get(version_id, project_id)
|
||||||
|
|
||||||
|
def get_versions(
|
||||||
|
self,
|
||||||
|
model_id: str,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[ModelVersionsFilter] = None,
|
||||||
|
) -> ResourceCollection[Version]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Get Versions"})
|
||||||
|
return super().get_versions(
|
||||||
|
model_id, project_id, limit=limit, cursor=cursor, filter=filter
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(self, input: CreateVersionInput) -> str:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Create"})
|
||||||
|
return super().create(input)
|
||||||
|
|
||||||
|
def update(self, input: UpdateVersionInput) -> Version:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Update"})
|
||||||
|
return super().update(input)
|
||||||
|
|
||||||
|
def move_to_model(self, input: MoveVersionsInput) -> str:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Move To Model"})
|
||||||
|
return super().move_to_model(input)
|
||||||
|
|
||||||
|
def delete(self, input: DeleteVersionsInput) -> bool:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Delete"})
|
||||||
|
return super().delete(input)
|
||||||
|
|
||||||
|
def received(self, input: MarkReceivedVersionInput) -> bool:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Version Received"})
|
||||||
|
return super().received(input)
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.resources import ActiveUserResource
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason="Renamed to ActiveUserResource", version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Resource(ActiveUserResource):
|
||||||
|
"""Renamed to ActiveUserResource"""
|
||||||
@@ -0,0 +1,108 @@
|
|||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.models import Branch
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.deprecated.branch import Resource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
def get(
|
||||||
|
self, stream_id: str, name: str, commits_limit: int = 10
|
||||||
|
) -> Union[Branch, None, SpeckleException]:
|
||||||
|
"""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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
@@ -0,0 +1,134 @@
|
|||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.models import Commit
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.deprecated.commit import Resource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
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
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
from typing import Dict, List
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.deprecated.object import Resource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.resources import OtherUserResource
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason="Renamed to OtherUserResource", version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Resource(OtherUserResource):
|
||||||
|
"""
|
||||||
|
Renamed to OtherUserResource
|
||||||
|
"""
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.resources import ServerResource
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason="Renamed to ServerResource", version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Resource(ServerResource):
|
||||||
|
"""Renamed to ServerResource"""
|
||||||
@@ -0,0 +1,322 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.models import PendingStreamCollaborator, Stream
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.deprecated.stream import Resource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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)
|
||||||
@@ -0,0 +1,107 @@
|
|||||||
|
from typing import Callable, Dict, List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
from graphql import DocumentNode
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.current.subscription_resource import check_wsclient
|
||||||
|
from specklepy.core.api.resources.deprecated.subscriptions import (
|
||||||
|
Resource as CoreResource,
|
||||||
|
)
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class Resource(CoreResource):
|
||||||
|
"""API Access class for subscriptions"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
@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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
@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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
@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)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
@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
|
||||||
@@ -0,0 +1,153 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.api.models import PendingStreamCollaborator, User
|
||||||
|
from specklepy.core.api.resources.deprecated.user import Resource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
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)
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.api.credentials import Account
|
||||||
|
from specklepy.core.api.wrapper import StreamWrapper as CoreStreamWrapper
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.transports.server.server import ServerTransport
|
||||||
|
|
||||||
|
|
||||||
|
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://app.speckle.systems/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)
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
"""
|
||||||
|
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,36 +1,44 @@
|
|||||||
import re
|
import re
|
||||||
from warnings import warn
|
|
||||||
from deprecated import deprecated
|
|
||||||
from specklepy.api.credentials import Account, get_account_from_token
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.logging.exceptions import (
|
|
||||||
SpeckleException,
|
|
||||||
SpeckleWarning,
|
|
||||||
)
|
|
||||||
from typing import Dict
|
from typing import Dict
|
||||||
|
from warnings import warn
|
||||||
|
|
||||||
from specklepy.api import resources
|
from deprecated import deprecated
|
||||||
from specklepy.api.resources import (
|
|
||||||
branch,
|
|
||||||
commit,
|
|
||||||
stream,
|
|
||||||
object,
|
|
||||||
server,
|
|
||||||
user,
|
|
||||||
subscriptions,
|
|
||||||
)
|
|
||||||
from specklepy.api.models import ServerInfo
|
|
||||||
from gql import Client
|
from gql import Client
|
||||||
|
from gql.transport.exceptions import TransportServerError
|
||||||
from gql.transport.requests import RequestsHTTPTransport
|
from gql.transport.requests import RequestsHTTPTransport
|
||||||
from gql.transport.websockets import WebsocketsTransport
|
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 (
|
||||||
|
ActiveUserResource,
|
||||||
|
ModelResource,
|
||||||
|
OtherUserResource,
|
||||||
|
ProjectInviteResource,
|
||||||
|
ProjectResource,
|
||||||
|
ServerResource,
|
||||||
|
SubscriptionResource,
|
||||||
|
VersionResource,
|
||||||
|
branch,
|
||||||
|
commit,
|
||||||
|
object,
|
||||||
|
stream,
|
||||||
|
subscriptions,
|
||||||
|
user,
|
||||||
|
)
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.logging.exceptions import SpeckleException, SpeckleWarning
|
||||||
|
|
||||||
|
|
||||||
class SpeckleClient:
|
class SpeckleClient:
|
||||||
"""
|
"""
|
||||||
The `SpeckleClient` is your entry point for interacting with your Speckle Server's GraphQL API.
|
The `SpeckleClient` is your entry point for interacting with
|
||||||
You'll need to have access to a server to use it, or you can use our public server `speckle.xyz`.
|
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 `app.speckle.systems`.
|
||||||
|
|
||||||
To authenticate the client, you'll need to have downloaded the [Speckle Manager](https://speckle.guide/#speckle-manager)
|
To authenticate the client, you'll need to have downloaded
|
||||||
|
the [Speckle Manager](https://speckle.guide/#speckle-manager)
|
||||||
and added your account.
|
and added your account.
|
||||||
|
|
||||||
```py
|
```py
|
||||||
@@ -38,7 +46,7 @@ class SpeckleClient:
|
|||||||
from specklepy.api.credentials import get_default_account
|
from specklepy.api.credentials import get_default_account
|
||||||
|
|
||||||
# initialise the client
|
# initialise the client
|
||||||
client = SpeckleClient(host="speckle.xyz") # or whatever your host is
|
client = SpeckleClient(host="app.speckle.systems") # or whatever your host is
|
||||||
# client = SpeckleClient(host="localhost:3000", use_ssl=False) or use local server
|
# 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)
|
# authenticate the client with an account (account has been added in Speckle Manager)
|
||||||
@@ -53,11 +61,17 @@ class SpeckleClient:
|
|||||||
```
|
```
|
||||||
"""
|
"""
|
||||||
|
|
||||||
DEFAULT_HOST = "speckle.xyz"
|
DEFAULT_HOST = "app.speckle.systems"
|
||||||
USE_SSL = True
|
USE_SSL = True
|
||||||
|
|
||||||
def __init__(self, host: str = DEFAULT_HOST, use_ssl: bool = USE_SSL) -> None:
|
def __init__(
|
||||||
metrics.track(metrics.CLIENT, custom_props={"name": "create"})
|
self,
|
||||||
|
host: str = DEFAULT_HOST,
|
||||||
|
use_ssl: bool = USE_SSL,
|
||||||
|
verify_certificate: bool = True,
|
||||||
|
connection_retries: int = 3,
|
||||||
|
connection_timeout: int = 10,
|
||||||
|
) -> None:
|
||||||
ws_protocol = "ws"
|
ws_protocol = "ws"
|
||||||
http_protocol = "http"
|
http_protocol = "http"
|
||||||
|
|
||||||
@@ -72,9 +86,17 @@ class SpeckleClient:
|
|||||||
self.graphql = f"{self.url}/graphql"
|
self.graphql = f"{self.url}/graphql"
|
||||||
self.ws_url = f"{ws_protocol}://{host}/graphql"
|
self.ws_url = f"{ws_protocol}://{host}/graphql"
|
||||||
self.account = Account()
|
self.account = Account()
|
||||||
|
self.verify_certificate = verify_certificate
|
||||||
|
self.connection_retries = connection_retries
|
||||||
|
self.connection_timeout = connection_timeout
|
||||||
|
|
||||||
self.httpclient = Client(
|
self.httpclient = Client(
|
||||||
transport=RequestsHTTPTransport(url=self.graphql, verify=True, retries=3)
|
transport=RequestsHTTPTransport(
|
||||||
|
url=self.graphql,
|
||||||
|
verify=self.verify_certificate,
|
||||||
|
retries=self.connection_retries,
|
||||||
|
timeout=self.connection_timeout,
|
||||||
|
)
|
||||||
)
|
)
|
||||||
self.wsclient = None
|
self.wsclient = None
|
||||||
|
|
||||||
@@ -93,52 +115,61 @@ class SpeckleClient:
|
|||||||
# ) from ex
|
# ) from ex
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return f"SpeckleClient( server: {self.url}, authenticated: {self.account.token is not None} )"
|
return (
|
||||||
|
f"SpeckleClient( server: {self.url}, authenticated:"
|
||||||
|
f" {self.account.token is not None} )"
|
||||||
|
)
|
||||||
|
|
||||||
@deprecated(
|
@deprecated(
|
||||||
version="2.6.0",
|
version="2.6.0",
|
||||||
reason="Renamed: please use `authenticate_with_account` or `authenticate_with_token` instead.",
|
reason=(
|
||||||
|
"Renamed: please use `authenticate_with_account` or"
|
||||||
|
" `authenticate_with_token` instead."
|
||||||
|
),
|
||||||
)
|
)
|
||||||
def authenticate(self, token: str) -> None:
|
def authenticate(self, token: str) -> None:
|
||||||
"""Authenticate the client using a personal access token
|
"""Authenticate the client using a personal access token
|
||||||
The token is saved in the client object and a synchronous GraphQL entrypoint is created
|
The token is saved in the client object and a synchronous GraphQL
|
||||||
|
entrypoint is created
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
token {str} -- an api token
|
token {str} -- an api token
|
||||||
"""
|
"""
|
||||||
self.authenticate_with_token(token)
|
self.authenticate_with_account(get_account_from_token(token))
|
||||||
self._set_up_client()
|
|
||||||
|
|
||||||
def authenticate_with_token(self, token: str) -> None:
|
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
|
Authenticate the client using a personal access token.
|
||||||
|
The token is saved in the client object and a synchronous GraphQL
|
||||||
|
entrypoint is created
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
token {str} -- an api token
|
token {str} -- an api token
|
||||||
"""
|
"""
|
||||||
self.account = get_account_from_token(token, self.url)
|
self.account = Account.from_token(token, self.url)
|
||||||
metrics.track(metrics.CLIENT, self.account, {"name": "authenticate with token"})
|
|
||||||
self._set_up_client()
|
self._set_up_client()
|
||||||
|
|
||||||
def authenticate_with_account(self, account: Account) -> None:
|
def authenticate_with_account(self, account: Account) -> None:
|
||||||
"""Authenticate the client using an Account object
|
"""Authenticate the client using an Account object
|
||||||
The account is saved in the client object and a synchronous GraphQL entrypoint is created
|
The account is saved in the client object and a synchronous GraphQL
|
||||||
|
entrypoint is created
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
account {Account} -- the account object which can be found with `get_default_account` or `get_local_accounts`
|
account {Account} -- the account object which can be found with
|
||||||
|
`get_default_account` or `get_local_accounts`
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.CLIENT, account, {"name": "authenticate with account"})
|
|
||||||
self.account = account
|
self.account = account
|
||||||
self._set_up_client()
|
self._set_up_client()
|
||||||
|
|
||||||
def _set_up_client(self) -> None:
|
def _set_up_client(self) -> None:
|
||||||
metrics.track(metrics.CLIENT, self.account, {"name": "set up client"})
|
|
||||||
headers = {
|
headers = {
|
||||||
"Authorization": f"Bearer {self.account.token}",
|
"Authorization": f"Bearer {self.account.token}",
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
|
"apollographql-client-name": metrics.HOST_APP,
|
||||||
|
"apollographql-client-version": metrics.HOST_APP_VERSION,
|
||||||
}
|
}
|
||||||
httptransport = RequestsHTTPTransport(
|
httptransport = RequestsHTTPTransport(
|
||||||
url=self.graphql, headers=headers, verify=True, retries=3
|
url=self.graphql, headers=headers, verify=self.verify_certificate, retries=3
|
||||||
)
|
)
|
||||||
wstransport = WebsocketsTransport(
|
wstransport = WebsocketsTransport(
|
||||||
url=self.ws_url,
|
url=self.ws_url,
|
||||||
@@ -149,25 +180,76 @@ class SpeckleClient:
|
|||||||
|
|
||||||
self._init_resources()
|
self._init_resources()
|
||||||
|
|
||||||
if self.user.get() is None:
|
try:
|
||||||
warn(
|
_ = self.active_user.get()
|
||||||
SpeckleWarning(
|
except SpeckleException as ex:
|
||||||
f"Possibly invalid token - could not authenticate Speckle Client for server {self.url}"
|
if isinstance(ex.exception, TransportServerError):
|
||||||
)
|
if ex.exception.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:
|
def execute_query(self, query: str) -> Dict:
|
||||||
return self.httpclient.execute(query)
|
return self.httpclient.execute(query)
|
||||||
|
|
||||||
def _init_resources(self) -> None:
|
def _init_resources(self) -> None:
|
||||||
self.server = server.Resource(
|
self.server = ServerResource(
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
account=self.account, basepath=self.url, client=self.httpclient
|
||||||
)
|
)
|
||||||
|
|
||||||
server_version = None
|
server_version = None
|
||||||
try:
|
try:
|
||||||
server_version = self.server.version()
|
server_version = self.server.version()
|
||||||
except:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
self.other_user = OtherUserResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.active_user = ActiveUserResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.project = ProjectResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.project_invite = ProjectInviteResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.model = ModelResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.version = VersionResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.subscription = SubscriptionResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.ws_url,
|
||||||
|
client=self.wsclient,
|
||||||
|
)
|
||||||
|
# Deprecated Resources
|
||||||
self.user = user.Resource(
|
self.user = user.Resource(
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.url,
|
basepath=self.url,
|
||||||
@@ -201,7 +283,7 @@ class SpeckleClient:
|
|||||||
return attr.Resource(
|
return attr.Resource(
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
account=self.account, basepath=self.url, client=self.httpclient
|
||||||
)
|
)
|
||||||
except:
|
except AttributeError:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
f"Method {name} is not supported by the SpeckleClient class"
|
f"Method {name} is not supported by the SpeckleClient class"
|
||||||
)
|
)
|
||||||
@@ -1,17 +1,22 @@
|
|||||||
import os
|
import os
|
||||||
from pydantic import BaseModel, Field # pylint: disable=no-name-in-module
|
from pathlib import Path
|
||||||
from typing import List, Optional
|
from typing import List, Optional
|
||||||
from specklepy.logging import metrics
|
from urllib.parse import urlparse
|
||||||
from specklepy.api.models import ServerInfo
|
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
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.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
|
||||||
|
|
||||||
class UserInfo(BaseModel):
|
class UserInfo(BaseModel):
|
||||||
name: Optional[str]
|
id: Optional[str] = None
|
||||||
email: Optional[str]
|
name: Optional[str] = None
|
||||||
company: Optional[str]
|
email: Optional[str] = None
|
||||||
id: Optional[str]
|
company: Optional[str] = None
|
||||||
|
avatar: Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
class Account(BaseModel):
|
class Account(BaseModel):
|
||||||
@@ -23,7 +28,10 @@ class Account(BaseModel):
|
|||||||
id: Optional[str] = None
|
id: Optional[str] = None
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
def __repr__(self) -> str:
|
||||||
return f"Account(email: {self.userInfo.email}, server: {self.serverInfo.url}, isDefault: {self.isDefault})"
|
return (
|
||||||
|
f"Account(email: {self.userInfo.email}, server: {self.serverInfo.url},"
|
||||||
|
f" isDefault: {self.isDefault})"
|
||||||
|
)
|
||||||
|
|
||||||
def __str__(self) -> str:
|
def __str__(self) -> str:
|
||||||
return self.__repr__()
|
return self.__repr__()
|
||||||
@@ -35,31 +43,45 @@ class Account(BaseModel):
|
|||||||
return acct
|
return acct
|
||||||
|
|
||||||
|
|
||||||
def get_local_accounts(base_path: str = None) -> List[Account]:
|
def get_local_accounts(base_path: Optional[str] = None) -> List[Account]:
|
||||||
"""Gets all the accounts present in this environment
|
"""Gets all the accounts present in this environment
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
base_path {str} -- custom base path if you are not using the system default
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
List[Account] -- list of all local accounts or an empty list if no accounts were found
|
List[Account] -- list of all local accounts or an empty list if
|
||||||
|
no accounts were found
|
||||||
"""
|
"""
|
||||||
account_storage = SQLiteTransport(scope="Accounts", base_path=base_path)
|
|
||||||
# pylint: disable=protected-access
|
|
||||||
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: List[Account] = []
|
accounts: List[Account] = []
|
||||||
res = account_storage.get_all_objects()
|
try:
|
||||||
account_storage.close()
|
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 res:
|
|
||||||
accounts.extend(Account.parse_raw(r[1]) for r in res)
|
|
||||||
if json_acct_files:
|
if json_acct_files:
|
||||||
try:
|
try:
|
||||||
accounts.extend(
|
accounts.extend(
|
||||||
Account.parse_file(os.path.join(json_path, json_file))
|
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
|
for json_file in json_acct_files
|
||||||
)
|
)
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
@@ -68,19 +90,13 @@ def get_local_accounts(base_path: str = None) -> List[Account]:
|
|||||||
ex,
|
ex,
|
||||||
) from ex
|
) from ex
|
||||||
|
|
||||||
metrics.track(
|
|
||||||
metrics.ACCOUNTS,
|
|
||||||
next(
|
|
||||||
(acc for acc in accounts if acc.isDefault),
|
|
||||||
accounts[0] if accounts else None,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
|
|
||||||
return accounts
|
return accounts
|
||||||
|
|
||||||
|
|
||||||
def get_default_account(base_path: str = None) -> Account:
|
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.
|
"""
|
||||||
|
Gets this environment's default account if any. If there is no default,
|
||||||
|
the first found will be returned and set as default.
|
||||||
Arguments:
|
Arguments:
|
||||||
base_path {str} -- custom base path if you are not using the system default
|
base_path {str} -- custom base path if you are not using the system default
|
||||||
|
|
||||||
@@ -95,7 +111,7 @@ def get_default_account(base_path: str = None) -> Account:
|
|||||||
if not default:
|
if not default:
|
||||||
default = accounts[0]
|
default = accounts[0]
|
||||||
default.isDefault = True
|
default.isDefault = True
|
||||||
metrics.initialise_tracker(default)
|
# metrics.initialise_tracker(default)
|
||||||
|
|
||||||
return default
|
return default
|
||||||
|
|
||||||
@@ -106,7 +122,8 @@ def get_account_from_token(token: str, server_url: str = None) -> Account:
|
|||||||
token {str} -- the api token
|
token {str} -- the api token
|
||||||
|
|
||||||
Returns:
|
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 -- 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()
|
accounts = get_local_accounts()
|
||||||
if not accounts:
|
if not accounts:
|
||||||
@@ -127,9 +144,34 @@ def get_account_from_token(token: str, server_url: str = None) -> Account:
|
|||||||
return Account.from_token(token, server_url)
|
return Account.from_token(token, server_url)
|
||||||
|
|
||||||
|
|
||||||
|
def get_accounts_for_server(host: str) -> List[Account]:
|
||||||
|
all_accounts = get_local_accounts()
|
||||||
|
filtered: List[Account] = []
|
||||||
|
|
||||||
|
for acc in all_accounts:
|
||||||
|
moved_from = (
|
||||||
|
acc.serverInfo.migration.movedFrom if acc.serverInfo.migration else None
|
||||||
|
)
|
||||||
|
|
||||||
|
if moved_from and host == urlparse(moved_from).netloc:
|
||||||
|
filtered.append(acc)
|
||||||
|
|
||||||
|
for acc in all_accounts:
|
||||||
|
if any([x for x in filtered if x.userInfo.id == acc.userInfo.id]):
|
||||||
|
continue
|
||||||
|
|
||||||
|
if host == urlparse(acc.serverInfo.url).netloc:
|
||||||
|
filtered.append(acc)
|
||||||
|
|
||||||
|
return filtered
|
||||||
|
|
||||||
|
|
||||||
class StreamWrapper:
|
class StreamWrapper:
|
||||||
def __init__(self, url: str = None) -> None:
|
def __init__(self, url: str = None) -> None:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
message="The StreamWrapper has moved as of v2.6.0! Please import from specklepy.api.wrapper",
|
message=(
|
||||||
exception=DeprecationWarning,
|
"The StreamWrapper has moved as of v2.6.0! Please import from"
|
||||||
|
" specklepy.api.wrapper"
|
||||||
|
),
|
||||||
|
exception=DeprecationWarning(),
|
||||||
)
|
)
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
from enum import Enum
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectVisibility(str, Enum):
|
||||||
|
PRIVATE = "PRIVATE"
|
||||||
|
PUBLIC = "PUBLIC"
|
||||||
|
UNLISTEd = "UNLISTED"
|
||||||
|
|
||||||
|
|
||||||
|
class UserProjectsUpdatedMessageType(str, Enum):
|
||||||
|
ADDED = "ADDED"
|
||||||
|
REMOVED = "REMOVED"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectModelsUpdatedMessageType(str, Enum):
|
||||||
|
CREATED = "CREATED"
|
||||||
|
DELETED = "DELETED"
|
||||||
|
UPDATED = "UPDATED"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectUpdatedMessageType(str, Enum):
|
||||||
|
DELETED = "DELETED"
|
||||||
|
UPDATED = "UPDATED"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectVersionsUpdatedMessageType(str, Enum):
|
||||||
|
CREATED = "CREATED"
|
||||||
|
DELETED = "DELETED"
|
||||||
|
UPDATED = "UPDATED"
|
||||||
@@ -0,0 +1,116 @@
|
|||||||
|
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)
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
from specklepy.core.api.inputs.model_inputs import (
|
||||||
|
CreateModelInput,
|
||||||
|
DeleteModelInput,
|
||||||
|
ModelVersionsFilter,
|
||||||
|
UpdateModelInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectCreateInput,
|
||||||
|
ProjectInviteCreateInput,
|
||||||
|
ProjectInviteUseInput,
|
||||||
|
ProjectModelsFilter,
|
||||||
|
ProjectUpdateInput,
|
||||||
|
ProjectUpdateRoleInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.inputs.user_inputs import UserProjectsFilter, UserUpdateInput
|
||||||
|
from specklepy.core.api.inputs.version_inputs import (
|
||||||
|
CreateVersionInput,
|
||||||
|
DeleteVersionsInput,
|
||||||
|
MarkReceivedVersionInput,
|
||||||
|
MoveVersionsInput,
|
||||||
|
UpdateVersionInput,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"CreateModelInput",
|
||||||
|
"DeleteModelInput",
|
||||||
|
"UpdateModelInput",
|
||||||
|
"ModelVersionsFilter",
|
||||||
|
"ProjectCreateInput",
|
||||||
|
"ProjectInviteCreateInput",
|
||||||
|
"ProjectInviteUseInput",
|
||||||
|
"ProjectModelsFilter",
|
||||||
|
"ProjectUpdateInput",
|
||||||
|
"ProjectUpdateRoleInput",
|
||||||
|
"UserProjectsFilter",
|
||||||
|
"UserUpdateInput",
|
||||||
|
"UpdateVersionInput",
|
||||||
|
"MoveVersionsInput",
|
||||||
|
"DeleteVersionsInput",
|
||||||
|
"CreateVersionInput",
|
||||||
|
"MarkReceivedVersionInput",
|
||||||
|
]
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
from typing import Optional, Sequence
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class CreateModelInput(BaseModel):
|
||||||
|
name: str
|
||||||
|
description: Optional[str] = None
|
||||||
|
projectId: str
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteModelInput(BaseModel):
|
||||||
|
id: str
|
||||||
|
projectId: str
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateModelInput(BaseModel):
|
||||||
|
id: str
|
||||||
|
name: Optional[str] = None
|
||||||
|
description: Optional[str] = None
|
||||||
|
projectId: str
|
||||||
|
|
||||||
|
|
||||||
|
class ModelVersionsFilter(BaseModel):
|
||||||
|
priorityIds: Sequence[str]
|
||||||
|
priorityIdsOnly: Optional[bool] = None
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
from typing import Optional, Sequence
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
from specklepy.core.api.enums import ProjectVisibility
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectCreateInput(BaseModel):
|
||||||
|
name: Optional[str]
|
||||||
|
description: Optional[str]
|
||||||
|
visibility: Optional[ProjectVisibility]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectInviteCreateInput(BaseModel):
|
||||||
|
email: Optional[str]
|
||||||
|
role: Optional[str]
|
||||||
|
serverRole: Optional[str]
|
||||||
|
userId: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectInviteUseInput(BaseModel):
|
||||||
|
accept: bool
|
||||||
|
projectId: str
|
||||||
|
token: str
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectModelsFilter(BaseModel):
|
||||||
|
contributors: Optional[Sequence[str]] = None
|
||||||
|
excludeIds: Optional[Sequence[str]] = None
|
||||||
|
ids: Optional[Sequence[str]] = None
|
||||||
|
onlyWithVersions: Optional[bool] = None
|
||||||
|
search: Optional[str] = None
|
||||||
|
sourceApps: Optional[Sequence[str]] = None
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectUpdateInput(BaseModel):
|
||||||
|
id: str
|
||||||
|
name: Optional[str] = None
|
||||||
|
description: Optional[str] = None
|
||||||
|
allowPublicComments: Optional[bool] = None
|
||||||
|
visibility: Optional[ProjectVisibility] = None
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectUpdateRoleInput(BaseModel):
|
||||||
|
userId: str
|
||||||
|
projectId: str
|
||||||
|
role: Optional[str]
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
from typing import Optional, Sequence
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class UserUpdateInput(BaseModel):
|
||||||
|
avatar: Optional[str] = None
|
||||||
|
bio: Optional[str] = None
|
||||||
|
company: Optional[str] = None
|
||||||
|
name: Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
|
class UserProjectsFilter(BaseModel):
|
||||||
|
search: str
|
||||||
|
onlyWithRoles: Optional[Sequence[str]] = None
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
from typing import Optional, Sequence
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateVersionInput(BaseModel):
|
||||||
|
versionId: str
|
||||||
|
projectId: str
|
||||||
|
message: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class MoveVersionsInput(BaseModel):
|
||||||
|
targetModelName: str
|
||||||
|
versionIds: Sequence[str]
|
||||||
|
projectId: str
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteVersionsInput(BaseModel):
|
||||||
|
versionIds: Sequence[str]
|
||||||
|
projectId: str
|
||||||
|
|
||||||
|
|
||||||
|
class CreateVersionInput(BaseModel):
|
||||||
|
objectId: str
|
||||||
|
modelId: str
|
||||||
|
projectId: str
|
||||||
|
message: Optional[str] = None
|
||||||
|
sourceApplication: Optional[str] = "py"
|
||||||
|
totalChildrenCount: Optional[int] = None
|
||||||
|
parents: Optional[Sequence[str]] = None
|
||||||
|
|
||||||
|
|
||||||
|
class MarkReceivedVersionInput(BaseModel):
|
||||||
|
versionId: str
|
||||||
|
projectId: str
|
||||||
|
sourceApplication: str
|
||||||
|
message: Optional[str] = None
|
||||||
@@ -0,0 +1,71 @@
|
|||||||
|
from specklepy.core.api.models.current import (
|
||||||
|
AuthStrategy,
|
||||||
|
LimitedUser,
|
||||||
|
Model,
|
||||||
|
ModelWithVersions,
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
Project,
|
||||||
|
ProjectCollaborator,
|
||||||
|
ProjectCommentCollection,
|
||||||
|
ProjectWithModels,
|
||||||
|
ProjectWithTeam,
|
||||||
|
ResourceCollection,
|
||||||
|
ServerConfiguration,
|
||||||
|
ServerInfo,
|
||||||
|
ServerMigration,
|
||||||
|
User,
|
||||||
|
UserSearchResultCollection,
|
||||||
|
Version,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
Activity,
|
||||||
|
ActivityCollection,
|
||||||
|
Branch,
|
||||||
|
Branches,
|
||||||
|
Collaborator,
|
||||||
|
Commit,
|
||||||
|
Commits,
|
||||||
|
Object,
|
||||||
|
Stream,
|
||||||
|
Streams,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.subscription_messages import (
|
||||||
|
ProjectModelsUpdatedMessage,
|
||||||
|
ProjectUpdatedMessage,
|
||||||
|
ProjectVersionsUpdatedMessage,
|
||||||
|
UserProjectsUpdatedMessage,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"User",
|
||||||
|
"ResourceCollection",
|
||||||
|
"ServerMigration",
|
||||||
|
"AuthStrategy",
|
||||||
|
"ServerConfiguration",
|
||||||
|
"ServerInfo",
|
||||||
|
"LimitedUser",
|
||||||
|
"PendingStreamCollaborator",
|
||||||
|
"ProjectCollaborator",
|
||||||
|
"Version",
|
||||||
|
"Model",
|
||||||
|
"ModelWithVersions",
|
||||||
|
"Project",
|
||||||
|
"ProjectWithModels",
|
||||||
|
"ProjectWithTeam",
|
||||||
|
"ProjectCommentCollection",
|
||||||
|
"UserSearchResultCollection",
|
||||||
|
"UserProjectsUpdatedMessage",
|
||||||
|
"ProjectModelsUpdatedMessage",
|
||||||
|
"ProjectUpdatedMessage",
|
||||||
|
"ProjectVersionsUpdatedMessage",
|
||||||
|
"Collaborator",
|
||||||
|
"Commit",
|
||||||
|
"Commits",
|
||||||
|
"Object",
|
||||||
|
"Branch",
|
||||||
|
"Branches",
|
||||||
|
"Stream",
|
||||||
|
"Streams",
|
||||||
|
"Activity",
|
||||||
|
"ActivityCollection",
|
||||||
|
]
|
||||||
@@ -0,0 +1,171 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import Generic, List, Optional, TypeVar
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
from specklepy.core.api.enums import ProjectVisibility
|
||||||
|
from specklepy.core.api.models.deprecated import Streams
|
||||||
|
|
||||||
|
T = TypeVar("T")
|
||||||
|
|
||||||
|
|
||||||
|
class User(BaseModel):
|
||||||
|
id: str
|
||||||
|
email: Optional[str] = None
|
||||||
|
name: str
|
||||||
|
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 ResourceCollection(BaseModel, Generic[T]):
|
||||||
|
totalCount: int
|
||||||
|
items: List[T]
|
||||||
|
cursor: Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
|
class ServerMigration(BaseModel):
|
||||||
|
movedFrom: Optional[str]
|
||||||
|
movedTo: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class AuthStrategy(BaseModel):
|
||||||
|
color: Optional[str]
|
||||||
|
icon: str
|
||||||
|
id: str
|
||||||
|
name: str
|
||||||
|
url: str
|
||||||
|
|
||||||
|
|
||||||
|
class ServerConfiguration(BaseModel):
|
||||||
|
blobSizeLimitBytes: int
|
||||||
|
objectMultipartUploadSizeLimitBytes: int
|
||||||
|
objectSizeLimitBytes: int
|
||||||
|
|
||||||
|
|
||||||
|
# Keeping this one all Optionals at the minute, because its used both as a deserialization model for GQL and Account Management
|
||||||
|
class ServerInfo(BaseModel):
|
||||||
|
name: Optional[str] = None
|
||||||
|
company: Optional[str] = None
|
||||||
|
url: Optional[str] = None
|
||||||
|
adminContact: Optional[str] = None
|
||||||
|
description: 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
|
||||||
|
frontend2: Optional[bool] = None
|
||||||
|
migration: Optional[ServerMigration] = None
|
||||||
|
|
||||||
|
# TODO separate gql model from account management model
|
||||||
|
|
||||||
|
|
||||||
|
class LimitedUser(BaseModel):
|
||||||
|
"""Limited user type, for showing public info about a user to another user."""
|
||||||
|
|
||||||
|
id: str
|
||||||
|
name: str
|
||||||
|
bio: Optional[str]
|
||||||
|
company: Optional[str]
|
||||||
|
avatar: Optional[str]
|
||||||
|
verified: Optional[bool]
|
||||||
|
role: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class PendingStreamCollaborator(BaseModel):
|
||||||
|
id: str
|
||||||
|
inviteId: str
|
||||||
|
streamId: Optional[str] = None
|
||||||
|
projectId: str
|
||||||
|
streamName: Optional[str] = None
|
||||||
|
projectName: str
|
||||||
|
title: str
|
||||||
|
role: str
|
||||||
|
invitedBy: LimitedUser
|
||||||
|
user: Optional[LimitedUser] = None
|
||||||
|
token: Optional[str]
|
||||||
|
|
||||||
|
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 ProjectCollaborator(BaseModel):
|
||||||
|
id: str
|
||||||
|
role: str
|
||||||
|
user: LimitedUser
|
||||||
|
|
||||||
|
|
||||||
|
class Version(BaseModel):
|
||||||
|
authorUser: Optional[LimitedUser]
|
||||||
|
createdAt: datetime
|
||||||
|
id: str
|
||||||
|
message: Optional[str]
|
||||||
|
previewUrl: str
|
||||||
|
referencedObject: str
|
||||||
|
sourceApplication: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class Model(BaseModel):
|
||||||
|
author: LimitedUser
|
||||||
|
createdAt: datetime
|
||||||
|
description: Optional[str]
|
||||||
|
displayName: str
|
||||||
|
id: str
|
||||||
|
name: str
|
||||||
|
previewUrl: Optional[str]
|
||||||
|
updatedAt: datetime
|
||||||
|
|
||||||
|
|
||||||
|
class ModelWithVersions(Model):
|
||||||
|
versions: ResourceCollection[Version]
|
||||||
|
|
||||||
|
|
||||||
|
class Project(BaseModel):
|
||||||
|
allowPublicComments: bool
|
||||||
|
createdAt: datetime
|
||||||
|
description: Optional[str]
|
||||||
|
id: str
|
||||||
|
name: str
|
||||||
|
role: Optional[str]
|
||||||
|
sourceApps: List[str]
|
||||||
|
updatedAt: datetime
|
||||||
|
visibility: ProjectVisibility
|
||||||
|
workspaceId: Optional[str]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectWithModels(Project):
|
||||||
|
models: ResourceCollection[Model]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectWithTeam(Project):
|
||||||
|
invitedTeam: List[PendingStreamCollaborator]
|
||||||
|
team: List[ProjectCollaborator]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectCommentCollection(ResourceCollection[T], Generic[T]):
|
||||||
|
totalArchivedCount: int
|
||||||
|
|
||||||
|
|
||||||
|
class UserSearchResultCollection(BaseModel):
|
||||||
|
items: List[LimitedUser]
|
||||||
|
cursor: Optional[str] = None
|
||||||
@@ -0,0 +1,144 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
from pydantic import BaseModel, Field
|
||||||
|
|
||||||
|
FE1_DEPRECATION_REASON = "Stream/Branch/Commit API is now deprecated, Use the new Project/Model/Version API functions in Client}"
|
||||||
|
FE1_DEPRECATION_VERSION = "2.20"
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Collaborator(BaseModel):
|
||||||
|
id: Optional[str] = None
|
||||||
|
name: Optional[str] = None
|
||||||
|
role: Optional[str] = None
|
||||||
|
avatar: Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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__()
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Commits(BaseModel):
|
||||||
|
totalCount: Optional[int] = None
|
||||||
|
cursor: Optional[datetime] = None
|
||||||
|
items: List[Commit] = []
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Object(BaseModel):
|
||||||
|
id: Optional[str] = None
|
||||||
|
speckleType: Optional[str] = None
|
||||||
|
applicationId: Optional[str] = None
|
||||||
|
totalChildrenCount: Optional[int] = None
|
||||||
|
createdAt: Optional[datetime] = None
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Branch(BaseModel):
|
||||||
|
id: Optional[str] = None
|
||||||
|
name: Optional[str] = None
|
||||||
|
description: Optional[str] = None
|
||||||
|
commits: Optional[Commits] = None
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Branches(BaseModel):
|
||||||
|
totalCount: Optional[int] = None
|
||||||
|
cursor: Optional[datetime] = None
|
||||||
|
items: List[Branch] = []
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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__()
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Streams(BaseModel):
|
||||||
|
totalCount: Optional[int] = None
|
||||||
|
cursor: Optional[datetime] = None
|
||||||
|
items: List[Stream] = []
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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__()
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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__()
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
from specklepy.core.api.enums import (
|
||||||
|
ProjectModelsUpdatedMessageType,
|
||||||
|
ProjectUpdatedMessageType,
|
||||||
|
ProjectVersionsUpdatedMessageType,
|
||||||
|
UserProjectsUpdatedMessageType,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.current import Model, Project, Version
|
||||||
|
|
||||||
|
|
||||||
|
class UserProjectsUpdatedMessage(BaseModel):
|
||||||
|
id: str
|
||||||
|
type: UserProjectsUpdatedMessageType
|
||||||
|
project: Optional[Project]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectModelsUpdatedMessage(BaseModel):
|
||||||
|
id: str
|
||||||
|
type: ProjectModelsUpdatedMessageType
|
||||||
|
model: Optional[Model]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectUpdatedMessage(BaseModel):
|
||||||
|
id: str
|
||||||
|
type: ProjectUpdatedMessageType
|
||||||
|
project: Optional[Project]
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectVersionsUpdatedMessage(BaseModel):
|
||||||
|
id: str
|
||||||
|
type: ProjectVersionsUpdatedMessageType
|
||||||
|
modelId: Optional[str]
|
||||||
|
version: Optional[Version]
|
||||||
@@ -1,16 +1,16 @@
|
|||||||
from typing import List
|
from typing import List, Optional
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.objects.base import Base
|
# from specklepy.logging import metrics
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
from specklepy.transports.server import ServerTransport
|
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.transports.abstract_transport import AbstractTransport
|
from specklepy.objects.base import Base
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
||||||
|
from specklepy.transports.abstract_transport import AbstractTransport
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
|
||||||
|
|
||||||
def send(
|
def send(
|
||||||
base: Base,
|
base: Base,
|
||||||
transports: List[AbstractTransport] = None,
|
transports: Optional[List[AbstractTransport]] = None,
|
||||||
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,7 +18,8 @@ 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. If set to false, it will only send to the provided transports
|
use_default_cache {bool} -- toggle for the default cache.
|
||||||
|
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
|
||||||
@@ -26,17 +27,17 @@ def send(
|
|||||||
|
|
||||||
if not transports and not use_default_cache:
|
if not transports and not use_default_cache:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
message="You need to provide at least one transport: cannot send with an empty transport list and no default cache"
|
message=(
|
||||||
|
"You need to provide at least one transport: cannot send with an empty"
|
||||||
|
" transport list and no default cache"
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
if isinstance(transports, AbstractTransport):
|
if isinstance(transports, AbstractTransport):
|
||||||
transports = [transports]
|
transports = [transports]
|
||||||
|
|
||||||
if transports is None:
|
if transports is None:
|
||||||
metrics.track(metrics.SEND)
|
|
||||||
transports = []
|
transports = []
|
||||||
else:
|
|
||||||
metrics.track(metrics.SEND, getattr(transports[0], "account", None))
|
|
||||||
|
|
||||||
if use_default_cache:
|
if use_default_cache:
|
||||||
transports.insert(0, SQLiteTransport())
|
transports.insert(0, SQLiteTransport())
|
||||||
@@ -50,8 +51,8 @@ def send(
|
|||||||
|
|
||||||
def receive(
|
def receive(
|
||||||
obj_id: str,
|
obj_id: str,
|
||||||
remote_transport: AbstractTransport = None,
|
remote_transport: Optional[AbstractTransport] = None,
|
||||||
local_transport: AbstractTransport = None,
|
local_transport: Optional[AbstractTransport] = None,
|
||||||
) -> Base:
|
) -> Base:
|
||||||
"""Receives an object from a transport.
|
"""Receives an object from a transport.
|
||||||
|
|
||||||
@@ -64,20 +65,22 @@ def receive(
|
|||||||
Returns:
|
Returns:
|
||||||
Base -- the base object
|
Base -- the base object
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.RECEIVE, getattr(remote_transport, "account", None))
|
|
||||||
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 deserialisation using the 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
|
||||||
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="Could not find the specified object using the local transport, and you didn't provide a fallback remote from which to pull it."
|
message=(
|
||||||
|
"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(
|
||||||
@@ -89,38 +92,48 @@ 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, the object will be serialized
|
Serialize a base object. If no write transports are provided,
|
||||||
|
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]} -- optional: the transports to write to
|
write_transports {List[AbstractTransport]}
|
||||||
|
-- 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(obj_string: str, read_transport: AbstractTransport = None) -> Base:
|
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.
|
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:
|
Arguments:
|
||||||
obj_string {str} -- the string object to deserialize
|
obj_string {str} -- the string object to deserialize
|
||||||
read_transport {AbstractTransport} -- the transport to fetch children objects from
|
read_transport {AbstractTransport}
|
||||||
(defaults to SQLiteTransport)
|
-- the transport to fetch children objects from
|
||||||
|
(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"]
|
||||||
@@ -1,15 +1,21 @@
|
|||||||
from graphql import DocumentNode
|
from threading import Lock
|
||||||
from specklepy.api.credentials import Account
|
from typing import Any, Dict, List, Optional, Tuple, Type, TypeVar, Union
|
||||||
from specklepy.transports.sqlite import SQLiteTransport
|
|
||||||
from typing import Any, Dict, List, Optional, Tuple, Type, Union
|
|
||||||
from gql.client import Client
|
from gql.client import Client
|
||||||
from gql.transport.exceptions import TransportQueryError
|
from gql.transport.exceptions import TransportQueryError
|
||||||
|
from graphql import DocumentNode
|
||||||
|
from pydantic import BaseModel
|
||||||
|
|
||||||
|
from specklepy.core.api.credentials import Account
|
||||||
from specklepy.logging.exceptions import (
|
from specklepy.logging.exceptions import (
|
||||||
GraphQLException,
|
GraphQLException,
|
||||||
SpeckleException,
|
SpeckleException,
|
||||||
UnsupportedException,
|
UnsupportedException,
|
||||||
)
|
)
|
||||||
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
from specklepy.serialization.base_object_serializer import BaseObjectSerializer
|
||||||
|
from specklepy.transports.sqlite import SQLiteTransport
|
||||||
|
|
||||||
|
T = TypeVar("T", bound=BaseModel)
|
||||||
|
|
||||||
|
|
||||||
class ResourceBase(object):
|
class ResourceBase(object):
|
||||||
@@ -27,6 +33,7 @@ class ResourceBase(object):
|
|||||||
self.name = name
|
self.name = name
|
||||||
self.server_version = server_version
|
self.server_version = server_version
|
||||||
self.schema: Optional[Type] = None
|
self.schema: Optional[Type] = None
|
||||||
|
self.__lock = Lock()
|
||||||
|
|
||||||
def _step_into_response(self, response: dict, return_type: Union[str, List, None]):
|
def _step_into_response(self, response: dict, return_type: Union[str, List, None]):
|
||||||
"""Step into the dict to get the relevant data"""
|
"""Step into the dict to get the relevant data"""
|
||||||
@@ -39,6 +46,35 @@ class ResourceBase(object):
|
|||||||
response = response[key]
|
response = response[key]
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
def make_request_and_parse_response(
|
||||||
|
self,
|
||||||
|
schema: Type[T],
|
||||||
|
query: DocumentNode,
|
||||||
|
variables: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> T:
|
||||||
|
try:
|
||||||
|
with self.__lock:
|
||||||
|
response = self.client.execute(query, variable_values=variables)
|
||||||
|
except TransportQueryError as ex:
|
||||||
|
raise GraphQLException(
|
||||||
|
message=(
|
||||||
|
f"Failed to execute the GraphQL {self.name} request. Errors:"
|
||||||
|
f" {ex.errors}"
|
||||||
|
),
|
||||||
|
errors=ex.errors,
|
||||||
|
data=ex.data,
|
||||||
|
) from ex
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException(
|
||||||
|
message=(
|
||||||
|
f"Failed to execute the GraphQL {self.name} request. Inner"
|
||||||
|
f" exception: {ex}"
|
||||||
|
),
|
||||||
|
exception=ex,
|
||||||
|
) from ex
|
||||||
|
|
||||||
|
return schema.model_validate(response)
|
||||||
|
|
||||||
def _parse_response(self, response: Union[dict, list, None], schema=None):
|
def _parse_response(self, response: Union[dict, list, None], schema=None):
|
||||||
"""Try to create a class instance from the response"""
|
"""Try to create a class instance from the response"""
|
||||||
if response is None:
|
if response is None:
|
||||||
@@ -46,11 +82,11 @@ class ResourceBase(object):
|
|||||||
if isinstance(response, list):
|
if isinstance(response, list):
|
||||||
return [self._parse_response(response=r, schema=schema) for r in response]
|
return [self._parse_response(response=r, schema=schema) for r in response]
|
||||||
if schema:
|
if schema:
|
||||||
return schema.parse_obj(response)
|
return schema.model_validate(response)
|
||||||
elif self.schema:
|
elif self.schema:
|
||||||
try:
|
try:
|
||||||
return self.schema.parse_obj(response)
|
return self.schema.model_validate(response)
|
||||||
except:
|
except Exception:
|
||||||
s = BaseObjectSerializer(read_transport=SQLiteTransport())
|
s = BaseObjectSerializer(read_transport=SQLiteTransport())
|
||||||
return s.recompose_base(response)
|
return s.recompose_base(response)
|
||||||
else:
|
else:
|
||||||
@@ -59,24 +95,33 @@ class ResourceBase(object):
|
|||||||
def make_request(
|
def make_request(
|
||||||
self,
|
self,
|
||||||
query: DocumentNode,
|
query: DocumentNode,
|
||||||
params: Dict = None,
|
params: Optional[Dict] = None,
|
||||||
return_type: Union[str, List, None] = None,
|
return_type: Union[str, List, None] = None,
|
||||||
schema=None,
|
schema=None,
|
||||||
parse_response: bool = True,
|
parse_response: bool = True,
|
||||||
) -> Any:
|
) -> Any:
|
||||||
"""Executes the GraphQL query"""
|
"""Executes the GraphQL query"""
|
||||||
|
# This method has quite complex and ambiguous typing, and counter-intuitive error handling
|
||||||
|
# We are going to phase it out in favour of `make_request_and_parse_response`
|
||||||
try:
|
try:
|
||||||
response = self.client.execute(query, variable_values=params)
|
with self.__lock:
|
||||||
|
response = self.client.execute(query, variable_values=params)
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
if isinstance(ex, TransportQueryError):
|
if isinstance(ex, TransportQueryError):
|
||||||
return GraphQLException(
|
return GraphQLException(
|
||||||
message=f"Failed to execute the GraphQL {self.name} request. Errors: {ex.errors}",
|
message=(
|
||||||
|
f"Failed to execute the GraphQL {self.name} request. Errors:"
|
||||||
|
f" {ex.errors}"
|
||||||
|
),
|
||||||
errors=ex.errors,
|
errors=ex.errors,
|
||||||
data=ex.data,
|
data=ex.data,
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return SpeckleException(
|
return SpeckleException(
|
||||||
message=f"Failed to execute the GraphQL {self.name} request. Inner exception: {ex}",
|
message=(
|
||||||
|
f"Failed to execute the GraphQL {self.name} request. Inner"
|
||||||
|
f" exception: {ex}"
|
||||||
|
),
|
||||||
exception=ex,
|
exception=ex,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -88,16 +133,23 @@ class ResourceBase(object):
|
|||||||
return response
|
return response
|
||||||
|
|
||||||
def _check_server_version_at_least(
|
def _check_server_version_at_least(
|
||||||
self, target_version: Tuple[Any, ...], unsupported_message: str = None
|
self, target_version: Tuple[Any, ...], unsupported_message: Optional[str] = None
|
||||||
):
|
):
|
||||||
"""Use this check to guard against making unsupported requests on older servers.
|
"""Use this check to guard against making unsupported requests on older servers.
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
target_version {tuple} -- the minimum server version in the format (major, minor, patch, (tag, build))
|
target_version {tuple}
|
||||||
eg (2, 6, 3) for a stable build and (2, 6, 4, 'alpha', 4711) for alpha
|
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:
|
if not unsupported_message:
|
||||||
unsupported_message = f"The client method used is not supported on Speckle Server versios prior to v{'.'.join(target_version)}"
|
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:
|
if self.server_version and self.server_version < target_version:
|
||||||
raise UnsupportedException(unsupported_message)
|
raise UnsupportedException(unsupported_message)
|
||||||
|
|
||||||
@@ -107,8 +159,7 @@ class ResourceBase(object):
|
|||||||
"""
|
"""
|
||||||
self._check_server_version_at_least(
|
self._check_server_version_at_least(
|
||||||
(2, 6, 4),
|
(2, 6, 4),
|
||||||
(
|
"Stream invites are only supported as of Speckle Server v2.6.4. Please"
|
||||||
"Stream invites are only supported as of Speckle Server v2.6.4. "
|
" update your Speckle Server to use this method or use the"
|
||||||
"Please update your Speckle Server to use this method or use the `grant_permission` flow instead."
|
" `grant_permission` flow instead.",
|
||||||
),
|
|
||||||
)
|
)
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
from specklepy.core.api.resources.current.active_user_resource import ActiveUserResource
|
||||||
|
from specklepy.core.api.resources.current.model_resource import ModelResource
|
||||||
|
from specklepy.core.api.resources.current.other_user_resource import OtherUserResource
|
||||||
|
from specklepy.core.api.resources.current.project_invite_resource import (
|
||||||
|
ProjectInviteResource,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.current.project_resource import ProjectResource
|
||||||
|
from specklepy.core.api.resources.current.server_resource import ServerResource
|
||||||
|
from specklepy.core.api.resources.current.subscription_resource import (
|
||||||
|
SubscriptionResource,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources.current.version_resource import VersionResource
|
||||||
|
from specklepy.core.api.resources.deprecated import (
|
||||||
|
active_user,
|
||||||
|
branch,
|
||||||
|
commit,
|
||||||
|
object,
|
||||||
|
other_user,
|
||||||
|
server,
|
||||||
|
stream,
|
||||||
|
subscriptions,
|
||||||
|
user,
|
||||||
|
)
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"ActiveUserResource",
|
||||||
|
"ModelResource",
|
||||||
|
"OtherUserResource",
|
||||||
|
"ProjectInviteResource",
|
||||||
|
"ProjectResource",
|
||||||
|
"ServerResource",
|
||||||
|
"SubscriptionResource",
|
||||||
|
"VersionResource",
|
||||||
|
"active_user",
|
||||||
|
"branch",
|
||||||
|
"commit",
|
||||||
|
"object",
|
||||||
|
"other_user",
|
||||||
|
"server",
|
||||||
|
"stream",
|
||||||
|
"subscriptions",
|
||||||
|
"user",
|
||||||
|
]
|
||||||
@@ -0,0 +1,410 @@
|
|||||||
|
from datetime import datetime, timezone
|
||||||
|
from typing import List, Optional, overload
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.user_inputs import UserProjectsFilter, UserUpdateInput
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ActivityCollection,
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
Project,
|
||||||
|
ResourceCollection,
|
||||||
|
User,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
from specklepy.logging.exceptions import GraphQLException
|
||||||
|
|
||||||
|
NAME = "active_user"
|
||||||
|
|
||||||
|
|
||||||
|
class ActiveUserResource(ResourceBase):
|
||||||
|
"""API Access class for the 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 = User
|
||||||
|
|
||||||
|
def get(self) -> Optional[User]:
|
||||||
|
"""Gets the currently active user profile (as extracted from the authorization header)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
User -- the requested user, or none if no authentication token is provided to the Client
|
||||||
|
"""
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query User {
|
||||||
|
data:activeUser {
|
||||||
|
id
|
||||||
|
email
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[Optional[User]], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def _update(self, input: UserUpdateInput) -> User:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ActiveUserMutations($input: UserUpdateInput!) {
|
||||||
|
data:activeUserMutations {
|
||||||
|
data:update(user: $input) {
|
||||||
|
id
|
||||||
|
email
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"input": input.model_dump(warnings="error")}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[User]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
@deprecated("Use UserUpdateInput overload", version=FE1_DEPRECATION_VERSION)
|
||||||
|
@overload
|
||||||
|
def update(
|
||||||
|
self,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
company: Optional[str] = None,
|
||||||
|
bio: Optional[str] = None,
|
||||||
|
avatar: Optional[str] = None,
|
||||||
|
) -> User: ...
|
||||||
|
|
||||||
|
@overload
|
||||||
|
def update(self, *, input: UserUpdateInput) -> User: ...
|
||||||
|
|
||||||
|
def update(
|
||||||
|
self,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
company: Optional[str] = None,
|
||||||
|
bio: Optional[str] = None,
|
||||||
|
avatar: Optional[str] = None,
|
||||||
|
*,
|
||||||
|
input: Optional[UserUpdateInput] = None,
|
||||||
|
) -> User:
|
||||||
|
if isinstance(input, UserUpdateInput):
|
||||||
|
return self._update(input=input)
|
||||||
|
else:
|
||||||
|
return self._update(
|
||||||
|
input=UserUpdateInput(
|
||||||
|
name=name,
|
||||||
|
company=company,
|
||||||
|
bio=bio,
|
||||||
|
avatar=avatar,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_projects(
|
||||||
|
self,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[UserProjectsFilter] = None,
|
||||||
|
) -> ResourceCollection[Project]:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query User($limit : Int!, $cursor: String, $filter: UserProjectsFilter) {
|
||||||
|
data:activeUser {
|
||||||
|
data:projects(limit: $limit, cursor: $cursor, filter: $filter) {
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"limit": limit,
|
||||||
|
"cursor": cursor,
|
||||||
|
"filter": filter.model_dump(warnings="error") if filter else None,
|
||||||
|
}
|
||||||
|
|
||||||
|
response = self.make_request_and_parse_response(
|
||||||
|
DataResponse[Optional[DataResponse[ResourceCollection[Project]]]],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.data is None:
|
||||||
|
raise GraphQLException(
|
||||||
|
"GraphQL response indicated that the ActiveUser could not be found"
|
||||||
|
)
|
||||||
|
|
||||||
|
return response.data.data
|
||||||
|
|
||||||
|
def get_project_invites(self) -> List[PendingStreamCollaborator]:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ProjectInvites {
|
||||||
|
data:activeUser {
|
||||||
|
data:projectInvites {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
invitedBy {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
role
|
||||||
|
title
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
avatar
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {}
|
||||||
|
|
||||||
|
response = self.make_request_and_parse_response(
|
||||||
|
DataResponse[Optional[DataResponse[List[PendingStreamCollaborator]]]],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
)
|
||||||
|
|
||||||
|
if response.data is None:
|
||||||
|
raise GraphQLException(
|
||||||
|
"GraphQL response indicated that the ActiveUser could not be found"
|
||||||
|
)
|
||||||
|
|
||||||
|
return response.data.data
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
def activity(
|
||||||
|
self,
|
||||||
|
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.
|
||||||
|
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,
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query,
|
||||||
|
return_type="streamInvites",
|
||||||
|
schema=PendingStreamCollaborator,
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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
|
||||||
|
inviteId
|
||||||
|
streamId
|
||||||
|
streamName
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
params = {"streamId": stream_id}
|
||||||
|
if token:
|
||||||
|
params["token"] = token
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query,
|
||||||
|
params=params,
|
||||||
|
return_type="streamInvite",
|
||||||
|
schema=PendingStreamCollaborator,
|
||||||
|
)
|
||||||
@@ -0,0 +1,280 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.model_inputs import (
|
||||||
|
CreateModelInput,
|
||||||
|
DeleteModelInput,
|
||||||
|
ModelVersionsFilter,
|
||||||
|
UpdateModelInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.inputs.project_inputs import ProjectModelsFilter
|
||||||
|
from specklepy.core.api.models import Model, ModelWithVersions, ResourceCollection
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
|
||||||
|
NAME = "model"
|
||||||
|
|
||||||
|
|
||||||
|
class ModelResource(ResourceBase):
|
||||||
|
"""API Access class for models"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
name=NAME,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, model_id: str, project_id: str) -> Model:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ModelGet($modelId: String!, $projectId: String!) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
data:model(id: $modelId) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
previewUrl
|
||||||
|
updatedAt
|
||||||
|
description
|
||||||
|
displayName
|
||||||
|
createdAt
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"modelId": model_id,
|
||||||
|
"projectId": project_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Model]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def get_with_versions(
|
||||||
|
self,
|
||||||
|
model_id: str,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
versions_limit: int = 25,
|
||||||
|
versions_cursor: Optional[str] = None,
|
||||||
|
versions_filter: Optional[ModelVersionsFilter] = None,
|
||||||
|
) -> ModelWithVersions:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ModelGetWithVersions($modelId: String!, $projectId: String!, $versionsLimit: Int!, $versionsCursor: String, $versionsFilter: ModelVersionsFilter) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
data:model(id: $modelId) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
previewUrl
|
||||||
|
updatedAt
|
||||||
|
versions(limit: $versionsLimit, cursor: $versionsCursor, filter: $versionsFilter) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
sourceApplication
|
||||||
|
createdAt
|
||||||
|
previewUrl
|
||||||
|
authorUser {
|
||||||
|
avatar
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
}
|
||||||
|
description
|
||||||
|
displayName
|
||||||
|
createdAt
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"modelId": model_id,
|
||||||
|
"versionsLimit": versions_limit,
|
||||||
|
"versionsCursor": versions_cursor,
|
||||||
|
"versionsFilter": (
|
||||||
|
versions_filter.model_dump(warnings="error")
|
||||||
|
if versions_filter
|
||||||
|
else None
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[ModelWithVersions]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def get_models(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
models_limit: int = 25,
|
||||||
|
models_cursor: Optional[str] = None,
|
||||||
|
models_filter: Optional[ProjectModelsFilter] = None,
|
||||||
|
) -> ResourceCollection[Model]:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ProjectGetWithModels($projectId: String!, $modelsLimit: Int!, $modelsCursor: String, $modelsFilter: ProjectModelsFilter) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
data:models(limit: $modelsLimit, cursor: $modelsCursor, filter: $modelsFilter) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
previewUrl
|
||||||
|
updatedAt
|
||||||
|
displayName
|
||||||
|
description
|
||||||
|
createdAt
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
totalCount
|
||||||
|
cursor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"modelsLimit": models_limit,
|
||||||
|
"modelsCursor": models_cursor,
|
||||||
|
"modelsFilter": (
|
||||||
|
models_filter.model_dump(warnings="error") if models_filter else None
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[ResourceCollection[Model]]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def create(self, input: CreateModelInput) -> Model:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ModelCreate($input: CreateModelInput!) {
|
||||||
|
data:modelMutations {
|
||||||
|
data:create(input: $input) {
|
||||||
|
id
|
||||||
|
displayName
|
||||||
|
name
|
||||||
|
description
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
previewUrl
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Model]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def delete(self, input: DeleteModelInput) -> bool:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ModelDelete($input: DeleteModelInput!) {
|
||||||
|
data:modelMutations {
|
||||||
|
data:delete(input: $input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"input": input.model_dump(warnings="error")}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[bool]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def update(self, input: UpdateModelInput) -> Model:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ModelUpdate($input: UpdateModelInput!) {
|
||||||
|
data:modelMutations {
|
||||||
|
data:update(input: $input) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
displayName
|
||||||
|
description
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
previewUrl
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Model]], QUERY, variables
|
||||||
|
).data.data
|
||||||
@@ -0,0 +1,239 @@
|
|||||||
|
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,
|
||||||
|
LimitedUser,
|
||||||
|
UserSearchResultCollection,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
NAME = "other_user"
|
||||||
|
|
||||||
|
|
||||||
|
class OtherUserResource(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) -> Optional[LimitedUser]:
|
||||||
|
"""
|
||||||
|
Gets the profile of another user.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
id {str} -- the user id
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
LimitedUser -- the retrieved profile of another user
|
||||||
|
"""
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query LimitedUser($id: String!) {
|
||||||
|
data:otherUser(id: $id){
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"id": id}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[Optional[LimitedUser]], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def user_search(
|
||||||
|
self,
|
||||||
|
query: str,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
archived: bool = False,
|
||||||
|
emailOnly: bool = False,
|
||||||
|
) -> UserSearchResultCollection:
|
||||||
|
"""Searches for a user on the server, 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
|
||||||
|
cursor {Optional[str]} --
|
||||||
|
archived {bool} --
|
||||||
|
emailOnly {bool} --
|
||||||
|
Returns:
|
||||||
|
ResourceCollection[LimitedUser] -- User objects that match the search query
|
||||||
|
"""
|
||||||
|
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query UserSearch($query: String!, $limit: Int!, $cursor: String, $archived: Boolean, $emailOnly: Boolean) {
|
||||||
|
data:userSearch(query: $query, limit: $limit, cursor: $cursor, archived: $archived, emailOnly: $emailOnly) {
|
||||||
|
cursor
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
variables = {
|
||||||
|
"query": query,
|
||||||
|
"limit": limit,
|
||||||
|
"cursor": cursor,
|
||||||
|
"archived": archived,
|
||||||
|
"emailOnly": emailOnly,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[UserSearchResultCollection], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
@deprecated(reason="Use user_search instead", version=FE1_DEPRECATION_VERSION)
|
||||||
|
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
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
params = {"search_query": search_query, "limit": limit}
|
||||||
|
|
||||||
|
return self.make_request(
|
||||||
|
query=query, params=params, return_type=["userSearch", "items"]
|
||||||
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
|
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,
|
||||||
|
)
|
||||||
@@ -0,0 +1,254 @@
|
|||||||
|
from typing import Any, Optional, Tuple
|
||||||
|
|
||||||
|
from gql import Client, gql
|
||||||
|
|
||||||
|
from specklepy.core.api.credentials import Account
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectInviteCreateInput,
|
||||||
|
ProjectInviteUseInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import PendingStreamCollaborator, ProjectWithTeam
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
|
||||||
|
NAME = "project_invite"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectInviteResource(ResourceBase):
|
||||||
|
"""API Access class for project invites"""
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
account: Account,
|
||||||
|
basepath: str,
|
||||||
|
client: Client,
|
||||||
|
server_version: Optional[Tuple[Any, ...]],
|
||||||
|
) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
name=NAME,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def create(
|
||||||
|
self, project_id: str, input: ProjectInviteCreateInput
|
||||||
|
) -> ProjectWithTeam:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectInviteCreate($projectId: ID!, $input: ProjectInviteCreateInput!) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:invites {
|
||||||
|
data:create(projectId: $projectId, input: $input) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
workspaceId
|
||||||
|
sourceApps
|
||||||
|
team {
|
||||||
|
id
|
||||||
|
role
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
invitedTeam {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[ProjectWithTeam]]], QUERY, variables
|
||||||
|
).data.data.data
|
||||||
|
|
||||||
|
def use(self, input: ProjectInviteUseInput) -> bool:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectInviteUse($input: ProjectInviteUseInput!) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:invites {
|
||||||
|
data:use(input: $input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[bool]]], QUERY, variables
|
||||||
|
).data.data.data
|
||||||
|
|
||||||
|
def get(
|
||||||
|
self, project_id: str, token: Optional[str]
|
||||||
|
) -> Optional[PendingStreamCollaborator]:
|
||||||
|
"""Returns: The invite, or None if no invite exists"""
|
||||||
|
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ProjectInvite($projectId: String!, $token: String) {
|
||||||
|
data:projectInvite(projectId: $projectId, token: $token) {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
invitedBy {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
role
|
||||||
|
title
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"token": token,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[Optional[PendingStreamCollaborator]], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def cancel(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
invite_id: str,
|
||||||
|
) -> ProjectWithTeam:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectInviteCancel($projectId: ID!, $inviteId: String!) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:invites {
|
||||||
|
data:cancel(projectId: $projectId, inviteId: $inviteId) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
team {
|
||||||
|
id
|
||||||
|
role
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
invitedTeam {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"inviteId": invite_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[ProjectWithTeam]]], QUERY, variables
|
||||||
|
).data.data.data
|
||||||
@@ -0,0 +1,336 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectCreateInput,
|
||||||
|
ProjectModelsFilter,
|
||||||
|
ProjectUpdateInput,
|
||||||
|
ProjectUpdateRoleInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import Project, ProjectWithModels, ProjectWithTeam
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
|
||||||
|
NAME = "project"
|
||||||
|
|
||||||
|
|
||||||
|
class ProjectResource(ResourceBase):
|
||||||
|
"""API Access class for projects"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
name=NAME,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, project_id: str) -> Project:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query Project($projectId: String!) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
allowPublicComments
|
||||||
|
createdAt
|
||||||
|
description
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
sourceApps
|
||||||
|
updatedAt
|
||||||
|
visibility
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[Project], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def get_with_models(
|
||||||
|
self,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
models_limit: int = 25,
|
||||||
|
models_cursor: Optional[str] = None,
|
||||||
|
models_filter: Optional[ProjectModelsFilter] = None,
|
||||||
|
) -> ProjectWithModels:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ProjectGetWithModels($projectId: String!, $modelsLimit: Int!, $modelsCursor: String, $modelsFilter: ProjectModelsFilter) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
models(limit: $modelsLimit, cursor: $modelsCursor, filter: $modelsFilter) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
previewUrl
|
||||||
|
updatedAt
|
||||||
|
displayName
|
||||||
|
description
|
||||||
|
createdAt
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cursor
|
||||||
|
totalCount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"modelsLimit": models_limit,
|
||||||
|
"modelsCursor": models_cursor,
|
||||||
|
"modelsFilter": (
|
||||||
|
models_filter.model_dump(warnings="error") if models_filter else None
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[ProjectWithModels], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def get_with_team(self, project_id: str) -> ProjectWithTeam:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query ProjectGetWithTeam($projectId: String!) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
workspaceId
|
||||||
|
sourceApps
|
||||||
|
team {
|
||||||
|
id
|
||||||
|
role
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
invitedTeam {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[ProjectWithTeam], QUERY, variables
|
||||||
|
).data
|
||||||
|
|
||||||
|
def create(self, input: ProjectCreateInput) -> Project:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectCreate($input: ProjectCreateInput) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:create(input: $input) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Project]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def update(self, input: ProjectUpdateInput) -> Project:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectUpdate($input: ProjectUpdateInput!) {
|
||||||
|
data:projectMutations{
|
||||||
|
data:update(update: $input) {
|
||||||
|
allowPublicComments
|
||||||
|
createdAt
|
||||||
|
description
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
sourceApps
|
||||||
|
updatedAt
|
||||||
|
visibility
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Project]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def delete(self, project_id: str) -> bool:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectDelete($projectId: String!) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:delete(id: $projectId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[bool]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def update_role(self, input: ProjectUpdateRoleInput) -> ProjectWithTeam:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation ProjectUpdateRole($input: ProjectUpdateRoleInput!) {
|
||||||
|
data:projectMutations {
|
||||||
|
data:updateRole(input: $input) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
team {
|
||||||
|
id
|
||||||
|
role
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
invitedTeam {
|
||||||
|
id
|
||||||
|
inviteId
|
||||||
|
projectId
|
||||||
|
projectName
|
||||||
|
title
|
||||||
|
role
|
||||||
|
token
|
||||||
|
user {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
invitedBy {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[ProjectWithTeam]], QUERY, variables
|
||||||
|
).data.data
|
||||||
+22
-13
@@ -1,16 +1,17 @@
|
|||||||
import re
|
import re
|
||||||
from typing import Any, Dict, List, Tuple
|
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
|
|
||||||
|
|
||||||
|
import requests
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.models import ServerInfo
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.logging.exceptions import GraphQLException
|
||||||
|
|
||||||
NAME = "server"
|
NAME = "server"
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class ServerResource(ResourceBase):
|
||||||
"""API Access class for the server"""
|
"""API Access class for the server"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, account, basepath, client) -> None:
|
||||||
@@ -27,7 +28,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
dict -- the server info in dictionary form
|
dict -- the server info in dictionary form
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.SERVER, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query Server {
|
query Server {
|
||||||
@@ -57,16 +57,28 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.make_request(
|
server_info = self.make_request(
|
||||||
query=query, return_type="serverInfo", schema=ServerInfo
|
query=query, return_type="serverInfo", schema=ServerInfo
|
||||||
)
|
)
|
||||||
|
if isinstance(server_info, ServerInfo) and isinstance(
|
||||||
|
server_info.canonicalUrl, str
|
||||||
|
):
|
||||||
|
r = requests.get(
|
||||||
|
server_info.canonicalUrl, headers={"User-Agent": "specklepy SDK"}
|
||||||
|
)
|
||||||
|
if "x-speckle-frontend-2" in r.headers:
|
||||||
|
server_info.frontend2 = True
|
||||||
|
else:
|
||||||
|
server_info.frontend2 = False
|
||||||
|
|
||||||
|
return server_info
|
||||||
|
|
||||||
def version(self) -> Tuple[Any, ...]:
|
def version(self) -> Tuple[Any, ...]:
|
||||||
"""Get the server version
|
"""Get the server version
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
tuple -- the server version in the format (major, minor, patch, (tag, build))
|
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
|
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
|
# not tracking as it will be called along with other mutations / queries as a check
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -100,7 +112,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
dict -- a dictionary of apps registered on the server
|
dict -- a dictionary of apps registered on the server
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.SERVER, self.account, {"name": "apps"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query Apps {
|
query Apps {
|
||||||
@@ -134,7 +145,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
str -- the new API token. note: this is the only time you'll see the token!
|
str -- the new API token. note: this is the only time you'll see the token!
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.SERVER, self.account, {"name": "create_token"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation TokenCreate($token: ApiTokenCreateInput!) {
|
mutation TokenCreate($token: ApiTokenCreateInput!) {
|
||||||
@@ -160,7 +170,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if the token was successfully deleted
|
bool -- True if the token was successfully deleted
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.SERVER, self.account, {"name": "revoke_token"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation TokenRevoke($token: String!) {
|
mutation TokenRevoke($token: String!) {
|
||||||
@@ -0,0 +1,218 @@
|
|||||||
|
from functools import wraps
|
||||||
|
from typing import Any, Callable, Dict, Optional, Sequence, Type
|
||||||
|
|
||||||
|
from gql import gql
|
||||||
|
from graphql import DocumentNode
|
||||||
|
from pydantic import BaseModel
|
||||||
|
from typing_extensions import TypeVar
|
||||||
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ProjectModelsUpdatedMessage,
|
||||||
|
ProjectUpdatedMessage,
|
||||||
|
ProjectVersionsUpdatedMessage,
|
||||||
|
UserProjectsUpdatedMessage,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
NAME = "subscribe"
|
||||||
|
|
||||||
|
TEventArgs = TypeVar("TEventArgs", bound=BaseModel)
|
||||||
|
|
||||||
|
|
||||||
|
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 SubscriptionResource(ResourceBase):
|
||||||
|
"""API Access class for subscriptions"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
name=NAME,
|
||||||
|
)
|
||||||
|
|
||||||
|
async def user_projects_updated(
|
||||||
|
self, callback: Callable[[UserProjectsUpdatedMessage], None]
|
||||||
|
) -> None:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
subscription UserProjectsUpdated {
|
||||||
|
data:userProjectsUpdated {
|
||||||
|
id
|
||||||
|
project {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
type
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
await self.subscribe_2(
|
||||||
|
DataResponse[UserProjectsUpdatedMessage],
|
||||||
|
QUERY,
|
||||||
|
None,
|
||||||
|
callback=lambda d: callback(d.data),
|
||||||
|
)
|
||||||
|
|
||||||
|
async def project_models_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectModelsUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
model_ids: Optional[Sequence[str]] = None,
|
||||||
|
) -> None:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
subscription ProjectModelsUpdated($id: String!, $modelIds: [String!]) {
|
||||||
|
data:projectModelsUpdated(id: $id, modelIds: $modelIds) {
|
||||||
|
id
|
||||||
|
model {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
previewUrl
|
||||||
|
updatedAt
|
||||||
|
description
|
||||||
|
displayName
|
||||||
|
createdAt
|
||||||
|
author {
|
||||||
|
avatar
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
id
|
||||||
|
name
|
||||||
|
role
|
||||||
|
verified
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"id": id, "modelIds": model_ids}
|
||||||
|
|
||||||
|
await self.subscribe_2(
|
||||||
|
DataResponse[ProjectModelsUpdatedMessage],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
callback=lambda d: callback(d.data),
|
||||||
|
)
|
||||||
|
|
||||||
|
async def project_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
) -> None:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
subscription ProjectUpdated($id: String!) {
|
||||||
|
data:projectUpdated(id: $id) {
|
||||||
|
id
|
||||||
|
project {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
description
|
||||||
|
visibility
|
||||||
|
allowPublicComments
|
||||||
|
role
|
||||||
|
createdAt
|
||||||
|
updatedAt
|
||||||
|
sourceApps
|
||||||
|
workspaceId
|
||||||
|
}
|
||||||
|
type
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"id": id}
|
||||||
|
|
||||||
|
await self.subscribe_2(
|
||||||
|
DataResponse[ProjectUpdatedMessage],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
callback=lambda d: callback(d.data),
|
||||||
|
)
|
||||||
|
|
||||||
|
async def project_versions_updated(
|
||||||
|
self,
|
||||||
|
callback: Callable[[ProjectVersionsUpdatedMessage], None],
|
||||||
|
id: str,
|
||||||
|
) -> None:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
subscription ProjectVersionsUpdated($id: String!) {
|
||||||
|
data:projectVersionsUpdated(id: $id) {
|
||||||
|
id
|
||||||
|
modelId
|
||||||
|
type
|
||||||
|
version {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
sourceApplication
|
||||||
|
createdAt
|
||||||
|
previewUrl
|
||||||
|
authorUser {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"id": id}
|
||||||
|
|
||||||
|
await self.subscribe_2(
|
||||||
|
DataResponse[ProjectVersionsUpdatedMessage],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
callback=lambda d: callback(d.data),
|
||||||
|
)
|
||||||
|
|
||||||
|
@check_wsclient
|
||||||
|
async def subscribe_2(
|
||||||
|
self,
|
||||||
|
response_type: Type[TEventArgs],
|
||||||
|
query: DocumentNode,
|
||||||
|
variables: Optional[Dict[str, Any]],
|
||||||
|
callback: Callable[[TEventArgs], None],
|
||||||
|
) -> None:
|
||||||
|
async with self.client as session:
|
||||||
|
self.session = session
|
||||||
|
gen = session.subscribe(query, variable_values=variables)
|
||||||
|
async for res in gen:
|
||||||
|
event_arg = response_type.model_validate(res)
|
||||||
|
callback(event_arg)
|
||||||
@@ -0,0 +1,234 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.model_inputs import ModelVersionsFilter
|
||||||
|
from specklepy.core.api.inputs.version_inputs import (
|
||||||
|
CreateVersionInput,
|
||||||
|
DeleteVersionsInput,
|
||||||
|
MarkReceivedVersionInput,
|
||||||
|
MoveVersionsInput,
|
||||||
|
UpdateVersionInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import ResourceCollection, Version
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.core.api.responses import DataResponse
|
||||||
|
|
||||||
|
NAME = "model"
|
||||||
|
|
||||||
|
|
||||||
|
class VersionResource(ResourceBase):
|
||||||
|
"""API Access class for model versions"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
name=NAME,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, version_id: str, project_id: str) -> Version:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query VersionGet($projectId: String!, $versionId: String!) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
data:version(id: $versionId) {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
sourceApplication
|
||||||
|
createdAt
|
||||||
|
previewUrl
|
||||||
|
authorUser {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"versionId": version_id,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Version]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def get_versions(
|
||||||
|
self,
|
||||||
|
model_id: str,
|
||||||
|
project_id: str,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[ModelVersionsFilter] = None,
|
||||||
|
) -> ResourceCollection[Version]:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
query VersionGetVersions($projectId: String!, $modelId: String!, $limit: Int!, $cursor: String, $filter: ModelVersionsFilter) {
|
||||||
|
data:project(id: $projectId) {
|
||||||
|
data:model(id: $modelId) {
|
||||||
|
data:versions(limit: $limit, cursor: $cursor, filter: $filter) {
|
||||||
|
items {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
sourceApplication
|
||||||
|
createdAt
|
||||||
|
previewUrl
|
||||||
|
authorUser {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cursor
|
||||||
|
totalCount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"projectId": project_id,
|
||||||
|
"modelId": model_id,
|
||||||
|
"limit": limit,
|
||||||
|
"cursor": cursor,
|
||||||
|
"filter": filter.model_dump(warnings="error") if filter else None,
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[ResourceCollection[Version]]]],
|
||||||
|
QUERY,
|
||||||
|
variables,
|
||||||
|
).data.data.data
|
||||||
|
|
||||||
|
def create(self, input: CreateVersionInput) -> str:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation Create($input: CreateVersionInput!) {
|
||||||
|
data:versionMutations {
|
||||||
|
data:create(input: $input) {
|
||||||
|
data:id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[str]]], QUERY, variables
|
||||||
|
).data.data.data
|
||||||
|
|
||||||
|
def update(self, input: UpdateVersionInput) -> Version:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation VersionUpdate($input: UpdateVersionInput!) {
|
||||||
|
data:versionMutations {
|
||||||
|
data:update(input: $input) {
|
||||||
|
id
|
||||||
|
referencedObject
|
||||||
|
message
|
||||||
|
sourceApplication
|
||||||
|
createdAt
|
||||||
|
previewUrl
|
||||||
|
authorUser {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
bio
|
||||||
|
company
|
||||||
|
verified
|
||||||
|
role
|
||||||
|
avatar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {"input": input.model_dump(warnings="error")}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[Version]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def move_to_model(self, input: MoveVersionsInput) -> str:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation VersionMoveToModel($input: MoveVersionsInput!) {
|
||||||
|
data:versionMutations {
|
||||||
|
data:moveToModel(input: $input) {
|
||||||
|
data:id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[DataResponse[str]]], QUERY, variables
|
||||||
|
).data.data.data
|
||||||
|
|
||||||
|
def delete(self, input: DeleteVersionsInput) -> bool:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation VersionDelete($input: DeleteVersionsInput!) {
|
||||||
|
data:versionMutations {
|
||||||
|
data:delete(input: $input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[bool]], QUERY, variables
|
||||||
|
).data.data
|
||||||
|
|
||||||
|
def received(self, input: MarkReceivedVersionInput) -> bool:
|
||||||
|
QUERY = gql(
|
||||||
|
"""
|
||||||
|
mutation MarkReceived($input: MarkReceivedVersionInput!) {
|
||||||
|
data:versionMutations {
|
||||||
|
data:markReceived(input: $input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
variables = {
|
||||||
|
"input": input.model_dump(warnings="error"),
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.make_request_and_parse_response(
|
||||||
|
DataResponse[DataResponse[bool]], QUERY, variables
|
||||||
|
).data.data
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
from specklepy.core.api.resources import ActiveUserResource
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(
|
||||||
|
reason="Class renamed to ActiveUserResource", version=FE1_DEPRECATION_VERSION
|
||||||
|
)
|
||||||
|
class Resource(ActiveUserResource):
|
||||||
|
"""
|
||||||
|
Class renamed to ActiveUserResource
|
||||||
|
"""
|
||||||
|
|
||||||
|
pass
|
||||||
+33
-12
@@ -1,13 +1,24 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.api.models import Branch
|
from specklepy.core.api.models.deprecated import (
|
||||||
from specklepy.logging import metrics
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
Branch,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
NAME = "branch"
|
NAME = "branch"
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for branches"""
|
"""
|
||||||
|
API Access class for branches
|
||||||
|
Branch resource is deprecated, please use model resource instead
|
||||||
|
"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, account, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
@@ -18,6 +29,7 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
self.schema = Branch
|
self.schema = Branch
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def create(
|
def create(
|
||||||
self, stream_id: str, name: str, description: str = "No description provided"
|
self, stream_id: str, name: str, description: str = "No description provided"
|
||||||
) -> str:
|
) -> str:
|
||||||
@@ -30,7 +42,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
id {str} -- the newly created branch's id
|
id {str} -- the newly created branch's id
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.BRANCH, self.account, {"name": "create"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation BranchCreate($branch: BranchCreateInput!) {
|
mutation BranchCreate($branch: BranchCreateInput!) {
|
||||||
@@ -38,6 +49,8 @@ class Resource(ResourceBase):
|
|||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
if len(name) < 3:
|
||||||
|
return SpeckleException(message="Branch Name must be at least 3 characters")
|
||||||
params = {
|
params = {
|
||||||
"branch": {
|
"branch": {
|
||||||
"streamId": stream_id,
|
"streamId": stream_id,
|
||||||
@@ -50,6 +63,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="branchCreate", parse_response=False
|
query=query, params=params, return_type="branchCreate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def get(self, stream_id: str, name: str, commits_limit: int = 10):
|
def get(self, stream_id: str, name: str, commits_limit: int = 10):
|
||||||
"""Get a branch by name from a stream
|
"""Get a branch by name from a stream
|
||||||
|
|
||||||
@@ -61,7 +75,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
Branch -- the fetched branch with its latest commits
|
Branch -- the fetched branch with its latest commits
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.BRANCH, self.account, {"name": "get"})
|
|
||||||
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 +99,7 @@ class Resource(ResourceBase):
|
|||||||
createdAt
|
createdAt
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
@@ -98,6 +111,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["stream", "branch"]
|
query=query, params=params, return_type=["stream", "branch"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def list(self, stream_id: str, branches_limit: int = 10, commits_limit: int = 10):
|
def list(self, stream_id: str, branches_limit: int = 10, commits_limit: int = 10):
|
||||||
"""Get a list of branches from a given stream
|
"""Get a list of branches from a given stream
|
||||||
|
|
||||||
@@ -109,10 +123,13 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
List[Branch] -- the branches on the stream
|
List[Branch] -- the branches on the stream
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.BRANCH, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query BranchesGet($stream_id: String!, $branches_limit: Int!, $commits_limit: Int!) {
|
query BranchesGet(
|
||||||
|
$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 {
|
||||||
@@ -150,8 +167,13 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["stream", "branches", "items"]
|
query=query, params=params, return_type=["stream", "branches", "items"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def update(
|
def update(
|
||||||
self, stream_id: str, branch_id: str, name: str = None, description: str = None
|
self,
|
||||||
|
stream_id: str,
|
||||||
|
branch_id: str,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
description: Optional[str] = None,
|
||||||
):
|
):
|
||||||
"""Update a branch
|
"""Update a branch
|
||||||
|
|
||||||
@@ -164,7 +186,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if update is successful
|
bool -- True if update is successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.BRANCH, self.account, {"name": "update"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation BranchUpdate($branch: BranchUpdateInput!) {
|
mutation BranchUpdate($branch: BranchUpdateInput!) {
|
||||||
@@ -188,6 +209,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="branchUpdate", parse_response=False
|
query=query, params=params, return_type="branchUpdate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def delete(self, stream_id: str, branch_id: str):
|
def delete(self, stream_id: str, branch_id: str):
|
||||||
"""Delete a branch
|
"""Delete a branch
|
||||||
|
|
||||||
@@ -198,7 +220,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if deletion is successful
|
bool -- True if deletion is successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.BRANCH, self.account, {"name": "delete"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation BranchDelete($branch: BranchDeleteInput!) {
|
mutation BranchDelete($branch: BranchDeleteInput!) {
|
||||||
+40
-21
@@ -1,15 +1,24 @@
|
|||||||
from typing import Optional, List
|
from typing import List, Optional, Union
|
||||||
from gql import gql
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.api.models import Commit
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
|
from gql import gql
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
Commit,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
NAME = "commit"
|
NAME = "commit"
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for commits"""
|
"""
|
||||||
|
API Access class for commits
|
||||||
|
Commit resource is deprecated, please use version resource instead
|
||||||
|
"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client) -> None:
|
def __init__(self, account, basepath, client) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
@@ -20,6 +29,7 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
self.schema = Commit
|
self.schema = Commit
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def get(self, stream_id: str, commit_id: str) -> Commit:
|
def get(self, stream_id: str, commit_id: str) -> Commit:
|
||||||
"""
|
"""
|
||||||
Gets a commit given a stream and the commit id
|
Gets a commit given a stream and the commit id
|
||||||
@@ -58,6 +68,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["stream", "commit"]
|
query=query, params=params, return_type=["stream", "commit"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def list(self, stream_id: str, limit: int = 10) -> List[Commit]:
|
def list(self, stream_id: str, limit: int = 10) -> List[Commit]:
|
||||||
"""
|
"""
|
||||||
Get a list of commits on a given stream
|
Get a list of commits on a given stream
|
||||||
@@ -69,7 +80,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
List[Commit] -- a list of the most recent commit objects
|
List[Commit] -- a list of the most recent commit objects
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.COMMIT, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query Commits($stream_id: String!, $limit: Int!) {
|
query Commits($stream_id: String!, $limit: Int!) {
|
||||||
@@ -100,6 +110,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["stream", "commits", "items"]
|
query=query, params=params, return_type=["stream", "commits", "items"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def create(
|
def create(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
@@ -107,26 +118,30 @@ class Resource(ResourceBase):
|
|||||||
branch_name: str = "main",
|
branch_name: str = "main",
|
||||||
message: str = "",
|
message: str = "",
|
||||||
source_application: str = "python",
|
source_application: str = "python",
|
||||||
parents: List[str] = None,
|
parents: Optional[List[str]] = None,
|
||||||
) -> str:
|
) -> Union[str, SpeckleException]:
|
||||||
"""
|
"""
|
||||||
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} -- the name of the branch to commit to (defaults to "main")
|
branch_name {str}
|
||||||
message {str} -- optional: a message to give more information about the commit
|
-- the name of the branch to commit to (defaults to "main")
|
||||||
source_application{str} -- optional: the application from which the commit was created (defaults to "python")
|
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
|
parents {List[str]} -- optional: the id of the parent commits
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
str -- the id of the created commit
|
str -- the id of the created commit
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.COMMIT, self.account, {"name": "create"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitCreate ($commit: CommitCreateInput!){ commitCreate(commit: $commit)}
|
mutation CommitCreate ($commit: CommitCreateInput!)
|
||||||
|
{ commitCreate(commit: $commit)}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {
|
params = {
|
||||||
@@ -145,22 +160,24 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="commitCreate", parse_response=False
|
query=query, params=params, return_type="commitCreate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def update(self, stream_id: str, commit_id: str, message: str) -> bool:
|
def update(self, stream_id: str, commit_id: str, message: str) -> bool:
|
||||||
"""
|
"""
|
||||||
Update a commit
|
Update a commit
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream that contains the commit you'd like to update
|
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
|
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
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
bool -- True if the operation succeeded
|
bool -- True if the operation succeeded
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.COMMIT, self.account, {"name": "update"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitUpdate($commit: CommitUpdateInput!){ commitUpdate(commit: $commit)}
|
mutation CommitUpdate($commit: CommitUpdateInput!)
|
||||||
|
{ commitUpdate(commit: $commit)}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {
|
params = {
|
||||||
@@ -171,21 +188,23 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="commitUpdate", parse_response=False
|
query=query, params=params, return_type="commitUpdate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def delete(self, stream_id: str, commit_id: str) -> bool:
|
def delete(self, stream_id: str, commit_id: str) -> bool:
|
||||||
"""
|
"""
|
||||||
Delete a commit
|
Delete a commit
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream that contains the commit you'd like to delete
|
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
|
commit_id {str} -- the id of the commit you'd like to delete
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
bool -- True if the operation succeeded
|
bool -- True if the operation succeeded
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.COMMIT, self.account, {"name": "delete"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitDelete($commit: CommitDeleteInput!){ commitDelete(commit: $commit)}
|
mutation CommitDelete($commit: CommitDeleteInput!)
|
||||||
|
{ commitDelete(commit: $commit)}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {"commit": {"streamId": stream_id, "id": commit_id}}
|
params = {"commit": {"streamId": stream_id, "id": commit_id}}
|
||||||
@@ -194,6 +213,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="commitDelete", parse_response=False
|
query=query, params=params, return_type="commitDelete", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def received(
|
def received(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
@@ -204,7 +224,6 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
Mark a commit object a received by the source application.
|
Mark a commit object a received by the source application.
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.COMMIT, self.account, {"name": "received"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation CommitReceive($receivedInput:CommitReceivedInput!){
|
mutation CommitReceive($receivedInput:CommitReceivedInput!){
|
||||||
+15
-6
@@ -1,6 +1,8 @@
|
|||||||
from typing import Dict, List
|
from typing import Dict, List
|
||||||
|
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
from specklepy.objects.base import Base
|
from specklepy.objects.base import Base
|
||||||
|
|
||||||
NAME = "object"
|
NAME = "object"
|
||||||
@@ -59,21 +61,28 @@ class Resource(ResourceBase):
|
|||||||
"""
|
"""
|
||||||
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. To send a base object, you can prepare it by running it through the
|
Create a new object on a stream.
|
||||||
`BaseObjectSerializer.traverse_base()` function to get a valid (serialisable) object to send.
|
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
|
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:
|
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]} -- a list of base dictionary objects (NOTE: must be json serialisable)
|
objects {List[Dict]}
|
||||||
|
-- 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!) { objectCreate(objectInput: $object_input) }
|
mutation ObjectCreate($object_input: ObjectCreateInput!) {
|
||||||
|
objectCreate(objectInput: $object_input)
|
||||||
|
}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
params = {"object_input": {"streamId": stream_id, "objects": objects}}
|
params = {"object_input": {"streamId": stream_id, "objects": objects}}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
from specklepy.core.api.resources import OtherUserResource
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(
|
||||||
|
reason="Class renamed to OtherUserResource", version=FE1_DEPRECATION_VERSION
|
||||||
|
)
|
||||||
|
class Resource(OtherUserResource):
|
||||||
|
"""
|
||||||
|
Class renamed to OtherUserResource
|
||||||
|
"""
|
||||||
|
|
||||||
|
pass
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
from deprecated import deprecated
|
||||||
|
|
||||||
|
from specklepy.core.api.models.deprecated import FE1_DEPRECATION_VERSION
|
||||||
|
from specklepy.core.api.resources import ServerResource
|
||||||
|
|
||||||
|
NAME = "server"
|
||||||
|
|
||||||
|
|
||||||
|
@deprecated(reason="Renamed to ServerResource", version=FE1_DEPRECATION_VERSION)
|
||||||
|
class Resource(ServerResource):
|
||||||
|
"""API Access class for the server"""
|
||||||
+144
-128
@@ -1,18 +1,29 @@
|
|||||||
from datetime import datetime, timezone
|
from datetime import datetime, timezone
|
||||||
from typing import List, Optional
|
from typing import List, Optional
|
||||||
|
|
||||||
from deprecated import deprecated
|
from deprecated import deprecated
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from specklepy.logging import metrics
|
|
||||||
from specklepy.api.models import ActivityCollection, PendingStreamCollaborator, Stream
|
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.logging.exceptions import UnsupportedException, SpeckleException
|
|
||||||
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ActivityCollection,
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
Stream,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
|
from specklepy.logging.exceptions import SpeckleException, UnsupportedException
|
||||||
|
|
||||||
NAME = "stream"
|
NAME = "stream"
|
||||||
|
|
||||||
|
|
||||||
class Resource(ResourceBase):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for streams"""
|
"""
|
||||||
|
API Access class for streams
|
||||||
|
Stream resource is deprecated, please use project resource instead
|
||||||
|
"""
|
||||||
|
|
||||||
def __init__(self, account, basepath, client, server_version) -> None:
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
super().__init__(
|
super().__init__(
|
||||||
@@ -25,6 +36,7 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
self.schema = Stream
|
self.schema = Stream
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def get(self, id: str, branch_limit: int = 10, commit_limit: int = 10) -> Stream:
|
def get(self, id: str, branch_limit: int = 10, commit_limit: int = 10) -> Stream:
|
||||||
"""Get the specified stream from the server
|
"""Get the specified stream from the server
|
||||||
|
|
||||||
@@ -36,7 +48,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
Stream -- the retrieved stream
|
Stream -- the retrieved stream
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query Stream($id: String!, $branch_limit: Int!, $commit_limit: Int!) {
|
query Stream($id: String!, $branch_limit: Int!, $commit_limit: Int!) {
|
||||||
@@ -87,6 +98,7 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="stream")
|
return self.make_request(query=query, params=params, return_type="stream")
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def list(self, stream_limit: int = 10) -> List[Stream]:
|
def list(self, stream_limit: int = 10) -> List[Stream]:
|
||||||
"""Get a list of the user's streams
|
"""Get a list of the user's streams
|
||||||
|
|
||||||
@@ -96,11 +108,10 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
List[Stream] -- A list of Stream objects
|
List[Stream] -- A list of Stream objects
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query User($stream_limit: Int!) {
|
query User($stream_limit: Int!) {
|
||||||
user {
|
activeUser {
|
||||||
id
|
id
|
||||||
bio
|
bio
|
||||||
name
|
name
|
||||||
@@ -138,9 +149,10 @@ class Resource(ResourceBase):
|
|||||||
params = {"stream_limit": stream_limit}
|
params = {"stream_limit": stream_limit}
|
||||||
|
|
||||||
return self.make_request(
|
return self.make_request(
|
||||||
query=query, params=params, return_type=["user", "streams", "items"]
|
query=query, params=params, return_type=["activeUser", "streams", "items"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def create(
|
def create(
|
||||||
self,
|
self,
|
||||||
name: str = "Anonymous Python Stream",
|
name: str = "Anonymous Python Stream",
|
||||||
@@ -152,12 +164,12 @@ class Resource(ResourceBase):
|
|||||||
Arguments:
|
Arguments:
|
||||||
name {str} -- the name of the string
|
name {str} -- the name of the string
|
||||||
description {str} -- a short description of the stream
|
description {str} -- a short description of the stream
|
||||||
is_public {bool} -- whether or not the stream can be viewed by anyone with the id
|
is_public {bool}
|
||||||
|
-- whether or not the stream can be viewed by anyone with the id
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
id {str} -- the id of the newly created stream
|
id {str} -- the id of the newly created stream
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "create"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamCreate($stream: StreamCreateInput!) {
|
mutation StreamCreate($stream: StreamCreateInput!) {
|
||||||
@@ -165,7 +177,8 @@ class Resource(ResourceBase):
|
|||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
if len(name) < 3 and len(name) != 0:
|
||||||
|
return SpeckleException(message="Stream Name must be at least 3 characters")
|
||||||
params = {
|
params = {
|
||||||
"stream": {"name": name, "description": description, "isPublic": is_public}
|
"stream": {"name": name, "description": description, "isPublic": is_public}
|
||||||
}
|
}
|
||||||
@@ -174,8 +187,13 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="streamCreate", parse_response=False
|
query=query, params=params, return_type="streamCreate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def update(
|
def update(
|
||||||
self, id: str, name: str = None, description: str = None, is_public: bool = None
|
self,
|
||||||
|
id: str,
|
||||||
|
name: Optional[str] = None,
|
||||||
|
description: Optional[str] = None,
|
||||||
|
is_public: Optional[bool] = None,
|
||||||
) -> bool:
|
) -> bool:
|
||||||
"""Update an existing stream
|
"""Update an existing stream
|
||||||
|
|
||||||
@@ -183,12 +201,12 @@ class Resource(ResourceBase):
|
|||||||
id {str} -- the id of the stream to be updated
|
id {str} -- the id of the stream to be updated
|
||||||
name {str} -- the name of the string
|
name {str} -- the name of the string
|
||||||
description {str} -- a short description of the stream
|
description {str} -- a short description of the stream
|
||||||
is_public {bool} -- whether or not the stream can be viewed by anyone with the id
|
is_public {bool}
|
||||||
|
-- whether or not the stream can be viewed by anyone with the id
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
bool -- whether the stream update was successful
|
bool -- whether the stream update was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "update"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamUpdate($stream: StreamUpdateInput!) {
|
mutation StreamUpdate($stream: StreamUpdateInput!) {
|
||||||
@@ -210,6 +228,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="streamUpdate", parse_response=False
|
query=query, params=params, return_type="streamUpdate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def delete(self, id: str) -> bool:
|
def delete(self, id: str) -> bool:
|
||||||
"""Delete a stream given its id
|
"""Delete a stream given its id
|
||||||
|
|
||||||
@@ -219,7 +238,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- whether the deletion was successful
|
bool -- whether the deletion was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "delete"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamDelete($id: String!) {
|
mutation StreamDelete($id: String!) {
|
||||||
@@ -234,6 +252,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type="streamDelete", parse_response=False
|
query=query, params=params, return_type="streamDelete", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def search(
|
def search(
|
||||||
self,
|
self,
|
||||||
search_query: str,
|
search_query: str,
|
||||||
@@ -252,10 +271,14 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
List[Stream] -- a list of Streams that match the search query
|
List[Stream] -- a list of Streams that match the search query
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "search"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query StreamSearch($search_query: String!,$limit: Int!, $branch_limit:Int!, $commit_limit:Int!) {
|
query StreamSearch(
|
||||||
|
$search_query: String!,
|
||||||
|
$limit: Int!,
|
||||||
|
$branch_limit:Int!,
|
||||||
|
$commit_limit:Int!
|
||||||
|
) {
|
||||||
streams(query: $search_query, limit: $limit) {
|
streams(query: $search_query, limit: $limit) {
|
||||||
items {
|
items {
|
||||||
id
|
id
|
||||||
@@ -309,17 +332,18 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["streams", "items"]
|
query=query, params=params, return_type=["streams", "items"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def favorite(self, stream_id: str, favorited: bool = True):
|
def favorite(self, stream_id: str, favorited: bool = True):
|
||||||
"""Favorite or unfavorite the given stream.
|
"""Favorite or unfavorite the given stream.
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream to favorite / unfavorite
|
stream_id {str} -- the id of the stream to favorite / unfavorite
|
||||||
favorited {bool} -- whether to favorite (True) or unfavorite (False) the stream
|
favorited {bool}
|
||||||
|
-- whether to favorite (True) or unfavorite (False) the stream
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Stream -- the stream with its `id`, `name`, and `favoritedDate`
|
Stream -- the stream with its `id`, `name`, and `favoritedDate`
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.STREAM, self.account, {"name": "favorite"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamFavorite($stream_id: String!, $favorited: Boolean!) {
|
mutation StreamFavorite($stream_id: String!, $favorited: Boolean!) {
|
||||||
@@ -342,59 +366,7 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["streamFavorite"]
|
query=query, params=params, return_type=["streamFavorite"]
|
||||||
)
|
)
|
||||||
|
|
||||||
@deprecated(
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
version="2.6.4",
|
|
||||||
reason=(
|
|
||||||
"As of Speckle Server v2.6.4, this method is deprecated. "
|
|
||||||
"Users need to be invited and accept the invite before being added to a stream"
|
|
||||||
),
|
|
||||||
)
|
|
||||||
def grant_permission(self, stream_id: str, user_id: str, role: str):
|
|
||||||
"""Grant permissions to a user on a given stream
|
|
||||||
|
|
||||||
Valid for Speckle Server version < 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.PERMISSION, self.account, {"name": "add", "role": role})
|
|
||||||
if self.server_version and self.server_version >= (2, 6, 4):
|
|
||||||
raise UnsupportedException(
|
|
||||||
(
|
|
||||||
"Server mutation `grant_permission` is no longer supported as of Speckle Server v2.6.4. "
|
|
||||||
"Please use the new `update_permission` method to change an existing user's permission "
|
|
||||||
"or use the `invite` method to invite a user to a stream."
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
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 get_all_pending_invites(
|
def get_all_pending_invites(
|
||||||
self, stream_id: str
|
self, stream_id: str
|
||||||
) -> List[PendingStreamCollaborator]:
|
) -> List[PendingStreamCollaborator]:
|
||||||
@@ -407,9 +379,9 @@ class Resource(ResourceBase):
|
|||||||
stream_id {str} -- the stream id from which to get the pending invites
|
stream_id {str} -- the stream id from which to get the pending invites
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
List[PendingStreamCollaborator] -- a list of pending invites for the specified stream
|
List[PendingStreamCollaborator]
|
||||||
|
-- a list of pending invites for the specified stream
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "get"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -422,19 +394,27 @@ class Resource(ResourceBase):
|
|||||||
inviteId
|
inviteId
|
||||||
streamId
|
streamId
|
||||||
streamName
|
streamName
|
||||||
|
projectName
|
||||||
|
projectId
|
||||||
title
|
title
|
||||||
role
|
role
|
||||||
invitedBy{
|
invitedBy{
|
||||||
id
|
id
|
||||||
name
|
name
|
||||||
|
bio
|
||||||
company
|
company
|
||||||
avatar
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
}
|
}
|
||||||
user {
|
user {
|
||||||
id
|
id
|
||||||
name
|
name
|
||||||
|
bio
|
||||||
company
|
company
|
||||||
avatar
|
avatar
|
||||||
|
verified
|
||||||
|
role
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -450,13 +430,14 @@ class Resource(ResourceBase):
|
|||||||
schema=PendingStreamCollaborator,
|
schema=PendingStreamCollaborator,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def invite(
|
def invite(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
email: str = None,
|
email: Optional[str] = None,
|
||||||
user_id: str = None,
|
user_id: Optional[str] = None,
|
||||||
role: str = "stream:contributor", # should default be reviewer?
|
role: str = "stream:contributor", # should default be reviewer?
|
||||||
message: str = None,
|
message: Optional[str] = None,
|
||||||
):
|
):
|
||||||
"""Invite someone to a stream using either their email or user id
|
"""Invite someone to a stream using either their email or user id
|
||||||
|
|
||||||
@@ -466,18 +447,20 @@ class Resource(ResourceBase):
|
|||||||
stream_id {str} -- the id of the stream to invite the user to
|
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`)
|
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`)
|
user_id {str} -- the id of the user to invite (use this OR `email`)
|
||||||
role {str} -- the role to assing to the user (defaults to `stream:contributor`)
|
role {str}
|
||||||
message {str} -- a message to send along with this invite to the specified user
|
-- 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:
|
Returns:
|
||||||
bool -- True if the operation was successful
|
bool -- True if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "create"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
if email is None and user_id is None:
|
if email is None and user_id is None:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"You must provide either an email or a user id to use the `stream.invite` method"
|
"You must provide either an email or a user id to use the"
|
||||||
|
" `stream.invite` method"
|
||||||
)
|
)
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -504,12 +487,13 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def invite_batch(
|
def invite_batch(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
emails: List[str] = None,
|
emails: Optional[List[str]] = None,
|
||||||
user_ids: List[None] = None,
|
user_ids: Optional[List[None]] = None,
|
||||||
message: str = None,
|
message: Optional[str] = None,
|
||||||
) -> bool:
|
) -> bool:
|
||||||
"""Invite a batch of users to a specified stream.
|
"""Invite a batch of users to a specified stream.
|
||||||
|
|
||||||
@@ -517,18 +501,21 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream to invite the user to
|
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`)
|
emails {List[str]}
|
||||||
user_id {List[str]} -- the id of the user to invite (use this and/or `emails`)
|
-- the email of the user to invite (use this and/or `user_ids`)
|
||||||
message {str} -- a message to send along with this invite to the specified user
|
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:
|
Returns:
|
||||||
bool -- True if the operation was successful
|
bool -- True if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "batch create"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
if emails is None and user_ids is None:
|
if emails is None and user_ids is None:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"You must provide either an email or a user id to use the `stream.invite` method"
|
"You must provide either an email or a user id to use the"
|
||||||
|
" `stream.invite` method"
|
||||||
)
|
)
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -541,14 +528,14 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
email_invites = [
|
email_invites = [
|
||||||
{"streamId": stream_id, "message": message, "email": email}
|
{"streamId": stream_id, "message": message, "email": email}
|
||||||
for email in emails
|
for email in (emails if emails is not None else [])
|
||||||
if emails is not None
|
if email is not None
|
||||||
]
|
]
|
||||||
|
|
||||||
user_invites = [
|
user_invites = [
|
||||||
{"streamId": stream_id, "message": message, "userId": user_id}
|
{"streamId": stream_id, "message": message, "userId": user_id}
|
||||||
for user_id in user_ids
|
for user_id in (user_ids if user_ids is not None else [])
|
||||||
if user_ids is not None
|
if user_id is not None
|
||||||
]
|
]
|
||||||
|
|
||||||
params = {"input": [*email_invites, *user_invites]}
|
params = {"input": [*email_invites, *user_invites]}
|
||||||
@@ -560,6 +547,7 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def invite_cancel(self, stream_id: str, invite_id: str) -> bool:
|
def invite_cancel(self, stream_id: str, invite_id: str) -> bool:
|
||||||
"""Cancel an existing stream invite
|
"""Cancel an existing stream invite
|
||||||
|
|
||||||
@@ -572,7 +560,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- true if the operation was successful
|
bool -- true if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "cancel"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -592,25 +579,30 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def invite_use(self, stream_id: str, token: str, accept: bool = True) -> bool:
|
def invite_use(self, stream_id: str, token: str, accept: bool = True) -> bool:
|
||||||
"""Accept or decline a stream invite
|
"""Accept or decline a stream invite
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
Requires Speckle Server version >= 2.6.4
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
stream_id {str} -- the id of the stream for which the user has a pending invite
|
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
|
token {str} -- the token of the invite to use
|
||||||
accept {bool} -- whether or not to accept the invite (defaults to True)
|
accept {bool} -- whether or not to accept the invite (defaults to True)
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
bool -- true if the operation was successful
|
bool -- true if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "use"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamInviteUse($accept: Boolean!, $streamId: String!, $token: String!) {
|
mutation StreamInviteUse(
|
||||||
|
$accept: Boolean!,
|
||||||
|
$streamId: String!,
|
||||||
|
$token: String!
|
||||||
|
) {
|
||||||
streamInviteUse(accept: $accept, streamId: $streamId, token: $token)
|
streamInviteUse(accept: $accept, streamId: $streamId, token: $token)
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
@@ -625,6 +617,7 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def update_permission(self, stream_id: str, user_id: str, role: str):
|
def update_permission(self, stream_id: str, user_id: str, role: str):
|
||||||
"""Updates permissions for a user on a given stream
|
"""Updates permissions for a user on a given stream
|
||||||
|
|
||||||
@@ -638,19 +631,19 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if the operation was successful
|
bool -- True if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(
|
if self.server_version and (
|
||||||
metrics.PERMISSION, self.account, {"name": "update", "role": role}
|
self.server_version != ("dev",) and self.server_version < (2, 6, 4)
|
||||||
)
|
):
|
||||||
if self.server_version and self.server_version < (2, 6, 4):
|
|
||||||
raise UnsupportedException(
|
raise UnsupportedException(
|
||||||
(
|
"Server mutation `update_permission` is only supported as of Speckle"
|
||||||
"Server mutation `update_permission` is only supported as of Speckle Server v2.6.4. "
|
" Server v2.6.4. Please update your Speckle Server to use this method"
|
||||||
"Please update your Speckle Server to use this method or use the `grant_permission` method instead."
|
" or use the `grant_permission` method instead."
|
||||||
)
|
|
||||||
)
|
)
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamUpdatePermission($permission_params: StreamUpdatePermissionInput !) {
|
mutation StreamUpdatePermission(
|
||||||
|
$permission_params: StreamUpdatePermissionInput!
|
||||||
|
) {
|
||||||
streamUpdatePermission(permissionParams: $permission_params)
|
streamUpdatePermission(permissionParams: $permission_params)
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
@@ -671,6 +664,7 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def revoke_permission(self, stream_id: str, user_id: str):
|
def revoke_permission(self, stream_id: str, user_id: str):
|
||||||
"""Revoke permissions from a user on a given stream
|
"""Revoke permissions from a user on a given stream
|
||||||
|
|
||||||
@@ -681,10 +675,11 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if the operation was successful
|
bool -- True if the operation was successful
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.PERMISSION, self.account, {"name": "revoke"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation StreamRevokePermission($permission_params: StreamRevokePermissionInput !) {
|
mutation StreamRevokePermission(
|
||||||
|
$permission_params: StreamRevokePermissionInput!
|
||||||
|
) {
|
||||||
streamRevokePermission(permissionParams: $permission_params)
|
streamRevokePermission(permissionParams: $permission_params)
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
@@ -699,32 +694,52 @@ class Resource(ResourceBase):
|
|||||||
parse_response=False,
|
parse_response=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
def activity(
|
def activity(
|
||||||
self,
|
self,
|
||||||
stream_id: str,
|
stream_id: str,
|
||||||
action_type: str = None,
|
action_type: Optional[str] = None,
|
||||||
limit: int = 20,
|
limit: int = 20,
|
||||||
before: datetime = None,
|
before: Optional[datetime] = None,
|
||||||
after: datetime = None,
|
after: Optional[datetime] = None,
|
||||||
cursor: 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.
|
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
|
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
|
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`)
|
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
|
limit {int} -- max number of Activity items to return
|
||||||
before {datetime} -- latest cutoff for activity (ie: return all activity _before_ this time)
|
before {datetime}
|
||||||
after {datetime} -- oldest cutoff for activity (ie: return all activity _after_ this time)
|
-- 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
|
cursor {datetime} -- timestamp cursor for pagination
|
||||||
"""
|
"""
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query StreamActivity($stream_id: String!, $action_type: String, $before:DateTime, $after: DateTime, $cursor: DateTime, $limit: Int){
|
query StreamActivity(
|
||||||
|
$stream_id: String!,
|
||||||
|
$action_type: String,
|
||||||
|
$before:DateTime,
|
||||||
|
$after: DateTime,
|
||||||
|
$cursor: DateTime,
|
||||||
|
$limit: Int
|
||||||
|
){
|
||||||
stream(id: $stream_id) {
|
stream(id: $stream_id) {
|
||||||
activity(actionType: $action_type, before: $before, after: $after, cursor: $cursor, limit: $limit) {
|
activity(
|
||||||
|
actionType: $action_type,
|
||||||
|
before: $before,
|
||||||
|
after: $after,
|
||||||
|
cursor: $cursor,
|
||||||
|
limit: $limit
|
||||||
|
) {
|
||||||
totalCount
|
totalCount
|
||||||
cursor
|
cursor
|
||||||
items {
|
items {
|
||||||
@@ -747,18 +762,19 @@ class Resource(ResourceBase):
|
|||||||
"stream_id": stream_id,
|
"stream_id": stream_id,
|
||||||
"limit": limit,
|
"limit": limit,
|
||||||
"action_type": action_type,
|
"action_type": action_type,
|
||||||
"before": before.astimezone(timezone.utc).isoformat()
|
"before": (
|
||||||
if before
|
before.astimezone(timezone.utc).isoformat() if before else before
|
||||||
else before,
|
),
|
||||||
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
"after": after.astimezone(timezone.utc).isoformat() if after else after,
|
||||||
"cursor": cursor.astimezone(timezone.utc).isoformat()
|
"cursor": (
|
||||||
if cursor
|
cursor.astimezone(timezone.utc).isoformat() if cursor else cursor
|
||||||
else cursor,
|
),
|
||||||
}
|
}
|
||||||
except AttributeError as e:
|
except AttributeError as e:
|
||||||
raise SpeckleException(
|
raise SpeckleException(
|
||||||
"Could not get stream activity - `before`, `after`, and `cursor` must be in `datetime` format if provided",
|
"Could not get stream activity - `before`, `after`, and `cursor` must"
|
||||||
ValueError,
|
" be in `datetime` format if provided",
|
||||||
|
ValueError(),
|
||||||
) from e
|
) from e
|
||||||
|
|
||||||
return self.make_request(
|
return self.make_request(
|
||||||
+35
-15
@@ -1,9 +1,16 @@
|
|||||||
from typing import Callable, Dict, List, Union
|
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
|
from typing import Callable, Dict, List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from graphql import DocumentNode
|
from graphql import DocumentNode
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.api.resources.stream import Stream
|
from specklepy.core.api.models.deprecated import (
|
||||||
|
FE1_DEPRECATION_REASON,
|
||||||
|
FE1_DEPRECATION_VERSION,
|
||||||
|
Stream,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
NAME = "subscribe"
|
NAME = "subscribe"
|
||||||
@@ -33,12 +40,15 @@ class Resource(ResourceBase):
|
|||||||
name=NAME,
|
name=NAME,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(reason=FE1_DEPRECATION_REASON, version=FE1_DEPRECATION_VERSION)
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_added(self, callback: Callable = None):
|
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.
|
"""Subscribes to new stream added event for your profile.
|
||||||
|
Use this to display an up-to-date list of streams.
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
callback {Callable[Stream]} -- a function that takes the updated stream as an argument and executes each time a stream is added
|
callback {Callable[Stream]} -- a function that takes the updated stream
|
||||||
|
as an argument and executes each time a stream is added
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Stream -- the update stream
|
Stream -- the update stream
|
||||||
@@ -53,12 +63,16 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_updated(self, id: str, callback: Callable = None):
|
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.
|
"""
|
||||||
|
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]} -- a function that takes the updated stream as an argument and executes each time the stream is updated
|
callback {Callable[Stream]}
|
||||||
|
-- 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
|
||||||
@@ -79,11 +93,17 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
@check_wsclient
|
@check_wsclient
|
||||||
async def stream_removed(self, callback: Callable = None):
|
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.
|
"""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:
|
Arguments:
|
||||||
callback {Callable[Dict]} -- a function that takes the returned dict as an argument and executes each time a stream is removed
|
callback {Callable[Dict]}
|
||||||
|
-- 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'
|
||||||
@@ -105,9 +125,9 @@ class Resource(ResourceBase):
|
|||||||
async def subscribe(
|
async def subscribe(
|
||||||
self,
|
self,
|
||||||
query: DocumentNode,
|
query: DocumentNode,
|
||||||
params: Dict = None,
|
params: Optional[Dict] = None,
|
||||||
callback: Callable = None,
|
callback: Optional[Callable] = None,
|
||||||
return_type: Union[str, List] = None,
|
return_type: Optional[Union[str, List]] = None,
|
||||||
schema=None,
|
schema=None,
|
||||||
parse_response: bool = True,
|
parse_response: bool = True,
|
||||||
):
|
):
|
||||||
+75
-30
@@ -1,13 +1,25 @@
|
|||||||
from typing import List, Optional, Union
|
|
||||||
from datetime import datetime, timezone
|
from datetime import datetime, timezone
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from deprecated import deprecated
|
||||||
from gql import gql
|
from gql import gql
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
ActivityCollection,
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
User,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resource import ResourceBase
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.api.resource import ResourceBase
|
|
||||||
from specklepy.api.models import ActivityCollection, PendingStreamCollaborator, User
|
|
||||||
|
|
||||||
NAME = "user"
|
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):
|
class Resource(ResourceBase):
|
||||||
"""API Access class for users"""
|
"""API Access class for users"""
|
||||||
@@ -22,8 +34,12 @@ class Resource(ResourceBase):
|
|||||||
)
|
)
|
||||||
self.schema = User
|
self.schema = User
|
||||||
|
|
||||||
def get(self, id: str = None) -> User:
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
"""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).
|
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:
|
Arguments:
|
||||||
id {str} -- the user id
|
id {str} -- the user id
|
||||||
@@ -31,7 +47,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
User -- the retrieved user
|
User -- the retrieved user
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.USER, self.account, {"name": "get"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query User($id: String) {
|
query User($id: String) {
|
||||||
@@ -54,10 +69,13 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
return self.make_request(query=query, params=params, return_type="user")
|
return self.make_request(query=query, params=params, return_type="user")
|
||||||
|
|
||||||
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
def search(
|
def search(
|
||||||
self, search_query: str, limit: int = 25
|
self, search_query: str, limit: int = 25
|
||||||
) -> Union[List[User], SpeckleException]:
|
) -> Union[List[User], SpeckleException]:
|
||||||
"""Searches for user by name or email. The search query must be at least 3 characters long
|
"""
|
||||||
|
Searches for user by name or email.
|
||||||
|
The search query must be at least 3 characters long
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
search_query {str} -- a string to search for
|
search_query {str} -- a string to search for
|
||||||
@@ -70,7 +88,6 @@ class Resource(ResourceBase):
|
|||||||
message="User search query must be at least 3 characters"
|
message="User search query must be at least 3 characters"
|
||||||
)
|
)
|
||||||
|
|
||||||
metrics.track(metrics.USER, self.account, {"name": "search"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query UserSearch($search_query: String!, $limit: Int!) {
|
query UserSearch($search_query: String!, $limit: Int!) {
|
||||||
@@ -93,8 +110,13 @@ class Resource(ResourceBase):
|
|||||||
query=query, params=params, return_type=["userSearch", "items"]
|
query=query, params=params, return_type=["userSearch", "items"]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
def update(
|
def update(
|
||||||
self, name: str = None, company: str = None, bio: str = None, avatar: str = None
|
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.
|
"""Updates your user profile. All arguments are optional.
|
||||||
|
|
||||||
@@ -107,7 +129,6 @@ class Resource(ResourceBase):
|
|||||||
Returns:
|
Returns:
|
||||||
bool -- True if your profile was updated successfully
|
bool -- True if your profile was updated successfully
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.USER, self.account, {"name": "update"})
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation UserUpdate($user: UserUpdateInput!) {
|
mutation UserUpdate($user: UserUpdateInput!) {
|
||||||
@@ -121,41 +142,63 @@ class Resource(ResourceBase):
|
|||||||
|
|
||||||
if not params["user"]:
|
if not params["user"]:
|
||||||
return SpeckleException(
|
return SpeckleException(
|
||||||
message="You must provide at least one field to update your user profile"
|
message=(
|
||||||
|
"You must provide at least one field to update your user profile"
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.make_request(
|
return self.make_request(
|
||||||
query=query, params=params, return_type="userUpdate", parse_response=False
|
query=query, params=params, return_type="userUpdate", parse_response=False
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
def activity(
|
def activity(
|
||||||
self,
|
self,
|
||||||
user_id: str = None,
|
user_id: Optional[str] = None,
|
||||||
limit: int = 20,
|
limit: int = 20,
|
||||||
action_type: str = None,
|
action_type: Optional[str] = None,
|
||||||
before: datetime = None,
|
before: Optional[datetime] = None,
|
||||||
after: datetime = None,
|
after: Optional[datetime] = None,
|
||||||
cursor: 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.
|
Get the activity from a given stream in an Activity collection.
|
||||||
If no id argument is provided, will return the current authenticated user's activity (as extracted from the authorization header).
|
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
|
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
|
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`)
|
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
|
limit {int} -- max number of Activity items to return
|
||||||
before {datetime} -- latest cutoff for activity (ie: return all activity _before_ this time)
|
before {datetime}
|
||||||
after {datetime} -- oldest cutoff for activity (ie: return all activity _after_ this time)
|
-- 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
|
cursor {datetime} -- timestamp cursor for pagination
|
||||||
"""
|
"""
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
query UserActivity($user_id: String, $action_type: String, $before:DateTime, $after: DateTime, $cursor: DateTime, $limit: Int){
|
query UserActivity(
|
||||||
|
$user_id: String,
|
||||||
|
$action_type: String,
|
||||||
|
$before:DateTime,
|
||||||
|
$after: DateTime,
|
||||||
|
$cursor: DateTime,
|
||||||
|
$limit: Int
|
||||||
|
){
|
||||||
user(id: $user_id) {
|
user(id: $user_id) {
|
||||||
activity(actionType: $action_type, before: $before, after: $after, cursor: $cursor, limit: $limit) {
|
activity(
|
||||||
|
actionType: $action_type,
|
||||||
|
before: $before,
|
||||||
|
after: $after,
|
||||||
|
cursor: $cursor,
|
||||||
|
limit: $limit
|
||||||
|
) {
|
||||||
totalCount
|
totalCount
|
||||||
cursor
|
cursor
|
||||||
items {
|
items {
|
||||||
@@ -190,15 +233,16 @@ class Resource(ResourceBase):
|
|||||||
schema=ActivityCollection,
|
schema=ActivityCollection,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
def get_all_pending_invites(self) -> List[PendingStreamCollaborator]:
|
||||||
"""Get all of the active user's pending stream invites
|
"""Get all of the active user's pending stream invites
|
||||||
|
|
||||||
Requires Speckle Server version >= 2.6.4
|
Requires Speckle Server version >= 2.6.4
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
List[PendingStreamCollaborator] -- a list of pending invites for the current user
|
List[PendingStreamCollaborator]
|
||||||
|
-- a list of pending invites for the current user
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "get"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -229,8 +273,9 @@ class Resource(ResourceBase):
|
|||||||
schema=PendingStreamCollaborator,
|
schema=PendingStreamCollaborator,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT)
|
||||||
def get_pending_invite(
|
def get_pending_invite(
|
||||||
self, stream_id: str, token: str = None
|
self, stream_id: str, token: Optional[str] = None
|
||||||
) -> Optional[PendingStreamCollaborator]:
|
) -> Optional[PendingStreamCollaborator]:
|
||||||
"""Get a particular pending invite for the active user on a given stream.
|
"""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.
|
If no invite_id is provided, any valid invite will be returned.
|
||||||
@@ -242,9 +287,9 @@ class Resource(ResourceBase):
|
|||||||
token {str} -- the token of the invite to look for (optional)
|
token {str} -- the token of the invite to look for (optional)
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
PendingStreamCollaborator -- the invite for the given stream (or None if it isn't found)
|
PendingStreamCollaborator
|
||||||
|
-- the invite for the given stream (or None if it isn't found)
|
||||||
"""
|
"""
|
||||||
metrics.track(metrics.INVITE, self.account, {"name": "get"})
|
|
||||||
self._check_invites_supported()
|
self._check_invites_supported()
|
||||||
|
|
||||||
query = gql(
|
query = gql(
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
schema: https://app.speckle.systems/graphql
|
||||||
|
documents: '**/*.graphql'
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user