diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml
new file mode 100644
index 0000000..1e261f1
--- /dev/null
+++ b/.github/workflows/release.yml
@@ -0,0 +1,69 @@
+name: Release
+
+on:
+ push:
+ branches: [release]
+ workflow_dispatch:
+ branches: [release]
+ workflow_call:
+ inputs:
+ QT_DEPENDENCIES:
+ description: 'Set to true to install Qt dependencies (Linux)'
+ required: false
+ default: 'false'
+ type: string
+
+jobs:
+ run-tests:
+ uses: nxt-dev/nxt/.github/workflows/unittests.yml@release
+ with:
+ QT_DEPENDENCIES: ${{ inputs.QT_DEPENDENCIES }}
+ build-and-deploy-pypi:
+ needs: [run-tests]
+ runs-on: ubuntu-latest
+ env:
+ TWINE_PASSWORD: ${{ secrets.TWINE_PASSWORD }}
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v4
+
+ - name: Checkout build graph (if needed)
+ if: github.repository != 'nxt-dev/nxt'
+ uses: actions/checkout@v4
+ with:
+ repository: nxt-dev/nxt
+ ref: release
+ path: nxt
+ sparse-checkout: build
+
+
+ - name: Install Python 3.10
+ uses: actions/setup-python@v5
+ with:
+ python-version: '3.10'
+
+ - name: Ensure PIP
+ run: python -m ensurepip --upgrade
+
+ - name: Install nxt-core (if needed)
+ if: github.repository != 'nxt-dev/nxt'
+ run: |
+ pip install git+https://github.com/nxt-dev/nxt.git@release
+
+ - name: Install this package and dependencies
+ run: |
+ pip install twine
+ pip install .
+
+ - name: Set NXT_FILE_ROOTS (if needed)
+ if: github.repository != 'nxt-dev/nxt'
+ env:
+ # Thepackaging graph expects nxt to be cloned next to nxt_editor
+ # this is not possible in GHA so we abuse the file fall backs to
+ # trick it into working
+ NXT_FILE_ROOTS: '../docs/api_docs'
+ run: echo "NXT_FILE_ROOTS=$NXT_FILE_ROOTS" >> $GITHUB_ENV
+
+ - name: Build package and upload
+ run: |
+ python -m nxt.cli -vv exec build/packaging.nxt -s /make_and_upload
diff --git a/.github/workflows/unittests.yml b/.github/workflows/unittests.yml
new file mode 100644
index 0000000..c23ed16
--- /dev/null
+++ b/.github/workflows/unittests.yml
@@ -0,0 +1,91 @@
+name: Unittests
+
+on:
+ workflow_dispatch:
+ branches-ignore: [release]
+ workflow_call:
+ inputs:
+ QT_DEPENDENCIES:
+ description: 'Set to true to install Qt dependencies (Linux)'
+ required: false
+ default: 'false'
+ type: string
+ pull_request:
+ branches-ignore: [release]
+ push:
+ branches-ignore: [release]
+
+jobs:
+ test-python:
+ env:
+ QT_DEPENDENCIES: ${{ inputs.QT_DEPENDENCIES }}
+ strategy:
+ matrix:
+ os: [ubuntu-latest, windows-latest]
+ python-version: ['3.7', '3.8', '3.9', '3.10']
+ fail-fast: true
+ runs-on: ${{ matrix.os }}
+
+ steps:
+ - name: Checkout code
+ uses: actions/checkout@v3
+
+ - name: Determine nxt-core branch to use
+ run: |
+ if [ "${{ github.ref_name }}" = "release" ]; then
+ echo "NXT_CORE_BRANCH=release" >> $GITHUB_ENV
+ else
+ echo "NXT_CORE_BRANCH=dev" >> $GITHUB_ENV
+ fi
+ shell: bash
+
+ - name: Install Python ${{ matrix.python-version }}
+ uses: actions/setup-python@v4
+ with:
+ python-version: ${{ matrix.python-version }}
+
+ - name: Install dependencies (Ubuntu)
+ if: runner.os == 'Linux' && env.QT_DEPENDENCIES == 'true'
+ run: |
+ sudo apt-get update
+ sudo apt-get install -y \
+ libxkbcommon-x11-0 \
+ libxrender1 \
+ libxext6 \
+ libxcb1 \
+ libxcb-icccm4 \
+ libxcb-image0 \
+ libxcb-keysyms1 \
+ libxcb-randr0 \
+ libxcb-render0 \
+ libxcb-render-util0 \
+ libxcb-shape0 \
+ libxcb-shm0 \
+ libxcb-sync1 \
+ libxcb-xfixes0 \
+ libxcb-xinerama0 \
+ libxcb-xkb1 \
+ libqt5x11extras5 \
+ xvfb
+
+ - name: Ensure PIP
+ run: python -m ensurepip --upgrade
+
+ - name: Install nxt-core
+ if: github.repository != 'nxt-dev/nxt'
+ run: |
+ pip install git+https://github.com/nxt-dev/nxt.git@${{ env.NXT_CORE_BRANCH }}
+
+ - name: Install this package
+ run: pip install .
+
+ - name: Run tests with xvfb (Ubuntu + Qt)
+ if: runner.os == 'Linux' && env.QT_DEPENDENCIES == 'true'
+ run: |
+ Xvfb :99 -screen 0 1024x768x24 &
+ export DISPLAY=:99
+ python -m nxt.cli test
+
+ - name: Run Unittests from CLI
+ if: env.QT_DEPENDENCIES != 'true'
+ run: python -m nxt.cli test
\ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
deleted file mode 100644
index b6603eb..0000000
--- a/.travis.yml
+++ /dev/null
@@ -1,34 +0,0 @@
-language: python
-branches:
- only:
- - release
- - dev
-python:
- - "3.9"
-
-before_install:
- - |
- if [ "$TRAVIS_BRANCH" = "dev" ] && [ "$TRAVIS_EVENT_TYPE" = "push" ]; then
- echo "Push to dev, not running tests until PR"
- exit 0
- else
- echo "Doing the build"
- fi
-
-install:
- - cd ..
- - pip install ./nxt
- - pip install importlib-metadata
- - pip install twine
-
-script:
- - |
- if [ "$TRAVIS_BRANCH" = "release" ] && [ "$TRAVIS_EVENT_TYPE" = "pull_request" ]; then
- python -m nxt.cli test
- exit $?
- fi
- - |
- if [ "$TRAVIS_BRANCH" = "release" ] && [ "$TRAVIS_EVENT_TYPE" = "push" ]; then
- python -m nxt.cli exec nxt/build/packaging.nxt -s /make_and_upload
- exit $?
- fi
\ No newline at end of file
diff --git a/README.md b/README.md
index f5abf05..a443cce 100644
--- a/README.md
+++ b/README.md
@@ -1,9 +1,17 @@
+
+
+![Release Status](https://github.com/nxt-dev/nxt/actions/workflows/release.yml/badge.svg?branch=release)
+![Dev Status](https://github.com/nxt-dev/nxt/actions/workflows/unittests.yml/badge.svg?branch=dev)
# NXT Python Core
+
+
+
**nxt** (**/ɛn·ɛks·ti/**) is a general purpose code compositor designed for rigging, scene assembly, and automation. (node execution tree)
[Installation/Usage](#installationusage) | [Docs](https://nxt-dev.github.io/) | [Contributing](CONTRIBUTING.md) | [Licensing](LICENSE)
# Installation/Usage
+
**This repo is the nxt core, it does not come with a UI. If you're looking for the UI, please see [the nxt editor](https://github.com/nxt-dev/nxt_editor).**
This package is designed for headless execution of graphs in a render farm or other headless environment.
Only clone this repo if you're contributing to the NXT codebase.
@@ -11,18 +19,21 @@ Only clone this repo if you're contributing to the NXT codebase.
#### Requirements
+
- Python >= [2.7.*](https://www.python.org/download/releases/2.7) <= [3.7.*](https://www.python.org/download/releases/3.7)
- We strongly recommend using a Python [virtual environment](https://docs.python.org/3.7/tutorial/venv.html)
*[Requirements for contributors](CONTRIBUTING.md#python-environment)*
### NXT Python Core
+
Our releases are hosted on [PyPi](https://pypi.org/project/nxt-editor/).
**Install:**
`pip install nxt-core`
**Execute Graph:**
+
```python
import nxt
nxt.execute_graph('path/to/graph.nxt')
@@ -34,11 +45,6 @@ nxt.execute_graph('path/to/graph.nxt')
## Special Thanks
-[Sunrise Productions](https://sunriseproductions.tv/) | [School of Visual Art and Design](https://www.southern.edu/visualartanddesign/)
----
-
-| Release | Dev |
-| :---: | :---: |
-| [![Build Status](https://travis-ci.com/nxt-dev/nxt.svg?token=rBRbAJTv2rq1c8WVEwGs&branch=release)](https://travis-ci.com/nxt-dev/nxt) | [![Build Status](https://travis-ci.com/nxt-dev/nxt.svg?token=rBRbAJTv2rq1c8WVEwGs&branch=dev)](https://travis-ci.com/nxt-dev/nxt) |
+[Sunrise Productions](https://sunriseproductions.tv/) | [School of Visual Art and Design](https://www.southern.edu/visualartanddesign/)
diff --git a/build/packaging.nxt b/build/packaging.nxt
index e6b6031..eed556e 100644
--- a/build/packaging.nxt
+++ b/build/packaging.nxt
@@ -7,16 +7,16 @@
"meta_data": {
"positions": {
"/make_and_test_upload": [
- -223,
- 306
+ 380.0,
+ 340.0
],
"/make_and_upload": [
- -221.61718055199395,
- 65.0
+ 400.0,
+ 100.0
],
"/make_package": [
- -223.0,
- -153.0
+ 380.0,
+ -120.0
]
}
},
@@ -45,7 +45,7 @@
"import subprocess",
"orig_cwd = os.getcwd()",
"os.chdir('${repo_root}')",
- "subprocess.call([sys.executable, '-m', 'twine' ,'upload', '--repository', 'testpypi', '${dest_dir}/*'])",
+ "subprocess.check_call([sys.executable, '-m', 'twine' ,'upload', '--repository', 'testpypi', '${dest_dir}/*'])",
"os.chdir(orig_cwd)"
]
},
diff --git a/nxt/nxt_log.py b/nxt/nxt_log.py
index 1327939..fa995b7 100644
--- a/nxt/nxt_log.py
+++ b/nxt/nxt_log.py
@@ -20,6 +20,7 @@
# Internal
from .constants import USER_DIR
+LOG_DIR = os.path.join(tempfile.gettempdir(), 'nxt_logs')
VERBOSE_ENV_VAR = 'NXT_VERBOSE'
"""Logging levels are as follows, with custom levels marked with *
@@ -248,10 +249,9 @@ def get_new_session_log_filename():
:rtype: str
"""
logger = logging.getLogger(__name__)
- log_dir = os.path.join(tempfile.gettempdir(), 'nxt_logs')
try:
- os.makedirs(log_dir)
- os.chmod(log_dir, 0o777)
+ os.makedirs(LOG_DIR)
+ os.chmod(LOG_DIR, 0o777)
except OSError as err:
if err.errno is errno.EEXIST:
logger.debug("Not creating logs directory because it exists.")
@@ -259,7 +259,7 @@ def get_new_session_log_filename():
raise
log_filename_template = "nxt_session_{time}.log"
log_filename = log_filename_template.format(time=str(int(time.time())))
- return os.path.join(log_dir, log_filename)
+ return os.path.join(LOG_DIR, log_filename)
def stop_session_log(filename):
diff --git a/nxt/plugin_loader.py b/nxt/plugin_loader.py
index 278fd02..836e8cf 100644
--- a/nxt/plugin_loader.py
+++ b/nxt/plugin_loader.py
@@ -1,6 +1,6 @@
# Built-in
import os
-import imp
+import importlib.util
import logging
# Internal
@@ -23,10 +23,11 @@ def load_plugins():
mod_name, _ = os.path.splitext(file_name)
if mod_name in _nxt_loaded_plugin_module_names:
continue
- try:
- found = imp.find_module(mod_name, [plugin_dir])
- except ImportError:
+ mod_path = os.path.join(plugin_dir, file_name)
+ spec = importlib.util.spec_from_file_location(mod_name, mod_path)
+ if not spec:
continue
- mod = imp.load_module(mod_name, *found)
+ mod = importlib.util.module_from_spec(spec)
_nxt_loaded_plugin_module_names += [mod_name]
_nxt_loaded_plugin_modules += [mod]
+ spec.loader.exec_module(mod)
diff --git a/nxt/test/test_plugins.py b/nxt/test/test_plugins.py
index b240a46..cf8386a 100644
--- a/nxt/test/test_plugins.py
+++ b/nxt/test/test_plugins.py
@@ -3,7 +3,7 @@
import os
# Internal
-from nxt.plugins.file_fallbacks import NXT_FILE_ROOTS
+from nxt.plugins.file_fallbacks import NXT_FILE_ROOTS, ROOT_PATH_SEP
from nxt.session import Session
from nxt.stage import DATA_STATE
@@ -218,7 +218,8 @@ def test_saved(self):
print('Test in a saved file with an environment, path:: '
'resolves to first real env root')
test_root = os.path.dirname(my_dir)
- os.environ[NXT_FILE_ROOTS] = 'a super fake root;' + test_root
+ test_roots = 'a super fake root' + ROOT_PATH_SEP + test_root
+ os.environ[NXT_FILE_ROOTS] = test_roots
expected = os.path.join(test_root, self.test_filename).replace(os.sep,
'/')
found = test_stage.get_node_attr_value(test_node, test_attr,
diff --git a/nxt/test/test_stage.py b/nxt/test/test_stage.py
index df5d11d..db2f5d1 100644
--- a/nxt/test/test_stage.py
+++ b/nxt/test/test_stage.py
@@ -278,7 +278,7 @@ def test_inherit_attrs(self):
self.assertIs(True, hasattr(runtime_child, 'parent'))
print("Test that the local attr from the parent node is IS accessible"
"from the EDITOR child node via `self`.")
- self.assertEquals(True, hasattr(self.child_node, 'parent'))
+ self.assertEqual(True, hasattr(self.child_node, 'parent'))
# Test that code is not inherited
print("Test that child code is NOT inherited FROM the PARENT.")
self.assertEqual(["# Code"], getattr(runtime_parent,
diff --git a/setup.py b/setup.py
index bb692d5..2ce3974 100644
--- a/setup.py
+++ b/setup.py
@@ -26,7 +26,7 @@
long_description_content_type="text/markdown",
url="https://github.com/nxt-dev/nxt",
packages=setuptools.find_packages(),
- python_requires='>=2.7, <3.11',
+ python_requires='>=3.7, <3.11',
entry_points={
'console_scripts': [
'nxt=nxt.cli:main',