Compare commits
239 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c04a97780c | |||
| 309c78da37 | |||
| ff812d5ad9 | |||
| 8edc0d5d78 | |||
| 78b3e99475 | |||
| ac9e081d49 | |||
| 4bc95441b9 | |||
| 0d74848b68 | |||
| 8a76006f9e | |||
| af42b09dd5 | |||
| e4453f0b04 | |||
| c9a0e45171 | |||
| f20fc7edb3 | |||
| 0cd0c3a1f6 | |||
| 2594ce0382 | |||
| ec67f5ba48 | |||
| db61d2e99c | |||
| 69090f6eb1 | |||
| 99f0b3516a | |||
| f69ee07a94 | |||
| 1d246c921a | |||
| 80b5982424 | |||
| d06f0b5b4e | |||
| a6790c7c70 | |||
| 7bc78b6bf9 | |||
| f584ad84ed | |||
| 55bc1b2fa5 | |||
| 87720c1d6c | |||
| ed8df12e54 | |||
| a8a5296d7e | |||
| 4f82c0f43d | |||
| f5e024c8ce | |||
| 3bcdf723b0 | |||
| adc1105b3a | |||
| fa9877b6da | |||
| 2929e2f93b | |||
| 6636950705 | |||
| 79c0106f57 | |||
| f4d73ff1ae | |||
| 7ea719141f | |||
| a47f568f69 | |||
| b174802451 | |||
| 87a7e7482d | |||
| e888339dda | |||
| 3417557405 | |||
| 8aba21de01 | |||
| 4ce61f4e89 | |||
| 6d6e1e7650 | |||
| 95de5cbb30 | |||
| 5f56818d63 | |||
| 825097e1a6 | |||
| d3ab26240a | |||
| ce6be1a98e | |||
| 213e73dfdd | |||
| 15129df7ce | |||
| 88519ce8b0 | |||
| d4f94450a5 | |||
| 4c46201526 | |||
| 75b064b3c7 | |||
| 1198f2e2ad | |||
| 7ab787bfb1 | |||
| bbbf373b50 | |||
| f34e4a2874 | |||
| 45ebc375ad | |||
| 4c41fa79fc | |||
| 0aa14ca077 | |||
| 6bfdf8850c | |||
| 22ecd2c2b3 | |||
| f7f9f73e7b | |||
| a7bada391b | |||
| 81ff5d82cb | |||
| d25edbb3d7 | |||
| 7dedff68f4 | |||
| 12b9602577 | |||
| d6e31a9752 | |||
| 09c61424d7 | |||
| e9bdf0ceb8 | |||
| 7e6174ebc1 | |||
| b8ae3ca8c8 | |||
| d690c45b35 | |||
| 5d3a824986 | |||
| 6f56ecb0c0 | |||
| c5cd69569e | |||
| e38249bc38 | |||
| 08fbf59c8a | |||
| e9cdd3e900 | |||
| 20bb0449e8 | |||
| ef5a570dd4 | |||
| 424d7d9caf | |||
| 6aa643837a | |||
| 32cbb33e10 | |||
| 51ae6f5978 | |||
| b64dde152a | |||
| d1b6755997 | |||
| da6e2d92e0 | |||
| 37e9c2372f | |||
| a620a358d3 | |||
| fd46fbd961 | |||
| 732f28e653 | |||
| 7671998541 | |||
| cab9674803 | |||
| 6c33c61a6d | |||
| 71afb1275f | |||
| 1b53410a86 | |||
| 1ba6983573 | |||
| d5a36fa5e3 | |||
| b6e47fb820 | |||
| 06e21154c4 | |||
| adc0c40ab7 | |||
| a44bb92ec4 | |||
| bd98244869 | |||
| 2acfa48b98 | |||
| a0283b6048 | |||
| 0e771a68b8 | |||
| 838f9d4969 | |||
| 88b17db901 | |||
| f98c804094 | |||
| 0382c246b8 | |||
| 0b38fb5a2a | |||
| 405972f681 | |||
| ff686b4361 | |||
| 7857451ec9 | |||
| 0fbfff54d4 | |||
| 826dadc8c8 | |||
| b9e4ee2b23 | |||
| 78c55b787f | |||
| 34f2dc2ab6 | |||
| a658e12cda | |||
| 85aa938fc2 | |||
| 010fb83ea6 | |||
| 7a291ce2f6 | |||
| 989c975c86 | |||
| 516eff4d8b | |||
| 0650210601 | |||
| b0b8140363 | |||
| d25f30b20d | |||
| b4e2f37b7f | |||
| b7ba2196f3 | |||
| 17cbcc38ba | |||
| 9afb2c5c1c | |||
| eb13c9bc70 | |||
| a33588f3af | |||
| 970cf62e50 | |||
| 513594c49f | |||
| 37c8e6dfb1 | |||
| 3859a88c4b | |||
| dfa8fc99d9 | |||
| ee97f3b718 | |||
| e0b48f6123 | |||
| 6fb6418d16 | |||
| ce104adb50 | |||
| fe0a8eb9f5 | |||
| 6279dd3885 | |||
| 811c5843a9 | |||
| 035cd089e2 | |||
| 6daef049bb | |||
| d526c8ce3e | |||
| 4c91032718 | |||
| ffb80457bc | |||
| d380e6eaf8 | |||
| ace7c390c1 | |||
| c052dfad46 | |||
| 66802726b9 | |||
| b8f4150fb7 | |||
| 255133010f | |||
| aea9bb3e1d | |||
| 5ca5334730 | |||
| ba5f40a749 | |||
| 04fc0fa715 | |||
| 2e80646d2c | |||
| fe6c18e97b | |||
| 7c9058172f | |||
| a82187589f | |||
| d811b010ff | |||
| e1e5d9dbb6 | |||
| 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 | |||
| 537a504121 | |||
| 6c03dc82c8 | |||
| 780126528d |
+9
-100
@@ -1,108 +1,17 @@
|
|||||||
version: 2.1
|
version: 2.1
|
||||||
|
|
||||||
orbs:
|
# Define the jobs we want to run for this project
|
||||||
codecov: codecov/codecov@3.3.0
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
pre-commit:
|
build:
|
||||||
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: speckle/pre-commit-runner:latest
|
- image: cimg/base:2023.03
|
||||||
resource_class: medium
|
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- run: echo "so long and thanks for all the fish"
|
||||||
- restore_cache:
|
|
||||||
keys:
|
|
||||||
- cache-pre-commit-<<parameters.cache_prefix>>-{{ checksum "<<parameters.config_file>>" }}
|
|
||||||
- run:
|
|
||||||
name: Install pre-commit hooks
|
|
||||||
command: pre-commit install-hooks --config <<parameters.config_file>>
|
|
||||||
- save_cache:
|
|
||||||
key: cache-pre-commit-<<parameters.cache_prefix>>-{{ checksum "<<parameters.config_file>>" }}
|
|
||||||
paths:
|
|
||||||
- ~/.cache/pre-commit
|
|
||||||
- run:
|
|
||||||
name: Run pre-commit
|
|
||||||
command: pre-commit run --all-files
|
|
||||||
- run:
|
|
||||||
command: git --no-pager diff
|
|
||||||
name: git diff
|
|
||||||
when: on_fail
|
|
||||||
|
|
||||||
test:
|
|
||||||
machine:
|
|
||||||
image: ubuntu-2204:2023.02.1
|
|
||||||
docker_layer_caching: false
|
|
||||||
resource_class: medium
|
|
||||||
parameters:
|
|
||||||
tag:
|
|
||||||
default: "3.11"
|
|
||||||
type: string
|
|
||||||
steps:
|
|
||||||
- checkout
|
|
||||||
- run:
|
|
||||||
name: Install python
|
|
||||||
command: |
|
|
||||||
pyenv install -s << parameters.tag >>
|
|
||||||
pyenv global << parameters.tag >>
|
|
||||||
- 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:
|
|
||||||
path: reports
|
|
||||||
- store_artifacts:
|
|
||||||
path: reports
|
|
||||||
- codecov/upload
|
|
||||||
|
|
||||||
deploy:
|
|
||||||
docker:
|
|
||||||
- image: "cimg/python:3.8"
|
|
||||||
steps:
|
|
||||||
- checkout
|
|
||||||
- run: python patch_version.py $CIRCLE_TAG
|
|
||||||
- run: poetry build
|
|
||||||
- run: poetry publish -u __token__ -p $PYPI_TOKEN
|
|
||||||
|
|
||||||
|
# Orchestrate our job run sequence
|
||||||
workflows:
|
workflows:
|
||||||
main:
|
build_and_test:
|
||||||
|
when:
|
||||||
|
false
|
||||||
jobs:
|
jobs:
|
||||||
- pre-commit:
|
- build
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
- test:
|
|
||||||
matrix:
|
|
||||||
parameters:
|
|
||||||
tag: ["3.11"]
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
- deploy:
|
|
||||||
context: pypi
|
|
||||||
requires:
|
|
||||||
- pre-commit
|
|
||||||
- test
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /[0-9]+(\.[0-9]+)*/
|
|
||||||
branches:
|
|
||||||
ignore: /.*/ # For testing only! /ci\/.*/
|
|
||||||
|
|||||||
@@ -1,27 +0,0 @@
|
|||||||
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/blob/main/containers/python-3/.devcontainer/base.Dockerfile
|
|
||||||
|
|
||||||
# [Choice] Python version: 3, 3.9, 3.8, 3.7, 3.6
|
|
||||||
ARG VARIANT="3.10"
|
|
||||||
FROM mcr.microsoft.com/vscode/devcontainers/python:${VARIANT}
|
|
||||||
|
|
||||||
# [Choice] Node.js version: none, lts/*, 16, 14, 12, 10
|
|
||||||
ARG NODE_VERSION="16"
|
|
||||||
RUN if [ "${NODE_VERSION}" != "none" ]; then su vscode -c "umask 0002 && . /usr/local/share/nvm/nvm.sh && nvm install ${NODE_VERSION} 2>&1"; fi
|
|
||||||
|
|
||||||
# [Optional] If your pip requirements rarely change, uncomment this section to add them to the image.
|
|
||||||
# COPY requirements.txt /tmp/pip-tmp/
|
|
||||||
# RUN pip3 --disable-pip-version-check --no-cache-dir install -r /tmp/pip-tmp/requirements.txt \
|
|
||||||
# && rm -rf /tmp/pip-tmp
|
|
||||||
|
|
||||||
# [Optional] Uncomment this section to install additional OS packages.
|
|
||||||
# RUN apt-get update && export DEBIAN_FRONTEND=noninteractive \
|
|
||||||
# && apt-get -y install --no-install-recommends <your-package-list-here>
|
|
||||||
|
|
||||||
# [Optional] Uncomment this line to install global node packages.
|
|
||||||
# RUN su vscode -c "source /usr/local/share/nvm/nvm.sh && npm install -g <your-package-here>" 2>&1
|
|
||||||
|
|
||||||
USER vscode
|
|
||||||
|
|
||||||
RUN curl -sSL https://install.python-poetry.org | python3 -
|
|
||||||
|
|
||||||
ENV PATH=$PATH:$HOME/.poetry/env
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
// For format details, see https://aka.ms/devcontainer.json. For config options, see the README at:
|
|
||||||
// https://github.com/microsoft/vscode-dev-containers/tree/v0.191.1/containers/python-3
|
|
||||||
{
|
|
||||||
"name": "Python 3",
|
|
||||||
// "build": {
|
|
||||||
// "dockerfile": "Dockerfile",
|
|
||||||
// "context": "..",
|
|
||||||
// "args": {
|
|
||||||
// // Update 'VARIANT' to pick a Python version: 3, 3.6, 3.7, 3.8, 3.9
|
|
||||||
// "VARIANT": "3.6",
|
|
||||||
// // Options
|
|
||||||
// "NODE_VERSION": "lts/*"
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
"dockerComposeFile": "./docker-compose.yaml",
|
|
||||||
"service": "specklepy",
|
|
||||||
"workspaceFolder": "/workspaces/specklepy",
|
|
||||||
"shutdownAction": "stopCompose",
|
|
||||||
// Set *default* container specific settings.json values on container create.
|
|
||||||
"settings": {
|
|
||||||
"python.pythonPath": "/usr/local/bin/python",
|
|
||||||
"python.languageServer": "Pylance",
|
|
||||||
"python.linting.enabled": true,
|
|
||||||
"python.linting.pylintEnabled": true,
|
|
||||||
"python.linting.pylintArgs": [
|
|
||||||
"--max-line-length=120"
|
|
||||||
],
|
|
||||||
"python.formatting.autopep8Path": "/usr/local/py-utils/bin/autopep8",
|
|
||||||
"python.formatting.blackPath": "/usr/local/py-utils/bin/black",
|
|
||||||
"python.formatting.yapfPath": "/usr/local/py-utils/bin/yapf",
|
|
||||||
"python.linting.banditPath": "/usr/local/py-utils/bin/bandit",
|
|
||||||
"python.linting.flake8Path": "/usr/local/py-utils/bin/flake8",
|
|
||||||
"python.linting.mypyPath": "/usr/local/py-utils/bin/mypy",
|
|
||||||
"python.linting.pycodestylePath": "/usr/local/py-utils/bin/pycodestyle",
|
|
||||||
"python.linting.pydocstylePath": "/usr/local/py-utils/bin/pydocstyle",
|
|
||||||
"python.linting.pylintPath": "/usr/local/py-utils/bin/pylint",
|
|
||||||
"python.testing.pytestArgs": [
|
|
||||||
"tests/",
|
|
||||||
"-s"
|
|
||||||
],
|
|
||||||
"python.testing.pytestEnabled": true,
|
|
||||||
"editor.formatOnSave": true,
|
|
||||||
},
|
|
||||||
// Add the IDs of extensions you want installed when the container is created.
|
|
||||||
"extensions": [
|
|
||||||
"ms-python.python",
|
|
||||||
"ms-python.vscode-pylance"
|
|
||||||
],
|
|
||||||
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
|
||||||
// "forwardPorts": [],
|
|
||||||
// Use 'postCreateCommand' to run commands after the container is created.
|
|
||||||
"postCreateCommand": "poetry config virtualenvs.create false && poetry install",
|
|
||||||
// Comment out connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
|
||||||
"remoteUser": "vscode"
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
version: "3.3" # optional since v1.27.0
|
|
||||||
services:
|
|
||||||
postgres:
|
|
||||||
image: cimg/postgres:14.2
|
|
||||||
environment:
|
|
||||||
POSTGRES_DB: speckle2_test
|
|
||||||
POSTGRES_PASSWORD: speckle
|
|
||||||
POSTGRES_USER: speckle
|
|
||||||
network_mode: host
|
|
||||||
redis:
|
|
||||||
image: cimg/redis:6.2
|
|
||||||
network_mode: host
|
|
||||||
speckle-server:
|
|
||||||
image: speckle/speckle-server:latest
|
|
||||||
command: ["bash", "-c", "/wait && node bin/www"]
|
|
||||||
environment:
|
|
||||||
POSTGRES_URL: "localhost"
|
|
||||||
POSTGRES_USER: "speckle"
|
|
||||||
POSTGRES_PASSWORD: "speckle"
|
|
||||||
POSTGRES_DB: "speckle2_test"
|
|
||||||
REDIS_URL: "redis://localhost"
|
|
||||||
SESSION_SECRET: "keyboard cat"
|
|
||||||
STRATEGY_LOCAL: "true"
|
|
||||||
CANONICAL_URL: "http://localhost:3000"
|
|
||||||
WAIT_HOSTS: localhost:5432, localhost:6379
|
|
||||||
DISABLE_FILE_UPLOADS: "true"
|
|
||||||
network_mode: host
|
|
||||||
|
|
||||||
specklepy:
|
|
||||||
build:
|
|
||||||
dockerfile: Dockerfile
|
|
||||||
context: .
|
|
||||||
args:
|
|
||||||
VARIANT: 3.9
|
|
||||||
NODE_VERSION: lts/*
|
|
||||||
volumes:
|
|
||||||
# Mounts the project folder to '/workspace'. While this file is in .devcontainer,
|
|
||||||
# mounts are relative to the first file in the list, which is a level up.
|
|
||||||
- ..:/workspaces/specklepy:cached
|
|
||||||
# Overrides default command so things don't shut down after the process ends.
|
|
||||||
command: /bin/sh -c "while sleep 1000; do :; done"
|
|
||||||
network_mode: host
|
|
||||||
# networks:
|
|
||||||
# default:
|
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
name: "Specklepy test"
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
secrets:
|
||||||
|
CODECOV_TOKEN:
|
||||||
|
required: true
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- "main"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test-internal: # Run integration tests against the internal server image
|
||||||
|
name: Test (internal)
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
python-version:
|
||||||
|
- "3.10"
|
||||||
|
- "3.11"
|
||||||
|
- "3.12"
|
||||||
|
- "3.13"
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install uv and set the python version
|
||||||
|
uses: astral-sh/setup-uv@v5
|
||||||
|
with:
|
||||||
|
python-version: ${{ matrix.python-version }}
|
||||||
|
enable-cache: true
|
||||||
|
cache-dependency-glob: "uv.lock"
|
||||||
|
|
||||||
|
- name: Install the project
|
||||||
|
run: uv sync --all-extras --dev
|
||||||
|
|
||||||
|
- name: Login to GitHub Container Registry
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: "ghcr.io"
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ github.token }}
|
||||||
|
|
||||||
|
- name: Run Speckle Server
|
||||||
|
run: docker compose --file docker-compose-internal.yml up --detach --wait
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
run: uv run pytest --cov --cov-report xml:reports/coverage.xml --junitxml=reports/test-results.xml
|
||||||
|
|
||||||
|
- uses: codecov/codecov-action@v5
|
||||||
|
if: matrix.python-version == 3.12
|
||||||
|
with:
|
||||||
|
fail_ci_if_error: true # optional (default = false)
|
||||||
|
files: ./reports/test-results.xml # optional
|
||||||
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
|
||||||
|
- name: Minimize uv cache
|
||||||
|
run: uv cache prune --ci
|
||||||
|
|
||||||
|
test-public: # Run integration tests against the public server image
|
||||||
|
name: Test (public)
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
python-version:
|
||||||
|
- "3.10"
|
||||||
|
- "3.11"
|
||||||
|
- "3.12"
|
||||||
|
- "3.13"
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install uv and set the python version
|
||||||
|
uses: astral-sh/setup-uv@v5
|
||||||
|
with:
|
||||||
|
python-version: ${{ matrix.python-version }}
|
||||||
|
enable-cache: true
|
||||||
|
cache-dependency-glob: "uv.lock"
|
||||||
|
|
||||||
|
- name: Install the project
|
||||||
|
run: uv sync --all-extras --dev
|
||||||
|
|
||||||
|
- uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.cache/pre-commit/
|
||||||
|
key: ${{ hashFiles('.pre-commit-config.yaml') }}
|
||||||
|
|
||||||
|
- name: Run pre-commit
|
||||||
|
run: uv run pre-commit run --all-files
|
||||||
|
|
||||||
|
- name: Run Speckle Server
|
||||||
|
run: docker compose --file docker-compose.yml up --detach --wait
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
run: uv run pytest --cov --cov-report xml:reports/coverage.xml --junitxml=reports/test-results.xml
|
||||||
|
|
||||||
|
- name: Minimize uv cache
|
||||||
|
run: uv cache prune --ci
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
name: "Publish Python Package"
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- "main"
|
||||||
|
tags:
|
||||||
|
- "3.*.*"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
uses: "./.github/workflows/pr.yml"
|
||||||
|
secrets:
|
||||||
|
CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
|
||||||
|
publish-package:
|
||||||
|
name: "Build and Publish Python Package"
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: test
|
||||||
|
|
||||||
|
# set the environment based on what triggered the workflow
|
||||||
|
environment:
|
||||||
|
name: ${{ github.ref_type == 'tag' && 'pypi' || 'testpypi' }}
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: "Install uv"
|
||||||
|
uses: astral-sh/setup-uv@v5
|
||||||
|
|
||||||
|
- name: "Checkout code"
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: "Build package artifacts"
|
||||||
|
run: uv build
|
||||||
|
|
||||||
|
# Logic for TestPyPI (on main branch push)
|
||||||
|
- name: "Publish to TestPyPI"
|
||||||
|
if: ${{ github.ref_type == 'branch' }}
|
||||||
|
run: uv publish --index test
|
||||||
|
|
||||||
|
- name: "Verify TestPyPI package installation"
|
||||||
|
if: ${{ github.ref_type == 'branch' }}
|
||||||
|
run: uv run --index test --with specklepy --no-project -- python -c "import specklepy"
|
||||||
|
|
||||||
|
# Logic for PyPI (on v3* tag creation)
|
||||||
|
- name: "Publish to PyPI"
|
||||||
|
if: ${{ github.ref_type == 'tag' }}
|
||||||
|
run: uv publish
|
||||||
|
|
||||||
|
- name: "Verify PyPI package installation"
|
||||||
|
if: ${{ github.ref_type == 'tag' }}
|
||||||
|
run: uv run --with specklepy --no-project -- python -c "import specklepy"
|
||||||
@@ -2,6 +2,8 @@
|
|||||||
.envrc
|
.envrc
|
||||||
reports/
|
reports/
|
||||||
|
|
||||||
|
.volumes/
|
||||||
|
|
||||||
# Byte-compiled / optimized / DLL files
|
# Byte-compiled / optimized / DLL files
|
||||||
__pycache__/
|
__pycache__/
|
||||||
*.py[cod]
|
*.py[cod]
|
||||||
|
|||||||
+15
-17
@@ -1,33 +1,31 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/charliermarsh/ruff-pre-commit
|
- repo: local
|
||||||
hooks:
|
hooks:
|
||||||
|
# Run the linter.
|
||||||
- id: ruff
|
- id: ruff
|
||||||
rev: v0.1.6
|
name: ruff lint
|
||||||
|
entry: uv run ruff check --force-exclude
|
||||||
|
language: system
|
||||||
|
types_or: [python, pyi]
|
||||||
|
# Run the formatter.
|
||||||
|
- id: ruff-format
|
||||||
|
name: ruff format
|
||||||
|
entry: uv run ruff format --force-exclude
|
||||||
|
language: system
|
||||||
|
types_or: [python, pyi]
|
||||||
|
|
||||||
|
|
||||||
- repo: https://github.com/commitizen-tools/commitizen
|
- repo: https://github.com/commitizen-tools/commitizen
|
||||||
hooks:
|
hooks:
|
||||||
- id: commitizen
|
- id: commitizen
|
||||||
- id: commitizen-branch
|
- id: commitizen-branch
|
||||||
stages:
|
stages:
|
||||||
- push
|
- pre-push
|
||||||
rev: v3.13.0
|
rev: v3.13.0
|
||||||
|
|
||||||
- repo: https://github.com/pycqa/isort
|
|
||||||
rev: 5.12.0
|
|
||||||
hooks:
|
|
||||||
- id: isort
|
|
||||||
|
|
||||||
- repo: https://github.com/psf/black
|
|
||||||
rev: 23.11.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
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v4.5.0
|
rev: v5.0.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: end-of-file-fixer
|
- id: end-of-file-fixer
|
||||||
- id: trailing-whitespace
|
- id: trailing-whitespace
|
||||||
|
|||||||
Vendored
+6
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"recommendations": [
|
||||||
|
"ms-python.python",
|
||||||
|
"charliermarsh.ruff"
|
||||||
|
]
|
||||||
|
}
|
||||||
Vendored
+3
-5
@@ -4,11 +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",
|
||||||
@@ -16,9 +14,9 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Pytest",
|
"name": "Pytest",
|
||||||
"type": "python",
|
"type": "debugpy",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"program": "pytest",
|
"module": "pytest",
|
||||||
"args": [],
|
"args": [],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"justMyCode": true
|
"justMyCode": true
|
||||||
|
|||||||
@@ -2,46 +2,20 @@
|
|||||||
<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://app.speckle.systems) ⇒ 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://pypi.org/project/specklepy/"><img alt="PyPI - Version" src="https://img.shields.io/pypi/v/specklepy"></a>
|
||||||
|
<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>
|
||||||
|
<a href="https://github.com/specklesystems/specklepy/blob/main/LICENSE"><img alt="License" src="https://img.shields.io/github/license/specklesystems/specklepy"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
# Repo structure
|
# Repo structure
|
||||||
|
|
||||||
@@ -55,25 +29,25 @@ Head to the [**📚 specklepy docs**](https://speckle.guide/dev/python.html) for
|
|||||||
|
|
||||||
### Installation
|
### Installation
|
||||||
|
|
||||||
This project uses python-poetry for dependency management, make sure you follow the official [docs](https://python-poetry.org/docs/#installation) to get poetry.
|
This project uses uv for dependency management, make sure you follow the official [docs](https://docs.astral.sh/uv/) to get it.
|
||||||
|
|
||||||
To bootstrap the project environment run `$ poetry install`. This will create a new virtual-env for the project and install both the package and dev dependencies.
|
To create a new virtual environment with uv run `$ uv venv` and follow the instructions on the screen to activate the virtual environment.
|
||||||
|
To bootstrap the project environment run `$ uv sync`. This will install both the package and dev dependencies.
|
||||||
|
|
||||||
If this is your first time using poetry and you're used to creating your venvs within the project directory, run `poetry config virtualenvs.in-project true` to configure poetry to do the same.
|
To execute any python script run `$ uv run python my_script.py`
|
||||||
|
|
||||||
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. Uv 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
|
### 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.
|
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`.
|
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.
|
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:
|
||||||
|
|
||||||
- Windows: `APPDATA` or `<USER>\AppData\Roaming\Speckle`
|
- Windows: `APPDATA` or `<USER>\AppData\Roaming\Speckle`
|
||||||
- Linux: `$XDG_DATA_HOME` or by default `~/.local/share/Speckle`
|
- Linux: `$XDG_DATA_HOME` or by default `~/.local/share/Speckle`
|
||||||
- Mac: `~/.config/Speckle`
|
- Mac: `~/.config/Speckle`
|
||||||
|
|||||||
@@ -0,0 +1,118 @@
|
|||||||
|
name: "speckle-server"
|
||||||
|
|
||||||
|
services:
|
||||||
|
####
|
||||||
|
# Speckle Server dependencies
|
||||||
|
#######
|
||||||
|
postgres:
|
||||||
|
image: "postgres:16.4-alpine3.20@sha256:d898b0b78a2627cb4ee63464a14efc9d296884f1b28c841b0ab7d7c42f1fffdf"
|
||||||
|
restart: always
|
||||||
|
environment:
|
||||||
|
POSTGRES_DB: speckle
|
||||||
|
POSTGRES_USER: speckle
|
||||||
|
POSTGRES_PASSWORD: speckle
|
||||||
|
volumes:
|
||||||
|
- ./.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: "valkey/valkey:8.1-alpine@sha256:0d27f0bca0249f61d060029a6aaf2e16b2c417d68d02a508e1dfb763fa2948b4"
|
||||||
|
restart: always
|
||||||
|
volumes:
|
||||||
|
- ./.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:
|
||||||
|
- ./.volumes/minio-data:/data
|
||||||
|
ports:
|
||||||
|
- '127.0.0.1:9000:9000'
|
||||||
|
- '127.0.0.1:9001:9001'
|
||||||
|
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:
|
||||||
|
image: ghcr.io/specklesystems/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(Number(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"
|
||||||
|
FRONTEND_ORIGIN: "http://127.0.0.1:8081"
|
||||||
|
|
||||||
|
# TODO: Change thvolumes:
|
||||||
|
REDIS_URL: "redis://redis"
|
||||||
|
|
||||||
|
S3_ENDPOINT: "http://minio:9000"
|
||||||
|
S3_PUBLIC_ENDPOINT: "http://127.0.0.1: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"
|
||||||
|
|
||||||
|
POSTGRES_URL: "postgres"
|
||||||
|
POSTGRES_USER: "speckle"
|
||||||
|
POSTGRES_PASSWORD: "speckle"
|
||||||
|
POSTGRES_DB: "speckle"
|
||||||
|
ENABLE_MP: "false"
|
||||||
|
|
||||||
|
LOG_PRETTY: "true"
|
||||||
|
|
||||||
|
FF_NEXT_GEN_FILE_IMPORTER_ENABLED: "true"
|
||||||
|
FF_LARGE_FILE_IMPORTS_ENABLED: "true"
|
||||||
|
|
||||||
|
networks:
|
||||||
|
default:
|
||||||
|
name: speckle-server
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
postgres-data:
|
||||||
|
redis-data:
|
||||||
|
minio-data:
|
||||||
+16
-18
@@ -1,4 +1,3 @@
|
|||||||
version: "3.9"
|
|
||||||
name: "speckle-server"
|
name: "speckle-server"
|
||||||
|
|
||||||
services:
|
services:
|
||||||
@@ -6,14 +5,14 @@ services:
|
|||||||
# Speckle Server dependencies
|
# Speckle Server dependencies
|
||||||
#######
|
#######
|
||||||
postgres:
|
postgres:
|
||||||
image: "postgres:14.5-alpine"
|
image: "postgres:16.4-alpine3.20@sha256:d898b0b78a2627cb4ee63464a14efc9d296884f1b28c841b0ab7d7c42f1fffdf"
|
||||||
restart: always
|
restart: always
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_DB: speckle
|
POSTGRES_DB: speckle
|
||||||
POSTGRES_USER: speckle
|
POSTGRES_USER: speckle
|
||||||
POSTGRES_PASSWORD: speckle
|
POSTGRES_PASSWORD: speckle
|
||||||
volumes:
|
volumes:
|
||||||
- postgres-data:/var/lib/postgresql/data/
|
- ./.volumes/postgres-data:/var/lib/postgresql/data/
|
||||||
healthcheck:
|
healthcheck:
|
||||||
# the -U user has to match the POSTGRES_USER value
|
# the -U user has to match the POSTGRES_USER value
|
||||||
test: ["CMD-SHELL", "pg_isready -U speckle"]
|
test: ["CMD-SHELL", "pg_isready -U speckle"]
|
||||||
@@ -22,10 +21,10 @@ services:
|
|||||||
retries: 30
|
retries: 30
|
||||||
|
|
||||||
redis:
|
redis:
|
||||||
image: "redis:6.0-alpine"
|
image: "valkey/valkey:8.1-alpine@sha256:0d27f0bca0249f61d060029a6aaf2e16b2c417d68d02a508e1dfb763fa2948b4"
|
||||||
restart: always
|
restart: always
|
||||||
volumes:
|
volumes:
|
||||||
- redis-data:/data
|
- ./.volumes/redis-data:/data
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test: ["CMD", "redis-cli", "--raw", "incr", "ping"]
|
test: ["CMD", "redis-cli", "--raw", "incr", "ping"]
|
||||||
interval: 5s
|
interval: 5s
|
||||||
@@ -37,7 +36,10 @@ services:
|
|||||||
command: server /data --console-address ":9001"
|
command: server /data --console-address ":9001"
|
||||||
restart: always
|
restart: always
|
||||||
volumes:
|
volumes:
|
||||||
- minio-data:/data
|
- ./.volumes/minio-data:/data
|
||||||
|
ports:
|
||||||
|
- '127.0.0.1:9000:9000'
|
||||||
|
- '127.0.0.1:9001:9001'
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test:
|
test:
|
||||||
[
|
[
|
||||||
@@ -49,16 +51,6 @@ services:
|
|||||||
retries: 30
|
retries: 30
|
||||||
start_period: 10s
|
start_period: 10s
|
||||||
|
|
||||||
####
|
|
||||||
# Speckle Server
|
|
||||||
#######
|
|
||||||
|
|
||||||
speckle-frontend:
|
|
||||||
image: speckle/speckle-frontend-2:latest
|
|
||||||
restart: always
|
|
||||||
ports:
|
|
||||||
- "0.0.0.0:8080:8080"
|
|
||||||
|
|
||||||
speckle-server:
|
speckle-server:
|
||||||
image: speckle/speckle-server:latest
|
image: speckle/speckle-server:latest
|
||||||
restart: always
|
restart: always
|
||||||
@@ -67,7 +59,7 @@ services:
|
|||||||
- CMD
|
- CMD
|
||||||
- /nodejs/bin/node
|
- /nodejs/bin/node
|
||||||
- -e
|
- -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); }"
|
- "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(Number(res.statusCode != 200 || body.toLowerCase().includes('error')));}); }).end(); } catch { process.exit(1); }"
|
||||||
interval: 10s
|
interval: 10s
|
||||||
timeout: 10s
|
timeout: 10s
|
||||||
retries: 3
|
retries: 3
|
||||||
@@ -85,11 +77,13 @@ services:
|
|||||||
# TODO: Change this to the URL of the speckle server, as accessed from the network
|
# TODO: Change this to the URL of the speckle server, as accessed from the network
|
||||||
CANONICAL_URL: "http://127.0.0.1:8080"
|
CANONICAL_URL: "http://127.0.0.1:8080"
|
||||||
SPECKLE_AUTOMATE_URL: "http://127.0.0.1:3030"
|
SPECKLE_AUTOMATE_URL: "http://127.0.0.1:3030"
|
||||||
|
FRONTEND_ORIGIN: "http://127.0.0.1:8081"
|
||||||
|
|
||||||
# TODO: Change thvolumes:
|
# TODO: Change thvolumes:
|
||||||
REDIS_URL: "redis://redis"
|
REDIS_URL: "redis://redis"
|
||||||
|
|
||||||
S3_ENDPOINT: "http://minio:9000"
|
S3_ENDPOINT: "http://minio:9000"
|
||||||
|
S3_PUBLIC_ENDPOINT: "http://127.0.0.1:9000"
|
||||||
S3_ACCESS_KEY: "minioadmin"
|
S3_ACCESS_KEY: "minioadmin"
|
||||||
S3_SECRET_KEY: "minioadmin"
|
S3_SECRET_KEY: "minioadmin"
|
||||||
S3_BUCKET: "speckle-server"
|
S3_BUCKET: "speckle-server"
|
||||||
@@ -102,7 +96,6 @@ services:
|
|||||||
SESSION_SECRET: "TODO:ReplaceWithLongString"
|
SESSION_SECRET: "TODO:ReplaceWithLongString"
|
||||||
|
|
||||||
STRATEGY_LOCAL: "true"
|
STRATEGY_LOCAL: "true"
|
||||||
DEBUG: "speckle:*"
|
|
||||||
|
|
||||||
POSTGRES_URL: "postgres"
|
POSTGRES_URL: "postgres"
|
||||||
POSTGRES_USER: "speckle"
|
POSTGRES_USER: "speckle"
|
||||||
@@ -110,6 +103,11 @@ services:
|
|||||||
POSTGRES_DB: "speckle"
|
POSTGRES_DB: "speckle"
|
||||||
ENABLE_MP: "false"
|
ENABLE_MP: "false"
|
||||||
|
|
||||||
|
LOG_PRETTY: "true"
|
||||||
|
|
||||||
|
FF_NEXT_GEN_FILE_IMPORTER_ENABLED: "true"
|
||||||
|
FF_LARGE_FILE_IMPORTS_ENABLED: "true"
|
||||||
|
|
||||||
networks:
|
networks:
|
||||||
default:
|
default:
|
||||||
name: speckle-server
|
name: speckle-server
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
from devtools import debug
|
from devtools import debug
|
||||||
|
|
||||||
from specklepy.api import operations
|
from specklepy.api import operations
|
||||||
from specklepy.objects.geometry import Base
|
from specklepy.objects_v2.geometry import Base
|
||||||
from specklepy.objects.units import Units
|
from specklepy.objects_v2.units import Units
|
||||||
|
|
||||||
dct = {
|
dct = {
|
||||||
"id": "1234abcd",
|
"id": "1234abcd",
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
[tools]
|
||||||
|
python = "3.13.7"
|
||||||
|
|
||||||
|
[settings]
|
||||||
|
experimental = true
|
||||||
|
python.uv_venv_auto = true
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
import re
|
|
||||||
import sys
|
|
||||||
|
|
||||||
|
|
||||||
def patch(tag):
|
|
||||||
print(f"Patching version: {tag}")
|
|
||||||
|
|
||||||
with open("pyproject.toml", "r") as f:
|
|
||||||
lines = f.readlines()
|
|
||||||
|
|
||||||
if "version" not in lines[2]:
|
|
||||||
raise Exception("Invalid pyproject.toml. Could not patch version.")
|
|
||||||
|
|
||||||
lines[2] = f'version = "{tag}"\n'
|
|
||||||
with open("pyproject.toml", "w") as file:
|
|
||||||
file.writelines(lines)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
if len(sys.argv) < 2:
|
|
||||||
return
|
|
||||||
|
|
||||||
tag = sys.argv[1]
|
|
||||||
if not re.match(r"[0-9]+(\.[0-9]+)*$", tag):
|
|
||||||
raise ValueError(f"Invalid tag provided: {tag}")
|
|
||||||
|
|
||||||
patch(tag)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
||||||
Generated
-2022
File diff suppressed because it is too large
Load Diff
+86
-60
@@ -1,74 +1,100 @@
|
|||||||
[tool.poetry]
|
[project]
|
||||||
|
dynamic = ["version"]
|
||||||
|
# version = "3.0.0a1"
|
||||||
name = "specklepy"
|
name = "specklepy"
|
||||||
version = "2.17.14"
|
description = "The Python SDK for Speckle"
|
||||||
description = "The Python SDK for Speckle 2.0"
|
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
authors = ["Speckle Systems <devops@speckle.systems>"]
|
authors = [{ name = "Speckle Systems", email = "devops@speckle.systems" }]
|
||||||
license = "Apache-2.0"
|
license = { text = "Apache-2.0" }
|
||||||
repository = "https://github.com/specklesystems/speckle-py"
|
requires-python = ">=3.10.0, <4.0"
|
||||||
documentation = "https://speckle.guide/dev/py-examples.html"
|
dependencies = [
|
||||||
homepage = "https://speckle.systems/"
|
"appdirs>=1.4.4",
|
||||||
packages = [
|
"attrs>=24.3.0",
|
||||||
{ include = "specklepy", from = "src" },
|
"deprecated>=1.2.15",
|
||||||
{ include = "speckle_automate", from = "src" },
|
"gql[requests,websockets]>=3.5.0,<4.0.0",
|
||||||
|
"httpx>=0.28.1",
|
||||||
|
"mkdocs>=1.6.1",
|
||||||
|
"mkdocs-material>=9.6.5",
|
||||||
|
"mkdocstrings>=0.28.1",
|
||||||
|
"mkdocstrings-python>=1.15.0",
|
||||||
|
"pydantic>=2.10.5",
|
||||||
|
"pydantic-settings>=2.7.1",
|
||||||
|
"ujson>=5.10.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[project.optional-dependencies]
|
||||||
|
speckleifc = ["ifcopenshell>=0.8.3.post2"]
|
||||||
|
|
||||||
[tool.poetry.dependencies]
|
[dependency-groups]
|
||||||
python = ">=3.8.0, <4.0"
|
dev = [
|
||||||
pydantic = "^2.5"
|
"commitizen>=4.1.0",
|
||||||
appdirs = "^1.4.4"
|
"devtools>=0.12.2",
|
||||||
gql = { extras = ["requests", "websockets"], version = "^3.3.0" }
|
"hatch>=1.14.0",
|
||||||
ujson = "^5.3.0"
|
"hatch-vcs>=0.4.0",
|
||||||
Deprecated = "^1.2.13"
|
"pre-commit>=4.0.1",
|
||||||
stringcase = "^1.2.0"
|
"pytest>=8.3.4",
|
||||||
attrs = "^23.1.0"
|
"pytest-asyncio>=0.25.2",
|
||||||
httpx = "^0.25.0"
|
"pytest-cov>=6.0.0",
|
||||||
|
"pytest-ordering>=0.6",
|
||||||
|
"ruff>=0.9.2",
|
||||||
|
"types-deprecated>=1.2.15.20241117",
|
||||||
|
"types-requests>=2.32.0.20241016",
|
||||||
|
"types-ujson>=5.10.0.20240515",
|
||||||
|
]
|
||||||
|
|
||||||
[tool.poetry.group.dev.dependencies]
|
[project.urls]
|
||||||
black = "23.11.0"
|
repository = "https://github.com/specklesystems/specklepy"
|
||||||
isort = "^5.7.0"
|
documentation = "https://speckle.guide/dev/py-examples.html"
|
||||||
pytest = "^7.1.3"
|
homepage = "https://speckle.systems/"
|
||||||
pytest-ordering = "^0.6"
|
|
||||||
pytest-cov = "^3.0.0"
|
|
||||||
devtools = "^0.8.0"
|
|
||||||
pylint = "^2.14.4"
|
|
||||||
pydantic-settings = "^2.3.0"
|
|
||||||
mypy = "^0.982"
|
|
||||||
pre-commit = "^2.20.0"
|
|
||||||
commitizen = "^2.38.0"
|
|
||||||
ruff = "^0.4.4"
|
|
||||||
types-deprecated = "^1.2.9"
|
|
||||||
types-ujson = "^5.6.0.0"
|
|
||||||
types-requests = "^2.28.11.5"
|
|
||||||
|
|
||||||
[tool.black]
|
[build-system]
|
||||||
exclude = '''
|
requires = ["hatchling", "hatch-vcs"]
|
||||||
/(
|
build-backend = "hatchling.build"
|
||||||
\.eggs
|
|
||||||
| \.git
|
|
||||||
| \.hg
|
|
||||||
| \.mypy_cache
|
|
||||||
| \.tox
|
|
||||||
| \.venv
|
|
||||||
| _build
|
|
||||||
| buck-out
|
|
||||||
| build
|
|
||||||
| dist
|
|
||||||
)/
|
|
||||||
'''
|
|
||||||
include = '\.pyi?$'
|
|
||||||
line-length = 88
|
|
||||||
target-version = ["py37", "py38", "py39", "py310", "py311"]
|
|
||||||
|
|
||||||
|
[tool.hatch.version]
|
||||||
|
source = "vcs"
|
||||||
|
|
||||||
|
[tool.hatch.build.targets.wheel]
|
||||||
|
only-include = ["src", "licenses"]
|
||||||
|
sources = ["src"]
|
||||||
|
|
||||||
|
[tool.hatch.build.targets.sdist]
|
||||||
|
include = ["src", "licenses"]
|
||||||
|
|
||||||
|
[tool.hatch.version.raw-options]
|
||||||
|
local_scheme = "no-local-version"
|
||||||
|
|
||||||
[tool.commitizen]
|
[tool.commitizen]
|
||||||
name = "cz_conventional_commits"
|
name = "cz_conventional_commits"
|
||||||
version = "2.9.2"
|
version = "2.9.2"
|
||||||
tag_format = "$version"
|
tag_format = "$version"
|
||||||
[build-system]
|
|
||||||
requires = ["poetry-core>=1.0.0"]
|
|
||||||
build-backend = "poetry.core.masonry.api"
|
|
||||||
|
|
||||||
[tool.isort]
|
[tool.ruff]
|
||||||
profile = "black"
|
exclude = [".venv", "**/*.yml"]
|
||||||
|
|
||||||
|
[tool.ruff.lint]
|
||||||
|
select = [
|
||||||
|
# pycodestyle
|
||||||
|
"E",
|
||||||
|
# Pyflakes
|
||||||
|
"F",
|
||||||
|
# pyupgrade
|
||||||
|
"UP",
|
||||||
|
# flake8-bugbear
|
||||||
|
"B",
|
||||||
|
# flake8-simplify
|
||||||
|
"SIM",
|
||||||
|
# isort
|
||||||
|
"I",
|
||||||
|
]
|
||||||
|
ignore = ["UP006", "UP007", "UP035"]
|
||||||
|
|
||||||
|
[[tool.uv.index]]
|
||||||
|
name = "pypi"
|
||||||
|
url = "https://pypi.org/simple/"
|
||||||
|
publish-url = "https://upload.pypi.org/legacy/"
|
||||||
|
|
||||||
|
[[tool.uv.index]]
|
||||||
|
name = "test"
|
||||||
|
url = "https://test.pypi.org/simple/"
|
||||||
|
publish-url = "https://test.pypi.org/legacy/"
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.api.client.SpeckleClient
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.data_objects.DataObject
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.data_objects.QgisObject
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IBlenderObject
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.ICurve
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IDataObject
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IDisplayValue
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IBlenderObject
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IHasArea
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IHasUnits
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IHasVolume
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.interfaces.IProperties
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.other.RenderMaterial
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.primitive.Interval
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.proxies.ColorProxy
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.proxies.GroupProxy
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.proxies.InstanceDefinitionProxy
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.proxies.InstanceProxy
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.proxies.RenderMaterialProxy
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.base.Base
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.arc.Arc
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.box.Box
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.circle.Circle
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.control_point.ControlPoint
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.ellipse.Ellipse
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.line.Line
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.mesh.Mesh
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.plane.Plane
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.point.Point
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.point_cloud.PointCloud
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.polycurve.Polycurve
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.polyline.Polyline
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.spiral.Spiral
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.surface.Surface
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: specklepy.objects.geometry.vector.Vector
|
||||||
Binary file not shown.
|
After Width: | Height: | Size: 386 B |
@@ -0,0 +1,29 @@
|
|||||||
|
# Introduction
|
||||||
|
|
||||||
|
Welcome to the Specklepy Developer Docs - a single source of documentation on everything Specklepy! If you're looking for info on how to use Speckle, check our [user guide](https://speckle.guide/).
|
||||||
|
|
||||||
|
### Code Repository
|
||||||
|
The Python SDK can be found in our [repository](//github.com/specklesystems/specklepy), its readme contains instructions on how to build it.
|
||||||
|
|
||||||
|
### Installation
|
||||||
|
You can install it using pip
|
||||||
|
```
|
||||||
|
pip install specklepy
|
||||||
|
```
|
||||||
|
|
||||||
|
### Key Components
|
||||||
|
|
||||||
|
SpecklePy has three main parts:
|
||||||
|
|
||||||
|
1. a `SpeckleClient` which allows you to interact with the server API
|
||||||
|
2. `operations` and `transports` for sending and receiving large objects
|
||||||
|
3. a `Base` object and accompaniying serializer for creating and customizing your own Speckle objects
|
||||||
|
|
||||||
|
|
||||||
|
### 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:
|
||||||
|
|
||||||
|
- Windows: `APPDATA` or `<USER>\AppData\Roaming\Speckle`
|
||||||
|
- Linux: `$XDG_DATA_HOME` or by default `~/.local/share/Speckle`
|
||||||
|
- Mac: `~/.config/Speckle`
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
::: speckle_automate.automation_context.AutomationContext
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
site_name: Specklepy Docs
|
||||||
|
theme:
|
||||||
|
name: material
|
||||||
|
favicon: assets/speckle_logo.png
|
||||||
|
logo: assets/speckle_logo.png
|
||||||
|
features:
|
||||||
|
- navigation.tabs
|
||||||
|
palette:
|
||||||
|
# Palette toggle for light mode
|
||||||
|
- scheme: default
|
||||||
|
primary: white
|
||||||
|
toggle:
|
||||||
|
icon: material/weather-night
|
||||||
|
name: Switch to dark mode
|
||||||
|
|
||||||
|
# Palette toggle for dark mode
|
||||||
|
- scheme: slate
|
||||||
|
primary: black
|
||||||
|
logo: assets/logo_white.png
|
||||||
|
toggle:
|
||||||
|
icon: material/weather-sunny
|
||||||
|
name: Switch to light mode
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
markdown_extensions:
|
||||||
|
- pymdownx.highlight:
|
||||||
|
anchor_linenums: true
|
||||||
|
line_spans: __span
|
||||||
|
pygments_lang_class: true
|
||||||
|
- pymdownx.inlinehilite
|
||||||
|
- pymdownx.snippets
|
||||||
|
- pymdownx.superfences
|
||||||
|
|
||||||
|
extra_css:
|
||||||
|
- css/mkdocstrings.css
|
||||||
|
|
||||||
|
plugins:
|
||||||
|
- search
|
||||||
|
- mkdocstrings:
|
||||||
|
handlers:
|
||||||
|
python:
|
||||||
|
paths: [.]
|
||||||
|
options:
|
||||||
|
parameter_headings: false
|
||||||
|
members_order: source
|
||||||
|
separate_signature: true
|
||||||
|
filters: ["!^_"] #Ignore _ prefixed properties
|
||||||
|
docstring_options:
|
||||||
|
ignore_init_summary: true
|
||||||
|
merge_init_into_class: true
|
||||||
|
show_signature_annotations: true
|
||||||
|
signature_crossrefs: true
|
||||||
|
show_if_no_docstring: true
|
||||||
|
show_labels: true
|
||||||
|
show_source: true
|
||||||
|
show_symbol_type_heading: true
|
||||||
|
show_symbol_type_toc: true
|
||||||
|
show_bases: false
|
||||||
|
heading_level: 3
|
||||||
|
|
||||||
|
inventories:
|
||||||
|
- url: https://docs.python.org/3/objects.inv
|
||||||
|
domains: [py, std]
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
import time
|
import time
|
||||||
from dataclasses import dataclass, field
|
from dataclasses import dataclass, field
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Any, Dict, List, Optional, Tuple, Union
|
from typing import Any, Dict, List, Optional, Union
|
||||||
|
|
||||||
import httpx
|
import httpx
|
||||||
from gql import gql
|
from gql import gql
|
||||||
@@ -18,7 +18,9 @@ from speckle_automate.schema import (
|
|||||||
)
|
)
|
||||||
from specklepy.api import operations
|
from specklepy.api import operations
|
||||||
from specklepy.api.client import SpeckleClient
|
from specklepy.api.client import SpeckleClient
|
||||||
from specklepy.core.api.models import Branch
|
from specklepy.core.api.inputs.model_inputs import CreateModelInput
|
||||||
|
from specklepy.core.api.inputs.version_inputs import CreateVersionInput
|
||||||
|
from specklepy.core.api.models.current import Model, Version
|
||||||
from specklepy.logging.exceptions import SpeckleException
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
from specklepy.objects.base import Base
|
from specklepy.objects.base import Base
|
||||||
from specklepy.transports.memory import MemoryTransport
|
from specklepy.transports.memory import MemoryTransport
|
||||||
@@ -71,7 +73,7 @@ class AutomationContext:
|
|||||||
speckle_client.authenticate_with_token(speckle_token)
|
speckle_client.authenticate_with_token(speckle_token)
|
||||||
if not speckle_client.account:
|
if not speckle_client.account:
|
||||||
msg = (
|
msg = (
|
||||||
f"Could not autenticate to {automation_run_data.speckle_server_url}",
|
f"Could not authenticate to {automation_run_data.speckle_server_url}",
|
||||||
"with the provided token",
|
"with the provided token",
|
||||||
)
|
)
|
||||||
raise ValueError(msg)
|
raise ValueError(msg)
|
||||||
@@ -96,61 +98,81 @@ class AutomationContext:
|
|||||||
|
|
||||||
def receive_version(self) -> Base:
|
def receive_version(self) -> Base:
|
||||||
"""Receive the Speckle project version that triggered this automation run."""
|
"""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
|
# 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
|
version_id = self.automation_run_data.triggers[0].payload.version_id
|
||||||
commit = self.speckle_client.commit.get(
|
try:
|
||||||
self.automation_run_data.project_id, version_id
|
version = self.speckle_client.version.get(
|
||||||
)
|
version_id, self.automation_run_data.project_id
|
||||||
if not commit.referencedObject:
|
)
|
||||||
raise ValueError("The commit has no referencedObject, cannot receive it.")
|
except SpeckleException as err:
|
||||||
|
raise ValueError(
|
||||||
|
f"""Could not receive specified version.
|
||||||
|
Is your environment configured correctly?
|
||||||
|
project_id: {self.automation_run_data.project_id}
|
||||||
|
model_id: {self.automation_run_data.triggers[0].payload.model_id}
|
||||||
|
version_id: {self.automation_run_data.triggers[0].payload.version_id}
|
||||||
|
"""
|
||||||
|
) from err
|
||||||
|
|
||||||
|
if not version.referenced_object:
|
||||||
|
raise Exception(
|
||||||
|
"This version is past the version history limit,",
|
||||||
|
" cannot execute an automation on it",
|
||||||
|
)
|
||||||
|
|
||||||
base = operations.receive(
|
base = operations.receive(
|
||||||
commit.referencedObject, self._server_transport, self._memory_transport
|
version.referenced_object, self._server_transport, self._memory_transport
|
||||||
)
|
)
|
||||||
|
# self._closure_tree = base["__closure"]
|
||||||
print(
|
print(
|
||||||
f"It took {self.elapsed():.2f} seconds to receive",
|
f"It took {self.elapsed():.2f} seconds to receive",
|
||||||
f" the speckle version {version_id}",
|
f" the speckle version {version_id}",
|
||||||
)
|
)
|
||||||
return base
|
return base
|
||||||
|
|
||||||
|
def create_new_model_in_project(
|
||||||
|
self, model_name: str, model_description: Optional[str] = None
|
||||||
|
) -> Model:
|
||||||
|
input = CreateModelInput(
|
||||||
|
name=model_name,
|
||||||
|
description=model_description,
|
||||||
|
project_id=self.automation_run_data.project_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.speckle_client.model.create(input)
|
||||||
|
|
||||||
|
def get_model(self, model_id: str) -> Model:
|
||||||
|
"""
|
||||||
|
Args:
|
||||||
|
model_id (str): The id of the model to get
|
||||||
|
"""
|
||||||
|
return self.speckle_client.model.get(
|
||||||
|
model_id, self.automation_run_data.project_id
|
||||||
|
)
|
||||||
|
|
||||||
def create_new_version_in_project(
|
def create_new_version_in_project(
|
||||||
self, root_object: Base, model_name: str, version_message: str = ""
|
self, root_object: Base, model_id: str, version_message: str = ""
|
||||||
) -> Tuple[str, str]:
|
) -> Version:
|
||||||
"""Save a base model to a new version on the project.
|
"""Save a base model to a new version on the project.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
root_object (Base): The Speckle base object for the new version.
|
root_object (Base): The Speckle base object for the new version.
|
||||||
model_id (str): For now please use a `branchName`!
|
model_id (str): Id of model to create the new version on.
|
||||||
version_message (str): The message for the new version.
|
version_message (str): The message for the new version.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
branch = self.speckle_client.branch.get(
|
matching_trigger = [
|
||||||
self.automation_run_data.project_id, model_name, 1
|
t
|
||||||
)
|
for t in self.automation_run_data.triggers
|
||||||
if isinstance(branch, Branch):
|
if t.payload.model_id == model_id
|
||||||
if not branch.id:
|
]
|
||||||
raise ValueError("Cannot use the branch without its id")
|
if matching_trigger:
|
||||||
matching_trigger = [
|
raise ValueError(
|
||||||
t
|
f"The target model: {model_id} cannot match the model"
|
||||||
for t in self.automation_run_data.triggers
|
f" that triggered this automation:"
|
||||||
if t.payload.model_id == branch.id
|
f" {matching_trigger[0].payload.model_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_id = operations.send(
|
||||||
root_object,
|
root_object,
|
||||||
@@ -158,19 +180,17 @@ class AutomationContext:
|
|||||||
use_default_cache=False,
|
use_default_cache=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
version_id = self.speckle_client.commit.create(
|
create_version_input = CreateVersionInput(
|
||||||
stream_id=self.automation_run_data.project_id,
|
|
||||||
object_id=root_object_id,
|
object_id=root_object_id,
|
||||||
branch_name=model_name,
|
model_id=model_id,
|
||||||
|
project_id=self.automation_run_data.project_id,
|
||||||
message=version_message,
|
message=version_message,
|
||||||
source_application="SpeckleAutomate",
|
source_application="SpeckleAutomate",
|
||||||
)
|
)
|
||||||
|
version = self.speckle_client.version.create(create_version_input)
|
||||||
|
|
||||||
if isinstance(version_id, SpeckleException):
|
self._automation_result.result_versions.append(version.id)
|
||||||
raise version_id
|
return version
|
||||||
|
|
||||||
self._automation_result.result_versions.append(version_id)
|
|
||||||
return model_id, version_id
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def context_view(self) -> Optional[str]:
|
def context_view(self) -> Optional[str]:
|
||||||
@@ -206,6 +226,7 @@ class AutomationContext:
|
|||||||
query = gql(
|
query = gql(
|
||||||
"""
|
"""
|
||||||
mutation AutomateFunctionRunStatusReport(
|
mutation AutomateFunctionRunStatusReport(
|
||||||
|
$projectId: String!
|
||||||
$functionRunId: String!
|
$functionRunId: String!
|
||||||
$status: AutomateRunStatus!
|
$status: AutomateRunStatus!
|
||||||
$statusMessage: String
|
$statusMessage: String
|
||||||
@@ -213,6 +234,7 @@ class AutomationContext:
|
|||||||
$contextView: String
|
$contextView: String
|
||||||
){
|
){
|
||||||
automateFunctionRunStatusReport(input: {
|
automateFunctionRunStatusReport(input: {
|
||||||
|
projectId: $projectId
|
||||||
functionRunId: $functionRunId
|
functionRunId: $functionRunId
|
||||||
status: $status
|
status: $status
|
||||||
statusMessage: $statusMessage
|
statusMessage: $statusMessage
|
||||||
@@ -223,29 +245,30 @@ class AutomationContext:
|
|||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
if self.run_status in [AutomationStatus.SUCCEEDED, AutomationStatus.FAILED]:
|
if self.run_status in [AutomationStatus.SUCCEEDED, AutomationStatus.FAILED]:
|
||||||
object_results = {
|
results_dict = self._automation_result.model_dump(by_alias=True)
|
||||||
"version": 1,
|
results = {
|
||||||
|
"version": 3,
|
||||||
"values": {
|
"values": {
|
||||||
"objectResults": self._automation_result.model_dump(by_alias=True)[
|
"objectResults": results_dict["objectResults"],
|
||||||
"objectResults"
|
"versionResult": results_dict["versionResult"],
|
||||||
],
|
|
||||||
"blobIds": self._automation_result.blobs,
|
"blobIds": self._automation_result.blobs,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
else:
|
else:
|
||||||
object_results = None
|
results = None
|
||||||
|
|
||||||
params = {
|
params = {
|
||||||
|
"projectId": self.automation_run_data.project_id,
|
||||||
"functionRunId": self.automation_run_data.function_run_id,
|
"functionRunId": self.automation_run_data.function_run_id,
|
||||||
"status": self.run_status.value,
|
"status": self.run_status.value,
|
||||||
"statusMessage": self._automation_result.status_message,
|
"statusMessage": self._automation_result.status_message,
|
||||||
"results": object_results,
|
"results": results,
|
||||||
"contextView": self._automation_result.result_view,
|
"contextView": self._automation_result.result_view,
|
||||||
}
|
}
|
||||||
print(f"Reporting run status with content: {params}")
|
print(f"Reporting run status with content: {params}")
|
||||||
self.speckle_client.httpclient.execute(query, params)
|
self.speckle_client.httpclient.execute(query, params)
|
||||||
|
|
||||||
def store_file_result(self, file_path: Union[Path, str]) -> None:
|
def store_file_result(self, file_path: Union[Path, str]) -> str:
|
||||||
"""Save a file attached to the project of this automation."""
|
"""Save a file attached to the project of this automation."""
|
||||||
path_obj = (
|
path_obj = (
|
||||||
Path(file_path).resolve() if isinstance(file_path, str) else file_path
|
Path(file_path).resolve() if isinstance(file_path, str) else file_path
|
||||||
@@ -261,7 +284,8 @@ class AutomationContext:
|
|||||||
|
|
||||||
if not path_obj.exists():
|
if not path_obj.exists():
|
||||||
raise ValueError("The given file path doesn't exist")
|
raise ValueError("The given file path doesn't exist")
|
||||||
files = {path_obj.name: open(str(path_obj), "rb")}
|
|
||||||
|
files = {path_obj.name: path_obj.open("rb")}
|
||||||
|
|
||||||
url = (
|
url = (
|
||||||
f"{self.automation_run_data.speckle_server_url}api/stream/"
|
f"{self.automation_run_data.speckle_server_url}api/stream/"
|
||||||
@@ -286,25 +310,51 @@ class AutomationContext:
|
|||||||
[upload_result.blob_id for upload_result in upload_response.upload_results]
|
[upload_result.blob_id for upload_result in upload_response.upload_results]
|
||||||
)
|
)
|
||||||
|
|
||||||
def mark_run_failed(self, status_message: str) -> None:
|
return upload_response.upload_results[0].blob_id
|
||||||
"""Mark the current run a failure."""
|
|
||||||
self._mark_run(AutomationStatus.FAILED, status_message)
|
def mark_run_failed(
|
||||||
|
self, status_message: str, version_result: dict[str, Any] | None = None
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Mark the current run a failure.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
status_message: Optional message to be displayed.
|
||||||
|
version_result: Optional data object,
|
||||||
|
that will be attached to the run results.
|
||||||
|
The dictionary should be JSON serializable
|
||||||
|
"""
|
||||||
|
self._mark_run(AutomationStatus.FAILED, status_message, version_result)
|
||||||
|
|
||||||
def mark_run_exception(self, status_message: str) -> None:
|
def mark_run_exception(self, status_message: str) -> None:
|
||||||
"""Mark the current run a failure."""
|
"""Mark the current run a failure."""
|
||||||
self._mark_run(AutomationStatus.EXCEPTION, status_message)
|
self._mark_run(AutomationStatus.EXCEPTION, status_message, None)
|
||||||
|
|
||||||
def mark_run_success(self, status_message: Optional[str]) -> None:
|
def mark_run_success(
|
||||||
"""Mark the current run a success with an optional message."""
|
self, status_message: str | None, version_result: dict[str, Any] | None = None
|
||||||
self._mark_run(AutomationStatus.SUCCEEDED, status_message)
|
) -> None:
|
||||||
|
"""
|
||||||
|
Mark the current run a success with an optional message.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
status_message: Optional message to be displayed.
|
||||||
|
version_result: Optional data object,
|
||||||
|
that will be attached to the run results.
|
||||||
|
The dictionary should be JSON serializable
|
||||||
|
"""
|
||||||
|
self._mark_run(AutomationStatus.SUCCEEDED, status_message, version_result)
|
||||||
|
|
||||||
def _mark_run(
|
def _mark_run(
|
||||||
self, status: AutomationStatus, status_message: Optional[str]
|
self,
|
||||||
|
status: AutomationStatus,
|
||||||
|
status_message: str | None,
|
||||||
|
version_result: dict[str, Any] | None,
|
||||||
) -> None:
|
) -> None:
|
||||||
duration = self.elapsed()
|
duration = self.elapsed()
|
||||||
self._automation_result.status_message = status_message
|
self._automation_result.status_message = status_message
|
||||||
self._automation_result.run_status = status
|
self._automation_result.run_status = status
|
||||||
self._automation_result.elapsed = duration
|
self._automation_result.elapsed = duration
|
||||||
|
self._automation_result.version_result = version_result
|
||||||
|
|
||||||
msg = f"Automation run {status.value} after {duration:.2f} seconds."
|
msg = f"Automation run {status.value} after {duration:.2f} seconds."
|
||||||
print("\n".join([msg, status_message]) if status_message else msg)
|
print("\n".join([msg, status_message]) if status_message else msg)
|
||||||
@@ -312,26 +362,24 @@ class AutomationContext:
|
|||||||
def attach_error_to_objects(
|
def attach_error_to_objects(
|
||||||
self,
|
self,
|
||||||
category: str,
|
category: str,
|
||||||
object_ids: Union[str, List[str]],
|
affected_objects: Union[Base, List[Base]],
|
||||||
message: Optional[str] = None,
|
message: Optional[str] = None,
|
||||||
metadata: Optional[Dict[str, Any]] = None,
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
visual_overrides: Optional[Dict[str, Any]] = None,
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Add a new error case to the run results.
|
"""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:
|
Args:
|
||||||
error_tag (str): A short tag for the error type.
|
category (str): A short tag for the event type.
|
||||||
causing_object_ids (str[]): A list of object_id-s that are causing the error
|
affected_objects (Union[Base, List[Base]]): A single object or a list of
|
||||||
error_messagge (Optional[str]): Optional error message.
|
objects that are causing the error case.
|
||||||
|
message (Optional[str]): Optional message.
|
||||||
metadata: User provided metadata key value pairs
|
metadata: User provided metadata key value pairs
|
||||||
visual_overrides: Case specific 3D visual overrides.
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
"""
|
"""
|
||||||
self.attach_result_to_objects(
|
self.attach_result_to_objects(
|
||||||
ObjectResultLevel.ERROR,
|
ObjectResultLevel.ERROR,
|
||||||
category,
|
category,
|
||||||
object_ids,
|
affected_objects,
|
||||||
message,
|
message,
|
||||||
metadata,
|
metadata,
|
||||||
visual_overrides,
|
visual_overrides,
|
||||||
@@ -340,16 +388,52 @@ class AutomationContext:
|
|||||||
def attach_warning_to_objects(
|
def attach_warning_to_objects(
|
||||||
self,
|
self,
|
||||||
category: str,
|
category: str,
|
||||||
object_ids: Union[str, List[str]],
|
affected_objects: Union[Base, List[Base]],
|
||||||
message: Optional[str] = None,
|
message: Optional[str] = None,
|
||||||
metadata: Optional[Dict[str, Any]] = None,
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
visual_overrides: Optional[Dict[str, Any]] = None,
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Add a new warning case to the run results."""
|
"""Add a new warning case to the run results.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
category (str): A short tag for the event type.
|
||||||
|
affected_objects (Union[Base, List[Base]]): A single object or a list of
|
||||||
|
objects that are causing the warning case.
|
||||||
|
message (Optional[str]): Optional message.
|
||||||
|
metadata: User provided metadata key value pairs
|
||||||
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
|
"""
|
||||||
self.attach_result_to_objects(
|
self.attach_result_to_objects(
|
||||||
ObjectResultLevel.WARNING,
|
ObjectResultLevel.WARNING,
|
||||||
category,
|
category,
|
||||||
object_ids,
|
affected_objects,
|
||||||
|
message,
|
||||||
|
metadata,
|
||||||
|
visual_overrides,
|
||||||
|
)
|
||||||
|
|
||||||
|
def attach_success_to_objects(
|
||||||
|
self,
|
||||||
|
category: str,
|
||||||
|
affected_objects: Union[Base, List[Base]],
|
||||||
|
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.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
category (str): A short tag for the event type.
|
||||||
|
affected_objects (Union[Base, List[Base]]): A single object or a list of
|
||||||
|
objects that are causing the success case.
|
||||||
|
message (Optional[str]): Optional message.
|
||||||
|
metadata: User provided metadata key value pairs
|
||||||
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
|
"""
|
||||||
|
self.attach_result_to_objects(
|
||||||
|
ObjectResultLevel.SUCCESS,
|
||||||
|
category,
|
||||||
|
affected_objects,
|
||||||
message,
|
message,
|
||||||
metadata,
|
metadata,
|
||||||
visual_overrides,
|
visual_overrides,
|
||||||
@@ -358,16 +442,25 @@ class AutomationContext:
|
|||||||
def attach_info_to_objects(
|
def attach_info_to_objects(
|
||||||
self,
|
self,
|
||||||
category: str,
|
category: str,
|
||||||
object_ids: Union[str, List[str]],
|
affected_objects: Union[Base, List[Base]],
|
||||||
message: Optional[str] = None,
|
message: Optional[str] = None,
|
||||||
metadata: Optional[Dict[str, Any]] = None,
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
visual_overrides: Optional[Dict[str, Any]] = None,
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Add a new info case to the run results."""
|
"""Add a new info case to the run results.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
category (str): A short tag for the event type.
|
||||||
|
affected_objects (Union[Base, List[Base]]): A single object or a list of
|
||||||
|
objects that are causing the info case.
|
||||||
|
message (Optional[str]): Optional message.
|
||||||
|
metadata: User provided metadata key value pairs
|
||||||
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
|
"""
|
||||||
self.attach_result_to_objects(
|
self.attach_result_to_objects(
|
||||||
ObjectResultLevel.INFO,
|
ObjectResultLevel.INFO,
|
||||||
category,
|
category,
|
||||||
object_ids,
|
affected_objects,
|
||||||
message,
|
message,
|
||||||
metadata,
|
metadata,
|
||||||
visual_overrides,
|
visual_overrides,
|
||||||
@@ -377,19 +470,39 @@ class AutomationContext:
|
|||||||
self,
|
self,
|
||||||
level: ObjectResultLevel,
|
level: ObjectResultLevel,
|
||||||
category: str,
|
category: str,
|
||||||
object_ids: Union[str, List[str]],
|
affected_objects: Union[Base, List[Base]],
|
||||||
message: Optional[str] = None,
|
message: Optional[str] = None,
|
||||||
metadata: Optional[Dict[str, Any]] = None,
|
metadata: Optional[Dict[str, Any]] = None,
|
||||||
visual_overrides: Optional[Dict[str, Any]] = None,
|
visual_overrides: Optional[Dict[str, Any]] = None,
|
||||||
) -> None:
|
) -> None:
|
||||||
if isinstance(object_ids, list):
|
"""Add a new result case to the run results.
|
||||||
if len(object_ids) < 1:
|
|
||||||
|
Args:
|
||||||
|
level: Result level.
|
||||||
|
category (str): A short tag for the event type.
|
||||||
|
affected_objects (Union[Base, List[Base]]): A single object or a list of
|
||||||
|
objects that are causing the info case.
|
||||||
|
message (Optional[str]): Optional message.
|
||||||
|
metadata: User provided metadata key value pairs
|
||||||
|
visual_overrides: Case specific 3D visual overrides.
|
||||||
|
"""
|
||||||
|
if isinstance(affected_objects, list):
|
||||||
|
if len(affected_objects) < 1:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"Need atleast one object_id to report a(n) {level.value.upper()}"
|
f"Need atleast one object to report a(n) {level.value.upper()}"
|
||||||
)
|
)
|
||||||
id_list = object_ids
|
object_list = affected_objects
|
||||||
else:
|
else:
|
||||||
id_list = [object_ids]
|
object_list = [affected_objects]
|
||||||
|
|
||||||
|
ids: Dict[str, Optional[str]] = {}
|
||||||
|
for o in object_list:
|
||||||
|
# validate that the Base.id is not None. If its a None, throw an Exception
|
||||||
|
if not o.id:
|
||||||
|
raise Exception(
|
||||||
|
f"You can only attach {level} results to objects with an id."
|
||||||
|
)
|
||||||
|
ids[o.id] = o.applicationId
|
||||||
print(
|
print(
|
||||||
f"Created new {level.value.upper()}"
|
f"Created new {level.value.upper()}"
|
||||||
f" category: {category} caused by: {message}"
|
f" category: {category} caused by: {message}"
|
||||||
@@ -398,7 +511,7 @@ class AutomationContext:
|
|||||||
ResultCase(
|
ResultCase(
|
||||||
category=category,
|
category=category,
|
||||||
level=level,
|
level=level,
|
||||||
object_ids=id_list,
|
object_app_ids=ids,
|
||||||
message=message,
|
message=message,
|
||||||
metadata=metadata,
|
metadata=metadata,
|
||||||
visual_overrides=visual_overrides,
|
visual_overrides=visual_overrides,
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
"""Some useful helpers for working with automation data."""
|
"""Some useful helpers for working with automation data."""
|
||||||
import secrets
|
|
||||||
import string
|
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from gql import gql
|
from gql import gql
|
||||||
@@ -90,10 +88,8 @@ def create_test_automation_run(
|
|||||||
|
|
||||||
print(result)
|
print(result)
|
||||||
|
|
||||||
return (
|
return TestAutomationRunData.model_validate(
|
||||||
result.get("projectMutations")
|
result["projectMutations"]["automationMutations"]["createTestAutomationRun"]
|
||||||
.get("automationMutations")
|
|
||||||
.get("createTestAutomationRun")
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -125,9 +121,9 @@ def create_test_automation_run_data(
|
|||||||
project_id=test_automation_environment.project_id,
|
project_id=test_automation_environment.project_id,
|
||||||
speckle_server_url=test_automation_environment.server_url,
|
speckle_server_url=test_automation_environment.server_url,
|
||||||
automation_id=test_automation_environment.automation_id,
|
automation_id=test_automation_environment.automation_id,
|
||||||
automation_run_id=test_automation_run_data["automationRunId"],
|
automation_run_id=test_automation_run_data.automation_run_id,
|
||||||
function_run_id=test_automation_run_data["functionRunId"],
|
function_run_id=test_automation_run_data.function_run_id,
|
||||||
triggers=test_automation_run_data["triggers"],
|
triggers=test_automation_run_data.triggers,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -139,12 +135,6 @@ def test_automation_run_data(
|
|||||||
return create_test_automation_run_data(speckle_client, test_automation_environment)
|
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__ = [
|
__all__ = [
|
||||||
"test_automation_environment",
|
"test_automation_environment",
|
||||||
"test_automation_token",
|
"test_automation_token",
|
||||||
|
|||||||
@@ -61,15 +61,13 @@ def _parse_input_data(
|
|||||||
def execute_automate_function(
|
def execute_automate_function(
|
||||||
automate_function: AutomateFunction[T],
|
automate_function: AutomateFunction[T],
|
||||||
input_schema: type[T],
|
input_schema: type[T],
|
||||||
) -> None:
|
) -> None: ...
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
def execute_automate_function(
|
def execute_automate_function(
|
||||||
automate_function: AutomateFunctionWithoutInputs,
|
automate_function: AutomateFunctionWithoutInputs,
|
||||||
) -> None:
|
) -> None: ...
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
class AutomateGenerateJsonSchema(GenerateJsonSchema):
|
class AutomateGenerateJsonSchema(GenerateJsonSchema):
|
||||||
@@ -130,7 +128,8 @@ def execute_automate_function(
|
|||||||
automate_function, # type: ignore
|
automate_function, # type: ignore
|
||||||
)
|
)
|
||||||
|
|
||||||
# if we've gotten this far, the execution should technically be completed as expected
|
# 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
|
# thus exiting with 0 is the schemantically correct thing to do
|
||||||
exit_code = (
|
exit_code = (
|
||||||
1 if automation_context.run_status == AutomationStatus.EXCEPTION else 0
|
1 if automation_context.run_status == AutomationStatus.EXCEPTION else 0
|
||||||
@@ -146,16 +145,14 @@ def run_function(
|
|||||||
automation_context: AutomationContext,
|
automation_context: AutomationContext,
|
||||||
automate_function: AutomateFunction[T],
|
automate_function: AutomateFunction[T],
|
||||||
inputs: T,
|
inputs: T,
|
||||||
) -> AutomationContext:
|
) -> AutomationContext: ...
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
def run_function(
|
def run_function(
|
||||||
automation_context: AutomationContext,
|
automation_context: AutomationContext,
|
||||||
automate_function: AutomateFunctionWithoutInputs,
|
automate_function: AutomateFunctionWithoutInputs,
|
||||||
) -> AutomationContext:
|
) -> AutomationContext: ...
|
||||||
...
|
|
||||||
|
|
||||||
|
|
||||||
def run_function(
|
def run_function(
|
||||||
@@ -194,4 +191,4 @@ def run_function(
|
|||||||
if not automation_context.context_view:
|
if not automation_context.context_view:
|
||||||
automation_context.set_context_view()
|
automation_context.set_context_view()
|
||||||
automation_context.report_run_status()
|
automation_context.report_run_status()
|
||||||
return automation_context
|
return automation_context
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
""""""
|
""""""
|
||||||
|
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
from typing import Any, Dict, List, Literal, Optional
|
from typing import Any, Literal
|
||||||
|
|
||||||
from pydantic import BaseModel, ConfigDict, Field
|
from pydantic import BaseModel, ConfigDict, Field
|
||||||
from stringcase import camelcase
|
from pydantic.alias_generators import to_camel
|
||||||
|
|
||||||
|
|
||||||
class AutomateBase(BaseModel):
|
class AutomateBase(BaseModel):
|
||||||
"""Use this class as a base model for automate related DTO."""
|
"""Use this class as a base model for automate related DTO."""
|
||||||
|
|
||||||
model_config = ConfigDict(alias_generator=camelcase, populate_by_name=True)
|
model_config = ConfigDict(alias_generator=to_camel, populate_by_name=True)
|
||||||
|
|
||||||
|
|
||||||
class VersionCreationTriggerPayload(AutomateBase):
|
class VersionCreationTriggerPayload(AutomateBase):
|
||||||
@@ -36,10 +36,10 @@ class AutomationRunData(BaseModel):
|
|||||||
automation_run_id: str
|
automation_run_id: str
|
||||||
function_run_id: str
|
function_run_id: str
|
||||||
|
|
||||||
triggers: List[VersionCreationTrigger]
|
triggers: list[VersionCreationTrigger]
|
||||||
|
|
||||||
model_config = ConfigDict(
|
model_config = ConfigDict(
|
||||||
alias_generator=camelcase, populate_by_name=True, protected_namespaces=()
|
alias_generator=to_camel, populate_by_name=True, protected_namespaces=()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -49,10 +49,10 @@ class TestAutomationRunData(BaseModel):
|
|||||||
automation_run_id: str
|
automation_run_id: str
|
||||||
function_run_id: str
|
function_run_id: str
|
||||||
|
|
||||||
triggers: List[VersionCreationTrigger]
|
triggers: list[VersionCreationTrigger]
|
||||||
|
|
||||||
model_config = ConfigDict(
|
model_config = ConfigDict(
|
||||||
alias_generator=camelcase, populate_by_name=True, protected_namespaces=()
|
alias_generator=to_camel, populate_by_name=True, protected_namespaces=()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -69,6 +69,7 @@ class AutomationStatus(str, Enum):
|
|||||||
class ObjectResultLevel(str, Enum):
|
class ObjectResultLevel(str, Enum):
|
||||||
"""Possible status message levels for object reports."""
|
"""Possible status message levels for object reports."""
|
||||||
|
|
||||||
|
SUCCESS = "SUCCESS"
|
||||||
INFO = "INFO"
|
INFO = "INFO"
|
||||||
WARNING = "WARNING"
|
WARNING = "WARNING"
|
||||||
ERROR = "ERROR"
|
ERROR = "ERROR"
|
||||||
@@ -79,19 +80,20 @@ class ResultCase(AutomateBase):
|
|||||||
|
|
||||||
category: str
|
category: str
|
||||||
level: ObjectResultLevel
|
level: ObjectResultLevel
|
||||||
object_ids: List[str]
|
object_app_ids: dict[str, str | None]
|
||||||
message: Optional[str]
|
message: str | None
|
||||||
metadata: Optional[Dict[str, Any]]
|
metadata: dict[str, Any] | None
|
||||||
visual_overrides: Optional[Dict[str, Any]]
|
visual_overrides: dict[str, Any] | None
|
||||||
|
|
||||||
|
|
||||||
class AutomationResult(AutomateBase):
|
class AutomationResult(AutomateBase):
|
||||||
"""Schema accepted by the Speckle server as a result for an automation run."""
|
"""Schema accepted by the Speckle server as a result for an automation run."""
|
||||||
|
|
||||||
elapsed: float = 0
|
elapsed: float = 0
|
||||||
result_view: Optional[str] = None
|
result_view: str | None = None
|
||||||
result_versions: List[str] = Field(default_factory=list)
|
result_versions: list[str] = Field(default_factory=list)
|
||||||
blobs: List[str] = Field(default_factory=list)
|
blobs: list[str] = Field(default_factory=list)
|
||||||
run_status: AutomationStatus = AutomationStatus.RUNNING
|
run_status: AutomationStatus = AutomationStatus.RUNNING
|
||||||
status_message: Optional[str] = None
|
status_message: str | None = None
|
||||||
object_results: list[ResultCase] = Field(default_factory=list)
|
object_results: list[ResultCase] = Field(default_factory=list)
|
||||||
|
version_result: dict[str, Any] | None = None
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import json
|
||||||
|
import time
|
||||||
|
import traceback
|
||||||
|
from argparse import ArgumentParser
|
||||||
|
from os import getenv
|
||||||
|
|
||||||
|
from speckleifc.main import open_and_convert_file
|
||||||
|
from specklepy.core.api.client import SpeckleClient
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
def cmd_line_import() -> None:
|
||||||
|
parser = ArgumentParser(
|
||||||
|
prog="speckleifc",
|
||||||
|
description="imports a file",
|
||||||
|
)
|
||||||
|
parser.add_argument("file_path")
|
||||||
|
parser.add_argument("output_path")
|
||||||
|
parser.add_argument("project_id")
|
||||||
|
parser.add_argument("version_message")
|
||||||
|
parser.add_argument("model_id")
|
||||||
|
# parser.add_argument("model_name")
|
||||||
|
# parser.add_argument("region_name")
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
TOKEN = getenv("USER_TOKEN")
|
||||||
|
assert TOKEN is not None
|
||||||
|
SERVER_URL = getenv("SPECKLE_SERVER_URL") or "http://127.0.0.1:3000"
|
||||||
|
|
||||||
|
metrics.set_host_app(
|
||||||
|
"ifc",
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
client = SpeckleClient(SERVER_URL, use_ssl=not SERVER_URL.startswith("http://"))
|
||||||
|
client.authenticate_with_token(TOKEN)
|
||||||
|
project = client.project.get(args.project_id)
|
||||||
|
|
||||||
|
version = open_and_convert_file(
|
||||||
|
args.file_path,
|
||||||
|
project,
|
||||||
|
args.version_message,
|
||||||
|
args.model_id,
|
||||||
|
client,
|
||||||
|
)
|
||||||
|
with open(args.output_path, "w") as f:
|
||||||
|
json.dump({"success": True, "commitId": version.id}, f)
|
||||||
|
except Exception as e:
|
||||||
|
error_msg = f"IFC Importer failed with exception:\n{traceback.format_exc()}"
|
||||||
|
print(error_msg)
|
||||||
|
|
||||||
|
# Write error result
|
||||||
|
with open(args.output_path, "w") as f:
|
||||||
|
json.dump({"success": False, "error": str(e)}, f)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
start = time.time()
|
||||||
|
cmd_line_import()
|
||||||
|
print(f"Total time (including cleanup): {(time.time() - start) * 1000}ms")
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
|
||||||
|
from speckleifc.property_extraction import extract_properties
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.objects.data_objects import DataObject
|
||||||
|
|
||||||
|
|
||||||
|
def data_object_to_speckle(
|
||||||
|
display_value: list[Base],
|
||||||
|
step_element: entity_instance,
|
||||||
|
children: list[Base],
|
||||||
|
current_storey: str | None = None,
|
||||||
|
) -> DataObject:
|
||||||
|
guid = cast(str, step_element.GlobalId)
|
||||||
|
name = cast(str, step_element.Name or guid)
|
||||||
|
|
||||||
|
properties = extract_properties(step_element)
|
||||||
|
|
||||||
|
# Add building storey information if available and not a building storey itself
|
||||||
|
if current_storey and not step_element.is_a("IfcBuildingStorey"):
|
||||||
|
properties["Building Storey"] = current_storey
|
||||||
|
|
||||||
|
data_object = DataObject(
|
||||||
|
applicationId=guid,
|
||||||
|
properties=properties,
|
||||||
|
name=name or guid,
|
||||||
|
displayValue=display_value,
|
||||||
|
)
|
||||||
|
|
||||||
|
data_object["@elements"] = children
|
||||||
|
data_object["ifcType"] = step_element.is_a()
|
||||||
|
|
||||||
|
return data_object
|
||||||
@@ -0,0 +1,130 @@
|
|||||||
|
from collections import defaultdict
|
||||||
|
from collections.abc import Sequence
|
||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from ifcopenshell.ifcopenshell_wrapper import (
|
||||||
|
Triangulation,
|
||||||
|
colour,
|
||||||
|
style,
|
||||||
|
)
|
||||||
|
|
||||||
|
from speckleifc.proxy_managers.render_material_proxy_manager import (
|
||||||
|
RenderMaterialProxyManager,
|
||||||
|
)
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.objects.geometry import Mesh
|
||||||
|
from specklepy.objects.other import RenderMaterial
|
||||||
|
|
||||||
|
|
||||||
|
def geometry_to_speckle(
|
||||||
|
geometry: Triangulation, render_material_manager: RenderMaterialProxyManager
|
||||||
|
) -> list[Base]:
|
||||||
|
materials = cast(Sequence[style], geometry.materials)
|
||||||
|
MESH_COUNT = max(len(materials), 1)
|
||||||
|
|
||||||
|
material_ids = cast(Sequence[int], geometry.material_ids)
|
||||||
|
faces = cast(Sequence[int], geometry.faces)
|
||||||
|
verts = cast(Sequence[float], geometry.verts)
|
||||||
|
normals = cast(Sequence[float], geometry.normals)
|
||||||
|
|
||||||
|
FACE_COUNT = len(material_ids)
|
||||||
|
|
||||||
|
if len(faces) != FACE_COUNT * 3 or FACE_COUNT == 0:
|
||||||
|
# Not really expected, but occasionally some meshes fail to triangulate
|
||||||
|
return []
|
||||||
|
|
||||||
|
mapped_meshes = _pre_alloc_mesh_lists(geometry, material_ids, MESH_COUNT)
|
||||||
|
for i, mesh in enumerate(mapped_meshes):
|
||||||
|
material = _material_to_speckle(materials[i])
|
||||||
|
render_material_manager.add_mesh_material_mapping(material, mesh)
|
||||||
|
|
||||||
|
mapped_faces_pointers = [0] * MESH_COUNT
|
||||||
|
mapped_vertices_pointers = [0] * MESH_COUNT
|
||||||
|
mapped_index_counters = [0] * MESH_COUNT
|
||||||
|
|
||||||
|
i = 0
|
||||||
|
face_index = 0
|
||||||
|
while i < FACE_COUNT:
|
||||||
|
mesh_index = material_ids[i]
|
||||||
|
mesh: Mesh = mapped_meshes[mesh_index]
|
||||||
|
|
||||||
|
face_ptr = mapped_faces_pointers[mesh_index]
|
||||||
|
vert_ptr = mapped_vertices_pointers[mesh_index]
|
||||||
|
|
||||||
|
# Add triangle
|
||||||
|
mesh.faces[face_ptr] = 3
|
||||||
|
for j in range(3):
|
||||||
|
# Add vert
|
||||||
|
mesh.faces[face_ptr + 1 + j] = mapped_index_counters[mesh_index] + j
|
||||||
|
vert_index = faces[face_index + j] * 3
|
||||||
|
mapped_vert_offset = vert_ptr + (j * 3)
|
||||||
|
|
||||||
|
mesh.vertices[mapped_vert_offset] = verts[vert_index]
|
||||||
|
mesh.vertices[mapped_vert_offset + 1] = verts[vert_index + 1]
|
||||||
|
mesh.vertices[mapped_vert_offset + 2] = verts[vert_index + 2]
|
||||||
|
|
||||||
|
mesh.vertexNormals[mapped_vert_offset] = normals[vert_index]
|
||||||
|
mesh.vertexNormals[mapped_vert_offset + 1] = normals[vert_index + 1]
|
||||||
|
mesh.vertexNormals[mapped_vert_offset + 2] = normals[vert_index + 2]
|
||||||
|
|
||||||
|
i += 1
|
||||||
|
face_index += 3 # number of items in the faces list we just jumped over
|
||||||
|
|
||||||
|
mapped_index_counters[mesh_index] += (
|
||||||
|
3 # number of verts we just added to the mesh.vertices i.e. the next index
|
||||||
|
)
|
||||||
|
mapped_faces_pointers[mesh_index] += (
|
||||||
|
4 # number of item's we've just added to the mesh.faces list
|
||||||
|
)
|
||||||
|
mapped_vertices_pointers[mesh_index] += (
|
||||||
|
9 # number of item's we've just added to the mesh.vertices list
|
||||||
|
)
|
||||||
|
|
||||||
|
return mapped_meshes # type: ignore
|
||||||
|
|
||||||
|
|
||||||
|
def _material_to_speckle(material: style) -> RenderMaterial:
|
||||||
|
return RenderMaterial(
|
||||||
|
applicationId=material.calc_hash(),
|
||||||
|
name=material.name,
|
||||||
|
diffuse=_color_to_argb(material.diffuse),
|
||||||
|
opacity=1 - material.transparency if material.has_transparency() else 1,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _color_to_argb(colour: colour) -> int:
|
||||||
|
# Clamp values to [0, 1] and convert to 0–255
|
||||||
|
a_int = 255
|
||||||
|
r_int = max(0, min(255, int(round(colour.r() * 255))))
|
||||||
|
g_int = max(0, min(255, int(round(colour.g() * 255))))
|
||||||
|
b_int = max(0, min(255, int(round(colour.b() * 255))))
|
||||||
|
|
||||||
|
return (a_int << 24) | (r_int << 16) | (g_int << 8) | b_int
|
||||||
|
|
||||||
|
|
||||||
|
def _pre_alloc_mesh_lists(
|
||||||
|
geometry: Triangulation, material_ids: Sequence[int], MESH_COUNT: int
|
||||||
|
) -> list[Mesh]:
|
||||||
|
"""
|
||||||
|
This is a performance optimisation to pre-size the lists
|
||||||
|
since we're expecting potential hundreds of thousands of verts in a single model
|
||||||
|
This is very much in the hot path, so worth the extra bit of convoluted logic
|
||||||
|
"""
|
||||||
|
appId = cast(str, geometry.id)
|
||||||
|
|
||||||
|
material_face_counts = defaultdict(int)
|
||||||
|
for mat_id in material_ids:
|
||||||
|
material_face_counts[mat_id] += 1
|
||||||
|
|
||||||
|
meshes = []
|
||||||
|
for mat_id in range(MESH_COUNT):
|
||||||
|
face_count = material_face_counts.get(mat_id, 0)
|
||||||
|
mesh = Mesh(
|
||||||
|
units="m",
|
||||||
|
vertices=[-1] * (face_count * 9),
|
||||||
|
vertexNormals=[-1] * (face_count * 9),
|
||||||
|
faces=[-1] * (face_count * 4), # 1 marker + 3 vertex indices
|
||||||
|
applicationId=f"{appId}_mat{mat_id}",
|
||||||
|
)
|
||||||
|
meshes.append(mesh)
|
||||||
|
return meshes
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.objects.models.collections.collection import Collection
|
||||||
|
|
||||||
|
|
||||||
|
def project_to_speckle(
|
||||||
|
step_element: entity_instance, children: list[Base]
|
||||||
|
) -> Collection:
|
||||||
|
guid = cast(str, step_element.GlobalId)
|
||||||
|
name = cast(str, step_element.Name or step_element.LongName or guid)
|
||||||
|
|
||||||
|
project = Collection(applicationId=guid, name=name, elements=children)
|
||||||
|
|
||||||
|
project["ifcType"] = step_element.is_a()
|
||||||
|
project["description"] = step_element.Description
|
||||||
|
project["objectType"] = step_element.ObjectType
|
||||||
|
project["longName"] = step_element.LongName
|
||||||
|
project["phase"] = step_element.Phase
|
||||||
|
|
||||||
|
return project
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
|
||||||
|
from speckleifc.property_extraction import extract_properties
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.objects.data_objects import DataObject
|
||||||
|
from specklepy.objects.models.collections.collection import Collection
|
||||||
|
|
||||||
|
|
||||||
|
def spatial_element_to_speckle(
|
||||||
|
display_value: list[Base],
|
||||||
|
step_element: entity_instance,
|
||||||
|
relational_children: list[Base],
|
||||||
|
current_storey: str | None = None,
|
||||||
|
) -> Collection:
|
||||||
|
direct_geometry = _convert_as_data_object(
|
||||||
|
display_value, step_element, current_storey
|
||||||
|
)
|
||||||
|
all_children = [direct_geometry] + relational_children
|
||||||
|
|
||||||
|
guid = cast(str, step_element.GlobalId)
|
||||||
|
name = cast(str, step_element.Name or step_element.LongName or guid)
|
||||||
|
|
||||||
|
data_object = Collection(applicationId=guid, name=name, elements=all_children)
|
||||||
|
data_object["ifcType"] = step_element.is_a()
|
||||||
|
|
||||||
|
return data_object
|
||||||
|
|
||||||
|
|
||||||
|
def _convert_as_data_object(
|
||||||
|
display_value: list[Base],
|
||||||
|
step_element: entity_instance,
|
||||||
|
current_storey: str | None = None,
|
||||||
|
) -> DataObject:
|
||||||
|
guid = cast(str, step_element.GlobalId)
|
||||||
|
name = cast(str, step_element.Name or step_element.LongName or guid)
|
||||||
|
|
||||||
|
properties = extract_properties(step_element)
|
||||||
|
|
||||||
|
# Add building storey information if available and not a building storey itself
|
||||||
|
if current_storey and not step_element.is_a("IfcBuildingStorey"):
|
||||||
|
properties["Building Storey"] = current_storey
|
||||||
|
|
||||||
|
data_object = DataObject(
|
||||||
|
applicationId=guid,
|
||||||
|
properties=properties,
|
||||||
|
name=name,
|
||||||
|
displayValue=display_value,
|
||||||
|
)
|
||||||
|
|
||||||
|
data_object["ifcType"] = step_element.is_a()
|
||||||
|
|
||||||
|
return data_object
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
import multiprocessing
|
||||||
|
|
||||||
|
from ifcopenshell import SchemaError, file, ifcopenshell_wrapper, open, sqlite
|
||||||
|
from ifcopenshell.geom import iterator, settings
|
||||||
|
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
|
||||||
|
|
||||||
|
def _create_iterator_settings() -> settings:
|
||||||
|
ifc_settings = settings()
|
||||||
|
# triangles for now, speckle does support n-gons, but may be less performant
|
||||||
|
ifc_settings.set("triangulation-type", ifcopenshell_wrapper.TRIANGLE_MESH)
|
||||||
|
# no need to weld verts
|
||||||
|
ifc_settings.set("weld-vertices", False)
|
||||||
|
#
|
||||||
|
ifc_settings.set("use-world-coords", False)
|
||||||
|
ifc_settings.set("permissive-shape-reuse", True)
|
||||||
|
|
||||||
|
# Tiny performance improvement,
|
||||||
|
ifc_settings.set("no-wire-intersection-check", True)
|
||||||
|
# Rendermaterials inherit the material names instead of type + unique id
|
||||||
|
ifc_settings.set("use-material-names", True)
|
||||||
|
|
||||||
|
# IfcOpenshell defaults to 0.001mm here, which leads to very dense meshes.
|
||||||
|
# lowering the mesh quality a bit here leads to meshes
|
||||||
|
# that are still much higher quality than webifc
|
||||||
|
|
||||||
|
# We still need to experiment with the affect on memory usage
|
||||||
|
# It may be desirable to lower this further, and increase the angular deflection
|
||||||
|
# to compensate. This would allow large meshes to be lower quality,
|
||||||
|
# while keeping small meshes relatively similar.
|
||||||
|
ifc_settings.set("mesher-linear-deflection", 0.2)
|
||||||
|
|
||||||
|
return ifc_settings
|
||||||
|
|
||||||
|
|
||||||
|
def open_ifc(file_path: str) -> file:
|
||||||
|
try:
|
||||||
|
ifc_file = open(file_path)
|
||||||
|
except SchemaError:
|
||||||
|
raise
|
||||||
|
except FileNotFoundError:
|
||||||
|
raise
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException("File could not be opened as an IFC file") from ex
|
||||||
|
|
||||||
|
if isinstance(ifc_file, file):
|
||||||
|
return ifc_file
|
||||||
|
else:
|
||||||
|
raise SpeckleException(f"file at {file_path} is not a compatible ifc file type")
|
||||||
|
|
||||||
|
|
||||||
|
def create_geometry_iterator(ifc_file: file | sqlite) -> iterator:
|
||||||
|
return iterator(_create_iterator_settings(), ifc_file, multiprocessing.cpu_count())
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
from collections.abc import Generator, Iterable
|
||||||
|
from itertools import chain
|
||||||
|
from typing import cast
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
|
||||||
|
|
||||||
|
def get_children(step_element: entity_instance) -> Generator[entity_instance]:
|
||||||
|
yield from chain(
|
||||||
|
get_spatial_children(step_element), get_aggregate_children(step_element)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_spatial_children(step_element: entity_instance) -> Generator[entity_instance]:
|
||||||
|
spatial_relations = cast(
|
||||||
|
Iterable[entity_instance] | None,
|
||||||
|
getattr(step_element, "ContainsElements", None),
|
||||||
|
)
|
||||||
|
if spatial_relations is not None:
|
||||||
|
for relation in spatial_relations:
|
||||||
|
yield from cast(Iterable[entity_instance], relation.RelatedElements)
|
||||||
|
|
||||||
|
|
||||||
|
def get_aggregate_children(step_element: entity_instance) -> Generator[entity_instance]:
|
||||||
|
aggregate_relations = cast(
|
||||||
|
Iterable[entity_instance] | None,
|
||||||
|
getattr(step_element, "IsDecomposedBy", None),
|
||||||
|
)
|
||||||
|
if aggregate_relations is not None:
|
||||||
|
for relation in aggregate_relations:
|
||||||
|
yield from cast(Iterable[entity_instance], relation.RelatedObjects)
|
||||||
@@ -0,0 +1,207 @@
|
|||||||
|
import time
|
||||||
|
from dataclasses import dataclass, field
|
||||||
|
from typing import List, cast
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
from ifcopenshell.geom import file
|
||||||
|
from ifcopenshell.ifcopenshell_wrapper import Triangulation, TriangulationElement
|
||||||
|
|
||||||
|
from speckleifc.converter.data_object_converter import data_object_to_speckle
|
||||||
|
from speckleifc.converter.geometry_converter import geometry_to_speckle
|
||||||
|
from speckleifc.converter.project_converter import project_to_speckle
|
||||||
|
from speckleifc.converter.spatial_element_converter import spatial_element_to_speckle
|
||||||
|
from speckleifc.ifc_geometry_processing import create_geometry_iterator
|
||||||
|
from speckleifc.ifc_openshell_helpers import get_children
|
||||||
|
from speckleifc.proxy_managers.instance_proxy_manager import InstanceProxyManager
|
||||||
|
from speckleifc.proxy_managers.level_proxy_manager import LevelProxyManager
|
||||||
|
from speckleifc.proxy_managers.render_material_proxy_manager import (
|
||||||
|
RenderMaterialProxyManager,
|
||||||
|
)
|
||||||
|
from specklepy.logging.exceptions import SpeckleException
|
||||||
|
from specklepy.objects import Base
|
||||||
|
from specklepy.objects.data_objects import DataObject
|
||||||
|
from specklepy.objects.models.collections.collection import Collection
|
||||||
|
from specklepy.objects.proxies import InstanceProxy
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class ImportJob:
|
||||||
|
ifc_file: file
|
||||||
|
|
||||||
|
_render_material_manager: RenderMaterialProxyManager = field(
|
||||||
|
default_factory=lambda: RenderMaterialProxyManager()
|
||||||
|
)
|
||||||
|
_level_proxy_manager: LevelProxyManager = field(
|
||||||
|
default_factory=lambda: LevelProxyManager()
|
||||||
|
)
|
||||||
|
_instance_proxy_manager: InstanceProxyManager = field(
|
||||||
|
default_factory=lambda: InstanceProxyManager()
|
||||||
|
)
|
||||||
|
geometries_count: int = 0
|
||||||
|
geometries_used: int = 0
|
||||||
|
_current_storey_data_object: DataObject | None = field(default=None, init=False)
|
||||||
|
|
||||||
|
_display_value_cache: dict[int, list[Base]] = field(default_factory=dict)
|
||||||
|
"""Maps an instance step ID to a list of instances"""
|
||||||
|
|
||||||
|
def convert_element(self, step_element: entity_instance) -> Base:
|
||||||
|
try:
|
||||||
|
return self._convert_element(step_element)
|
||||||
|
except SpeckleException:
|
||||||
|
raise
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Failed to convert {step_element.is_a()} #{step_element.id()}"
|
||||||
|
) from ex
|
||||||
|
|
||||||
|
def _convert_element(self, step_element: entity_instance) -> Base:
|
||||||
|
# Track current storey context and store for level proxies
|
||||||
|
previous_storey_data_object = self._current_storey_data_object
|
||||||
|
if step_element.is_a("IfcBuildingStorey"):
|
||||||
|
# Convert the building storey to a DataObject for the level proxy
|
||||||
|
storey_display_value = self._display_value_cache.get(step_element.id(), [])
|
||||||
|
self._current_storey_data_object = data_object_to_speckle(
|
||||||
|
storey_display_value, step_element, []
|
||||||
|
)
|
||||||
|
|
||||||
|
children = self._convert_children(step_element)
|
||||||
|
id = step_element.id()
|
||||||
|
display_value = self._display_value_cache.get(id, [])
|
||||||
|
|
||||||
|
if display_value:
|
||||||
|
self.geometries_used += 1
|
||||||
|
|
||||||
|
# Extract current storey name from DataObject if available
|
||||||
|
current_storey_name = (
|
||||||
|
self._current_storey_data_object.name
|
||||||
|
if self._current_storey_data_object
|
||||||
|
else None
|
||||||
|
)
|
||||||
|
|
||||||
|
if step_element.is_a("IfcProject"):
|
||||||
|
result = project_to_speckle(step_element, children)
|
||||||
|
elif step_element.is_a("IfcSpatialStructureElement"):
|
||||||
|
result = spatial_element_to_speckle(
|
||||||
|
display_value, step_element, children, current_storey_name
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
result = data_object_to_speckle(
|
||||||
|
display_value, step_element, children, current_storey_name
|
||||||
|
)
|
||||||
|
# Associate non-spatial elements with current storey for level proxies
|
||||||
|
if self._current_storey_data_object is not None and result.applicationId:
|
||||||
|
self._level_proxy_manager.add_element_level_mapping(
|
||||||
|
self._current_storey_data_object, result.applicationId
|
||||||
|
)
|
||||||
|
|
||||||
|
# Restore previous storey context
|
||||||
|
self._current_storey_data_object = previous_storey_data_object
|
||||||
|
return result
|
||||||
|
|
||||||
|
def _convert_children(self, step_element: entity_instance) -> list[Base]:
|
||||||
|
return [
|
||||||
|
self.convert_element(i)
|
||||||
|
for i in get_children(step_element)
|
||||||
|
if self._should_convert(i)
|
||||||
|
]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _should_convert(step_element: entity_instance) -> bool:
|
||||||
|
# We only consider IfcRoot objects convertible
|
||||||
|
# This is the super class for root level entities that have a GUID...
|
||||||
|
# This will ignore some types like IfcGridAxis
|
||||||
|
s = step_element.is_a("IfcRoot")
|
||||||
|
if not s:
|
||||||
|
print(
|
||||||
|
f"Skipping #{step_element.id()} because it's type ({step_element.is_a()}) it not an IfcRoot" # noqa: E501
|
||||||
|
)
|
||||||
|
return s
|
||||||
|
|
||||||
|
def convert(self) -> Base:
|
||||||
|
start = time.time()
|
||||||
|
self.pre_process_geometry()
|
||||||
|
print(f"Geometry conversion complete after {(time.time() - start) * 1000}ms")
|
||||||
|
print(f"Created {self.geometries_count} geometries")
|
||||||
|
|
||||||
|
start = time.time()
|
||||||
|
root = self._convert_project_tree()
|
||||||
|
print(f"Object tree conversion complete after {(time.time() - start) * 1000}ms")
|
||||||
|
print(f"Used {self.geometries_used} geometries")
|
||||||
|
return root
|
||||||
|
|
||||||
|
def pre_process_geometry(self) -> None:
|
||||||
|
iterator = create_geometry_iterator(self.ifc_file)
|
||||||
|
if not iterator.initialize():
|
||||||
|
raise SpeckleException("Failed to find any geometry in file")
|
||||||
|
self.geometries_count = 0
|
||||||
|
while True:
|
||||||
|
shape = cast(TriangulationElement, iterator.get())
|
||||||
|
self.geometries_count += 1
|
||||||
|
id = cast(int, shape.id)
|
||||||
|
try:
|
||||||
|
display_value = self._create_display_value(shape)
|
||||||
|
self._display_value_cache[id] = display_value
|
||||||
|
except Exception as ex:
|
||||||
|
raise SpeckleException(
|
||||||
|
f"Failed to convert geometry with id: {id}"
|
||||||
|
) from ex
|
||||||
|
if not iterator.next():
|
||||||
|
break
|
||||||
|
|
||||||
|
def _create_display_value(self, shape: TriangulationElement) -> List[Base]:
|
||||||
|
geometry = cast(Triangulation, shape.geometry)
|
||||||
|
display_value_geometry = geometry_to_speckle(
|
||||||
|
geometry, self._render_material_manager
|
||||||
|
)
|
||||||
|
|
||||||
|
definition_ids = self._instance_proxy_manager.add_display_value_definitions(
|
||||||
|
display_value_geometry
|
||||||
|
)
|
||||||
|
matrix = shape.transformation.matrix
|
||||||
|
transposed = [
|
||||||
|
matrix[0], matrix[4], matrix[8], matrix[12],
|
||||||
|
matrix[1], matrix[5], matrix[9], matrix[13],
|
||||||
|
matrix[2], matrix[6], matrix[10], matrix[14],
|
||||||
|
matrix[3], matrix[7], matrix[11], matrix[15],
|
||||||
|
] # fmt: skip
|
||||||
|
|
||||||
|
return [
|
||||||
|
cast(
|
||||||
|
Base,
|
||||||
|
InstanceProxy(
|
||||||
|
units="m",
|
||||||
|
definitionId=definition_id,
|
||||||
|
transform=transposed,
|
||||||
|
maxDepth=0,
|
||||||
|
applicationId=f"{shape.guid}:{definition_id}",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
for definition_id in definition_ids
|
||||||
|
]
|
||||||
|
|
||||||
|
def _convert_project_tree(self) -> Base:
|
||||||
|
projects = self.ifc_file.by_type("IfcProject", False)
|
||||||
|
if len(projects) != 1:
|
||||||
|
raise SpeckleException("Expected exactly one IfcProject in file")
|
||||||
|
project = projects[0]
|
||||||
|
|
||||||
|
tree = self.convert_element(project)
|
||||||
|
if not isinstance(tree, Collection):
|
||||||
|
raise TypeError("Expected root object to convert to a Collection")
|
||||||
|
|
||||||
|
tree["renderMaterialProxies"] = list(
|
||||||
|
self._render_material_manager.render_material_proxies.values()
|
||||||
|
)
|
||||||
|
tree["levelProxies"] = list(self._level_proxy_manager.level_proxies.values())
|
||||||
|
tree["instanceDefinitionProxies"] = list(
|
||||||
|
self._instance_proxy_manager.instance_definition_proxies.values()
|
||||||
|
)
|
||||||
|
tree.elements.append(
|
||||||
|
Collection(
|
||||||
|
name="definitionGeometry",
|
||||||
|
elements=list(self._instance_proxy_manager.instance_geometry.values()),
|
||||||
|
)
|
||||||
|
)
|
||||||
|
tree["version"] = 3
|
||||||
|
|
||||||
|
return tree
|
||||||
@@ -0,0 +1,621 @@
|
|||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The GNU General Public License is a free, copyleft license for
|
||||||
|
software and other kinds of works.
|
||||||
|
|
||||||
|
The licenses for most software and other practical works are designed
|
||||||
|
to take away your freedom to share and change the works. By contrast,
|
||||||
|
the GNU General Public License is intended to guarantee your freedom to
|
||||||
|
share and change all versions of a program--to make sure it remains free
|
||||||
|
software for all its users. We, the Free Software Foundation, use the
|
||||||
|
GNU General Public License for most of our software; it applies also to
|
||||||
|
any other work released this way by its authors. You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
them if you wish), that you receive source code or can get it if you
|
||||||
|
want it, that you can change the software or use pieces of it in new
|
||||||
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to prevent others from denying you
|
||||||
|
these rights or asking you to surrender the rights. Therefore, you have
|
||||||
|
certain responsibilities if you distribute copies of the software, or if
|
||||||
|
you modify it: responsibilities to respect the freedom of others.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must pass on to the recipients the same
|
||||||
|
freedoms that you received. You must make sure that they, too, receive
|
||||||
|
or can get the source code. And you must show them these terms so they
|
||||||
|
know their rights.
|
||||||
|
|
||||||
|
Developers that use the GNU GPL protect your rights with two steps:
|
||||||
|
(1) assert copyright on the software, and (2) offer you this License
|
||||||
|
giving you legal permission to copy, distribute and/or modify it.
|
||||||
|
|
||||||
|
For the developers' and authors' protection, the GPL clearly explains
|
||||||
|
that there is no warranty for this free software. For both users' and
|
||||||
|
authors' sake, the GPL requires that modified versions be marked as
|
||||||
|
changed, so that their problems will not be attributed erroneously to
|
||||||
|
authors of previous versions.
|
||||||
|
|
||||||
|
Some devices are designed to deny users access to install or run
|
||||||
|
modified versions of the software inside them, although the manufacturer
|
||||||
|
can do so. This is fundamentally incompatible with the aim of
|
||||||
|
protecting users' freedom to change the software. The systematic
|
||||||
|
pattern of such abuse occurs in the area of products for individuals to
|
||||||
|
use, which is precisely where it is most unacceptable. Therefore, we
|
||||||
|
have designed this version of the GPL to prohibit the practice for those
|
||||||
|
products. If such problems arise substantially in other domains, we
|
||||||
|
stand ready to extend this provision to those domains in future versions
|
||||||
|
of the GPL, as needed to protect the freedom of users.
|
||||||
|
|
||||||
|
Finally, every program is threatened constantly by software patents.
|
||||||
|
States should not allow patents to restrict development and use of
|
||||||
|
software on general-purpose computers, but in those that do, we wish to
|
||||||
|
avoid the special danger that patents applied to a free program could
|
||||||
|
make it effectively proprietary. To prevent this, the GPL assures that
|
||||||
|
patents cannot be used to render the program non-free.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
0. Definitions.
|
||||||
|
|
||||||
|
"This License" refers to version 3 of the GNU General Public License.
|
||||||
|
|
||||||
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
|
works, such as semiconductor masks.
|
||||||
|
|
||||||
|
"The Program" refers to any copyrightable work licensed under this
|
||||||
|
License. Each licensee is addressed as "you". "Licensees" and
|
||||||
|
"recipients" may be individuals or organizations.
|
||||||
|
|
||||||
|
To "modify" a work means to copy from or adapt all or part of the work
|
||||||
|
in a fashion requiring copyright permission, other than the making of an
|
||||||
|
exact copy. The resulting work is called a "modified version" of the
|
||||||
|
earlier work or a work "based on" the earlier work.
|
||||||
|
|
||||||
|
A "covered work" means either the unmodified Program or a work based
|
||||||
|
on the Program.
|
||||||
|
|
||||||
|
To "propagate" a work means to do anything with it that, without
|
||||||
|
permission, would make you directly or secondarily liable for
|
||||||
|
infringement under applicable copyright law, except executing it on a
|
||||||
|
computer or modifying a private copy. Propagation includes copying,
|
||||||
|
distribution (with or without modification), making available to the
|
||||||
|
public, and in some countries other activities as well.
|
||||||
|
|
||||||
|
To "convey" a work means any kind of propagation that enables other
|
||||||
|
parties to make or receive copies. Mere interaction with a user through
|
||||||
|
a computer network, with no transfer of a copy, is not conveying.
|
||||||
|
|
||||||
|
An interactive user interface displays "Appropriate Legal Notices"
|
||||||
|
to the extent that it includes a convenient and prominently visible
|
||||||
|
feature that (1) displays an appropriate copyright notice, and (2)
|
||||||
|
tells the user that there is no warranty for the work (except to the
|
||||||
|
extent that warranties are provided), that licensees may convey the
|
||||||
|
work under this License, and how to view a copy of this License. If
|
||||||
|
the interface presents a list of user commands or options, such as a
|
||||||
|
menu, a prominent item in the list meets this criterion.
|
||||||
|
|
||||||
|
1. Source Code.
|
||||||
|
|
||||||
|
The "source code" for a work means the preferred form of the work
|
||||||
|
for making modifications to it. "Object code" means any non-source
|
||||||
|
form of a work.
|
||||||
|
|
||||||
|
A "Standard Interface" means an interface that either is an official
|
||||||
|
standard defined by a recognized standards body, or, in the case of
|
||||||
|
interfaces specified for a particular programming language, one that
|
||||||
|
is widely used among developers working in that language.
|
||||||
|
|
||||||
|
The "System Libraries" of an executable work include anything, other
|
||||||
|
than the work as a whole, that (a) is included in the normal form of
|
||||||
|
packaging a Major Component, but which is not part of that Major
|
||||||
|
Component, and (b) serves only to enable use of the work with that
|
||||||
|
Major Component, or to implement a Standard Interface for which an
|
||||||
|
implementation is available to the public in source code form. A
|
||||||
|
"Major Component", in this context, means a major essential component
|
||||||
|
(kernel, window system, and so on) of the specific operating system
|
||||||
|
(if any) on which the executable work runs, or a compiler used to
|
||||||
|
produce the work, or an object code interpreter used to run it.
|
||||||
|
|
||||||
|
The "Corresponding Source" for a work in object code form means all
|
||||||
|
the source code needed to generate, install, and (for an executable
|
||||||
|
work) run the object code and to modify the work, including scripts to
|
||||||
|
control those activities. However, it does not include the work's
|
||||||
|
System Libraries, or general-purpose tools or generally available free
|
||||||
|
programs which are used unmodified in performing those activities but
|
||||||
|
which are not part of the work. For example, Corresponding Source
|
||||||
|
includes interface definition files associated with source files for
|
||||||
|
the work, and the source code for shared libraries and dynamically
|
||||||
|
linked subprograms that the work is specifically designed to require,
|
||||||
|
such as by intimate data communication or control flow between those
|
||||||
|
subprograms and other parts of the work.
|
||||||
|
|
||||||
|
The Corresponding Source need not include anything that users
|
||||||
|
can regenerate automatically from other parts of the Corresponding
|
||||||
|
Source.
|
||||||
|
|
||||||
|
The Corresponding Source for a work in source code form is that
|
||||||
|
same work.
|
||||||
|
|
||||||
|
2. Basic Permissions.
|
||||||
|
|
||||||
|
All rights granted under this License are granted for the term of
|
||||||
|
copyright on the Program, and are irrevocable provided the stated
|
||||||
|
conditions are met. This License explicitly affirms your unlimited
|
||||||
|
permission to run the unmodified Program. The output from running a
|
||||||
|
covered work is covered by this License only if the output, given its
|
||||||
|
content, constitutes a covered work. This License acknowledges your
|
||||||
|
rights of fair use or other equivalent, as provided by copyright law.
|
||||||
|
|
||||||
|
You may make, run and propagate covered works that you do not
|
||||||
|
convey, without conditions so long as your license otherwise remains
|
||||||
|
in force. You may convey covered works to others for the sole purpose
|
||||||
|
of having them make modifications exclusively for you, or provide you
|
||||||
|
with facilities for running those works, provided that you comply with
|
||||||
|
the terms of this License in conveying all material for which you do
|
||||||
|
not control copyright. Those thus making or running the covered works
|
||||||
|
for you must do so exclusively on your behalf, under your direction
|
||||||
|
and control, on terms that prohibit them from making any copies of
|
||||||
|
your copyrighted material outside their relationship with you.
|
||||||
|
|
||||||
|
Conveying under any other circumstances is permitted solely under
|
||||||
|
the conditions stated below. Sublicensing is not allowed; section 10
|
||||||
|
makes it unnecessary.
|
||||||
|
|
||||||
|
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||||
|
|
||||||
|
No covered work shall be deemed part of an effective technological
|
||||||
|
measure under any applicable law fulfilling obligations under article
|
||||||
|
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||||
|
similar laws prohibiting or restricting circumvention of such
|
||||||
|
measures.
|
||||||
|
|
||||||
|
When you convey a covered work, you waive any legal power to forbid
|
||||||
|
circumvention of technological measures to the extent such circumvention
|
||||||
|
is effected by exercising rights under this License with respect to
|
||||||
|
the covered work, and you disclaim any intention to limit operation or
|
||||||
|
modification of the work as a means of enforcing, against the work's
|
||||||
|
users, your or third parties' legal rights to forbid circumvention of
|
||||||
|
technological measures.
|
||||||
|
|
||||||
|
4. Conveying Verbatim Copies.
|
||||||
|
|
||||||
|
You may convey verbatim copies of the Program's source code as you
|
||||||
|
receive it, in any medium, provided that you conspicuously and
|
||||||
|
appropriately publish on each copy an appropriate copyright notice;
|
||||||
|
keep intact all notices stating that this License and any
|
||||||
|
non-permissive terms added in accord with section 7 apply to the code;
|
||||||
|
keep intact all notices of the absence of any warranty; and give all
|
||||||
|
recipients a copy of this License along with the Program.
|
||||||
|
|
||||||
|
You may charge any price or no price for each copy that you convey,
|
||||||
|
and you may offer support or warranty protection for a fee.
|
||||||
|
|
||||||
|
5. Conveying Modified Source Versions.
|
||||||
|
|
||||||
|
You may convey a work based on the Program, or the modifications to
|
||||||
|
produce it from the Program, in the form of source code under the
|
||||||
|
terms of section 4, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The work must carry prominent notices stating that you modified
|
||||||
|
it, and giving a relevant date.
|
||||||
|
|
||||||
|
b) The work must carry prominent notices stating that it is
|
||||||
|
released under this License and any conditions added under section
|
||||||
|
7. This requirement modifies the requirement in section 4 to
|
||||||
|
"keep intact all notices".
|
||||||
|
|
||||||
|
c) You must license the entire work, as a whole, under this
|
||||||
|
License to anyone who comes into possession of a copy. This
|
||||||
|
License will therefore apply, along with any applicable section 7
|
||||||
|
additional terms, to the whole of the work, and all its parts,
|
||||||
|
regardless of how they are packaged. This License gives no
|
||||||
|
permission to license the work in any other way, but it does not
|
||||||
|
invalidate such permission if you have separately received it.
|
||||||
|
|
||||||
|
d) If the work has interactive user interfaces, each must display
|
||||||
|
Appropriate Legal Notices; however, if the Program has interactive
|
||||||
|
interfaces that do not display Appropriate Legal Notices, your
|
||||||
|
work need not make them do so.
|
||||||
|
|
||||||
|
A compilation of a covered work with other separate and independent
|
||||||
|
works, which are not by their nature extensions of the covered work,
|
||||||
|
and which are not combined with it such as to form a larger program,
|
||||||
|
in or on a volume of a storage or distribution medium, is called an
|
||||||
|
"aggregate" if the compilation and its resulting copyright are not
|
||||||
|
used to limit the access or legal rights of the compilation's users
|
||||||
|
beyond what the individual works permit. Inclusion of a covered work
|
||||||
|
in an aggregate does not cause this License to apply to the other
|
||||||
|
parts of the aggregate.
|
||||||
|
|
||||||
|
6. Conveying Non-Source Forms.
|
||||||
|
|
||||||
|
You may convey a covered work in object code form under the terms
|
||||||
|
of sections 4 and 5, provided that you also convey the
|
||||||
|
machine-readable Corresponding Source under the terms of this License,
|
||||||
|
in one of these ways:
|
||||||
|
|
||||||
|
a) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by the
|
||||||
|
Corresponding Source fixed on a durable physical medium
|
||||||
|
customarily used for software interchange.
|
||||||
|
|
||||||
|
b) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by a
|
||||||
|
written offer, valid for at least three years and valid for as
|
||||||
|
long as you offer spare parts or customer support for that product
|
||||||
|
model, to give anyone who possesses the object code either (1) a
|
||||||
|
copy of the Corresponding Source for all the software in the
|
||||||
|
product that is covered by this License, on a durable physical
|
||||||
|
medium customarily used for software interchange, for a price no
|
||||||
|
more than your reasonable cost of physically performing this
|
||||||
|
conveying of source, or (2) access to copy the
|
||||||
|
Corresponding Source from a network server at no charge.
|
||||||
|
|
||||||
|
c) Convey individual copies of the object code with a copy of the
|
||||||
|
written offer to provide the Corresponding Source. This
|
||||||
|
alternative is allowed only occasionally and noncommercially, and
|
||||||
|
only if you received the object code with such an offer, in accord
|
||||||
|
with subsection 6b.
|
||||||
|
|
||||||
|
d) Convey the object code by offering access from a designated
|
||||||
|
place (gratis or for a charge), and offer equivalent access to the
|
||||||
|
Corresponding Source in the same way through the same place at no
|
||||||
|
further charge. You need not require recipients to copy the
|
||||||
|
Corresponding Source along with the object code. If the place to
|
||||||
|
copy the object code is a network server, the Corresponding Source
|
||||||
|
may be on a different server (operated by you or a third party)
|
||||||
|
that supports equivalent copying facilities, provided you maintain
|
||||||
|
clear directions next to the object code saying where to find the
|
||||||
|
Corresponding Source. Regardless of what server hosts the
|
||||||
|
Corresponding Source, you remain obligated to ensure that it is
|
||||||
|
available for as long as needed to satisfy these requirements.
|
||||||
|
|
||||||
|
e) Convey the object code using peer-to-peer transmission, provided
|
||||||
|
you inform other peers where the object code and Corresponding
|
||||||
|
Source of the work are being offered to the general public at no
|
||||||
|
charge under subsection 6d.
|
||||||
|
|
||||||
|
A separable portion of the object code, whose source code is excluded
|
||||||
|
from the Corresponding Source as a System Library, need not be
|
||||||
|
included in conveying the object code work.
|
||||||
|
|
||||||
|
A "User Product" is either (1) a "consumer product", which means any
|
||||||
|
tangible personal property which is normally used for personal, family,
|
||||||
|
or household purposes, or (2) anything designed or sold for incorporation
|
||||||
|
into a dwelling. In determining whether a product is a consumer product,
|
||||||
|
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||||
|
product received by a particular user, "normally used" refers to a
|
||||||
|
typical or common use of that class of product, regardless of the status
|
||||||
|
of the particular user or of the way in which the particular user
|
||||||
|
actually uses, or expects or is expected to use, the product. A product
|
||||||
|
is a consumer product regardless of whether the product has substantial
|
||||||
|
commercial, industrial or non-consumer uses, unless such uses represent
|
||||||
|
the only significant mode of use of the product.
|
||||||
|
|
||||||
|
"Installation Information" for a User Product means any methods,
|
||||||
|
procedures, authorization keys, or other information required to install
|
||||||
|
and execute modified versions of a covered work in that User Product from
|
||||||
|
a modified version of its Corresponding Source. The information must
|
||||||
|
suffice to ensure that the continued functioning of the modified object
|
||||||
|
code is in no case prevented or interfered with solely because
|
||||||
|
modification has been made.
|
||||||
|
|
||||||
|
If you convey an object code work under this section in, or with, or
|
||||||
|
specifically for use in, a User Product, and the conveying occurs as
|
||||||
|
part of a transaction in which the right of possession and use of the
|
||||||
|
User Product is transferred to the recipient in perpetuity or for a
|
||||||
|
fixed term (regardless of how the transaction is characterized), the
|
||||||
|
Corresponding Source conveyed under this section must be accompanied
|
||||||
|
by the Installation Information. But this requirement does not apply
|
||||||
|
if neither you nor any third party retains the ability to install
|
||||||
|
modified object code on the User Product (for example, the work has
|
||||||
|
been installed in ROM).
|
||||||
|
|
||||||
|
The requirement to provide Installation Information does not include a
|
||||||
|
requirement to continue to provide support service, warranty, or updates
|
||||||
|
for a work that has been modified or installed by the recipient, or for
|
||||||
|
the User Product in which it has been modified or installed. Access to a
|
||||||
|
network may be denied when the modification itself materially and
|
||||||
|
adversely affects the operation of the network or violates the rules and
|
||||||
|
protocols for communication across the network.
|
||||||
|
|
||||||
|
Corresponding Source conveyed, and Installation Information provided,
|
||||||
|
in accord with this section must be in a format that is publicly
|
||||||
|
documented (and with an implementation available to the public in
|
||||||
|
source code form), and must require no special password or key for
|
||||||
|
unpacking, reading or copying.
|
||||||
|
|
||||||
|
7. Additional Terms.
|
||||||
|
|
||||||
|
"Additional permissions" are terms that supplement the terms of this
|
||||||
|
License by making exceptions from one or more of its conditions.
|
||||||
|
Additional permissions that are applicable to the entire Program shall
|
||||||
|
be treated as though they were included in this License, to the extent
|
||||||
|
that they are valid under applicable law. If additional permissions
|
||||||
|
apply only to part of the Program, that part may be used separately
|
||||||
|
under those permissions, but the entire Program remains governed by
|
||||||
|
this License without regard to the additional permissions.
|
||||||
|
|
||||||
|
When you convey a copy of a covered work, you may at your option
|
||||||
|
remove any additional permissions from that copy, or from any part of
|
||||||
|
it. (Additional permissions may be written to require their own
|
||||||
|
removal in certain cases when you modify the work.) You may place
|
||||||
|
additional permissions on material, added by you to a covered work,
|
||||||
|
for which you have or can give appropriate copyright permission.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, for material you
|
||||||
|
add to a covered work, you may (if authorized by the copyright holders of
|
||||||
|
that material) supplement the terms of this License with terms:
|
||||||
|
|
||||||
|
a) Disclaiming warranty or limiting liability differently from the
|
||||||
|
terms of sections 15 and 16 of this License; or
|
||||||
|
|
||||||
|
b) Requiring preservation of specified reasonable legal notices or
|
||||||
|
author attributions in that material or in the Appropriate Legal
|
||||||
|
Notices displayed by works containing it; or
|
||||||
|
|
||||||
|
c) Prohibiting misrepresentation of the origin of that material, or
|
||||||
|
requiring that modified versions of such material be marked in
|
||||||
|
reasonable ways as different from the original version; or
|
||||||
|
|
||||||
|
d) Limiting the use for publicity purposes of names of licensors or
|
||||||
|
authors of the material; or
|
||||||
|
|
||||||
|
e) Declining to grant rights under trademark law for use of some
|
||||||
|
trade names, trademarks, or service marks; or
|
||||||
|
|
||||||
|
f) Requiring indemnification of licensors and authors of that
|
||||||
|
material by anyone who conveys the material (or modified versions of
|
||||||
|
it) with contractual assumptions of liability to the recipient, for
|
||||||
|
any liability that these contractual assumptions directly impose on
|
||||||
|
those licensors and authors.
|
||||||
|
|
||||||
|
All other non-permissive additional terms are considered "further
|
||||||
|
restrictions" within the meaning of section 10. If the Program as you
|
||||||
|
received it, or any part of it, contains a notice stating that it is
|
||||||
|
governed by this License along with a term that is a further
|
||||||
|
restriction, you may remove that term. If a license document contains
|
||||||
|
a further restriction but permits relicensing or conveying under this
|
||||||
|
License, you may add to a covered work material governed by the terms
|
||||||
|
of that license document, provided that the further restriction does
|
||||||
|
not survive such relicensing or conveying.
|
||||||
|
|
||||||
|
If you add terms to a covered work in accord with this section, you
|
||||||
|
must place, in the relevant source files, a statement of the
|
||||||
|
additional terms that apply to those files, or a notice indicating
|
||||||
|
where to find the applicable terms.
|
||||||
|
|
||||||
|
Additional terms, permissive or non-permissive, may be stated in the
|
||||||
|
form of a separately written license, or stated as exceptions;
|
||||||
|
the above requirements apply either way.
|
||||||
|
|
||||||
|
8. Termination.
|
||||||
|
|
||||||
|
You may not propagate or modify a covered work except as expressly
|
||||||
|
provided under this License. Any attempt otherwise to propagate or
|
||||||
|
modify it is void, and will automatically terminate your rights under
|
||||||
|
this License (including any patent licenses granted under the third
|
||||||
|
paragraph of section 11).
|
||||||
|
|
||||||
|
However, if you cease all violation of this License, then your
|
||||||
|
license from a particular copyright holder is reinstated (a)
|
||||||
|
provisionally, unless and until the copyright holder explicitly and
|
||||||
|
finally terminates your license, and (b) permanently, if the copyright
|
||||||
|
holder fails to notify you of the violation by some reasonable means
|
||||||
|
prior to 60 days after the cessation.
|
||||||
|
|
||||||
|
Moreover, your license from a particular copyright holder is
|
||||||
|
reinstated permanently if the copyright holder notifies you of the
|
||||||
|
violation by some reasonable means, this is the first time you have
|
||||||
|
received notice of violation of this License (for any work) from that
|
||||||
|
copyright holder, and you cure the violation prior to 30 days after
|
||||||
|
your receipt of the notice.
|
||||||
|
|
||||||
|
Termination of your rights under this section does not terminate the
|
||||||
|
licenses of parties who have received copies or rights from you under
|
||||||
|
this License. If your rights have been terminated and not permanently
|
||||||
|
reinstated, you do not qualify to receive new licenses for the same
|
||||||
|
material under section 10.
|
||||||
|
|
||||||
|
9. Acceptance Not Required for Having Copies.
|
||||||
|
|
||||||
|
You are not required to accept this License in order to receive or
|
||||||
|
run a copy of the Program. Ancillary propagation of a covered work
|
||||||
|
occurring solely as a consequence of using peer-to-peer transmission
|
||||||
|
to receive a copy likewise does not require acceptance. However,
|
||||||
|
nothing other than this License grants you permission to propagate or
|
||||||
|
modify any covered work. These actions infringe copyright if you do
|
||||||
|
not accept this License. Therefore, by modifying or propagating a
|
||||||
|
covered work, you indicate your acceptance of this License to do so.
|
||||||
|
|
||||||
|
10. Automatic Licensing of Downstream Recipients.
|
||||||
|
|
||||||
|
Each time you convey a covered work, the recipient automatically
|
||||||
|
receives a license from the original licensors, to run, modify and
|
||||||
|
propagate that work, subject to this License. You are not responsible
|
||||||
|
for enforcing compliance by third parties with this License.
|
||||||
|
|
||||||
|
An "entity transaction" is a transaction transferring control of an
|
||||||
|
organization, or substantially all assets of one, or subdividing an
|
||||||
|
organization, or merging organizations. If propagation of a covered
|
||||||
|
work results from an entity transaction, each party to that
|
||||||
|
transaction who receives a copy of the work also receives whatever
|
||||||
|
licenses to the work the party's predecessor in interest had or could
|
||||||
|
give under the previous paragraph, plus a right to possession of the
|
||||||
|
Corresponding Source of the work from the predecessor in interest, if
|
||||||
|
the predecessor has it or can get it with reasonable efforts.
|
||||||
|
|
||||||
|
You may not impose any further restrictions on the exercise of the
|
||||||
|
rights granted or affirmed under this License. For example, you may
|
||||||
|
not impose a license fee, royalty, or other charge for exercise of
|
||||||
|
rights granted under this License, and you may not initiate litigation
|
||||||
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||||
|
any patent claim is infringed by making, using, selling, offering for
|
||||||
|
sale, or importing the Program or any portion of it.
|
||||||
|
|
||||||
|
11. Patents.
|
||||||
|
|
||||||
|
A "contributor" is a copyright holder who authorizes use under this
|
||||||
|
License of the Program or a work on which the Program is based. The
|
||||||
|
work thus licensed is called the contributor's "contributor version".
|
||||||
|
|
||||||
|
A contributor's "essential patent claims" are all patent claims
|
||||||
|
owned or controlled by the contributor, whether already acquired or
|
||||||
|
hereafter acquired, that would be infringed by some manner, permitted
|
||||||
|
by this License, of making, using, or selling its contributor version,
|
||||||
|
but do not include claims that would be infringed only as a
|
||||||
|
consequence of further modification of the contributor version. For
|
||||||
|
purposes of this definition, "control" includes the right to grant
|
||||||
|
patent sublicenses in a manner consistent with the requirements of
|
||||||
|
this License.
|
||||||
|
|
||||||
|
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||||
|
patent license under the contributor's essential patent claims, to
|
||||||
|
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||||
|
propagate the contents of its contributor version.
|
||||||
|
|
||||||
|
In the following three paragraphs, a "patent license" is any express
|
||||||
|
agreement or commitment, however denominated, not to enforce a patent
|
||||||
|
(such as an express permission to practice a patent or covenant not to
|
||||||
|
sue for patent infringement). To "grant" such a patent license to a
|
||||||
|
party means to make such an agreement or commitment not to enforce a
|
||||||
|
patent against the party.
|
||||||
|
|
||||||
|
If you convey a covered work, knowingly relying on a patent license,
|
||||||
|
and the Corresponding Source of the work is not available for anyone
|
||||||
|
to copy, free of charge and under the terms of this License, through a
|
||||||
|
publicly available network server or other readily accessible means,
|
||||||
|
then you must either (1) cause the Corresponding Source to be so
|
||||||
|
available, or (2) arrange to deprive yourself of the benefit of the
|
||||||
|
patent license for this particular work, or (3) arrange, in a manner
|
||||||
|
consistent with the requirements of this License, to extend the patent
|
||||||
|
license to downstream recipients. "Knowingly relying" means you have
|
||||||
|
actual knowledge that, but for the patent license, your conveying the
|
||||||
|
covered work in a country, or your recipient's use of the covered work
|
||||||
|
in a country, would infringe one or more identifiable patents in that
|
||||||
|
country that you have reason to believe are valid.
|
||||||
|
|
||||||
|
If, pursuant to or in connection with a single transaction or
|
||||||
|
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||||
|
covered work, and grant a patent license to some of the parties
|
||||||
|
receiving the covered work authorizing them to use, propagate, modify
|
||||||
|
or convey a specific copy of the covered work, then the patent license
|
||||||
|
you grant is automatically extended to all recipients of the covered
|
||||||
|
work and works based on it.
|
||||||
|
|
||||||
|
A patent license is "discriminatory" if it does not include within
|
||||||
|
the scope of its coverage, prohibits the exercise of, or is
|
||||||
|
conditioned on the non-exercise of one or more of the rights that are
|
||||||
|
specifically granted under this License. You may not convey a covered
|
||||||
|
work if you are a party to an arrangement with a third party that is
|
||||||
|
in the business of distributing software, under which you make payment
|
||||||
|
to the third party based on the extent of your activity of conveying
|
||||||
|
the work, and under which the third party grants, to any of the
|
||||||
|
parties who would receive the covered work from you, a discriminatory
|
||||||
|
patent license (a) in connection with copies of the covered work
|
||||||
|
conveyed by you (or copies made from those copies), or (b) primarily
|
||||||
|
for and in connection with specific products or compilations that
|
||||||
|
contain the covered work, unless you entered into that arrangement,
|
||||||
|
or that patent license was granted, prior to 28 March 2007.
|
||||||
|
|
||||||
|
Nothing in this License shall be construed as excluding or limiting
|
||||||
|
any implied license or other defenses to infringement that may
|
||||||
|
otherwise be available to you under applicable patent law.
|
||||||
|
|
||||||
|
12. No Surrender of Others' Freedom.
|
||||||
|
|
||||||
|
If conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot convey a
|
||||||
|
covered work so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you may
|
||||||
|
not convey it at all. For example, if you agree to terms that obligate you
|
||||||
|
to collect a royalty for further conveying from those to whom you convey
|
||||||
|
the Program, the only way you could satisfy both those terms and this
|
||||||
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
|
13. Use with the GNU Affero General Public License.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, you have
|
||||||
|
permission to link or combine any covered work with a work licensed
|
||||||
|
under version 3 of the GNU Affero General Public License into a single
|
||||||
|
combined work, and to convey the resulting work. The terms of this
|
||||||
|
License will continue to apply to the part which is the covered work,
|
||||||
|
but the special requirements of the GNU Affero General Public License,
|
||||||
|
section 13, concerning interaction through a network will apply to the
|
||||||
|
combination as such.
|
||||||
|
|
||||||
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
|
the GNU General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Program specifies that a certain numbered version of the GNU General
|
||||||
|
Public License "or any later version" applies to it, you have the
|
||||||
|
option of following the terms and conditions either of that numbered
|
||||||
|
version or of any later version published by the Free Software
|
||||||
|
Foundation. If the Program does not specify a version number of the
|
||||||
|
GNU General Public License, you may choose any version ever published
|
||||||
|
by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Program specifies that a proxy can decide which future
|
||||||
|
versions of the GNU General Public License can be used, that proxy's
|
||||||
|
public statement of acceptance of a version permanently authorizes you
|
||||||
|
to choose that version for the Program.
|
||||||
|
|
||||||
|
Later license versions may give you additional or different
|
||||||
|
permissions. However, no additional obligations are imposed on any
|
||||||
|
author or copyright holder as a result of your choosing to follow a
|
||||||
|
later version.
|
||||||
|
|
||||||
|
15. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||||
|
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||||
|
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||||
|
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||||
|
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||||
|
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. Limitation of Liability.
|
||||||
|
|
||||||
|
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||||
|
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||||
|
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||||
|
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||||
|
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||||
|
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||||
|
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||||
|
SUCH DAMAGES.
|
||||||
|
|
||||||
|
17. Interpretation of Sections 15 and 16.
|
||||||
|
|
||||||
|
If the disclaimer of warranty and limitation of liability provided
|
||||||
|
above cannot be given local legal effect according to their terms,
|
||||||
|
reviewing courts shall apply local law that most closely approximates
|
||||||
|
an absolute waiver of all civil liability in connection with the
|
||||||
|
Program, unless a warranty or assumption of liability accompanies a
|
||||||
|
copy of the Program in return for a fee.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
@@ -0,0 +1,165 @@
|
|||||||
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
|
||||||
|
This version of the GNU Lesser General Public License incorporates
|
||||||
|
the terms and conditions of version 3 of the GNU General Public
|
||||||
|
License, supplemented by the additional permissions listed below.
|
||||||
|
|
||||||
|
0. Additional Definitions.
|
||||||
|
|
||||||
|
As used herein, "this License" refers to version 3 of the GNU Lesser
|
||||||
|
General Public License, and the "GNU GPL" refers to version 3 of the GNU
|
||||||
|
General Public License.
|
||||||
|
|
||||||
|
"The Library" refers to a covered work governed by this License,
|
||||||
|
other than an Application or a Combined Work as defined below.
|
||||||
|
|
||||||
|
An "Application" is any work that makes use of an interface provided
|
||||||
|
by the Library, but which is not otherwise based on the Library.
|
||||||
|
Defining a subclass of a class defined by the Library is deemed a mode
|
||||||
|
of using an interface provided by the Library.
|
||||||
|
|
||||||
|
A "Combined Work" is a work produced by combining or linking an
|
||||||
|
Application with the Library. The particular version of the Library
|
||||||
|
with which the Combined Work was made is also called the "Linked
|
||||||
|
Version".
|
||||||
|
|
||||||
|
The "Minimal Corresponding Source" for a Combined Work means the
|
||||||
|
Corresponding Source for the Combined Work, excluding any source code
|
||||||
|
for portions of the Combined Work that, considered in isolation, are
|
||||||
|
based on the Application, and not on the Linked Version.
|
||||||
|
|
||||||
|
The "Corresponding Application Code" for a Combined Work means the
|
||||||
|
object code and/or source code for the Application, including any data
|
||||||
|
and utility programs needed for reproducing the Combined Work from the
|
||||||
|
Application, but excluding the System Libraries of the Combined Work.
|
||||||
|
|
||||||
|
1. Exception to Section 3 of the GNU GPL.
|
||||||
|
|
||||||
|
You may convey a covered work under sections 3 and 4 of this License
|
||||||
|
without being bound by section 3 of the GNU GPL.
|
||||||
|
|
||||||
|
2. Conveying Modified Versions.
|
||||||
|
|
||||||
|
If you modify a copy of the Library, and, in your modifications, a
|
||||||
|
facility refers to a function or data to be supplied by an Application
|
||||||
|
that uses the facility (other than as an argument passed when the
|
||||||
|
facility is invoked), then you may convey a copy of the modified
|
||||||
|
version:
|
||||||
|
|
||||||
|
a) under this License, provided that you make a good faith effort to
|
||||||
|
ensure that, in the event an Application does not supply the
|
||||||
|
function or data, the facility still operates, and performs
|
||||||
|
whatever part of its purpose remains meaningful, or
|
||||||
|
|
||||||
|
b) under the GNU GPL, with none of the additional permissions of
|
||||||
|
this License applicable to that copy.
|
||||||
|
|
||||||
|
3. Object Code Incorporating Material from Library Header Files.
|
||||||
|
|
||||||
|
The object code form of an Application may incorporate material from
|
||||||
|
a header file that is part of the Library. You may convey such object
|
||||||
|
code under terms of your choice, provided that, if the incorporated
|
||||||
|
material is not limited to numerical parameters, data structure
|
||||||
|
layouts and accessors, or small macros, inline functions and templates
|
||||||
|
(ten or fewer lines in length), you do both of the following:
|
||||||
|
|
||||||
|
a) Give prominent notice with each copy of the object code that the
|
||||||
|
Library is used in it and that the Library and its use are
|
||||||
|
covered by this License.
|
||||||
|
|
||||||
|
b) Accompany the object code with a copy of the GNU GPL and this license
|
||||||
|
document.
|
||||||
|
|
||||||
|
4. Combined Works.
|
||||||
|
|
||||||
|
You may convey a Combined Work under terms of your choice that,
|
||||||
|
taken together, effectively do not restrict modification of the
|
||||||
|
portions of the Library contained in the Combined Work and reverse
|
||||||
|
engineering for debugging such modifications, if you also do each of
|
||||||
|
the following:
|
||||||
|
|
||||||
|
a) Give prominent notice with each copy of the Combined Work that
|
||||||
|
the Library is used in it and that the Library and its use are
|
||||||
|
covered by this License.
|
||||||
|
|
||||||
|
b) Accompany the Combined Work with a copy of the GNU GPL and this license
|
||||||
|
document.
|
||||||
|
|
||||||
|
c) For a Combined Work that displays copyright notices during
|
||||||
|
execution, include the copyright notice for the Library among
|
||||||
|
these notices, as well as a reference directing the user to the
|
||||||
|
copies of the GNU GPL and this license document.
|
||||||
|
|
||||||
|
d) Do one of the following:
|
||||||
|
|
||||||
|
0) Convey the Minimal Corresponding Source under the terms of this
|
||||||
|
License, and the Corresponding Application Code in a form
|
||||||
|
suitable for, and under terms that permit, the user to
|
||||||
|
recombine or relink the Application with a modified version of
|
||||||
|
the Linked Version to produce a modified Combined Work, in the
|
||||||
|
manner specified by section 6 of the GNU GPL for conveying
|
||||||
|
Corresponding Source.
|
||||||
|
|
||||||
|
1) Use a suitable shared library mechanism for linking with the
|
||||||
|
Library. A suitable mechanism is one that (a) uses at run time
|
||||||
|
a copy of the Library already present on the user's computer
|
||||||
|
system, and (b) will operate properly with a modified version
|
||||||
|
of the Library that is interface-compatible with the Linked
|
||||||
|
Version.
|
||||||
|
|
||||||
|
e) Provide Installation Information, but only if you would otherwise
|
||||||
|
be required to provide such information under section 6 of the
|
||||||
|
GNU GPL, and only to the extent that such information is
|
||||||
|
necessary to install and execute a modified version of the
|
||||||
|
Combined Work produced by recombining or relinking the
|
||||||
|
Application with a modified version of the Linked Version. (If
|
||||||
|
you use option 4d0, the Installation Information must accompany
|
||||||
|
the Minimal Corresponding Source and Corresponding Application
|
||||||
|
Code. If you use option 4d1, you must provide the Installation
|
||||||
|
Information in the manner specified by section 6 of the GNU GPL
|
||||||
|
for conveying Corresponding Source.)
|
||||||
|
|
||||||
|
5. Combined Libraries.
|
||||||
|
|
||||||
|
You may place library facilities that are a work based on the
|
||||||
|
Library side by side in a single library together with other library
|
||||||
|
facilities that are not Applications and are not covered by this
|
||||||
|
License, and convey such a combined library under terms of your
|
||||||
|
choice, if you do both of the following:
|
||||||
|
|
||||||
|
a) Accompany the combined library with a copy of the same work based
|
||||||
|
on the Library, uncombined with any other library facilities,
|
||||||
|
conveyed under the terms of this License.
|
||||||
|
|
||||||
|
b) Give prominent notice with the combined library that part of it
|
||||||
|
is a work based on the Library, and explaining where to find the
|
||||||
|
accompanying uncombined form of the same work.
|
||||||
|
|
||||||
|
6. Revised Versions of the GNU Lesser General Public License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the GNU Lesser General Public License from time to time. Such new
|
||||||
|
versions will be similar in spirit to the present version, but may
|
||||||
|
differ in detail to address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Library as you received it specifies that a certain numbered version
|
||||||
|
of the GNU Lesser General Public License "or any later version"
|
||||||
|
applies to it, you have the option of following the terms and
|
||||||
|
conditions either of that published version or of any later version
|
||||||
|
published by the Free Software Foundation. If the Library as you
|
||||||
|
received it does not specify a version number of the GNU Lesser
|
||||||
|
General Public License, you may choose any version of the GNU Lesser
|
||||||
|
General Public License ever published by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Library as you received it specifies that a proxy can decide
|
||||||
|
whether future versions of the GNU Lesser General Public License shall
|
||||||
|
apply, that proxy's public statement of acceptance of any version is
|
||||||
|
permanent authorization for you to choose that version for the
|
||||||
|
Library.
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
# Third-Party Licenses
|
||||||
|
|
||||||
|
This directory contains license files for third-party dependencies used by specklepy.
|
||||||
|
|
||||||
|
## IfcOpenShell
|
||||||
|
|
||||||
|
IfcOpenShell is an optional dependency available through the `speckleifc` extra:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pip install specklepy[speckleifc]
|
||||||
|
```
|
||||||
|
|
||||||
|
IfcOpenShell is dual-licensed under:
|
||||||
|
|
||||||
|
- **LGPL-3.0** (`IfcOpenShell-LGPL-3.0.txt`) - GNU Lesser General Public License v3.0
|
||||||
|
- **GPL-3.0** (`IfcOpenShell-GPL-3.0.txt`) - GNU General Public License v3.0
|
||||||
|
|
||||||
|
### About IfcOpenShell
|
||||||
|
|
||||||
|
IfcOpenShell is an open source software library for working with Industry Foundation Classes (IFC). It provides complete parsing support for IFC2x3 TC1, IFC4 Add2 TC1, IFC4x1, IFC4x2, and IFC4x3 Add2.
|
||||||
|
|
||||||
|
- **Project**: https://github.com/IfcOpenShell/IfcOpenShell
|
||||||
|
- **Documentation**: https://docs.ifcopenshell.org/
|
||||||
|
- **License**: LGPL-3.0-or-later, GPL-3.0-or-later
|
||||||
|
|
||||||
|
When using specklepy with IfcOpenShell, you must comply with the terms of these licenses.
|
||||||
|
|
||||||
|
## License Compatibility
|
||||||
|
|
||||||
|
specklepy is licensed under Apache-2.0, which is compatible with LGPL-3.0 for dynamic linking scenarios (which is how IfcOpenShell is used as an optional dependency).
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import time
|
||||||
|
|
||||||
|
from speckleifc.ifc_geometry_processing import open_ifc
|
||||||
|
from speckleifc.importer import ImportJob
|
||||||
|
from specklepy.core.api.client import SpeckleClient
|
||||||
|
from specklepy.core.api.inputs.version_inputs import CreateVersionInput
|
||||||
|
from specklepy.core.api.models.current import Project, Version
|
||||||
|
from specklepy.core.api.operations import send
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
from specklepy.transports.server import ServerTransport
|
||||||
|
|
||||||
|
|
||||||
|
def open_and_convert_file(
|
||||||
|
file_path: str,
|
||||||
|
project: Project,
|
||||||
|
version_message: str | None,
|
||||||
|
model_id: str,
|
||||||
|
client: SpeckleClient,
|
||||||
|
) -> Version:
|
||||||
|
start = time.time()
|
||||||
|
very_start = start
|
||||||
|
|
||||||
|
account = client.account
|
||||||
|
server_url = account.serverInfo.url
|
||||||
|
assert server_url
|
||||||
|
remote_transport = ServerTransport(project.id, account=account)
|
||||||
|
|
||||||
|
ifc_file = open_ifc(file_path) # pyright: ignore[reportUnknownVariableType]
|
||||||
|
|
||||||
|
import_job = ImportJob(ifc_file) # pyright: ignore[reportUnknownArgumentType]
|
||||||
|
data = import_job.convert()
|
||||||
|
|
||||||
|
print(f"File conversion complete after {(time.time() - start) * 1000}ms")
|
||||||
|
|
||||||
|
start = time.time()
|
||||||
|
|
||||||
|
root_id = send(data, transports=[remote_transport], use_default_cache=False)
|
||||||
|
print(f"Sending to speckle complete after: {(time.time() - start) * 1000}ms")
|
||||||
|
|
||||||
|
start = time.time()
|
||||||
|
|
||||||
|
create_version = CreateVersionInput(
|
||||||
|
object_id=root_id,
|
||||||
|
model_id=model_id,
|
||||||
|
project_id=project.id,
|
||||||
|
message=version_message,
|
||||||
|
source_application="ifc",
|
||||||
|
)
|
||||||
|
version = client.version.create(create_version)
|
||||||
|
end = time.time()
|
||||||
|
print(f"Version committed after: {(end - start) * 1000}ms")
|
||||||
|
|
||||||
|
print(f"Total time (to commit): {(end - very_start) * 1000}ms")
|
||||||
|
del ifc_file
|
||||||
|
|
||||||
|
custom_properties = {"ui": "dui3", "actionSource": "import"}
|
||||||
|
if project.workspace_id:
|
||||||
|
custom_properties["workspace_id"] = project.workspace_id
|
||||||
|
metrics.track(metrics.SEND, account, custom_properties, send_sync=True)
|
||||||
|
|
||||||
|
return version
|
||||||
@@ -0,0 +1,197 @@
|
|||||||
|
import math
|
||||||
|
from typing import Any, Tuple
|
||||||
|
|
||||||
|
from ifcopenshell.entity_instance import entity_instance
|
||||||
|
from ifcopenshell.util.element import get_type
|
||||||
|
from ifcopenshell.util.unit import get_full_unit_name, get_project_unit
|
||||||
|
|
||||||
|
UNIT_MAPPING = {
|
||||||
|
"IfcQuantityLength": "LENGTHUNIT",
|
||||||
|
"IfcQuantityArea": "AREAUNIT",
|
||||||
|
"IfcQuantityVolume": "VOLUMEUNIT",
|
||||||
|
"IfcQuantityCount": None, # Count quantities have no units
|
||||||
|
"IfcQuantityWeight": "MASSUNIT",
|
||||||
|
"IfcQuantityTime": "TIMEUNIT",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def extract_properties(element: entity_instance) -> dict[str, object]:
|
||||||
|
(psets, qtos) = _get_ifc_object_properties(element)
|
||||||
|
|
||||||
|
properties: dict[str, object] = {
|
||||||
|
"Attributes": _get_attributes(element),
|
||||||
|
"Property Sets": psets,
|
||||||
|
}
|
||||||
|
|
||||||
|
if qtos:
|
||||||
|
properties["Quantities"] = qtos
|
||||||
|
|
||||||
|
if (ifc_type := get_type(element)) is not None:
|
||||||
|
properties["Element Type Property Sets"] = _get_ifc_element_type_properties(
|
||||||
|
ifc_type,
|
||||||
|
)
|
||||||
|
properties["Element Type Attributes"] = _get_attributes(
|
||||||
|
ifc_type,
|
||||||
|
)
|
||||||
|
|
||||||
|
return properties
|
||||||
|
|
||||||
|
|
||||||
|
def _get_attributes(element: entity_instance) -> dict[str, object]:
|
||||||
|
return element.get_info(True, False, scalar_only=True)
|
||||||
|
|
||||||
|
|
||||||
|
def _get_ifc_element_type_properties(element: entity_instance) -> dict[str, object]:
|
||||||
|
result: dict[str, object] = {}
|
||||||
|
for definition in element.HasPropertySets or []:
|
||||||
|
if not definition.is_a("IfcPropertySet"):
|
||||||
|
continue
|
||||||
|
|
||||||
|
result[definition.Name] = _get_properties(definition.HasProperties)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def _get_ifc_object_properties(
|
||||||
|
element: entity_instance,
|
||||||
|
) -> Tuple[dict[str, object], dict[str, object]]:
|
||||||
|
psets: dict[str, object] = {}
|
||||||
|
qtos: dict[str, object] = {}
|
||||||
|
|
||||||
|
for rel in getattr(element, "IsDefinedBy", []):
|
||||||
|
if not rel.is_a("IfcRelDefinesByProperties"):
|
||||||
|
continue
|
||||||
|
|
||||||
|
definition: entity_instance | None = rel.RelatingPropertyDefinition
|
||||||
|
if not definition:
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
if definition.is_a("IfcPropertySet"):
|
||||||
|
set_name = definition.Name
|
||||||
|
properties = _get_properties(definition.HasProperties)
|
||||||
|
|
||||||
|
if properties:
|
||||||
|
psets[set_name] = properties
|
||||||
|
|
||||||
|
elif definition.is_a("IfcElementQuantity"):
|
||||||
|
quantities_data = _get_quantities(definition.Quantities, element)
|
||||||
|
if not quantities_data:
|
||||||
|
continue
|
||||||
|
quantities_data["id"] = definition.id()
|
||||||
|
qtos[definition.Name] = quantities_data
|
||||||
|
|
||||||
|
except (KeyError, AttributeError):
|
||||||
|
# If entity access fails, skip this quantity set
|
||||||
|
print(f"Skipping {definition}")
|
||||||
|
continue
|
||||||
|
|
||||||
|
return (psets, qtos)
|
||||||
|
|
||||||
|
|
||||||
|
def _get_properties(properties: entity_instance) -> dict[str, Any]:
|
||||||
|
"""
|
||||||
|
There already exists a canonical way to get properties
|
||||||
|
`ifcopenshell.util.element.get_properties` but it's very verbose
|
||||||
|
and we don't want to bloat our selves with supporting complex property types
|
||||||
|
|
||||||
|
This is a slimmed down version, only supporting a couple of property types
|
||||||
|
"""
|
||||||
|
result: dict[str, Any] = {}
|
||||||
|
|
||||||
|
for prop in properties:
|
||||||
|
name = prop.Name
|
||||||
|
if prop.is_a("IfcPropertySingleValue"):
|
||||||
|
val = prop.NominalValue
|
||||||
|
if val is not None:
|
||||||
|
result[name] = val.wrappedValue if hasattr(val, "wrappedValue") else val
|
||||||
|
elif prop.is_a("IfcPropertyListValue"):
|
||||||
|
values = getattr(prop, "ListValues", None)
|
||||||
|
if values:
|
||||||
|
result[name] = [
|
||||||
|
v.wrappedValue if hasattr(v, "wrappedValue") else v for v in values
|
||||||
|
]
|
||||||
|
elif prop.is_a("IfcPropertyEnumeratedValue"):
|
||||||
|
values = getattr(prop, "EnumerationValues", None)
|
||||||
|
if values:
|
||||||
|
result[name] = [
|
||||||
|
v.wrappedValue if hasattr(v, "wrappedValue") else v for v in values
|
||||||
|
]
|
||||||
|
|
||||||
|
# elif prop.is_a("IfcPropertyTableValue"):
|
||||||
|
# properties[name] = #not sure if we want to support these...
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def _get_quantities(
|
||||||
|
quantities: list[entity_instance], element: entity_instance
|
||||||
|
) -> dict[str, Any]:
|
||||||
|
"""Extract quantity values from IfcPhysicalQuantity entities."""
|
||||||
|
results: dict[str, Any] = {}
|
||||||
|
for quantity in quantities or []:
|
||||||
|
quantity_name = quantity.Name
|
||||||
|
|
||||||
|
if quantity.is_a("IfcPhysicalSimpleQuantity"):
|
||||||
|
# Get the quantity value (3rd attribute for simple quantities)
|
||||||
|
value = getattr(quantity, quantity.attribute_name(3))
|
||||||
|
unit_info = _get_unit_info(element, quantity)
|
||||||
|
|
||||||
|
# Server does not consider `NaN` valid json
|
||||||
|
if math.isnan(value):
|
||||||
|
value = None
|
||||||
|
|
||||||
|
if unit_info:
|
||||||
|
# Create structured quantity object with units
|
||||||
|
results[quantity_name] = {
|
||||||
|
"name": quantity_name,
|
||||||
|
"value": value,
|
||||||
|
**unit_info,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
# No unit info available, keep as simple value with name
|
||||||
|
results[quantity_name] = {"name": quantity_name, "value": value}
|
||||||
|
|
||||||
|
elif quantity.is_a("IfcPhysicalComplexQuantity"):
|
||||||
|
# Handle complex quantities
|
||||||
|
data = {
|
||||||
|
k: v
|
||||||
|
for k, v in quantity.get_info().items()
|
||||||
|
if v is not None and k != "Name"
|
||||||
|
}
|
||||||
|
data["properties"] = _get_quantities(quantity.HasQuantities, element)
|
||||||
|
del data["HasQuantities"]
|
||||||
|
results[quantity_name] = data
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def _get_unit_info(
|
||||||
|
element: entity_instance, quantity: entity_instance
|
||||||
|
) -> dict[str, str]:
|
||||||
|
"""Get unit information for a quantity."""
|
||||||
|
# Early return for count quantities - they don't have units
|
||||||
|
quantity_type = quantity.is_a()
|
||||||
|
if quantity_type == "IfcQuantityCount":
|
||||||
|
return {}
|
||||||
|
|
||||||
|
unit = getattr(element, "Unit", None)
|
||||||
|
if unit:
|
||||||
|
# Quantity has its own unit
|
||||||
|
unit_name = get_full_unit_name(unit)
|
||||||
|
formatted_unit_name = unit_name.replace("_", " ").title() if unit_name else ""
|
||||||
|
return {"units": formatted_unit_name}
|
||||||
|
|
||||||
|
else:
|
||||||
|
# Fall back to project unit based on quantity type
|
||||||
|
unit_type = UNIT_MAPPING.get(quantity_type)
|
||||||
|
if not unit_type:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
# Get the project unit for this unit type
|
||||||
|
project_unit = get_project_unit(element.file, unit_type, use_cache=True)
|
||||||
|
if not project_unit:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
# Get unit name and format
|
||||||
|
unit_name = get_full_unit_name(project_unit)
|
||||||
|
formatted_unit_name = unit_name.replace("_", " ").title() if unit_name else ""
|
||||||
|
|
||||||
|
return {"units": formatted_unit_name}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
from typing import Sequence
|
||||||
|
|
||||||
|
from specklepy.objects.base import Base
|
||||||
|
from specklepy.objects.proxies import InstanceDefinitionProxy
|
||||||
|
|
||||||
|
|
||||||
|
class InstanceProxyManager:
|
||||||
|
def __init__(self):
|
||||||
|
self._instance_definition_proxies: dict[str, InstanceDefinitionProxy] = {}
|
||||||
|
"""definition proxies to be added directly to the root"""
|
||||||
|
self._instance_geometry: dict[str, Base] = {}
|
||||||
|
"""The geometry that will be added in it's own collection under the root"""
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_definition_proxies(self) -> dict[str, InstanceDefinitionProxy]:
|
||||||
|
return self._instance_definition_proxies
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_geometry(self) -> dict[str, Base]:
|
||||||
|
return self._instance_geometry
|
||||||
|
|
||||||
|
def add_display_value_definitions(self, geometry: Sequence[Base]) -> list[str]:
|
||||||
|
result: list[str] = []
|
||||||
|
for m in geometry:
|
||||||
|
if not m.applicationId:
|
||||||
|
raise ValueError("geometry with no applicationId cannot be proxied ")
|
||||||
|
definition_id = f"DEFINITION:{m.applicationId}"
|
||||||
|
result.append(definition_id)
|
||||||
|
self._add_definition(definition_id, [m.applicationId], 0)
|
||||||
|
self._instance_geometry[m.applicationId] = m
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
def _add_definition(
|
||||||
|
self, definition_id: str, objects: list[str], max_depth: int
|
||||||
|
) -> None:
|
||||||
|
proxy = InstanceDefinitionProxy(
|
||||||
|
applicationId=definition_id,
|
||||||
|
name=definition_id,
|
||||||
|
objects=objects,
|
||||||
|
maxDepth=max_depth,
|
||||||
|
)
|
||||||
|
self._instance_definition_proxies[definition_id] = proxy
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
from specklepy.objects.data_objects import DataObject
|
||||||
|
from specklepy.objects.proxies import LevelProxy
|
||||||
|
|
||||||
|
|
||||||
|
class LevelProxyManager:
|
||||||
|
def __init__(self):
|
||||||
|
self._level_proxies: dict[str, LevelProxy] = {}
|
||||||
|
|
||||||
|
@property
|
||||||
|
def level_proxies(self):
|
||||||
|
return self._level_proxies
|
||||||
|
|
||||||
|
def add_element_level_mapping(
|
||||||
|
self, level_data_object: DataObject, element_application_id: str
|
||||||
|
) -> None:
|
||||||
|
level_id = level_data_object.applicationId
|
||||||
|
assert level_id is not None
|
||||||
|
|
||||||
|
proxy = self._level_proxies.get(level_id, None)
|
||||||
|
if proxy is not None:
|
||||||
|
proxy.objects.append(element_application_id)
|
||||||
|
else:
|
||||||
|
self._level_proxies[level_id] = LevelProxy(
|
||||||
|
objects=[element_application_id],
|
||||||
|
value=level_data_object,
|
||||||
|
applicationId=level_id,
|
||||||
|
)
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
from specklepy.objects.geometry import Mesh
|
||||||
|
from specklepy.objects.other import RenderMaterial
|
||||||
|
from specklepy.objects.proxies import RenderMaterialProxy
|
||||||
|
|
||||||
|
|
||||||
|
class RenderMaterialProxyManager:
|
||||||
|
def __init__(self):
|
||||||
|
self._render_material_proxies: dict[str, RenderMaterialProxy] = {}
|
||||||
|
|
||||||
|
@property
|
||||||
|
def render_material_proxies(self):
|
||||||
|
return self._render_material_proxies
|
||||||
|
|
||||||
|
def add_mesh_material_mapping(
|
||||||
|
self, render_material: RenderMaterial, mesh: Mesh
|
||||||
|
) -> None:
|
||||||
|
material_id = render_material.applicationId
|
||||||
|
assert material_id is not None
|
||||||
|
mesh_id = mesh.applicationId
|
||||||
|
assert mesh_id is not None
|
||||||
|
|
||||||
|
proxy = self._render_material_proxies.get(material_id, None)
|
||||||
|
if proxy is not None:
|
||||||
|
proxy.objects.append(mesh_id)
|
||||||
|
else:
|
||||||
|
self._render_material_proxies[material_id] = RenderMaterialProxy(
|
||||||
|
objects=[mesh_id], value=render_material
|
||||||
|
)
|
||||||
@@ -1,3 +1,3 @@
|
|||||||
from specklepy import objects
|
# from specklepy import objects
|
||||||
|
|
||||||
__all__ = ["objects"]
|
# __all__ = ["objects"]
|
||||||
|
|||||||
+51
-50
@@ -1,16 +1,17 @@
|
|||||||
from deprecated import deprecated
|
import contextlib
|
||||||
|
|
||||||
from specklepy.api.credentials import Account
|
from specklepy.api.credentials import Account
|
||||||
from specklepy.api.resources import (
|
from specklepy.api.resources import (
|
||||||
active_user,
|
ActiveUserResource,
|
||||||
branch,
|
FileImportResource,
|
||||||
commit,
|
ModelResource,
|
||||||
object,
|
OtherUserResource,
|
||||||
other_user,
|
ProjectInviteResource,
|
||||||
server,
|
ProjectResource,
|
||||||
stream,
|
ServerResource,
|
||||||
subscriptions,
|
SubscriptionResource,
|
||||||
user,
|
VersionResource,
|
||||||
|
WorkspaceResource,
|
||||||
)
|
)
|
||||||
from specklepy.core.api.client import SpeckleClient as CoreSpeckleClient
|
from specklepy.core.api.client import SpeckleClient as CoreSpeckleClient
|
||||||
from specklepy.logging import metrics
|
from specklepy.logging import metrics
|
||||||
@@ -29,21 +30,24 @@ class SpeckleClient(CoreSpeckleClient):
|
|||||||
|
|
||||||
```py
|
```py
|
||||||
from specklepy.api.client import SpeckleClient
|
from specklepy.api.client import SpeckleClient
|
||||||
|
from specklepy.core.api.inputs.project_inputs import ProjectCreateInput
|
||||||
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="app.speckle.systems") # 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)
|
||||||
account = get_default_account()
|
account = get_default_account()
|
||||||
client.authenticate_with_account(account)
|
client.authenticate_with_account(account)
|
||||||
|
|
||||||
# create a new stream. this returns the stream id
|
# create a new project
|
||||||
new_stream_id = client.stream.create(name="a shiny new stream")
|
input = ProjectCreateInput(name="a shiny new project")
|
||||||
|
project = self.project.create(input)
|
||||||
|
|
||||||
# use that stream id to get the stream from the server
|
# or, use a project id to get an existing project from the server
|
||||||
new_stream = client.stream.get(id=new_stream_id)
|
new_stream = client.project.get("abcdefghij")
|
||||||
```
|
```
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@@ -64,73 +68,70 @@ class SpeckleClient(CoreSpeckleClient):
|
|||||||
self.account = Account()
|
self.account = Account()
|
||||||
|
|
||||||
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:
|
|
||||||
|
with contextlib.suppress(Exception):
|
||||||
server_version = self.server.version()
|
server_version = self.server.version()
|
||||||
except Exception:
|
|
||||||
pass
|
self.other_user = OtherUserResource(
|
||||||
self.user = user.Resource(
|
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.url,
|
basepath=self.url,
|
||||||
client=self.httpclient,
|
client=self.httpclient,
|
||||||
server_version=server_version,
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.other_user = other_user.Resource(
|
self.active_user = ActiveUserResource(
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.url,
|
basepath=self.url,
|
||||||
client=self.httpclient,
|
client=self.httpclient,
|
||||||
server_version=server_version,
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.active_user = active_user.Resource(
|
self.project = ProjectResource(
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.url,
|
basepath=self.url,
|
||||||
client=self.httpclient,
|
client=self.httpclient,
|
||||||
server_version=server_version,
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.stream = stream.Resource(
|
self.project_invite = ProjectInviteResource(
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.url,
|
basepath=self.url,
|
||||||
client=self.httpclient,
|
client=self.httpclient,
|
||||||
server_version=server_version,
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.commit = commit.Resource(
|
self.model = ModelResource(
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.branch = branch.Resource(
|
self.version = VersionResource(
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.object = object.Resource(
|
self.workspace = WorkspaceResource(
|
||||||
account=self.account, basepath=self.url, client=self.httpclient
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
)
|
)
|
||||||
self.subscribe = subscriptions.Resource(
|
self.file_import = FileImportResource(
|
||||||
|
account=self.account,
|
||||||
|
basepath=self.url,
|
||||||
|
client=self.httpclient,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
self.subscription = SubscriptionResource(
|
||||||
account=self.account,
|
account=self.account,
|
||||||
basepath=self.ws_url,
|
basepath=self.ws_url,
|
||||||
client=self.wsclient,
|
client=self.wsclient,
|
||||||
|
# todo: why doesn't this take a server version
|
||||||
)
|
)
|
||||||
|
|
||||||
@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:
|
def authenticate_with_token(self, token: str) -> None:
|
||||||
"""
|
"""
|
||||||
Authenticate the client using a personal access token.
|
Authenticate the client using a personal access token.
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
# following imports seem to be unnecessary, but they need to stay
|
# following imports seem to be unnecessary, but they need to stay
|
||||||
# to not break the scripts using these functions as non-core
|
# 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 ( # noqa: F401
|
||||||
from specklepy.core.api.credentials import Account, UserInfo # noqa: F401
|
Account,
|
||||||
|
StreamWrapper, # noqa: F401
|
||||||
|
UserInfo,
|
||||||
|
)
|
||||||
from specklepy.core.api.credentials import (
|
from specklepy.core.api.credentials import (
|
||||||
get_account_from_token as core_get_account_from_token,
|
get_account_from_token as core_get_account_from_token,
|
||||||
)
|
)
|
||||||
@@ -11,7 +12,7 @@ from specklepy.core.api.credentials import get_local_accounts as core_get_local_
|
|||||||
from specklepy.logging import metrics
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
def get_local_accounts(base_path: Optional[str] = None) -> List[Account]:
|
def get_local_accounts(base_path: str | None = None) -> list[Account]:
|
||||||
"""Gets all the accounts present in this environment
|
"""Gets all the accounts present in this environment
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
@@ -35,7 +36,7 @@ def get_local_accounts(base_path: Optional[str] = None) -> List[Account]:
|
|||||||
return accounts
|
return accounts
|
||||||
|
|
||||||
|
|
||||||
def get_default_account(base_path: Optional[str] = None) -> Optional[Account]:
|
def get_default_account(base_path: str | None = None) -> Account | None:
|
||||||
"""
|
"""
|
||||||
Gets this environment's default account if any. If there is no default,
|
Gets this environment's default account if any. If there is no default,
|
||||||
the first found will be returned and set as default.
|
the first found will be returned and set as default.
|
||||||
@@ -58,7 +59,7 @@ def get_default_account(base_path: Optional[str] = None) -> Optional[Account]:
|
|||||||
return default
|
return default
|
||||||
|
|
||||||
|
|
||||||
def get_account_from_token(token: str, server_url: str = None) -> Account:
|
def get_account_from_token(token: str, server_url: str | None = None) -> Account:
|
||||||
"""Gets the local account for the token if it exists
|
"""Gets the local account for the token if it exists
|
||||||
Arguments:
|
Arguments:
|
||||||
token {str} -- the api token
|
token {str} -- the api token
|
||||||
|
|||||||
@@ -1,35 +1,15 @@
|
|||||||
# following imports seem to be unnecessary, but they need to stay
|
# following imports seem to be unnecessary, but they need to stay
|
||||||
# to not break the scripts using these functions as non-core
|
# to not break the scripts using these functions as non-core
|
||||||
from specklepy.core.api.models import (
|
from specklepy.core.api.models import (
|
||||||
Activity,
|
|
||||||
ActivityCollection,
|
|
||||||
Branch,
|
|
||||||
Branches,
|
|
||||||
Collaborator,
|
|
||||||
Commit,
|
|
||||||
Commits,
|
|
||||||
LimitedUser,
|
LimitedUser,
|
||||||
Object,
|
|
||||||
PendingStreamCollaborator,
|
PendingStreamCollaborator,
|
||||||
ServerInfo,
|
ServerInfo,
|
||||||
Stream,
|
|
||||||
Streams,
|
|
||||||
User,
|
User,
|
||||||
)
|
)
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
"Activity",
|
|
||||||
"ActivityCollection",
|
|
||||||
"Branch",
|
|
||||||
"Branches",
|
|
||||||
"Collaborator",
|
|
||||||
"Commit",
|
|
||||||
"Commits",
|
|
||||||
"LimitedUser",
|
"LimitedUser",
|
||||||
"Object",
|
|
||||||
"PendingStreamCollaborator",
|
"PendingStreamCollaborator",
|
||||||
"ServerInfo",
|
"ServerInfo",
|
||||||
"Stream",
|
|
||||||
"Streams",
|
|
||||||
"User",
|
"User",
|
||||||
]
|
]
|
||||||
@@ -53,7 +53,9 @@ def receive(
|
|||||||
return _untracked_receive(obj_id, remote_transport, local_transport)
|
return _untracked_receive(obj_id, remote_transport, local_transport)
|
||||||
|
|
||||||
|
|
||||||
def serialize(base: Base, write_transports: List[AbstractTransport] = []) -> str:
|
def serialize(
|
||||||
|
base: Base, write_transports: List[AbstractTransport] | None = None
|
||||||
|
) -> str:
|
||||||
"""
|
"""
|
||||||
Serialize a base object. If no write transports are provided,
|
Serialize a base object. If no write transports are provided,
|
||||||
the object will be serialized
|
the object will be serialized
|
||||||
@@ -67,6 +69,8 @@ def serialize(base: Base, write_transports: List[AbstractTransport] = []) -> str
|
|||||||
Returns:
|
Returns:
|
||||||
str -- the serialized object
|
str -- the serialized object
|
||||||
"""
|
"""
|
||||||
|
if not write_transports:
|
||||||
|
write_transports = []
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "Serialize"})
|
metrics.track(metrics.SDK, custom_props={"name": "Serialize"})
|
||||||
return core_serialize(base, write_transports)
|
return core_serialize(base, write_transports)
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,25 @@
|
|||||||
import pkgutil
|
from specklepy.api.resources.current.active_user_resource import ActiveUserResource
|
||||||
import sys
|
from specklepy.api.resources.current.file_import_resource import FileImportResource
|
||||||
from importlib import import_module
|
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.current.workspace_resource import WorkspaceResource
|
||||||
|
|
||||||
for _, name, _ in pkgutil.iter_modules(__path__):
|
__all__ = [
|
||||||
imported_module = import_module("." + name, package=__name__)
|
"FileImportResource",
|
||||||
|
"ActiveUserResource",
|
||||||
if hasattr(imported_module, "Resource"):
|
"ModelResource",
|
||||||
setattr(sys.modules[__name__], name, imported_module)
|
"OtherUserResource",
|
||||||
|
"ProjectInviteResource",
|
||||||
|
"ProjectResource",
|
||||||
|
"ServerResource",
|
||||||
|
"SubscriptionResource",
|
||||||
|
"VersionResource",
|
||||||
|
"WorkspaceResource",
|
||||||
|
]
|
||||||
|
|||||||
@@ -1,105 +0,0 @@
|
|||||||
from datetime import datetime
|
|
||||||
from typing import List, Optional
|
|
||||||
|
|
||||||
from specklepy.api.models import PendingStreamCollaborator, User
|
|
||||||
from specklepy.core.api.resources.active_user import Resource as CoreResource
|
|
||||||
from specklepy.logging import metrics
|
|
||||||
|
|
||||||
|
|
||||||
class Resource(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) -> User:
|
|
||||||
"""Gets the profile of the current authenticated user's profile
|
|
||||||
(as extracted from the authorization header).
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
User -- the retrieved user
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, custom_props={"name": "User Active Get"})
|
|
||||||
return super().get()
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
company: Optional[str] = None,
|
|
||||||
bio: Optional[str] = None,
|
|
||||||
avatar: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Updates your user profile. All arguments are optional.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
name (Optional[str]): The user's name.
|
|
||||||
company (Optional[str]): The company the user works for.
|
|
||||||
bio (Optional[str]): A brief user biography.
|
|
||||||
avatar (Optional[str]): A URL to an avatar image for the user.
|
|
||||||
|
|
||||||
Returns @deprecated(version=DEPRECATION_VERSION, reason=DEPRECATION_TEXT):
|
|
||||||
bool -- True if your profile was updated successfully
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "User Active Update"})
|
|
||||||
return super().update(name, company, bio, avatar)
|
|
||||||
|
|
||||||
def activity(
|
|
||||||
self,
|
|
||||||
limit: int = 20,
|
|
||||||
action_type: Optional[str] = None,
|
|
||||||
before: Optional[datetime] = None,
|
|
||||||
after: Optional[datetime] = None,
|
|
||||||
cursor: Optional[datetime] = None,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
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)
|
|
||||||
|
|
||||||
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()
|
|
||||||
|
|
||||||
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)
|
|
||||||
@@ -1,97 +0,0 @@
|
|||||||
from typing import Optional, Union
|
|
||||||
|
|
||||||
from specklepy.api.models import Branch
|
|
||||||
from specklepy.core.api.resources.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
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self, stream_id: str, name: str, description: str = "No description provided"
|
|
||||||
) -> str:
|
|
||||||
"""Create a new branch on this stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
name {str} -- the name of the new branch
|
|
||||||
description {str} -- a short description of the branch
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
id {str} -- the newly created branch's id
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Create"})
|
|
||||||
return super().create(stream_id, name, description)
|
|
||||||
|
|
||||||
def get(
|
|
||||||
self, stream_id: str, name: str, commits_limit: int = 10
|
|
||||||
) -> 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)
|
|
||||||
|
|
||||||
def list(self, stream_id: str, branches_limit: int = 10, commits_limit: int = 10):
|
|
||||||
"""Get a list of branches from a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream to get the branches from
|
|
||||||
branches_limit {int} -- maximum number of branches to get
|
|
||||||
commits_limit {int} -- maximum number of commits to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Branch] -- the branches on the stream
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch List"})
|
|
||||||
return super().list(stream_id, branches_limit, commits_limit)
|
|
||||||
|
|
||||||
def update(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
branch_id: str,
|
|
||||||
name: Optional[str] = None,
|
|
||||||
description: Optional[str] = None,
|
|
||||||
):
|
|
||||||
"""Update a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream containing the branch to update
|
|
||||||
branch_id {str} -- the id of the branch to update
|
|
||||||
name {str} -- optional: the updated branch name
|
|
||||||
description {str} -- optional: the updated branch description
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if update is successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Update"})
|
|
||||||
return super().update(stream_id, branch_id, name, description)
|
|
||||||
|
|
||||||
def delete(self, stream_id: str, branch_id: str):
|
|
||||||
"""Delete a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream containing the branch to delete
|
|
||||||
branch_id {str} -- the branch to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if deletion is successful
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Branch Delete"})
|
|
||||||
return super().delete(stream_id, branch_id)
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
from typing import List, Optional, Union
|
|
||||||
|
|
||||||
from specklepy.api.models import Commit
|
|
||||||
from specklepy.core.api.resources.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
|
|
||||||
|
|
||||||
def get(self, stream_id: str, commit_id: str) -> Commit:
|
|
||||||
"""
|
|
||||||
Gets a commit given a stream and the commit id
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream where we can find the commit
|
|
||||||
commit_id {str} -- the id of the commit you want to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Commit -- the retrieved commit object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Get"})
|
|
||||||
return super().get(stream_id, commit_id)
|
|
||||||
|
|
||||||
def list(self, stream_id: str, limit: int = 10) -> List[Commit]:
|
|
||||||
"""
|
|
||||||
Get a list of commits on a given stream
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream where the commits are
|
|
||||||
limit {int} -- the maximum number of commits to fetch (default = 10)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
List[Commit] -- a list of the most recent commit objects
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit List"})
|
|
||||||
return super().list(stream_id, limit)
|
|
||||||
|
|
||||||
def create(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
object_id: str,
|
|
||||||
branch_name: str = "main",
|
|
||||||
message: str = "",
|
|
||||||
source_application: str = "python",
|
|
||||||
parents: Optional[List[str]] = None,
|
|
||||||
) -> Union[str, SpeckleException]:
|
|
||||||
"""
|
|
||||||
Creates a commit on a branch
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the stream you want to commit to
|
|
||||||
object_id {str} -- the hash of your commit object
|
|
||||||
branch_name {str}
|
|
||||||
-- the name of the branch to commit to (defaults to "main")
|
|
||||||
message {str}
|
|
||||||
-- optional: a message to give more information about the commit
|
|
||||||
source_application{str}
|
|
||||||
-- optional: the application from which the commit was created
|
|
||||||
(defaults to "python")
|
|
||||||
parents {List[str]} -- optional: the id of the parent commits
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the id of the created commit
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Create"})
|
|
||||||
return super().create(
|
|
||||||
stream_id, object_id, branch_name, message, source_application, parents
|
|
||||||
)
|
|
||||||
|
|
||||||
def update(self, stream_id: str, commit_id: str, message: str) -> bool:
|
|
||||||
"""
|
|
||||||
Update a commit
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream that contains the commit you'd like to update
|
|
||||||
commit_id {str} -- the id of the commit you'd like to update
|
|
||||||
message {str} -- the updated commit message
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation succeeded
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Update"})
|
|
||||||
return super().update(stream_id, commit_id, message)
|
|
||||||
|
|
||||||
def delete(self, stream_id: str, commit_id: str) -> bool:
|
|
||||||
"""
|
|
||||||
Delete a commit
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str}
|
|
||||||
-- the id of the stream that contains the commit you'd like to delete
|
|
||||||
commit_id {str} -- the id of the commit you'd like to delete
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
bool -- True if the operation succeeded
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Delete"})
|
|
||||||
return super().delete(stream_id, commit_id)
|
|
||||||
|
|
||||||
def received(
|
|
||||||
self,
|
|
||||||
stream_id: str,
|
|
||||||
commit_id: str,
|
|
||||||
source_application: str = "python",
|
|
||||||
message: Optional[str] = None,
|
|
||||||
) -> bool:
|
|
||||||
"""
|
|
||||||
Mark a commit object a received by the source application.
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Commit Received"})
|
|
||||||
return super().received(stream_id, commit_id, source_application, message)
|
|
||||||
@@ -0,0 +1,102 @@
|
|||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.user_inputs import (
|
||||||
|
UserProjectsFilter,
|
||||||
|
UserUpdateInput,
|
||||||
|
UserWorkspacesFilter,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
PendingStreamCollaborator,
|
||||||
|
Project,
|
||||||
|
ResourceCollection,
|
||||||
|
User,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.current import (
|
||||||
|
LimitedWorkspace,
|
||||||
|
PermissionCheckResult,
|
||||||
|
ProjectWithPermissions,
|
||||||
|
Workspace,
|
||||||
|
)
|
||||||
|
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()
|
||||||
|
|
||||||
|
def update(
|
||||||
|
self,
|
||||||
|
input: UserUpdateInput,
|
||||||
|
) -> User:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Active User Update"})
|
||||||
|
|
||||||
|
return super().update(input=input)
|
||||||
|
|
||||||
|
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_projects_with_permissions(
|
||||||
|
self,
|
||||||
|
*,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[UserProjectsFilter] = None,
|
||||||
|
) -> ResourceCollection[ProjectWithPermissions]:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK,
|
||||||
|
self.account,
|
||||||
|
{"name": "Active User Get Projects With Permissions"},
|
||||||
|
)
|
||||||
|
return super().get_projects_with_permissions(
|
||||||
|
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()
|
||||||
|
|
||||||
|
def can_create_personal_projects(self) -> PermissionCheckResult:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK,
|
||||||
|
self.account,
|
||||||
|
{"name": "Active User Can Create Personal Projects Check"},
|
||||||
|
)
|
||||||
|
return super().can_create_personal_projects()
|
||||||
|
|
||||||
|
def get_workspaces(
|
||||||
|
self,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[UserWorkspacesFilter] = None,
|
||||||
|
) -> ResourceCollection[Workspace]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Active User Get Workspaces"})
|
||||||
|
return super().get_workspaces(limit, cursor, filter)
|
||||||
|
|
||||||
|
def get_active_workspace(self) -> Optional[LimitedWorkspace]:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Active User Get Active Workspace"}
|
||||||
|
)
|
||||||
|
return super().get_active_workspace()
|
||||||
@@ -0,0 +1,87 @@
|
|||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from typing_extensions import override
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs import (
|
||||||
|
FinishFileImportInput,
|
||||||
|
GenerateFileUploadUrlInput,
|
||||||
|
StartFileImportInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import FileImport, FileUploadUrl
|
||||||
|
from specklepy.core.api.models.current import ResourceCollection
|
||||||
|
from specklepy.core.api.resources import FileImportResource as CoreResource
|
||||||
|
from specklepy.core.api.resources.current.file_import_resource import UploadFileResponse
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class FileImportResource(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,
|
||||||
|
)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def start_file_import(self, input: StartFileImportInput) -> FileImport:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "File Import Start"})
|
||||||
|
return super().start_file_import(input)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def generate_upload_url(self, input: GenerateFileUploadUrlInput) -> FileUploadUrl:
|
||||||
|
"""
|
||||||
|
Get a file upload url from the Speckle server.
|
||||||
|
|
||||||
|
This method asks the server to create a pre-signed S3 url,
|
||||||
|
which can be used as a short term authenticated route,
|
||||||
|
to put a file to the server.
|
||||||
|
"""
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "File Import Generate Upload Url"}
|
||||||
|
)
|
||||||
|
return super().generate_upload_url(input)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def upload_file(self, file: Path, url: str) -> UploadFileResponse:
|
||||||
|
"""
|
||||||
|
Uploads a file to the given S3 url.
|
||||||
|
|
||||||
|
This method should be used together with the generate_upload_url method,
|
||||||
|
which generates a pre-signed S3 url, that can be used to upload the file to.
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "File Import Upload File"})
|
||||||
|
return super().upload_file(file, url)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def download_file(self, project_id: str, file_id: str, target_file: Path) -> Path:
|
||||||
|
"""Download a file blob attached to the project, to the target path."""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "File Import Download File"})
|
||||||
|
return super().download_file(project_id, file_id, target_file)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def finish_file_import_job(self, input: FinishFileImportInput) -> bool:
|
||||||
|
"""
|
||||||
|
This is mostly an internal api, that marks a file import job finished.
|
||||||
|
|
||||||
|
Only use this if you are writing a file importer, that is responsible for
|
||||||
|
processing file import jobs.
|
||||||
|
"""
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "File Import Finish Job"})
|
||||||
|
return super().finish_file_import_job(input)
|
||||||
|
|
||||||
|
@override
|
||||||
|
def get_model_file_import_jobs(
|
||||||
|
self,
|
||||||
|
*,
|
||||||
|
project_id: str,
|
||||||
|
model_id: str,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: str | None = None,
|
||||||
|
) -> ResourceCollection[FileImport]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "File Import Get Model Jobs"})
|
||||||
|
return super().get_model_file_import_jobs(
|
||||||
|
project_id=project_id, model_id=model_id, limit=limit, cursor=cursor
|
||||||
|
)
|
||||||
@@ -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,45 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
LimitedUser,
|
||||||
|
UserSearchResultCollection,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources import OtherUserResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
|
)
|
||||||
@@ -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,79 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.project_inputs import (
|
||||||
|
ProjectCreateInput,
|
||||||
|
ProjectModelsFilter,
|
||||||
|
ProjectUpdateInput,
|
||||||
|
ProjectUpdateRoleInput,
|
||||||
|
WorkspaceProjectCreateInput,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models import (
|
||||||
|
Project,
|
||||||
|
ProjectWithModels,
|
||||||
|
ProjectWithTeam,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.models.current import ProjectPermissionChecks
|
||||||
|
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_permissions(self, project_id: str) -> ProjectPermissionChecks:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK, self.account, {"name": "Project Project Permissions "}
|
||||||
|
)
|
||||||
|
return super().get_permissions(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 create_in_workspace(self, input: WorkspaceProjectCreateInput) -> Project:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Workspace Project Create"})
|
||||||
|
return super().create_in_workspace(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)
|
||||||
+4
-3
@@ -1,11 +1,11 @@
|
|||||||
from typing import Any, Dict, List, Tuple
|
from typing import Any, Dict, List, Tuple
|
||||||
|
|
||||||
from specklepy.api.models import ServerInfo
|
from specklepy.api.models import ServerInfo
|
||||||
from specklepy.core.api.resources.server import Resource as CoreResource
|
from specklepy.core.api.resources import ServerResource as CoreResource
|
||||||
from specklepy.logging import metrics
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
class Resource(CoreResource):
|
class ServerResource(CoreResource):
|
||||||
"""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:
|
||||||
@@ -31,7 +31,8 @@ class Resource(CoreResource):
|
|||||||
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
|
||||||
return super().version()
|
return super().version()
|
||||||
|
|
||||||
def apps(self) -> Dict:
|
def apps(self) -> Dict:
|
||||||
@@ -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) -> Version:
|
||||||
|
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,53 @@
|
|||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
from specklepy.core.api.inputs.project_inputs import WorksaceProjectsFilter
|
||||||
|
from specklepy.core.api.models.current import (
|
||||||
|
Project,
|
||||||
|
ProjectWithPermissions,
|
||||||
|
ResourceCollection,
|
||||||
|
Workspace,
|
||||||
|
)
|
||||||
|
from specklepy.core.api.resources import WorkspaceResource as CoreResource
|
||||||
|
from specklepy.logging import metrics
|
||||||
|
|
||||||
|
|
||||||
|
class WorkspaceResource(CoreResource):
|
||||||
|
"""API Access class for workspace"""
|
||||||
|
|
||||||
|
def __init__(self, account, basepath, client, server_version) -> None:
|
||||||
|
super().__init__(
|
||||||
|
account=account,
|
||||||
|
basepath=basepath,
|
||||||
|
client=client,
|
||||||
|
server_version=server_version,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get(self, workspace_id: str) -> Workspace:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Workspace Get"})
|
||||||
|
return super().get(workspace_id)
|
||||||
|
|
||||||
|
def get_projects(
|
||||||
|
self,
|
||||||
|
workspace_id: str,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[WorksaceProjectsFilter] = None,
|
||||||
|
) -> ResourceCollection[Project]:
|
||||||
|
metrics.track(metrics.SDK, self.account, {"name": "Workspace Get Projects"})
|
||||||
|
return super().get_projects(workspace_id, limit, cursor, filter)
|
||||||
|
|
||||||
|
def get_projects_with_permissions(
|
||||||
|
self,
|
||||||
|
workspace_id: str,
|
||||||
|
limit: int = 25,
|
||||||
|
cursor: Optional[str] = None,
|
||||||
|
filter: Optional[WorksaceProjectsFilter] = None,
|
||||||
|
) -> ResourceCollection[ProjectWithPermissions]:
|
||||||
|
metrics.track(
|
||||||
|
metrics.SDK,
|
||||||
|
self.account,
|
||||||
|
{"name": "Workspace Get Projects With Permissions"},
|
||||||
|
)
|
||||||
|
return super().get_projects_with_permissions(
|
||||||
|
workspace_id, limit, cursor, filter
|
||||||
|
)
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
from typing import Dict, List
|
|
||||||
|
|
||||||
from specklepy.core.api.resources.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
|
|
||||||
|
|
||||||
def get(self, stream_id: str, object_id: str) -> Base:
|
|
||||||
"""
|
|
||||||
Get a stream object
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream for the object
|
|
||||||
object_id {str} -- the hash of the object you want to get
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Base -- the returned Base object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Object Get"})
|
|
||||||
return super().get(stream_id, object_id)
|
|
||||||
|
|
||||||
def create(self, stream_id: str, objects: List[Dict]) -> str:
|
|
||||||
"""
|
|
||||||
Not advised - generally, you want to use `operations.send()`.
|
|
||||||
|
|
||||||
Create a new object on a stream.
|
|
||||||
To send a base object, you can prepare it by running it through the
|
|
||||||
`BaseObjectSerializer.traverse_base()` function to get a valid (serialisable)
|
|
||||||
object to send.
|
|
||||||
|
|
||||||
NOTE: this does not create a commit - you can create one with
|
|
||||||
`SpeckleClient.commit.create`.
|
|
||||||
Dynamic fields will be located in the 'data' dict of the received `Base` object
|
|
||||||
|
|
||||||
Arguments:
|
|
||||||
stream_id {str} -- the id of the stream you want to send the object to
|
|
||||||
objects {List[Dict]}
|
|
||||||
-- a list of base dictionary objects (NOTE: must be json serialisable)
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
str -- the id of the object
|
|
||||||
"""
|
|
||||||
metrics.track(metrics.SDK, self.account, {"name": "Object Create"})
|
|
||||||
return super().create(stream_id, objects)
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user