Initial working version
This commit is contained in:
parent
2c987058e7
commit
17e35fff4a
27 changed files with 2731 additions and 2 deletions
72
.github/workflows/build.yml
vendored
Normal file
72
.github/workflows/build.yml
vendored
Normal file
|
|
@ -0,0 +1,72 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
name: Build CI
|
||||
|
||||
on: [pull_request, push]
|
||||
|
||||
jobs:
|
||||
test:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Dump GitHub context
|
||||
env:
|
||||
GITHUB_CONTEXT: ${{ toJson(github) }}
|
||||
run: echo "$GITHUB_CONTEXT"
|
||||
- name: Translate Repo Name For Build Tools filename_prefix
|
||||
id: repo-name
|
||||
run: |
|
||||
echo ::set-output name=repo-name::$(
|
||||
echo ${{ github.repository }} |
|
||||
awk -F '\/' '{ print tolower($2) }' |
|
||||
tr '_' '-'
|
||||
)
|
||||
- name: Set up Python 3.6
|
||||
uses: actions/setup-python@v1
|
||||
with:
|
||||
python-version: 3.6
|
||||
- name: Versions
|
||||
run: |
|
||||
python3 --version
|
||||
- name: Checkout Current Repo
|
||||
uses: actions/checkout@v1
|
||||
with:
|
||||
submodules: true
|
||||
- name: Checkout tools repo
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
repository: adafruit/actions-ci-circuitpython-libs
|
||||
path: actions-ci
|
||||
- name: Install dependencies
|
||||
# (e.g. - apt-get: gettext, etc; pip: circuitpython-build-tools, requirements.txt; etc.)
|
||||
run: |
|
||||
source actions-ci/install.sh
|
||||
- name: Pip install pylint, Sphinx, pre-commit
|
||||
run: |
|
||||
pip install --force-reinstall pylint Sphinx sphinx-rtd-theme pre-commit
|
||||
- name: Library version
|
||||
run: git describe --dirty --always --tags
|
||||
- name: Pre-commit hooks
|
||||
run: |
|
||||
pre-commit run --all-files
|
||||
- name: PyLint
|
||||
run: |
|
||||
pylint $( find . -path './adafruit*.py' )
|
||||
([[ ! -d "examples" ]] || pylint --disable=missing-docstring,invalid-name,bad-whitespace $( find . -path "./examples/*.py" ))
|
||||
- name: Build assets
|
||||
run: circuitpython-build-bundles --filename_prefix ${{ steps.repo-name.outputs.repo-name }} --library_location .
|
||||
- name: Build docs
|
||||
working-directory: docs
|
||||
run: sphinx-build -E -W -b html . _build/html
|
||||
- name: Check For setup.py
|
||||
id: need-pypi
|
||||
run: |
|
||||
echo ::set-output name=setup-py::$( find . -wholename './setup.py' )
|
||||
- name: Build Python package
|
||||
if: contains(steps.need-pypi.outputs.setup-py, 'setup.py')
|
||||
run: |
|
||||
pip install --upgrade setuptools wheel twine readme_renderer testresources
|
||||
python setup.py sdist
|
||||
python setup.py bdist_wheel --universal
|
||||
twine check dist/*
|
||||
85
.github/workflows/release.yml
vendored
Normal file
85
.github/workflows/release.yml
vendored
Normal file
|
|
@ -0,0 +1,85 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
name: Release Actions
|
||||
|
||||
on:
|
||||
release:
|
||||
types: [published]
|
||||
|
||||
jobs:
|
||||
upload-release-assets:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Dump GitHub context
|
||||
env:
|
||||
GITHUB_CONTEXT: ${{ toJson(github) }}
|
||||
run: echo "$GITHUB_CONTEXT"
|
||||
- name: Translate Repo Name For Build Tools filename_prefix
|
||||
id: repo-name
|
||||
run: |
|
||||
echo ::set-output name=repo-name::$(
|
||||
echo ${{ github.repository }} |
|
||||
awk -F '\/' '{ print tolower($2) }' |
|
||||
tr '_' '-'
|
||||
)
|
||||
- name: Set up Python 3.6
|
||||
uses: actions/setup-python@v1
|
||||
with:
|
||||
python-version: 3.6
|
||||
- name: Versions
|
||||
run: |
|
||||
python3 --version
|
||||
- name: Checkout Current Repo
|
||||
uses: actions/checkout@v1
|
||||
with:
|
||||
submodules: true
|
||||
- name: Checkout tools repo
|
||||
uses: actions/checkout@v2
|
||||
with:
|
||||
repository: adafruit/actions-ci-circuitpython-libs
|
||||
path: actions-ci
|
||||
- name: Install deps
|
||||
run: |
|
||||
source actions-ci/install.sh
|
||||
- name: Build assets
|
||||
run: circuitpython-build-bundles --filename_prefix ${{ steps.repo-name.outputs.repo-name }} --library_location .
|
||||
- name: Upload Release Assets
|
||||
# the 'official' actions version does not yet support dynamically
|
||||
# supplying asset names to upload. @csexton's version chosen based on
|
||||
# discussion in the issue below, as its the simplest to implement and
|
||||
# allows for selecting files with a pattern.
|
||||
# https://github.com/actions/upload-release-asset/issues/4
|
||||
#uses: actions/upload-release-asset@v1.0.1
|
||||
uses: csexton/release-asset-action@master
|
||||
with:
|
||||
pattern: "bundles/*"
|
||||
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
||||
upload-pypi:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v1
|
||||
- name: Check For setup.py
|
||||
id: need-pypi
|
||||
run: |
|
||||
echo ::set-output name=setup-py::$( find . -wholename './setup.py' )
|
||||
- name: Set up Python
|
||||
if: contains(steps.need-pypi.outputs.setup-py, 'setup.py')
|
||||
uses: actions/setup-python@v1
|
||||
with:
|
||||
python-version: '3.x'
|
||||
- name: Install dependencies
|
||||
if: contains(steps.need-pypi.outputs.setup-py, 'setup.py')
|
||||
run: |
|
||||
python -m pip install --upgrade pip
|
||||
pip install setuptools wheel twine
|
||||
- name: Build and publish
|
||||
if: contains(steps.need-pypi.outputs.setup-py, 'setup.py')
|
||||
env:
|
||||
TWINE_USERNAME: ${{ secrets.pypi_username }}
|
||||
TWINE_PASSWORD: ${{ secrets.pypi_password }}
|
||||
run: |
|
||||
python setup.py sdist
|
||||
twine upload dist/*
|
||||
18
.gitignore
vendored
Normal file
18
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,18 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: Unlicense
|
||||
|
||||
*.mpy
|
||||
.idea
|
||||
__pycache__
|
||||
_build
|
||||
*.pyc
|
||||
.env
|
||||
.python-version
|
||||
build*/
|
||||
bundles
|
||||
*.DS_Store
|
||||
.eggs
|
||||
dist
|
||||
**/*.egg-info
|
||||
.vscode
|
||||
19
.pre-commit-config.yaml
Normal file
19
.pre-commit-config.yaml
Normal file
|
|
@ -0,0 +1,19 @@
|
|||
# SPDX-FileCopyrightText: 2020 Diego Elio Pettenò
|
||||
#
|
||||
# SPDX-License-Identifier: Unlicense
|
||||
|
||||
repos:
|
||||
- repo: https://github.com/python/black
|
||||
rev: stable
|
||||
hooks:
|
||||
- id: black
|
||||
- repo: https://github.com/fsfe/reuse-tool
|
||||
rev: latest
|
||||
hooks:
|
||||
- id: reuse
|
||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||
rev: v2.3.0
|
||||
hooks:
|
||||
- id: check-yaml
|
||||
- id: end-of-file-fixer
|
||||
- id: trailing-whitespace
|
||||
437
.pylintrc
Normal file
437
.pylintrc
Normal file
|
|
@ -0,0 +1,437 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: Unlicense
|
||||
|
||||
[MASTER]
|
||||
|
||||
# A comma-separated list of package or module names from where C extensions may
|
||||
# be loaded. Extensions are loading into the active Python interpreter and may
|
||||
# run arbitrary code
|
||||
extension-pkg-whitelist=
|
||||
|
||||
# Add files or directories to the blacklist. They should be base names, not
|
||||
# paths.
|
||||
ignore=CVS
|
||||
|
||||
# Add files or directories matching the regex patterns to the blacklist. The
|
||||
# regex matches against base names, not paths.
|
||||
ignore-patterns=
|
||||
|
||||
# Python code to execute, usually for sys.path manipulation such as
|
||||
# pygtk.require().
|
||||
#init-hook=
|
||||
|
||||
# Use multiple processes to speed up Pylint.
|
||||
# jobs=1
|
||||
jobs=2
|
||||
|
||||
# List of plugins (as comma separated values of python modules names) to load,
|
||||
# usually to register additional checkers.
|
||||
load-plugins=
|
||||
|
||||
# Pickle collected data for later comparisons.
|
||||
persistent=yes
|
||||
|
||||
# Specify a configuration file.
|
||||
#rcfile=
|
||||
|
||||
# Allow loading of arbitrary C extensions. Extensions are imported into the
|
||||
# active Python interpreter and may run arbitrary code.
|
||||
unsafe-load-any-extension=no
|
||||
|
||||
|
||||
[MESSAGES CONTROL]
|
||||
|
||||
# Only show warnings with the listed confidence levels. Leave empty to show
|
||||
# all. Valid levels: HIGH, INFERENCE, INFERENCE_FAILURE, UNDEFINED
|
||||
confidence=
|
||||
|
||||
# Disable the message, report, category or checker with the given id(s). You
|
||||
# can either give multiple identifiers separated by comma (,) or put this
|
||||
# option multiple times (only on the command line, not in the configuration
|
||||
# file where it should appear only once).You can also use "--disable=all" to
|
||||
# disable everything first and then reenable specific checks. For example, if
|
||||
# you want to run only the similarities checker, you can use "--disable=all
|
||||
# --enable=similarities". If you want to run only the classes checker, but have
|
||||
# no Warning level messages displayed, use"--disable=all --enable=classes
|
||||
# --disable=W"
|
||||
# disable=import-error,print-statement,parameter-unpacking,unpacking-in-except,old-raise-syntax,backtick,long-suffix,old-ne-operator,old-octal-literal,import-star-module-level,raw-checker-failed,bad-inline-option,locally-disabled,locally-enabled,file-ignored,suppressed-message,useless-suppression,deprecated-pragma,apply-builtin,basestring-builtin,buffer-builtin,cmp-builtin,coerce-builtin,execfile-builtin,file-builtin,long-builtin,raw_input-builtin,reduce-builtin,standarderror-builtin,unicode-builtin,xrange-builtin,coerce-method,delslice-method,getslice-method,setslice-method,no-absolute-import,old-division,dict-iter-method,dict-view-method,next-method-called,metaclass-assignment,indexing-exception,raising-string,reload-builtin,oct-method,hex-method,nonzero-method,cmp-method,input-builtin,round-builtin,intern-builtin,unichr-builtin,map-builtin-not-iterating,zip-builtin-not-iterating,range-builtin-not-iterating,filter-builtin-not-iterating,using-cmp-argument,eq-without-hash,div-method,idiv-method,rdiv-method,exception-message-attribute,invalid-str-codec,sys-max-int,bad-python3-import,deprecated-string-function,deprecated-str-translate-call
|
||||
disable=print-statement,parameter-unpacking,unpacking-in-except,old-raise-syntax,backtick,long-suffix,old-ne-operator,old-octal-literal,import-star-module-level,raw-checker-failed,bad-inline-option,locally-disabled,locally-enabled,file-ignored,suppressed-message,useless-suppression,deprecated-pragma,apply-builtin,basestring-builtin,buffer-builtin,cmp-builtin,coerce-builtin,execfile-builtin,file-builtin,long-builtin,raw_input-builtin,reduce-builtin,standarderror-builtin,unicode-builtin,xrange-builtin,coerce-method,delslice-method,getslice-method,setslice-method,no-absolute-import,old-division,dict-iter-method,dict-view-method,next-method-called,metaclass-assignment,indexing-exception,raising-string,reload-builtin,oct-method,hex-method,nonzero-method,cmp-method,input-builtin,round-builtin,intern-builtin,unichr-builtin,map-builtin-not-iterating,zip-builtin-not-iterating,range-builtin-not-iterating,filter-builtin-not-iterating,using-cmp-argument,eq-without-hash,div-method,idiv-method,rdiv-method,exception-message-attribute,invalid-str-codec,sys-max-int,bad-python3-import,deprecated-string-function,deprecated-str-translate-call,import-error,bad-continuation
|
||||
|
||||
# Enable the message, report, category or checker with the given id(s). You can
|
||||
# either give multiple identifier separated by comma (,) or put this option
|
||||
# multiple time (only on the command line, not in the configuration file where
|
||||
# it should appear only once). See also the "--disable" option for examples.
|
||||
enable=
|
||||
|
||||
|
||||
[REPORTS]
|
||||
|
||||
# Python expression which should return a note less than 10 (10 is the highest
|
||||
# note). You have access to the variables errors warning, statement which
|
||||
# respectively contain the number of errors / warnings messages and the total
|
||||
# number of statements analyzed. This is used by the global evaluation report
|
||||
# (RP0004).
|
||||
evaluation=10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10)
|
||||
|
||||
# Template used to display messages. This is a python new-style format string
|
||||
# used to format the message information. See doc for all details
|
||||
#msg-template=
|
||||
|
||||
# Set the output format. Available formats are text, parseable, colorized, json
|
||||
# and msvs (visual studio).You can also give a reporter class, eg
|
||||
# mypackage.mymodule.MyReporterClass.
|
||||
output-format=text
|
||||
|
||||
# Tells whether to display a full report or only the messages
|
||||
reports=no
|
||||
|
||||
# Activate the evaluation score.
|
||||
score=yes
|
||||
|
||||
|
||||
[REFACTORING]
|
||||
|
||||
# Maximum number of nested blocks for function / method body
|
||||
max-nested-blocks=5
|
||||
|
||||
|
||||
[LOGGING]
|
||||
|
||||
# Logging modules to check that the string format arguments are in logging
|
||||
# function parameter format
|
||||
logging-modules=logging
|
||||
|
||||
|
||||
[SPELLING]
|
||||
|
||||
# Spelling dictionary name. Available dictionaries: none. To make it working
|
||||
# install python-enchant package.
|
||||
spelling-dict=
|
||||
|
||||
# List of comma separated words that should not be checked.
|
||||
spelling-ignore-words=
|
||||
|
||||
# A path to a file that contains private dictionary; one word per line.
|
||||
spelling-private-dict-file=
|
||||
|
||||
# Tells whether to store unknown words to indicated private dictionary in
|
||||
# --spelling-private-dict-file option instead of raising a message.
|
||||
spelling-store-unknown-words=no
|
||||
|
||||
|
||||
[MISCELLANEOUS]
|
||||
|
||||
# List of note tags to take in consideration, separated by a comma.
|
||||
# notes=FIXME,XXX,TODO
|
||||
notes=FIXME,XXX
|
||||
|
||||
|
||||
[TYPECHECK]
|
||||
|
||||
# List of decorators that produce context managers, such as
|
||||
# contextlib.contextmanager. Add to this list to register other decorators that
|
||||
# produce valid context managers.
|
||||
contextmanager-decorators=contextlib.contextmanager
|
||||
|
||||
# List of members which are set dynamically and missed by pylint inference
|
||||
# system, and so shouldn't trigger E1101 when accessed. Python regular
|
||||
# expressions are accepted.
|
||||
generated-members=
|
||||
|
||||
# Tells whether missing members accessed in mixin class should be ignored. A
|
||||
# mixin class is detected if its name ends with "mixin" (case insensitive).
|
||||
ignore-mixin-members=yes
|
||||
|
||||
# This flag controls whether pylint should warn about no-member and similar
|
||||
# checks whenever an opaque object is returned when inferring. The inference
|
||||
# can return multiple potential results while evaluating a Python object, but
|
||||
# some branches might not be evaluated, which results in partial inference. In
|
||||
# that case, it might be useful to still emit no-member and other checks for
|
||||
# the rest of the inferred objects.
|
||||
ignore-on-opaque-inference=yes
|
||||
|
||||
# List of class names for which member attributes should not be checked (useful
|
||||
# for classes with dynamically set attributes). This supports the use of
|
||||
# qualified names.
|
||||
ignored-classes=optparse.Values,thread._local,_thread._local
|
||||
|
||||
# List of module names for which member attributes should not be checked
|
||||
# (useful for modules/projects where namespaces are manipulated during runtime
|
||||
# and thus existing member attributes cannot be deduced by static analysis. It
|
||||
# supports qualified module names, as well as Unix pattern matching.
|
||||
ignored-modules=board
|
||||
|
||||
# Show a hint with possible names when a member name was not found. The aspect
|
||||
# of finding the hint is based on edit distance.
|
||||
missing-member-hint=yes
|
||||
|
||||
# The minimum edit distance a name should have in order to be considered a
|
||||
# similar match for a missing member name.
|
||||
missing-member-hint-distance=1
|
||||
|
||||
# The total number of similar names that should be taken in consideration when
|
||||
# showing a hint for a missing member.
|
||||
missing-member-max-choices=1
|
||||
|
||||
|
||||
[VARIABLES]
|
||||
|
||||
# List of additional names supposed to be defined in builtins. Remember that
|
||||
# you should avoid to define new builtins when possible.
|
||||
additional-builtins=
|
||||
|
||||
# Tells whether unused global variables should be treated as a violation.
|
||||
allow-global-unused-variables=yes
|
||||
|
||||
# List of strings which can identify a callback function by name. A callback
|
||||
# name must start or end with one of those strings.
|
||||
callbacks=cb_,_cb
|
||||
|
||||
# A regular expression matching the name of dummy variables (i.e. expectedly
|
||||
# not used).
|
||||
dummy-variables-rgx=_+$|(_[a-zA-Z0-9_]*[a-zA-Z0-9]+?$)|dummy|^ignored_|^unused_
|
||||
|
||||
# Argument names that match this expression will be ignored. Default to name
|
||||
# with leading underscore
|
||||
ignored-argument-names=_.*|^ignored_|^unused_
|
||||
|
||||
# Tells whether we should check for unused import in __init__ files.
|
||||
init-import=no
|
||||
|
||||
# List of qualified module names which can have objects that can redefine
|
||||
# builtins.
|
||||
redefining-builtins-modules=six.moves,future.builtins
|
||||
|
||||
|
||||
[FORMAT]
|
||||
|
||||
# Expected format of line ending, e.g. empty (any line ending), LF or CRLF.
|
||||
# expected-line-ending-format=
|
||||
expected-line-ending-format=LF
|
||||
|
||||
# Regexp for a line that is allowed to be longer than the limit.
|
||||
ignore-long-lines=^\s*(# )?<?https?://\S+>?$
|
||||
|
||||
# Number of spaces of indent required inside a hanging or continued line.
|
||||
indent-after-paren=4
|
||||
|
||||
# String used as indentation unit. This is usually " " (4 spaces) or "\t" (1
|
||||
# tab).
|
||||
indent-string=' '
|
||||
|
||||
# Maximum number of characters on a single line.
|
||||
max-line-length=100
|
||||
|
||||
# Maximum number of lines in a module
|
||||
max-module-lines=1000
|
||||
|
||||
# List of optional constructs for which whitespace checking is disabled. `dict-
|
||||
# separator` is used to allow tabulation in dicts, etc.: {1 : 1,\n222: 2}.
|
||||
# `trailing-comma` allows a space between comma and closing bracket: (a, ).
|
||||
# `empty-line` allows space-only lines.
|
||||
no-space-check=trailing-comma,dict-separator
|
||||
|
||||
# Allow the body of a class to be on the same line as the declaration if body
|
||||
# contains single statement.
|
||||
single-line-class-stmt=no
|
||||
|
||||
# Allow the body of an if to be on the same line as the test if there is no
|
||||
# else.
|
||||
single-line-if-stmt=no
|
||||
|
||||
|
||||
[SIMILARITIES]
|
||||
|
||||
# Ignore comments when computing similarities.
|
||||
ignore-comments=yes
|
||||
|
||||
# Ignore docstrings when computing similarities.
|
||||
ignore-docstrings=yes
|
||||
|
||||
# Ignore imports when computing similarities.
|
||||
ignore-imports=no
|
||||
|
||||
# Minimum lines number of a similarity.
|
||||
min-similarity-lines=4
|
||||
|
||||
|
||||
[BASIC]
|
||||
|
||||
# Naming hint for argument names
|
||||
argument-name-hint=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Regular expression matching correct argument names
|
||||
argument-rgx=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Naming hint for attribute names
|
||||
attr-name-hint=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Regular expression matching correct attribute names
|
||||
attr-rgx=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Bad variable names which should always be refused, separated by a comma
|
||||
bad-names=foo,bar,baz,toto,tutu,tata
|
||||
|
||||
# Naming hint for class attribute names
|
||||
class-attribute-name-hint=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
||||
|
||||
# Regular expression matching correct class attribute names
|
||||
class-attribute-rgx=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
||||
|
||||
# Naming hint for class names
|
||||
# class-name-hint=[A-Z_][a-zA-Z0-9]+$
|
||||
class-name-hint=[A-Z_][a-zA-Z0-9_]+$
|
||||
|
||||
# Regular expression matching correct class names
|
||||
# class-rgx=[A-Z_][a-zA-Z0-9]+$
|
||||
class-rgx=[A-Z_][a-zA-Z0-9_]+$
|
||||
|
||||
# Naming hint for constant names
|
||||
const-name-hint=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
||||
|
||||
# Regular expression matching correct constant names
|
||||
const-rgx=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
||||
|
||||
# Minimum line length for functions/classes that require docstrings, shorter
|
||||
# ones are exempt.
|
||||
docstring-min-length=-1
|
||||
|
||||
# Naming hint for function names
|
||||
function-name-hint=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Regular expression matching correct function names
|
||||
function-rgx=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Good variable names which should always be accepted, separated by a comma
|
||||
# good-names=i,j,k,ex,Run,_
|
||||
good-names=r,g,b,w,i,j,k,n,x,y,z,ex,ok,Run,_
|
||||
|
||||
# Include a hint for the correct naming format with invalid-name
|
||||
include-naming-hint=no
|
||||
|
||||
# Naming hint for inline iteration names
|
||||
inlinevar-name-hint=[A-Za-z_][A-Za-z0-9_]*$
|
||||
|
||||
# Regular expression matching correct inline iteration names
|
||||
inlinevar-rgx=[A-Za-z_][A-Za-z0-9_]*$
|
||||
|
||||
# Naming hint for method names
|
||||
method-name-hint=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Regular expression matching correct method names
|
||||
method-rgx=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Naming hint for module names
|
||||
module-name-hint=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
||||
|
||||
# Regular expression matching correct module names
|
||||
module-rgx=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
||||
|
||||
# Colon-delimited sets of names that determine each other's naming style when
|
||||
# the name regexes allow several styles.
|
||||
name-group=
|
||||
|
||||
# Regular expression which should only match function or class names that do
|
||||
# not require a docstring.
|
||||
no-docstring-rgx=^_
|
||||
|
||||
# List of decorators that produce properties, such as abc.abstractproperty. Add
|
||||
# to this list to register other decorators that produce valid properties.
|
||||
property-classes=abc.abstractproperty
|
||||
|
||||
# Naming hint for variable names
|
||||
variable-name-hint=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
# Regular expression matching correct variable names
|
||||
variable-rgx=(([a-z][a-z0-9_]{2,30})|(_[a-z0-9_]*))$
|
||||
|
||||
|
||||
[IMPORTS]
|
||||
|
||||
# Allow wildcard imports from modules that define __all__.
|
||||
allow-wildcard-with-all=no
|
||||
|
||||
# Analyse import fallback blocks. This can be used to support both Python 2 and
|
||||
# 3 compatible code, which means that the block might have code that exists
|
||||
# only in one or another interpreter, leading to false positives when analysed.
|
||||
analyse-fallback-blocks=no
|
||||
|
||||
# Deprecated modules which should not be used, separated by a comma
|
||||
deprecated-modules=optparse,tkinter.tix
|
||||
|
||||
# Create a graph of external dependencies in the given file (report RP0402 must
|
||||
# not be disabled)
|
||||
ext-import-graph=
|
||||
|
||||
# Create a graph of every (i.e. internal and external) dependencies in the
|
||||
# given file (report RP0402 must not be disabled)
|
||||
import-graph=
|
||||
|
||||
# Create a graph of internal dependencies in the given file (report RP0402 must
|
||||
# not be disabled)
|
||||
int-import-graph=
|
||||
|
||||
# Force import order to recognize a module as part of the standard
|
||||
# compatibility libraries.
|
||||
known-standard-library=
|
||||
|
||||
# Force import order to recognize a module as part of a third party library.
|
||||
known-third-party=enchant
|
||||
|
||||
|
||||
[CLASSES]
|
||||
|
||||
# List of method names used to declare (i.e. assign) instance attributes.
|
||||
defining-attr-methods=__init__,__new__,setUp
|
||||
|
||||
# List of member names, which should be excluded from the protected access
|
||||
# warning.
|
||||
exclude-protected=_asdict,_fields,_replace,_source,_make
|
||||
|
||||
# List of valid names for the first argument in a class method.
|
||||
valid-classmethod-first-arg=cls
|
||||
|
||||
# List of valid names for the first argument in a metaclass class method.
|
||||
valid-metaclass-classmethod-first-arg=mcs
|
||||
|
||||
|
||||
[DESIGN]
|
||||
|
||||
# Maximum number of arguments for function / method
|
||||
max-args=5
|
||||
|
||||
# Maximum number of attributes for a class (see R0902).
|
||||
# max-attributes=7
|
||||
max-attributes=11
|
||||
|
||||
# Maximum number of boolean expressions in a if statement
|
||||
max-bool-expr=5
|
||||
|
||||
# Maximum number of branch for function / method body
|
||||
max-branches=12
|
||||
|
||||
# Maximum number of locals for function / method body
|
||||
max-locals=15
|
||||
|
||||
# Maximum number of parents for a class (see R0901).
|
||||
max-parents=7
|
||||
|
||||
# Maximum number of public methods for a class (see R0904).
|
||||
max-public-methods=20
|
||||
|
||||
# Maximum number of return / yield for function / method body
|
||||
max-returns=6
|
||||
|
||||
# Maximum number of statements in function / method body
|
||||
max-statements=50
|
||||
|
||||
# Minimum number of public methods for a class (see R0903).
|
||||
min-public-methods=1
|
||||
|
||||
|
||||
[EXCEPTIONS]
|
||||
|
||||
# Exceptions that will emit a warning when being caught. Defaults to
|
||||
# "Exception"
|
||||
overgeneral-exceptions=Exception
|
||||
7
.readthedocs.yml
Normal file
7
.readthedocs.yml
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: Unlicense
|
||||
|
||||
python:
|
||||
version: 3
|
||||
requirements_file: requirements.txt
|
||||
137
CODE_OF_CONDUCT.md
Normal file
137
CODE_OF_CONDUCT.md
Normal file
|
|
@ -0,0 +1,137 @@
|
|||
<!--
|
||||
SPDX-FileCopyrightText: 2014 Coraline Ada Ehmke
|
||||
SPDX-FileCopyrightText: 2019 Kattni Rembor for Adafruit Industries
|
||||
|
||||
SPDX-License-Identifier: CC-BY-4.0
|
||||
-->
|
||||
# Adafruit Community Code of Conduct
|
||||
|
||||
## Our Pledge
|
||||
|
||||
In the interest of fostering an open and welcoming environment, we as
|
||||
contributors and leaders pledge to making participation in our project and
|
||||
our community a harassment-free experience for everyone, regardless of age, body
|
||||
size, disability, ethnicity, gender identity and expression, level or type of
|
||||
experience, education, socio-economic status, nationality, personal appearance,
|
||||
race, religion, or sexual identity and orientation.
|
||||
|
||||
## Our Standards
|
||||
|
||||
We are committed to providing a friendly, safe and welcoming environment for
|
||||
all.
|
||||
|
||||
Examples of behavior that contributes to creating a positive environment
|
||||
include:
|
||||
|
||||
* Be kind and courteous to others
|
||||
* Using welcoming and inclusive language
|
||||
* Being respectful of differing viewpoints and experiences
|
||||
* Collaborating with other community members
|
||||
* Gracefully accepting constructive criticism
|
||||
* Focusing on what is best for the community
|
||||
* Showing empathy towards other community members
|
||||
|
||||
Examples of unacceptable behavior by participants include:
|
||||
|
||||
* The use of sexualized language or imagery and sexual attention or advances
|
||||
* The use of inappropriate images, including in a community member's avatar
|
||||
* The use of inappropriate language, including in a community member's nickname
|
||||
* Any spamming, flaming, baiting or other attention-stealing behavior
|
||||
* Excessive or unwelcome helping; answering outside the scope of the question
|
||||
asked
|
||||
* Trolling, insulting/derogatory comments, and personal or political attacks
|
||||
* Promoting or spreading disinformation, lies, or conspiracy theories against
|
||||
a person, group, organisation, project, or community
|
||||
* Public or private harassment
|
||||
* Publishing others' private information, such as a physical or electronic
|
||||
address, without explicit permission
|
||||
* Other conduct which could reasonably be considered inappropriate
|
||||
|
||||
The goal of the standards and moderation guidelines outlined here is to build
|
||||
and maintain a respectful community. We ask that you don’t just aim to be
|
||||
"technically unimpeachable", but rather try to be your best self.
|
||||
|
||||
We value many things beyond technical expertise, including collaboration and
|
||||
supporting others within our community. Providing a positive experience for
|
||||
other community members can have a much more significant impact than simply
|
||||
providing the correct answer.
|
||||
|
||||
## Our Responsibilities
|
||||
|
||||
Project leaders are responsible for clarifying the standards of acceptable
|
||||
behavior and are expected to take appropriate and fair corrective action in
|
||||
response to any instances of unacceptable behavior.
|
||||
|
||||
Project leaders have the right and responsibility to remove, edit, or
|
||||
reject messages, comments, commits, code, issues, and other contributions
|
||||
that are not aligned to this Code of Conduct, or to ban temporarily or
|
||||
permanently any community member for other behaviors that they deem
|
||||
inappropriate, threatening, offensive, or harmful.
|
||||
|
||||
## Moderation
|
||||
|
||||
Instances of behaviors that violate the Adafruit Community Code of Conduct
|
||||
may be reported by any member of the community. Community members are
|
||||
encouraged to report these situations, including situations they witness
|
||||
involving other community members.
|
||||
|
||||
You may report in the following ways:
|
||||
|
||||
In any situation, you may send an email to <support@adafruit.com>.
|
||||
|
||||
On the Adafruit Discord, you may send an open message from any channel
|
||||
to all Community Moderators by tagging @community moderators. You may
|
||||
also send an open message from any channel, or a direct message to
|
||||
@kattni#1507, @tannewt#4653, @danh#1614, @cater#2442,
|
||||
@sommersoft#0222, @Mr. Certainly#0472 or @Andon#8175.
|
||||
|
||||
Email and direct message reports will be kept confidential.
|
||||
|
||||
In situations on Discord where the issue is particularly egregious, possibly
|
||||
illegal, requires immediate action, or violates the Discord terms of service,
|
||||
you should also report the message directly to Discord.
|
||||
|
||||
These are the steps for upholding our community’s standards of conduct.
|
||||
|
||||
1. Any member of the community may report any situation that violates the
|
||||
Adafruit Community Code of Conduct. All reports will be reviewed and
|
||||
investigated.
|
||||
2. If the behavior is an egregious violation, the community member who
|
||||
committed the violation may be banned immediately, without warning.
|
||||
3. Otherwise, moderators will first respond to such behavior with a warning.
|
||||
4. Moderators follow a soft "three strikes" policy - the community member may
|
||||
be given another chance, if they are receptive to the warning and change their
|
||||
behavior.
|
||||
5. If the community member is unreceptive or unreasonable when warned by a
|
||||
moderator, or the warning goes unheeded, they may be banned for a first or
|
||||
second offense. Repeated offenses will result in the community member being
|
||||
banned.
|
||||
|
||||
## Scope
|
||||
|
||||
This Code of Conduct and the enforcement policies listed above apply to all
|
||||
Adafruit Community venues. This includes but is not limited to any community
|
||||
spaces (both public and private), the entire Adafruit Discord server, and
|
||||
Adafruit GitHub repositories. Examples of Adafruit Community spaces include
|
||||
but are not limited to meet-ups, audio chats on the Adafruit Discord, or
|
||||
interaction at a conference.
|
||||
|
||||
This Code of Conduct applies both within project spaces and in public spaces
|
||||
when an individual is representing the project or its community. As a community
|
||||
member, you are representing our community, and are expected to behave
|
||||
accordingly.
|
||||
|
||||
## Attribution
|
||||
|
||||
This Code of Conduct is adapted from the [Contributor Covenant],
|
||||
version 1.4, available at
|
||||
<https://www.contributor-covenant.org/version/1/4/code-of-conduct.html>,
|
||||
and the [Rust Code of Conduct](https://www.rust-lang.org/en-US/conduct.html).
|
||||
|
||||
For other projects adopting the Adafruit Community Code of
|
||||
Conduct, please contact the maintainers of those projects for enforcement.
|
||||
If you wish to use this code of conduct for your own project, consider
|
||||
explicitly mentioning your moderation policy or making a copy with your
|
||||
own moderation policy so as to avoid confusion.
|
||||
|
||||
[Contributor Covenant]: https://www.contributor-covenant.org
|
||||
21
LICENSE
Normal file
21
LICENSE
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
The MIT License (MIT)
|
||||
|
||||
Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
324
LICENSES/CC-BY-4.0.txt
Normal file
324
LICENSES/CC-BY-4.0.txt
Normal file
|
|
@ -0,0 +1,324 @@
|
|||
Creative Commons Attribution 4.0 International Creative Commons Corporation
|
||||
("Creative Commons") is not a law firm and does not provide legal services
|
||||
or legal advice. Distribution of Creative Commons public licenses does not
|
||||
create a lawyer-client or other relationship. Creative Commons makes its licenses
|
||||
and related information available on an "as-is" basis. Creative Commons gives
|
||||
no warranties regarding its licenses, any material licensed under their terms
|
||||
and conditions, or any related information. Creative Commons disclaims all
|
||||
liability for damages resulting from their use to the fullest extent possible.
|
||||
|
||||
Using Creative Commons Public Licenses
|
||||
|
||||
Creative Commons public licenses provide a standard set of terms and conditions
|
||||
that creators and other rights holders may use to share original works of
|
||||
authorship and other material subject to copyright and certain other rights
|
||||
specified in the public license below. The following considerations are for
|
||||
informational purposes only, are not exhaustive, and do not form part of our
|
||||
licenses.
|
||||
|
||||
Considerations for licensors: Our public licenses are intended for use by
|
||||
those authorized to give the public permission to use material in ways otherwise
|
||||
restricted by copyright and certain other rights. Our licenses are irrevocable.
|
||||
Licensors should read and understand the terms and conditions of the license
|
||||
they choose before applying it. Licensors should also secure all rights necessary
|
||||
before applying our licenses so that the public can reuse the material as
|
||||
expected. Licensors should clearly mark any material not subject to the license.
|
||||
This includes other CC-licensed material, or material used under an exception
|
||||
or limitation to copyright. More considerations for licensors : wiki.creativecommons.org/Considerations_for_licensors
|
||||
|
||||
Considerations for the public: By using one of our public licenses, a licensor
|
||||
grants the public permission to use the licensed material under specified
|
||||
terms and conditions. If the licensor's permission is not necessary for any
|
||||
reason–for example, because of any applicable exception or limitation to copyright–then
|
||||
that use is not regulated by the license. Our licenses grant only permissions
|
||||
under copyright and certain other rights that a licensor has authority to
|
||||
grant. Use of the licensed material may still be restricted for other reasons,
|
||||
including because others have copyright or other rights in the material. A
|
||||
licensor may make special requests, such as asking that all changes be marked
|
||||
or described. Although not required by our licenses, you are encouraged to
|
||||
respect those requests where reasonable. More considerations for the public
|
||||
: wiki.creativecommons.org/Considerations_for_licensees Creative Commons Attribution
|
||||
4.0 International Public License
|
||||
|
||||
By exercising the Licensed Rights (defined below), You accept and agree to
|
||||
be bound by the terms and conditions of this Creative Commons Attribution
|
||||
4.0 International Public License ("Public License"). To the extent this Public
|
||||
License may be interpreted as a contract, You are granted the Licensed Rights
|
||||
in consideration of Your acceptance of these terms and conditions, and the
|
||||
Licensor grants You such rights in consideration of benefits the Licensor
|
||||
receives from making the Licensed Material available under these terms and
|
||||
conditions.
|
||||
|
||||
Section 1 – Definitions.
|
||||
|
||||
a. Adapted Material means material subject to Copyright and Similar Rights
|
||||
that is derived from or based upon the Licensed Material and in which the
|
||||
Licensed Material is translated, altered, arranged, transformed, or otherwise
|
||||
modified in a manner requiring permission under the Copyright and Similar
|
||||
Rights held by the Licensor. For purposes of this Public License, where the
|
||||
Licensed Material is a musical work, performance, or sound recording, Adapted
|
||||
Material is always produced where the Licensed Material is synched in timed
|
||||
relation with a moving image.
|
||||
|
||||
b. Adapter's License means the license You apply to Your Copyright and Similar
|
||||
Rights in Your contributions to Adapted Material in accordance with the terms
|
||||
and conditions of this Public License.
|
||||
|
||||
c. Copyright and Similar Rights means copyright and/or similar rights closely
|
||||
related to copyright including, without limitation, performance, broadcast,
|
||||
sound recording, and Sui Generis Database Rights, without regard to how the
|
||||
rights are labeled or categorized. For purposes of this Public License, the
|
||||
rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights.
|
||||
|
||||
d. Effective Technological Measures means those measures that, in the absence
|
||||
of proper authority, may not be circumvented under laws fulfilling obligations
|
||||
under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996,
|
||||
and/or similar international agreements.
|
||||
|
||||
e. Exceptions and Limitations means fair use, fair dealing, and/or any other
|
||||
exception or limitation to Copyright and Similar Rights that applies to Your
|
||||
use of the Licensed Material.
|
||||
|
||||
f. Licensed Material means the artistic or literary work, database, or other
|
||||
material to which the Licensor applied this Public License.
|
||||
|
||||
g. Licensed Rights means the rights granted to You subject to the terms and
|
||||
conditions of this Public License, which are limited to all Copyright and
|
||||
Similar Rights that apply to Your use of the Licensed Material and that the
|
||||
Licensor has authority to license.
|
||||
|
||||
h. Licensor means the individual(s) or entity(ies) granting rights under this
|
||||
Public License.
|
||||
|
||||
i. Share means to provide material to the public by any means or process that
|
||||
requires permission under the Licensed Rights, such as reproduction, public
|
||||
display, public performance, distribution, dissemination, communication, or
|
||||
importation, and to make material available to the public including in ways
|
||||
that members of the public may access the material from a place and at a time
|
||||
individually chosen by them.
|
||||
|
||||
j. Sui Generis Database Rights means rights other than copyright resulting
|
||||
from Directive 96/9/EC of the European Parliament and of the Council of 11
|
||||
March 1996 on the legal protection of databases, as amended and/or succeeded,
|
||||
as well as other essentially equivalent rights anywhere in the world.
|
||||
|
||||
k. You means the individual or entity exercising the Licensed Rights under
|
||||
this Public License. Your has a corresponding meaning.
|
||||
|
||||
Section 2 – Scope.
|
||||
|
||||
a. License grant.
|
||||
|
||||
1. Subject to the terms and conditions of this Public License, the Licensor
|
||||
hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive,
|
||||
irrevocable license to exercise the Licensed Rights in the Licensed Material
|
||||
to:
|
||||
|
||||
A. reproduce and Share the Licensed Material, in whole or in part; and
|
||||
|
||||
B. produce, reproduce, and Share Adapted Material.
|
||||
|
||||
2. Exceptions and Limitations. For the avoidance of doubt, where Exceptions
|
||||
and Limitations apply to Your use, this Public License does not apply, and
|
||||
You do not need to comply with its terms and conditions.
|
||||
|
||||
3. Term. The term of this Public License is specified in Section 6(a).
|
||||
|
||||
4. Media and formats; technical modifications allowed. The Licensor authorizes
|
||||
You to exercise the Licensed Rights in all media and formats whether now known
|
||||
or hereafter created, and to make technical modifications necessary to do
|
||||
so. The Licensor waives and/or agrees not to assert any right or authority
|
||||
to forbid You from making technical modifications necessary to exercise the
|
||||
Licensed Rights, including technical modifications necessary to circumvent
|
||||
Effective Technological Measures. For purposes of this Public License, simply
|
||||
making modifications authorized by this Section 2(a)(4) never produces Adapted
|
||||
Material.
|
||||
|
||||
5. Downstream recipients.
|
||||
|
||||
A. Offer from the Licensor – Licensed Material. Every recipient of the Licensed
|
||||
Material automatically receives an offer from the Licensor to exercise the
|
||||
Licensed Rights under the terms and conditions of this Public License.
|
||||
|
||||
B. No downstream restrictions. You may not offer or impose any additional
|
||||
or different terms or conditions on, or apply any Effective Technological
|
||||
Measures to, the Licensed Material if doing so restricts exercise of the Licensed
|
||||
Rights by any recipient of the Licensed Material.
|
||||
|
||||
6. No endorsement. Nothing in this Public License constitutes or may be construed
|
||||
as permission to assert or imply that You are, or that Your use of the Licensed
|
||||
Material is, connected with, or sponsored, endorsed, or granted official status
|
||||
by, the Licensor or others designated to receive attribution as provided in
|
||||
Section 3(a)(1)(A)(i).
|
||||
|
||||
b. Other rights.
|
||||
|
||||
1. Moral rights, such as the right of integrity, are not licensed under this
|
||||
Public License, nor are publicity, privacy, and/or other similar personality
|
||||
rights; however, to the extent possible, the Licensor waives and/or agrees
|
||||
not to assert any such rights held by the Licensor to the limited extent necessary
|
||||
to allow You to exercise the Licensed Rights, but not otherwise.
|
||||
|
||||
2. Patent and trademark rights are not licensed under this Public License.
|
||||
|
||||
3. To the extent possible, the Licensor waives any right to collect royalties
|
||||
from You for the exercise of the Licensed Rights, whether directly or through
|
||||
a collecting society under any voluntary or waivable statutory or compulsory
|
||||
licensing scheme. In all other cases the Licensor expressly reserves any right
|
||||
to collect such royalties.
|
||||
|
||||
Section 3 – License Conditions.
|
||||
|
||||
Your exercise of the Licensed Rights is expressly made subject to the following
|
||||
conditions.
|
||||
|
||||
a. Attribution.
|
||||
|
||||
1. If You Share the Licensed Material (including in modified form), You must:
|
||||
|
||||
A. retain the following if it is supplied by the Licensor with the Licensed
|
||||
Material:
|
||||
|
||||
i. identification of the creator(s) of the Licensed Material and any others
|
||||
designated to receive attribution, in any reasonable manner requested by the
|
||||
Licensor (including by pseudonym if designated);
|
||||
|
||||
ii. a copyright notice;
|
||||
|
||||
iii. a notice that refers to this Public License;
|
||||
|
||||
iv. a notice that refers to the disclaimer of warranties;
|
||||
|
||||
v. a URI or hyperlink to the Licensed Material to the extent reasonably practicable;
|
||||
|
||||
B. indicate if You modified the Licensed Material and retain an indication
|
||||
of any previous modifications; and
|
||||
|
||||
C. indicate the Licensed Material is licensed under this Public License, and
|
||||
include the text of, or the URI or hyperlink to, this Public License.
|
||||
|
||||
2. You may satisfy the conditions in Section 3(a)(1) in any reasonable manner
|
||||
based on the medium, means, and context in which You Share the Licensed Material.
|
||||
For example, it may be reasonable to satisfy the conditions by providing a
|
||||
URI or hyperlink to a resource that includes the required information.
|
||||
|
||||
3. If requested by the Licensor, You must remove any of the information required
|
||||
by Section 3(a)(1)(A) to the extent reasonably practicable.
|
||||
|
||||
4. If You Share Adapted Material You produce, the Adapter's License You apply
|
||||
must not prevent recipients of the Adapted Material from complying with this
|
||||
Public License.
|
||||
|
||||
Section 4 – Sui Generis Database Rights.
|
||||
|
||||
Where the Licensed Rights include Sui Generis Database Rights that apply to
|
||||
Your use of the Licensed Material:
|
||||
|
||||
a. for the avoidance of doubt, Section 2(a)(1) grants You the right to extract,
|
||||
reuse, reproduce, and Share all or a substantial portion of the contents of
|
||||
the database;
|
||||
|
||||
b. if You include all or a substantial portion of the database contents in
|
||||
a database in which You have Sui Generis Database Rights, then the database
|
||||
in which You have Sui Generis Database Rights (but not its individual contents)
|
||||
is Adapted Material; and
|
||||
|
||||
c. You must comply with the conditions in Section 3(a) if You Share all or
|
||||
a substantial portion of the contents of the database.
|
||||
|
||||
For the avoidance of doubt, this Section 4 supplements and does not replace
|
||||
Your obligations under this Public License where the Licensed Rights include
|
||||
other Copyright and Similar Rights.
|
||||
|
||||
Section 5 – Disclaimer of Warranties and Limitation of Liability.
|
||||
|
||||
a. Unless otherwise separately undertaken by the Licensor, to the extent possible,
|
||||
the Licensor offers the Licensed Material as-is and as-available, and makes
|
||||
no representations or warranties of any kind concerning the Licensed Material,
|
||||
whether express, implied, statutory, or other. This includes, without limitation,
|
||||
warranties of title, merchantability, fitness for a particular purpose, non-infringement,
|
||||
absence of latent or other defects, accuracy, or the presence or absence of
|
||||
errors, whether or not known or discoverable. Where disclaimers of warranties
|
||||
are not allowed in full or in part, this disclaimer may not apply to You.
|
||||
|
||||
b. To the extent possible, in no event will the Licensor be liable to You
|
||||
on any legal theory (including, without limitation, negligence) or otherwise
|
||||
for any direct, special, indirect, incidental, consequential, punitive, exemplary,
|
||||
or other losses, costs, expenses, or damages arising out of this Public License
|
||||
or use of the Licensed Material, even if the Licensor has been advised of
|
||||
the possibility of such losses, costs, expenses, or damages. Where a limitation
|
||||
of liability is not allowed in full or in part, this limitation may not apply
|
||||
to You.
|
||||
|
||||
c. The disclaimer of warranties and limitation of liability provided above
|
||||
shall be interpreted in a manner that, to the extent possible, most closely
|
||||
approximates an absolute disclaimer and waiver of all liability.
|
||||
|
||||
Section 6 – Term and Termination.
|
||||
|
||||
a. This Public License applies for the term of the Copyright and Similar Rights
|
||||
licensed here. However, if You fail to comply with this Public License, then
|
||||
Your rights under this Public License terminate automatically.
|
||||
|
||||
b. Where Your right to use the Licensed Material has terminated under Section
|
||||
6(a), it reinstates:
|
||||
|
||||
1. automatically as of the date the violation is cured, provided it is cured
|
||||
within 30 days of Your discovery of the violation; or
|
||||
|
||||
2. upon express reinstatement by the Licensor.
|
||||
|
||||
c. For the avoidance of doubt, this Section 6(b) does not affect any right
|
||||
the Licensor may have to seek remedies for Your violations of this Public
|
||||
License.
|
||||
|
||||
d. For the avoidance of doubt, the Licensor may also offer the Licensed Material
|
||||
under separate terms or conditions or stop distributing the Licensed Material
|
||||
at any time; however, doing so will not terminate this Public License.
|
||||
|
||||
e. Sections 1, 5, 6, 7, and 8 survive termination of this Public License.
|
||||
|
||||
Section 7 – Other Terms and Conditions.
|
||||
|
||||
a. The Licensor shall not be bound by any additional or different terms or
|
||||
conditions communicated by You unless expressly agreed.
|
||||
|
||||
b. Any arrangements, understandings, or agreements regarding the Licensed
|
||||
Material not stated herein are separate from and independent of the terms
|
||||
and conditions of this Public License.
|
||||
|
||||
Section 8 – Interpretation.
|
||||
|
||||
a. For the avoidance of doubt, this Public License does not, and shall not
|
||||
be interpreted to, reduce, limit, restrict, or impose conditions on any use
|
||||
of the Licensed Material that could lawfully be made without permission under
|
||||
this Public License.
|
||||
|
||||
b. To the extent possible, if any provision of this Public License is deemed
|
||||
unenforceable, it shall be automatically reformed to the minimum extent necessary
|
||||
to make it enforceable. If the provision cannot be reformed, it shall be severed
|
||||
from this Public License without affecting the enforceability of the remaining
|
||||
terms and conditions.
|
||||
|
||||
c. No term or condition of this Public License will be waived and no failure
|
||||
to comply consented to unless expressly agreed to by the Licensor.
|
||||
|
||||
d. Nothing in this Public License constitutes or may be interpreted as a limitation
|
||||
upon, or waiver of, any privileges and immunities that apply to the Licensor
|
||||
or You, including from the legal processes of any jurisdiction or authority.
|
||||
|
||||
Creative Commons is not a party to its public licenses. Notwithstanding, Creative
|
||||
Commons may elect to apply one of its public licenses to material it publishes
|
||||
and in those instances will be considered the "Licensor." The text of the
|
||||
Creative Commons public licenses is dedicated to the public domain under the
|
||||
CC0 Public Domain Dedication. Except for the limited purpose of indicating
|
||||
that material is shared under a Creative Commons public license or as otherwise
|
||||
permitted by the Creative Commons policies published at creativecommons.org/policies,
|
||||
Creative Commons does not authorize the use of the trademark "Creative Commons"
|
||||
or any other trademark or logo of Creative Commons without its prior written
|
||||
consent including, without limitation, in connection with any unauthorized
|
||||
modifications to any of its public licenses or any other arrangements, understandings,
|
||||
or agreements concerning use of licensed material. For the avoidance of doubt,
|
||||
this paragraph does not form part of the public licenses.
|
||||
|
||||
Creative Commons may be contacted at creativecommons.org.
|
||||
19
LICENSES/MIT.txt
Normal file
19
LICENSES/MIT.txt
Normal file
|
|
@ -0,0 +1,19 @@
|
|||
MIT License Copyright (c) <year> <copyright holders>
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is furnished
|
||||
to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice (including the next
|
||||
paragraph) shall be included in all copies or substantial portions of the
|
||||
Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS
|
||||
OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
|
||||
OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
20
LICENSES/Unlicense.txt
Normal file
20
LICENSES/Unlicense.txt
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
This is free and unencumbered software released into the public domain.
|
||||
|
||||
Anyone is free to copy, modify, publish, use, compile, sell, or distribute
|
||||
this software, either in source code form or as a compiled binary, for any
|
||||
purpose, commercial or non-commercial, and by any means.
|
||||
|
||||
In jurisdictions that recognize copyright laws, the author or authors of this
|
||||
software dedicate any and all copyright interest in the software to the public
|
||||
domain. We make this dedication for the benefit of the public at large and
|
||||
to the detriment of our heirs and successors. We intend this dedication to
|
||||
be an overt act of relinquishment in perpetuity of all present and future
|
||||
rights to this software under copyright law.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS
|
||||
BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH
|
||||
THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. For more information,
|
||||
please refer to <https://unlicense.org/>
|
||||
|
|
@ -1,2 +0,0 @@
|
|||
# Adafruit_CircuitPython_PortalBase
|
||||
Base Library for the Portal-style libraries.
|
||||
71
README.rst
Normal file
71
README.rst
Normal file
|
|
@ -0,0 +1,71 @@
|
|||
Introduction
|
||||
============
|
||||
|
||||
.. image:: https://readthedocs.org/projects/adafruit-circuitpython-portalbase/badge/?version=latest
|
||||
:target: https://circuitpython.readthedocs.io/projects/portalbase/en/latest/
|
||||
:alt: Documentation Status
|
||||
|
||||
.. image:: https://img.shields.io/discord/327254708534116352.svg
|
||||
:target: https://adafru.it/discord
|
||||
:alt: Discord
|
||||
|
||||
.. image:: https://github.com/adafruit/Adafruit_CircuitPython_PortalBase/workflows/Build%20CI/badge.svg
|
||||
:target: https://github.com/adafruit/Adafruit_CircuitPython_PortalBase/actions
|
||||
:alt: Build Status
|
||||
|
||||
.. image:: https://img.shields.io/badge/code%20style-black-000000.svg
|
||||
:target: https://github.com/psf/black
|
||||
:alt: Code Style: Black
|
||||
|
||||
Base Library for the Portal-style libraries. This library only contains base classes and is not
|
||||
intended to be run on its own.
|
||||
|
||||
|
||||
Dependencies
|
||||
=============
|
||||
This driver depends on:
|
||||
|
||||
* `Adafruit CircuitPython <https://github.com/adafruit/circuitpython>`_
|
||||
|
||||
Please ensure all dependencies are available on the CircuitPython filesystem.
|
||||
This is easily achieved by downloading
|
||||
`the Adafruit library and driver bundle <https://circuitpython.org/libraries>`_.
|
||||
|
||||
Installing from PyPI
|
||||
=====================
|
||||
.. todo:: Remove the above note if PyPI version is/will be available at time of release.
|
||||
If the library is not planned for PyPI, remove the entire 'Installing from PyPI' section.
|
||||
|
||||
On supported GNU/Linux systems like the Raspberry Pi, you can install the driver locally `from
|
||||
PyPI <https://pypi.org/project/adafruit-circuitpython-portalbase/>`_. To install for current user:
|
||||
|
||||
.. code-block:: shell
|
||||
|
||||
pip3 install adafruit-circuitpython-portalbase
|
||||
|
||||
To install system-wide (this may be required in some cases):
|
||||
|
||||
.. code-block:: shell
|
||||
|
||||
sudo pip3 install adafruit-circuitpython-portalbase
|
||||
|
||||
To install in a virtual environment in your current project:
|
||||
|
||||
.. code-block:: shell
|
||||
|
||||
mkdir project-name && cd project-name
|
||||
python3 -m venv .env
|
||||
source .env/bin/activate
|
||||
pip3 install adafruit-circuitpython-portalbase
|
||||
|
||||
Contributing
|
||||
============
|
||||
|
||||
Contributions are welcome! Please read our `Code of Conduct
|
||||
<https://github.com/adafruit/Adafruit_CircuitPython_PortalBase/blob/master/CODE_OF_CONDUCT.md>`_
|
||||
before contributing to help this project stay welcoming.
|
||||
|
||||
Documentation
|
||||
=============
|
||||
|
||||
For information on building library documentation, please check out `this guide <https://learn.adafruit.com/creating-and-sharing-a-circuitpython-library/sharing-our-docs-on-readthedocs#sphinx-5-1>`_.
|
||||
3
README.rst.license
Normal file
3
README.rst.license
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
|
||||
SPDX-License-Identifier: MIT
|
||||
460
adafruit_portalbase/__init__.py
Executable file
460
adafruit_portalbase/__init__.py
Executable file
|
|
@ -0,0 +1,460 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
# SPDX-FileCopyrightText: Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
"""
|
||||
`adafruit_portalbase`
|
||||
================================================================================
|
||||
|
||||
Base Library for the Portal-style libraries.
|
||||
|
||||
|
||||
* Author(s): Melissa LeBlanc-Williams
|
||||
|
||||
Implementation Notes
|
||||
--------------------
|
||||
|
||||
**Software and Dependencies:**
|
||||
|
||||
* Adafruit CircuitPython firmware for the supported boards:
|
||||
https://github.com/adafruit/circuitpython/releases
|
||||
|
||||
"""
|
||||
import gc
|
||||
import time
|
||||
import terminalio
|
||||
from adafruit_bitmap_font import bitmap_font
|
||||
from adafruit_display_text.label import Label
|
||||
from adafruit_display_text import wrap_text_to_lines
|
||||
|
||||
__version__ = "0.0.0-auto.0"
|
||||
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_PortalBase.git"
|
||||
|
||||
|
||||
class PortalBase:
|
||||
"""Class representing the Adafruit MagTag.
|
||||
|
||||
:param network: An initialized network class instance.
|
||||
:param graphics: An initialized graphics class instance.
|
||||
:param url: The URL of your data source. Defaults to ``None``.
|
||||
:param headers: The headers for authentication, typically used by Azure API's.
|
||||
:param json_path: The list of json traversal to get data out of. Can be list of lists for
|
||||
multiple data points. Defaults to ``None`` to not use json.
|
||||
:param regexp_path: The list of regexp strings to get data out (use a single regexp group).
|
||||
Can be list of regexps for multiple data points. Defaults to ``None``
|
||||
to not use regexp.
|
||||
:param default_bg: The path to your default background image file or a hex color.
|
||||
Defaults to 0x000000.
|
||||
:param status_neopixel: The pin for the status NeoPixel. Use ``board.NEOPIXEL`` for the
|
||||
on-board NeoPixel. Defaults to ``None``, to not use the status LED
|
||||
:param json_transform: A function or a list of functions to call with the parsed JSON.
|
||||
Changes and additions are permitted for the ``dict`` object.
|
||||
:param debug: Turn on debug print outs. Defaults to False.
|
||||
|
||||
"""
|
||||
|
||||
# pylint: disable=too-many-instance-attributes, too-many-branches
|
||||
def __init__(
|
||||
self,
|
||||
network,
|
||||
graphics,
|
||||
*,
|
||||
url=None,
|
||||
headers=None,
|
||||
json_path=None,
|
||||
regexp_path=None,
|
||||
json_transform=None,
|
||||
debug=False,
|
||||
):
|
||||
|
||||
self.network = network
|
||||
self.graphics = graphics
|
||||
self.splash = self.graphics.splash
|
||||
self.display = self.graphics.display
|
||||
|
||||
# Font Cache
|
||||
self._fonts = {}
|
||||
self._text = []
|
||||
|
||||
try:
|
||||
import alarm # pylint: disable=import-outside-toplevel
|
||||
|
||||
self._alarm = alarm
|
||||
except ImportError:
|
||||
self._alarm = None
|
||||
|
||||
self._debug = debug
|
||||
self._url = None
|
||||
self.url = url
|
||||
self._headers = headers
|
||||
self._json_path = None
|
||||
self.json_path = json_path
|
||||
|
||||
self._regexp_path = regexp_path
|
||||
|
||||
# Add any JSON translators
|
||||
if json_transform:
|
||||
self.network.add_json_transform(json_transform)
|
||||
|
||||
def _load_font(self, font):
|
||||
"""
|
||||
Load and cache a font if not previously loaded
|
||||
Return the key of the cached font
|
||||
|
||||
:param font: Either terminalio.FONT or the path to the bdf font file
|
||||
|
||||
"""
|
||||
if font is terminalio.FONT:
|
||||
if "terminal" not in self._fonts:
|
||||
self._fonts["terminal"] = terminalio.FONT
|
||||
return "terminal"
|
||||
if font not in self._fonts:
|
||||
self._fonts[font] = bitmap_font.load_font(font)
|
||||
return font
|
||||
|
||||
@staticmethod
|
||||
def html_color_convert(color):
|
||||
"""Convert an HTML color code to an integer
|
||||
|
||||
:param color: The color value to be converted
|
||||
|
||||
"""
|
||||
if isinstance(color, str):
|
||||
if color[0] == "#":
|
||||
color = color.lstrip("#")
|
||||
return int(color, 16)
|
||||
return color # Return unconverted
|
||||
|
||||
@staticmethod
|
||||
def wrap_nicely(string, max_chars):
|
||||
"""A helper that will return a list of lines with word-break wrapping.
|
||||
|
||||
:param str string: The text to be wrapped.
|
||||
:param int max_chars: The maximum number of characters on a line before wrapping.
|
||||
|
||||
"""
|
||||
return wrap_text_to_lines(string, max_chars)
|
||||
|
||||
# pylint: disable=too-many-arguments
|
||||
def add_text(
|
||||
self,
|
||||
text_position=(0, 0),
|
||||
text_font=terminalio.FONT,
|
||||
text_color=0x000000,
|
||||
text_wrap=0,
|
||||
text_maxlen=0,
|
||||
text_transform=None,
|
||||
text_scale=1,
|
||||
line_spacing=1.25,
|
||||
text_anchor_point=(0, 0.5),
|
||||
is_data=True,
|
||||
):
|
||||
"""
|
||||
Add text labels with settings
|
||||
|
||||
:param str text_font: The path to your font file for your data text display.
|
||||
:param text_position: The position of your extracted text on the display in an (x, y) tuple.
|
||||
Can be a list of tuples for when there's a list of json_paths, for
|
||||
example.
|
||||
:param text_color: The color of the text, in 0xRRGGBB format. Can be a list of colors for
|
||||
when there's multiple texts. Defaults to ``None``.
|
||||
:param text_wrap: When non-zero, the maximum number of characters on each line before text
|
||||
is wrapped. (for long text data chunks). Defaults to 0, no wrapping.
|
||||
:param text_maxlen: The max length of the text. If non-zero, it will be truncated to this
|
||||
length. Defaults to 0.
|
||||
:param text_transform: A function that will be called on the text before display
|
||||
:param int text_scale: The factor to scale the default size of the text by
|
||||
:param float line_spacing: The factor to space the lines apart
|
||||
:param (float, float) text_anchor_point: Values between 0 and 1 to indicate where the text
|
||||
position is relative to the label
|
||||
:param bool is_data: If True, fetch will attempt to update the label
|
||||
"""
|
||||
if not text_wrap:
|
||||
text_wrap = 0
|
||||
if not text_maxlen:
|
||||
text_maxlen = 0
|
||||
if not text_transform:
|
||||
text_transform = None
|
||||
if not isinstance(text_scale, (int, float)) or text_scale < 1:
|
||||
text_scale = 1
|
||||
if not isinstance(text_anchor_point, (tuple, list)):
|
||||
text_anchor_point = (0, 0.5)
|
||||
if not 0 <= text_anchor_point[0] <= 1 or not 0 <= text_anchor_point[1] <= 1:
|
||||
raise ValueError("Text anchor point values should be between 0 and 1.")
|
||||
text_scale = round(text_scale)
|
||||
gc.collect()
|
||||
|
||||
if self._debug:
|
||||
print("Init text area")
|
||||
text_field = {
|
||||
"label": None,
|
||||
"font": self._load_font(text_font),
|
||||
"color": self.html_color_convert(text_color),
|
||||
"position": text_position,
|
||||
"wrap": text_wrap,
|
||||
"maxlen": text_maxlen,
|
||||
"transform": text_transform,
|
||||
"scale": text_scale,
|
||||
"line_spacing": line_spacing,
|
||||
"anchor_point": text_anchor_point,
|
||||
"is_data": bool(is_data),
|
||||
}
|
||||
self._text.append(text_field)
|
||||
|
||||
return len(self._text) - 1
|
||||
|
||||
# pylint: enable=too-many-arguments
|
||||
|
||||
def set_text(self, val, index=0):
|
||||
"""Display text, with indexing into our list of text boxes.
|
||||
|
||||
:param str val: The text to be displayed
|
||||
:param index: Defaults to 0.
|
||||
|
||||
"""
|
||||
# Make sure at least a single label exists
|
||||
if not self._text:
|
||||
self.add_text()
|
||||
string = str(val)
|
||||
if self._text[index]["maxlen"]:
|
||||
if len(string) >= 3:
|
||||
# too long! shorten it
|
||||
string = string[: self._text[index]["maxlen"] - 3] + "..."
|
||||
else:
|
||||
string = string[: self._text[index]["maxlen"]]
|
||||
index_in_splash = None
|
||||
|
||||
if len(string) > 0 and self._text[index]["wrap"]:
|
||||
if self._debug:
|
||||
print("Wrapping text with length of", self._text[index]["wrap"])
|
||||
lines = self.wrap_nicely(string, self._text[index]["wrap"])
|
||||
string = "\n".join(lines)
|
||||
|
||||
if self._text[index]["label"] is not None:
|
||||
if self._debug:
|
||||
print("Replacing text area with :", string)
|
||||
index_in_splash = self.splash.index(self._text[index]["label"])
|
||||
elif self._debug:
|
||||
print("Creating text area with :", string)
|
||||
|
||||
if len(string) > 0:
|
||||
self._text[index]["label"] = Label(
|
||||
self._fonts[self._text[index]["font"]],
|
||||
text=string,
|
||||
scale=self._text[index]["scale"],
|
||||
)
|
||||
self._text[index]["label"].color = self._text[index]["color"]
|
||||
self._text[index]["label"].anchor_point = self._text[index]["anchor_point"]
|
||||
self._text[index]["label"].anchored_position = self._text[index]["position"]
|
||||
self._text[index]["label"].line_spacing = self._text[index]["line_spacing"]
|
||||
elif index_in_splash is not None:
|
||||
self._text[index]["label"] = None
|
||||
|
||||
if index_in_splash is not None:
|
||||
if self._text[index]["label"] is not None:
|
||||
self.splash[index_in_splash] = self._text[index]["label"]
|
||||
else:
|
||||
del self.splash[index_in_splash]
|
||||
elif self._text[index]["label"] is not None:
|
||||
self.splash.append(self._text[index]["label"])
|
||||
|
||||
def preload_font(self, glyphs=None, index=0):
|
||||
# pylint: disable=line-too-long
|
||||
"""Preload font.
|
||||
|
||||
:param glyphs: The font glyphs to load. Defaults to ``None``, uses alphanumeric glyphs if
|
||||
None.
|
||||
"""
|
||||
# pylint: enable=line-too-long
|
||||
if not glyphs:
|
||||
glyphs = b"0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ-!,. \"'?!"
|
||||
print("Preloading font glyphs:", glyphs)
|
||||
if self._fonts[self._text[index]["font"]] is not terminalio.FONT:
|
||||
self._fonts[self._text[index]["font"]].load_glyphs(glyphs)
|
||||
|
||||
def set_headers(self, headers):
|
||||
"""Set the headers used by fetch().
|
||||
|
||||
:param headers: The new header dictionary
|
||||
|
||||
"""
|
||||
self._headers = headers
|
||||
|
||||
def set_background(self, file_or_color, position=None):
|
||||
"""The background image to a bitmap file.
|
||||
|
||||
:param file_or_color: The filename of the chosen background image, or a hex color.
|
||||
|
||||
"""
|
||||
self.graphics.set_background(file_or_color, position)
|
||||
|
||||
def set_text_color(self, color, index=0):
|
||||
"""Update the text color, with indexing into our list of text boxes.
|
||||
|
||||
:param int color: The color value to be used
|
||||
:param index: Defaults to 0.
|
||||
|
||||
"""
|
||||
if self._text[index]:
|
||||
color = self.html_color_convert(color)
|
||||
self._text[index]["color"] = color
|
||||
self._text[index]["label"].color = color
|
||||
|
||||
def exit_and_deep_sleep(self, sleep_time):
|
||||
"""
|
||||
Stops the current program and enters deep sleep. The program is restarted from the beginning
|
||||
after a certain period of time.
|
||||
|
||||
See https://circuitpython.readthedocs.io/en/latest/shared-bindings/alarm/index.html for more
|
||||
details.
|
||||
|
||||
:param float sleep_time: The amount of time to sleep in seconds
|
||||
|
||||
"""
|
||||
if self._alarm:
|
||||
pause = self._alarm.time.TimeAlarm(
|
||||
monotonic_time=time.monotonic() + sleep_time
|
||||
)
|
||||
self._alarm.exit_and_deep_sleep_until_alarms(pause)
|
||||
else:
|
||||
raise NotImplementedError(
|
||||
"Deep sleep not supported. Make sure you have the latest CircuitPython."
|
||||
)
|
||||
|
||||
def enter_light_sleep(self, sleep_time):
|
||||
"""
|
||||
Enter light sleep and resume the program after a certain period of time.
|
||||
|
||||
See https://circuitpython.readthedocs.io/en/latest/shared-bindings/alarm/index.html for more
|
||||
details.
|
||||
|
||||
:param float sleep_time: The amount of time to sleep in seconds
|
||||
|
||||
"""
|
||||
if self._alarm:
|
||||
pause = self._alarm.time.TimeAlarm(
|
||||
monotonic_time=time.monotonic() + sleep_time
|
||||
)
|
||||
self._alarm.light_sleep_until_alarms(pause)
|
||||
else:
|
||||
raise NotImplementedError(
|
||||
"Hardware light sleep not supported. Make sure you have the latest CircuitPython."
|
||||
)
|
||||
|
||||
def _fetch_set_text(self, val, index=0):
|
||||
self.set_text(val, index=index)
|
||||
|
||||
def fetch(self, refresh_url=None, timeout=10):
|
||||
"""Fetch data from the url we initialized with, perfom any parsing,
|
||||
and display text or graphics. This function does pretty much everything
|
||||
Optionally update the URL
|
||||
|
||||
:param str refresh_url: The overriding URL to fetch from. Defaults to ``None``.
|
||||
:param int timeout: The timeout period in seconds.
|
||||
|
||||
"""
|
||||
if refresh_url:
|
||||
self._url = refresh_url
|
||||
values = []
|
||||
|
||||
values = self.network.fetch_data(
|
||||
self._url,
|
||||
headers=self._headers,
|
||||
json_path=self._json_path,
|
||||
regexp_path=self._regexp_path,
|
||||
timeout=timeout,
|
||||
)
|
||||
|
||||
# fill out all the text blocks
|
||||
if self._text:
|
||||
value_index = 0 # In case values and text is not the same
|
||||
for i in range(len(self._text)):
|
||||
if not self._text[i]["is_data"]:
|
||||
continue
|
||||
string = None
|
||||
if self._text[i]["transform"]:
|
||||
func = self._text[i]["transform"]
|
||||
string = func(values[value_index])
|
||||
else:
|
||||
try:
|
||||
string = "{:,d}".format(int(values[value_index]))
|
||||
except (TypeError, ValueError):
|
||||
string = values[value_index] # ok it's a string
|
||||
self._fetch_set_text(string, index=i)
|
||||
value_index += 1
|
||||
if len(values) == 1:
|
||||
return values[0]
|
||||
return values
|
||||
|
||||
def get_local_time(self, location=None):
|
||||
"""Accessor function for get_local_time()"""
|
||||
return self.network.get_local_time(location=location)
|
||||
|
||||
def push_to_io(self, feed_key, data):
|
||||
"""Push data to an adafruit.io feed
|
||||
|
||||
:param str feed_key: Name of feed key to push data to.
|
||||
:param data: data to send to feed
|
||||
|
||||
"""
|
||||
|
||||
self.network.push_to_io(feed_key, data)
|
||||
|
||||
def get_io_data(self, feed_key):
|
||||
"""Return all values from the Adafruit IO Feed Data that matches the feed key
|
||||
|
||||
:param str feed_key: Name of feed key to receive data from.
|
||||
|
||||
"""
|
||||
|
||||
return self.network.get_io_data(feed_key)
|
||||
|
||||
def get_io_feed(self, feed_key, detailed=False):
|
||||
"""Return the Adafruit IO Feed that matches the feed key
|
||||
|
||||
:param str feed_key: Name of feed key to match.
|
||||
:param bool detailed: Whether to return additional detailed information
|
||||
|
||||
"""
|
||||
return self.network.get_io_feed(feed_key, detailed)
|
||||
|
||||
def get_io_group(self, group_key):
|
||||
"""Return the Adafruit IO Group that matches the group key
|
||||
|
||||
:param str group_key: Name of group key to match.
|
||||
|
||||
"""
|
||||
return self.network.get_io_group(group_key)
|
||||
|
||||
@property
|
||||
def url(self):
|
||||
"""
|
||||
Get or set the URL of your data source.
|
||||
"""
|
||||
return self._url
|
||||
|
||||
@url.setter
|
||||
def url(self, value):
|
||||
self._url = value
|
||||
if value and not self.network.uselocal:
|
||||
self.network.connect()
|
||||
# if self._debug:
|
||||
# print("My IP address is", self.network.ip_address)
|
||||
|
||||
@property
|
||||
def json_path(self):
|
||||
"""
|
||||
Get or set the list of json traversal to get data out of. Can be list
|
||||
of lists for multiple data points.
|
||||
"""
|
||||
return self._json_path
|
||||
|
||||
@json_path.setter
|
||||
def json_path(self, value):
|
||||
if value:
|
||||
if isinstance(value[0], (list, tuple)):
|
||||
self._json_path = value
|
||||
else:
|
||||
self._json_path = (value,)
|
||||
else:
|
||||
self._json_path = None
|
||||
164
adafruit_portalbase/graphics.py
Executable file
164
adafruit_portalbase/graphics.py
Executable file
|
|
@ -0,0 +1,164 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
# SPDX-FileCopyrightText: Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
"""
|
||||
`adafruit_portalbase.graphics`
|
||||
================================================================================
|
||||
|
||||
Base Library for the Portal-style libraries.
|
||||
|
||||
|
||||
* Author(s): Melissa LeBlanc-Williams
|
||||
|
||||
Implementation Notes
|
||||
--------------------
|
||||
|
||||
**Software and Dependencies:**
|
||||
|
||||
* Adafruit CircuitPython firmware for the supported boards:
|
||||
https://github.com/adafruit/circuitpython/releases
|
||||
|
||||
"""
|
||||
|
||||
import gc
|
||||
import displayio
|
||||
|
||||
__version__ = "0.0.0-auto.0"
|
||||
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_PortalBase.git"
|
||||
|
||||
|
||||
class GraphicsBase:
|
||||
"""Graphics Base Class for the Portal-style libraries.
|
||||
|
||||
:param display: An initialized display.
|
||||
:param default_bg: The path to your default background image file or a hex color.
|
||||
Defaults to 0x000000.
|
||||
:param bool auto_refresh: Automatically refresh the eInk after writing to displayio.
|
||||
Defaults to True.
|
||||
:param debug: Turn on debug print outs. Defaults to False.
|
||||
|
||||
"""
|
||||
|
||||
# pylint: disable=too-many-instance-attributes, too-many-locals, too-many-branches, too-many-statements
|
||||
def __init__(self, display, *, default_bg=0x000000, debug=False):
|
||||
|
||||
self._debug = debug
|
||||
self.display = display
|
||||
|
||||
if self._debug:
|
||||
print("Init display")
|
||||
self.splash = displayio.Group(max_size=15)
|
||||
self._qr_group = None
|
||||
if self._debug:
|
||||
print("Init background")
|
||||
self._bg_group = displayio.Group(max_size=1)
|
||||
self._bg_file = None
|
||||
self.splash.append(self._bg_group)
|
||||
self.display.show(self.splash)
|
||||
|
||||
# set the default background
|
||||
if default_bg is not None:
|
||||
self.set_background(default_bg)
|
||||
|
||||
gc.collect()
|
||||
|
||||
def set_background(self, file_or_color, position=None):
|
||||
"""The background image to a bitmap file.
|
||||
|
||||
:param file_or_color: The filename of the chosen background image, or a hex color.
|
||||
:param tuple position: Optional x and y coordinates to place the background at.
|
||||
|
||||
"""
|
||||
while self._bg_group:
|
||||
self._bg_group.pop()
|
||||
|
||||
if not position:
|
||||
position = (0, 0) # default in top corner
|
||||
|
||||
if not file_or_color:
|
||||
return # we're done, no background desired
|
||||
if self._bg_file:
|
||||
self._bg_file.close()
|
||||
if isinstance(file_or_color, str): # its a filenme:
|
||||
self._bg_file = open(file_or_color, "rb")
|
||||
background = displayio.OnDiskBitmap(self._bg_file)
|
||||
self._bg_sprite = displayio.TileGrid(
|
||||
background,
|
||||
pixel_shader=displayio.ColorConverter(),
|
||||
x=position[0],
|
||||
y=position[1],
|
||||
)
|
||||
elif isinstance(file_or_color, int):
|
||||
# Make a background color fill
|
||||
color_bitmap = displayio.Bitmap(self.display.width, self.display.height, 1)
|
||||
color_palette = displayio.Palette(1)
|
||||
color_palette[0] = file_or_color
|
||||
self._bg_sprite = displayio.TileGrid(
|
||||
color_bitmap,
|
||||
pixel_shader=color_palette,
|
||||
x=position[0],
|
||||
y=position[1],
|
||||
)
|
||||
else:
|
||||
raise RuntimeError("Unknown type of background")
|
||||
self._bg_group.append(self._bg_sprite)
|
||||
gc.collect()
|
||||
|
||||
def qrcode(
|
||||
self, qr_data, *, qr_size=1, x=0, y=0, qr_color=0x000000
|
||||
): # pylint: disable=invalid-name
|
||||
"""Display a QR code on the eInk
|
||||
|
||||
:param qr_data: The data for the QR code.
|
||||
:param int qr_size: The scale of the QR code.
|
||||
:param x: The x position of upper left corner of the QR code on the display.
|
||||
:param y: The y position of upper left corner of the QR code on the display.
|
||||
|
||||
"""
|
||||
import adafruit_miniqr # pylint: disable=import-outside-toplevel
|
||||
|
||||
# generate the QR code
|
||||
for qrtype in range(1, 5):
|
||||
try:
|
||||
qrcode = adafruit_miniqr.QRCode(qr_type=qrtype)
|
||||
qrcode.add_data(qr_data)
|
||||
qrcode.make()
|
||||
break
|
||||
except RuntimeError:
|
||||
pass
|
||||
# print("Trying with larger code")
|
||||
else:
|
||||
raise RuntimeError("Could not make QR code")
|
||||
# monochrome (2 color) palette
|
||||
palette = displayio.Palette(2)
|
||||
palette[0] = 0xFFFFFF
|
||||
palette[1] = qr_color
|
||||
|
||||
# pylint: disable=invalid-name
|
||||
# bitmap the size of the matrix, plus border, monochrome (2 colors)
|
||||
qr_bitmap = displayio.Bitmap(
|
||||
qrcode.matrix.width + 2, qrcode.matrix.height + 2, 2
|
||||
)
|
||||
for i in range(qr_bitmap.width * qr_bitmap.height):
|
||||
qr_bitmap[i] = 0
|
||||
|
||||
# transcribe QR code into bitmap
|
||||
for xx in range(qrcode.matrix.width):
|
||||
for yy in range(qrcode.matrix.height):
|
||||
qr_bitmap[xx + 1, yy + 1] = 1 if qrcode.matrix[xx, yy] else 0
|
||||
|
||||
# display the QR code
|
||||
qr_sprite = displayio.TileGrid(qr_bitmap, pixel_shader=palette)
|
||||
if self._qr_group:
|
||||
try:
|
||||
self._qr_group.pop()
|
||||
except IndexError: # later test if empty
|
||||
pass
|
||||
else:
|
||||
self._qr_group = displayio.Group()
|
||||
self.splash.append(self._qr_group)
|
||||
self._qr_group.scale = qr_size
|
||||
self._qr_group.x = x
|
||||
self._qr_group.y = y
|
||||
self._qr_group.append(qr_sprite)
|
||||
539
adafruit_portalbase/network.py
Executable file
539
adafruit_portalbase/network.py
Executable file
|
|
@ -0,0 +1,539 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
# SPDX-FileCopyrightText: Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
"""
|
||||
`adafruit_portalbase.network`
|
||||
================================================================================
|
||||
|
||||
Base Library for the Portal-style libraries.
|
||||
|
||||
|
||||
* Author(s): Melissa LeBlanc-Williams
|
||||
|
||||
Implementation Notes
|
||||
--------------------
|
||||
|
||||
**Software and Dependencies:**
|
||||
|
||||
* Adafruit CircuitPython firmware for the supported boards:
|
||||
https://github.com/adafruit/circuitpython/releases
|
||||
|
||||
"""
|
||||
|
||||
import os
|
||||
import time
|
||||
import gc
|
||||
from micropython import const
|
||||
from adafruit_io.adafruit_io import IO_HTTP, AdafruitIO_RequestError
|
||||
import supervisor
|
||||
import rtc
|
||||
from adafruit_fakerequests import Fake_Requests
|
||||
|
||||
|
||||
try:
|
||||
from secrets import secrets
|
||||
except ImportError:
|
||||
print(
|
||||
"""WiFi settings are kept in secrets.py, please add them there!
|
||||
the secrets dictionary must contain 'ssid' and 'password' at a minimum"""
|
||||
)
|
||||
raise
|
||||
|
||||
__version__ = "0.0.0-auto.0"
|
||||
__repo__ = "https://github.com/adafruit/Adafruit_CircuitPython_PortalBase.git"
|
||||
|
||||
# pylint: disable=line-too-long, too-many-lines
|
||||
# you'll need to pass in an io username and key
|
||||
TIME_SERVICE = (
|
||||
"https://io.adafruit.com/api/v2/%s/integrations/time/strftime?x-aio-key=%s"
|
||||
)
|
||||
# our strftime is %Y-%m-%d %H:%M:%S.%L %j %u %z %Z see http://strftime.net/ for decoding details
|
||||
# See https://apidock.com/ruby/DateTime/strftime for full options
|
||||
TIME_SERVICE_STRFTIME = (
|
||||
"&fmt=%25Y-%25m-%25d+%25H%3A%25M%3A%25S.%25L+%25j+%25u+%25z+%25Z"
|
||||
)
|
||||
LOCALFILE = "local.txt"
|
||||
# pylint: enable=line-too-long
|
||||
|
||||
STATUS_NO_CONNECTION = (100, 0, 0)
|
||||
STATUS_CONNECTING = (0, 0, 100)
|
||||
STATUS_FETCHING = (200, 100, 0)
|
||||
STATUS_DOWNLOADING = (0, 100, 100)
|
||||
STATUS_CONNECTED = (0, 100, 0)
|
||||
STATUS_DATA_RECEIVED = (0, 0, 100)
|
||||
STATUS_OFF = (0, 0, 0)
|
||||
|
||||
CONTENT_TEXT = const(1)
|
||||
CONTENT_JSON = const(2)
|
||||
CONTENT_IMAGE = const(3)
|
||||
|
||||
|
||||
class HttpError(Exception):
|
||||
"""HTTP Specific Error"""
|
||||
|
||||
|
||||
class NetworkBase:
|
||||
"""Network Base Class for the Portal-style libraries.
|
||||
|
||||
:param wifi_module: An initialized WiFi Module that encapsulates the WiFi communications
|
||||
:param bool extract_values: If true, single-length fetched values are automatically extracted
|
||||
from lists and tuples. Defaults to ``True``.
|
||||
:param debug: Turn on debug print outs. Defaults to False.
|
||||
|
||||
"""
|
||||
|
||||
# pylint: disable=too-many-instance-attributes, too-many-locals, too-many-branches, too-many-statements
|
||||
def __init__(
|
||||
self,
|
||||
wifi_module,
|
||||
*,
|
||||
extract_values=True,
|
||||
debug=False,
|
||||
):
|
||||
self._wifi = wifi_module
|
||||
self._debug = debug
|
||||
self.json_transform = []
|
||||
self._extract_values = extract_values
|
||||
|
||||
# This may be removed. Using for testing
|
||||
self.requests = None
|
||||
|
||||
try:
|
||||
os.stat(LOCALFILE)
|
||||
self.uselocal = True
|
||||
except OSError:
|
||||
self.uselocal = False
|
||||
|
||||
gc.collect()
|
||||
|
||||
def neo_status(self, value):
|
||||
"""The status NeoPixel.
|
||||
|
||||
:param value: The color to change the NeoPixel.
|
||||
|
||||
"""
|
||||
self._wifi.neo_status(value)
|
||||
|
||||
@staticmethod
|
||||
def json_traverse(json, path):
|
||||
"""
|
||||
Traverse down the specified JSON path and return the value or values
|
||||
|
||||
:param json: JSON data to traverse
|
||||
:param list path: The path that we want to follow
|
||||
|
||||
"""
|
||||
value = json
|
||||
if not isinstance(path, (list, tuple)):
|
||||
raise ValueError(
|
||||
"The json_path parameter should be enclosed in a list or tuple."
|
||||
)
|
||||
for x in path:
|
||||
try:
|
||||
value = value[x]
|
||||
except (TypeError, KeyError, IndexError) as error:
|
||||
raise ValueError(
|
||||
"The specified json_path was not found in the results."
|
||||
) from error
|
||||
gc.collect()
|
||||
return value
|
||||
|
||||
def add_json_transform(self, json_transform):
|
||||
"""Add a function that is applied to JSON data when data is fetched
|
||||
|
||||
:param json_transform: A function or a list of functions to call with the parsed JSON.
|
||||
Changes and additions are permitted for the ``dict`` object.
|
||||
"""
|
||||
if callable(json_transform):
|
||||
self.json_transform.append(json_transform)
|
||||
else:
|
||||
self.json_transform.extend(filter(callable, json_transform))
|
||||
|
||||
def get_local_time(self, location=None):
|
||||
# pylint: disable=line-too-long
|
||||
"""
|
||||
Fetch and "set" the local time of this microcontroller to the local time at the location, using an internet time API.
|
||||
|
||||
:param str location: Your city and country, e.g. ``"New York, US"``.
|
||||
|
||||
"""
|
||||
# pylint: enable=line-too-long
|
||||
self.connect()
|
||||
api_url = None
|
||||
try:
|
||||
aio_username = secrets["aio_username"]
|
||||
aio_key = secrets["aio_key"]
|
||||
except KeyError:
|
||||
raise KeyError(
|
||||
"\n\nOur time service requires a login/password to rate-limit. Please register for a free adafruit.io account and place the user/key in your secrets file under 'aio_username' and 'aio_key'" # pylint: disable=line-too-long
|
||||
) from KeyError
|
||||
|
||||
location = secrets.get("timezone", location)
|
||||
if location:
|
||||
print("Getting time for timezone", location)
|
||||
api_url = (TIME_SERVICE + "&tz=%s") % (aio_username, aio_key, location)
|
||||
else: # we'll try to figure it out from the IP address
|
||||
print("Getting time from IP address")
|
||||
api_url = TIME_SERVICE % (aio_username, aio_key)
|
||||
api_url += TIME_SERVICE_STRFTIME
|
||||
try:
|
||||
response = self._wifi.requests.get(api_url, timeout=10)
|
||||
if response.status_code != 200:
|
||||
error_message = (
|
||||
"Error connection to Adafruit IO. The response was: "
|
||||
+ response.text
|
||||
)
|
||||
raise ValueError(error_message)
|
||||
if self._debug:
|
||||
print("Time request: ", api_url)
|
||||
print("Time reply: ", response.text)
|
||||
times = response.text.split(" ")
|
||||
the_date = times[0]
|
||||
the_time = times[1]
|
||||
year_day = int(times[2])
|
||||
week_day = int(times[3])
|
||||
is_dst = None # no way to know yet
|
||||
except KeyError:
|
||||
raise KeyError(
|
||||
"Was unable to lookup the time, try setting secrets['timezone'] according to http://worldtimeapi.org/timezones" # pylint: disable=line-too-long
|
||||
) from KeyError
|
||||
year, month, mday = [int(x) for x in the_date.split("-")]
|
||||
the_time = the_time.split(".")[0]
|
||||
hours, minutes, seconds = [int(x) for x in the_time.split(":")]
|
||||
now = time.struct_time(
|
||||
(year, month, mday, hours, minutes, seconds, week_day, year_day, is_dst)
|
||||
)
|
||||
rtc.RTC().datetime = now
|
||||
|
||||
# now clean up
|
||||
response.close()
|
||||
response = None
|
||||
gc.collect()
|
||||
|
||||
def wget(self, url, filename, *, chunk_size=12000):
|
||||
"""Download a url and save to filename location, like the command wget.
|
||||
|
||||
:param url: The URL from which to obtain the data.
|
||||
:param filename: The name of the file to save the data to.
|
||||
:param chunk_size: how much data to read/write at a time.
|
||||
|
||||
"""
|
||||
print("Fetching stream from", url)
|
||||
|
||||
self.neo_status(STATUS_FETCHING)
|
||||
response = self._wifi.requests.get(url, stream=True)
|
||||
|
||||
headers = {}
|
||||
for title, content in response.headers.items():
|
||||
headers[title.lower()] = content
|
||||
|
||||
if response.status_code == 200:
|
||||
print("Reply is OK!")
|
||||
self.neo_status((0, 0, 100)) # green = got data
|
||||
else:
|
||||
if self._debug:
|
||||
if "content-length" in headers:
|
||||
print("Content-Length: {}".format(int(headers["content-length"])))
|
||||
if "date" in headers:
|
||||
print("Date: {}".format(headers["date"]))
|
||||
self.neo_status((100, 0, 0)) # red = http error
|
||||
raise HttpError(
|
||||
"Code {}: {}".format(
|
||||
response.status_code, response.reason.decode("utf-8")
|
||||
)
|
||||
)
|
||||
|
||||
if self._debug:
|
||||
print(response.headers)
|
||||
if "content-length" in headers:
|
||||
content_length = int(headers["content-length"])
|
||||
else:
|
||||
raise RuntimeError("Content-Length missing from headers")
|
||||
remaining = content_length
|
||||
print("Saving data to ", filename)
|
||||
stamp = time.monotonic()
|
||||
file = open(filename, "wb")
|
||||
for i in response.iter_content(min(remaining, chunk_size)): # huge chunks!
|
||||
self.neo_status(STATUS_DOWNLOADING)
|
||||
remaining -= len(i)
|
||||
file.write(i)
|
||||
if self._debug:
|
||||
print(
|
||||
"Read %d bytes, %d remaining"
|
||||
% (content_length - remaining, remaining)
|
||||
)
|
||||
else:
|
||||
print(".", end="")
|
||||
if not remaining:
|
||||
break
|
||||
self.neo_status(STATUS_FETCHING)
|
||||
file.close()
|
||||
|
||||
response.close()
|
||||
stamp = time.monotonic() - stamp
|
||||
print(
|
||||
"Created file of %d bytes in %0.1f seconds" % (os.stat(filename)[6], stamp)
|
||||
)
|
||||
self.neo_status(STATUS_OFF)
|
||||
if not content_length == os.stat(filename)[6]:
|
||||
raise RuntimeError
|
||||
|
||||
def connect(self):
|
||||
"""
|
||||
Connect to WiFi using the settings found in secrets.py
|
||||
"""
|
||||
self._wifi.neo_status(STATUS_CONNECTING)
|
||||
while not self._wifi.is_connected:
|
||||
# secrets dictionary must contain 'ssid' and 'password' at a minimum
|
||||
print("Connecting to AP", secrets["ssid"])
|
||||
if secrets["ssid"] == "CHANGE ME" or secrets["password"] == "CHANGE ME":
|
||||
change_me = "\n" + "*" * 45
|
||||
change_me += "\nPlease update the 'secrets.py' file on your\n"
|
||||
change_me += "CIRCUITPY drive to include your local WiFi\n"
|
||||
change_me += "access point SSID name in 'ssid' and SSID\n"
|
||||
change_me += "password in 'password'. Then save to reload!\n"
|
||||
change_me += "*" * 45
|
||||
raise OSError(change_me)
|
||||
self._wifi.neo_status(STATUS_NO_CONNECTION) # red = not connected
|
||||
try:
|
||||
self._wifi.connect(secrets["ssid"], secrets["password"])
|
||||
self.requests = self._wifi.requests
|
||||
except RuntimeError as error:
|
||||
print("Could not connect to internet", error)
|
||||
print("Retrying in 3 seconds...")
|
||||
time.sleep(3)
|
||||
|
||||
def _get_io_client(self):
|
||||
self.connect()
|
||||
|
||||
try:
|
||||
aio_username = secrets["aio_username"]
|
||||
aio_key = secrets["aio_key"]
|
||||
except KeyError:
|
||||
raise KeyError(
|
||||
"Adafruit IO secrets are kept in secrets.py, please add them there!\n\n"
|
||||
) from KeyError
|
||||
|
||||
return IO_HTTP(aio_username, aio_key, self._wifi.requests)
|
||||
|
||||
def push_to_io(self, feed_key, data):
|
||||
"""Push data to an adafruit.io feed
|
||||
|
||||
:param str feed_key: Name of feed key to push data to.
|
||||
:param data: data to send to feed
|
||||
|
||||
"""
|
||||
|
||||
io_client = self._get_io_client()
|
||||
|
||||
while True:
|
||||
try:
|
||||
feed_id = io_client.get_feed(feed_key)
|
||||
except AdafruitIO_RequestError:
|
||||
# If no feed exists, create one
|
||||
feed_id = io_client.create_new_feed(feed_key)
|
||||
except RuntimeError as exception:
|
||||
print("An error occured, retrying! 1 -", exception)
|
||||
continue
|
||||
break
|
||||
|
||||
while True:
|
||||
try:
|
||||
io_client.send_data(feed_id["key"], data)
|
||||
except RuntimeError as exception:
|
||||
print("An error occured, retrying! 2 -", exception)
|
||||
continue
|
||||
except NameError as exception:
|
||||
print(feed_id["key"], data, exception)
|
||||
continue
|
||||
break
|
||||
|
||||
def get_io_feed(self, feed_key, detailed=False):
|
||||
"""Return the Adafruit IO Feed that matches the feed key
|
||||
|
||||
:param str feed_key: Name of feed key to match.
|
||||
:param bool detailed: Whether to return additional detailed information
|
||||
|
||||
"""
|
||||
io_client = self._get_io_client()
|
||||
|
||||
while True:
|
||||
try:
|
||||
return io_client.get_feed(feed_key, detailed=detailed)
|
||||
except RuntimeError as exception:
|
||||
print("An error occured, retrying! 1 -", exception)
|
||||
continue
|
||||
break
|
||||
|
||||
def get_io_group(self, group_key):
|
||||
"""Return the Adafruit IO Group that matches the group key
|
||||
|
||||
:param str group_key: Name of group key to match.
|
||||
|
||||
"""
|
||||
io_client = self._get_io_client()
|
||||
|
||||
while True:
|
||||
try:
|
||||
return io_client.get_group(group_key)
|
||||
except RuntimeError as exception:
|
||||
print("An error occured, retrying! 1 -", exception)
|
||||
continue
|
||||
break
|
||||
|
||||
def get_io_data(self, feed_key):
|
||||
"""Return all values from Adafruit IO Feed Data that matches the feed key
|
||||
|
||||
:param str feed_key: Name of feed key to receive data from.
|
||||
|
||||
"""
|
||||
io_client = self._get_io_client()
|
||||
|
||||
while True:
|
||||
try:
|
||||
return io_client.receive_all_data(feed_key)
|
||||
except RuntimeError as exception:
|
||||
print("An error occured, retrying! 1 -", exception)
|
||||
continue
|
||||
break
|
||||
|
||||
def fetch(self, url, *, headers=None, timeout=10):
|
||||
"""Fetch data from the specified url and return a response object
|
||||
|
||||
:param str url: The URL to fetch from.
|
||||
:param list headers: Extra headers to include in the request.
|
||||
:param int timeout: The timeout period in seconds.
|
||||
|
||||
"""
|
||||
gc.collect()
|
||||
if self._debug:
|
||||
print("Free mem: ", gc.mem_free()) # pylint: disable=no-member
|
||||
|
||||
response = None
|
||||
if self.uselocal:
|
||||
print("*** USING LOCALFILE FOR DATA - NOT INTERNET!!! ***")
|
||||
response = Fake_Requests(LOCALFILE)
|
||||
|
||||
if not response:
|
||||
self.connect()
|
||||
# great, lets get the data
|
||||
print("Retrieving data...", end="")
|
||||
self.neo_status(STATUS_FETCHING) # yellow = fetching data
|
||||
gc.collect()
|
||||
response = self._wifi.requests.get(url, headers=headers, timeout=timeout)
|
||||
gc.collect()
|
||||
|
||||
return response
|
||||
|
||||
def fetch_data(
|
||||
self,
|
||||
url,
|
||||
*,
|
||||
headers=None,
|
||||
json_path=None,
|
||||
regexp_path=None,
|
||||
timeout=10,
|
||||
):
|
||||
"""Fetch data from the specified url and perfom any parsing
|
||||
|
||||
:param str url: The URL to fetch from.
|
||||
:param list headers: Extra headers to include in the request.
|
||||
:param json_path: The path to drill down into the JSON data.
|
||||
:param regexp_path: The path formatted as a regular expression to drill down
|
||||
into the JSON data.
|
||||
:param int timeout: The timeout period in seconds.
|
||||
|
||||
"""
|
||||
json_out = None
|
||||
values = []
|
||||
content_type = CONTENT_TEXT
|
||||
|
||||
response = self.fetch(url, headers=headers, timeout=timeout)
|
||||
|
||||
headers = {}
|
||||
for title, content in response.headers.items():
|
||||
headers[title.lower()] = content
|
||||
gc.collect()
|
||||
if self._debug:
|
||||
print("Headers:", headers)
|
||||
if response.status_code == 200:
|
||||
print("Reply is OK!")
|
||||
self.neo_status(STATUS_DATA_RECEIVED) # green = got data
|
||||
if "content-type" in headers:
|
||||
if "image/" in headers["content-type"]:
|
||||
content_type = CONTENT_IMAGE
|
||||
elif "application/json" in headers["content-type"]:
|
||||
content_type = CONTENT_JSON
|
||||
elif "application/javascript" in headers["content-type"]:
|
||||
content_type = CONTENT_JSON
|
||||
else:
|
||||
if self._debug:
|
||||
if "content-length" in headers:
|
||||
print("Content-Length: {}".format(int(headers["content-length"])))
|
||||
if "date" in headers:
|
||||
print("Date: {}".format(headers["date"]))
|
||||
self.neo_status((100, 0, 0)) # red = http error
|
||||
raise HttpError(
|
||||
"Code {}: {}".format(
|
||||
response.status_code, response.reason.decode("utf-8")
|
||||
)
|
||||
)
|
||||
|
||||
if content_type == CONTENT_JSON and json_path is not None:
|
||||
if isinstance(json_path, (list, tuple)) and (
|
||||
not json_path or not isinstance(json_path[0], (list, tuple))
|
||||
):
|
||||
json_path = (json_path,)
|
||||
try:
|
||||
gc.collect()
|
||||
json_out = response.json()
|
||||
if self._debug:
|
||||
print(json_out)
|
||||
gc.collect()
|
||||
except ValueError: # failed to parse?
|
||||
print("Couldn't parse json: ", response.text)
|
||||
raise
|
||||
except MemoryError:
|
||||
supervisor.reload()
|
||||
|
||||
if regexp_path:
|
||||
import re # pylint: disable=import-outside-toplevel
|
||||
|
||||
# optional JSON post processing, apply any transformations
|
||||
# these MAY change/add element
|
||||
for idx, json_transform in enumerate(self.json_transform):
|
||||
try:
|
||||
json_transform(json_out)
|
||||
except Exception as error:
|
||||
print("Exception from json_transform: ", idx, error)
|
||||
raise
|
||||
|
||||
# extract desired text/values from json
|
||||
if json_out is not None and json_path:
|
||||
for path in json_path:
|
||||
try:
|
||||
values.append(self.json_traverse(json_out, path))
|
||||
except KeyError:
|
||||
print(json_out)
|
||||
raise
|
||||
elif content_type == CONTENT_TEXT and regexp_path:
|
||||
for regexp in regexp_path:
|
||||
values.append(re.search(regexp, response.text).group(1))
|
||||
else:
|
||||
if json_out:
|
||||
# No path given, so return JSON as string for compatibility
|
||||
import json # pylint: disable=import-outside-toplevel
|
||||
|
||||
values = json.dumps(response.json())
|
||||
else:
|
||||
values = response.text
|
||||
|
||||
# we're done with the requests object, lets delete it so we can do more!
|
||||
json_out = None
|
||||
response = None
|
||||
gc.collect()
|
||||
if self._extract_values and len(values) == 1:
|
||||
return values[0]
|
||||
|
||||
return values
|
||||
BIN
docs/_static/favicon.ico
vendored
Normal file
BIN
docs/_static/favicon.ico
vendored
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.3 KiB |
3
docs/_static/favicon.ico.license
vendored
Normal file
3
docs/_static/favicon.ico.license
vendored
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
SPDX-FileCopyrightText: 2018 Phillip Torrone for Adafruit Industries
|
||||
|
||||
SPDX-License-Identifier: CC-BY-4.0
|
||||
9
docs/api.rst
Normal file
9
docs/api.rst
Normal file
|
|
@ -0,0 +1,9 @@
|
|||
|
||||
.. automodule:: adafruit_portalbase
|
||||
:members:
|
||||
|
||||
.. automodule:: adafruit_portalbase.graphics
|
||||
:members:
|
||||
|
||||
.. automodule:: adafruit_portalbase.network
|
||||
:members:
|
||||
3
docs/api.rst.license
Normal file
3
docs/api.rst.license
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
|
||||
SPDX-License-Identifier: MIT
|
||||
190
docs/conf.py
Normal file
190
docs/conf.py
Normal file
|
|
@ -0,0 +1,190 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
import os
|
||||
import sys
|
||||
|
||||
sys.path.insert(0, os.path.abspath(".."))
|
||||
|
||||
# -- General configuration ------------------------------------------------
|
||||
|
||||
# Add any Sphinx extension module names here, as strings. They can be
|
||||
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
|
||||
# ones.
|
||||
extensions = [
|
||||
"sphinx.ext.autodoc",
|
||||
"sphinx.ext.intersphinx",
|
||||
"sphinx.ext.napoleon",
|
||||
"sphinx.ext.todo",
|
||||
]
|
||||
|
||||
# Uncomment the below if you use native CircuitPython modules such as
|
||||
# digitalio, micropython and busio. List the modules you use. Without it, the
|
||||
# autodoc module docs will fail to generate with a warning.
|
||||
autodoc_mock_imports = [
|
||||
"supervisor",
|
||||
"rtc",
|
||||
"ssl",
|
||||
"secrets",
|
||||
]
|
||||
|
||||
intersphinx_mapping = {
|
||||
"python": ("https://docs.python.org/3.4", None),
|
||||
"CircuitPython": ("https://circuitpython.readthedocs.io/en/latest/", None),
|
||||
}
|
||||
|
||||
# Show the docstring from both the class and its __init__() method.
|
||||
autoclass_content = "both"
|
||||
|
||||
# Add any paths that contain templates here, relative to this directory.
|
||||
templates_path = ["_templates"]
|
||||
|
||||
source_suffix = ".rst"
|
||||
|
||||
# The master toctree document.
|
||||
master_doc = "index"
|
||||
|
||||
# General information about the project.
|
||||
project = "Adafruit PortalBase Library"
|
||||
copyright = "2020 Melissa LeBlanc-Williams"
|
||||
author = "Melissa LeBlanc-Williams"
|
||||
|
||||
# The version info for the project you're documenting, acts as replacement for
|
||||
# |version| and |release|, also used in various other places throughout the
|
||||
# built documents.
|
||||
#
|
||||
# The short X.Y version.
|
||||
version = "1.0"
|
||||
# The full version, including alpha/beta/rc tags.
|
||||
release = "1.0"
|
||||
|
||||
# The language for content autogenerated by Sphinx. Refer to documentation
|
||||
# for a list of supported languages.
|
||||
#
|
||||
# This is also used if you do content translation via gettext catalogs.
|
||||
# Usually you set "language" from the command line for these cases.
|
||||
language = None
|
||||
|
||||
# List of patterns, relative to source directory, that match files and
|
||||
# directories to ignore when looking for source files.
|
||||
# This patterns also effect to html_static_path and html_extra_path
|
||||
exclude_patterns = [
|
||||
"_build",
|
||||
"Thumbs.db",
|
||||
".DS_Store",
|
||||
".env",
|
||||
"CODE_OF_CONDUCT.md",
|
||||
]
|
||||
|
||||
# The reST default role (used for this markup: `text`) to use for all
|
||||
# documents.
|
||||
#
|
||||
default_role = "any"
|
||||
|
||||
# If true, '()' will be appended to :func: etc. cross-reference text.
|
||||
#
|
||||
add_function_parentheses = True
|
||||
|
||||
# The name of the Pygments (syntax highlighting) style to use.
|
||||
pygments_style = "sphinx"
|
||||
|
||||
# If true, `todo` and `todoList` produce output, else they produce nothing.
|
||||
todo_include_todos = False
|
||||
|
||||
# If this is True, todo emits a warning for each TODO entries. The default is False.
|
||||
todo_emit_warnings = True
|
||||
|
||||
napoleon_numpy_docstring = False
|
||||
|
||||
# -- Options for HTML output ----------------------------------------------
|
||||
|
||||
# The theme to use for HTML and HTML Help pages. See the documentation for
|
||||
# a list of builtin themes.
|
||||
#
|
||||
on_rtd = os.environ.get("READTHEDOCS", None) == "True"
|
||||
|
||||
if not on_rtd: # only import and set the theme if we're building docs locally
|
||||
try:
|
||||
import sphinx_rtd_theme
|
||||
|
||||
html_theme = "sphinx_rtd_theme"
|
||||
html_theme_path = [sphinx_rtd_theme.get_html_theme_path(), "."]
|
||||
except:
|
||||
html_theme = "default"
|
||||
html_theme_path = ["."]
|
||||
else:
|
||||
html_theme_path = ["."]
|
||||
|
||||
# Add any paths that contain custom static files (such as style sheets) here,
|
||||
# relative to this directory. They are copied after the builtin static files,
|
||||
# so a file named "default.css" will overwrite the builtin "default.css".
|
||||
html_static_path = ["_static"]
|
||||
|
||||
# The name of an image file (relative to this directory) to use as a favicon of
|
||||
# the docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32
|
||||
# pixels large.
|
||||
#
|
||||
html_favicon = "_static/favicon.ico"
|
||||
|
||||
# Output file base name for HTML help builder.
|
||||
htmlhelp_basename = "AdafruitPortalbaseLibrarydoc"
|
||||
|
||||
# -- Options for LaTeX output ---------------------------------------------
|
||||
|
||||
latex_elements = {
|
||||
# The paper size ('letterpaper' or 'a4paper').
|
||||
# 'papersize': 'letterpaper',
|
||||
# The font size ('10pt', '11pt' or '12pt').
|
||||
# 'pointsize': '10pt',
|
||||
# Additional stuff for the LaTeX preamble.
|
||||
# 'preamble': '',
|
||||
# Latex figure (float) alignment
|
||||
# 'figure_align': 'htbp',
|
||||
}
|
||||
|
||||
# Grouping the document tree into LaTeX files. List of tuples
|
||||
# (source start file, target name, title,
|
||||
# author, documentclass [howto, manual, or own class]).
|
||||
latex_documents = [
|
||||
(
|
||||
master_doc,
|
||||
"AdafruitPortalBaseLibrary.tex",
|
||||
"AdafruitPortalBase Library Documentation",
|
||||
author,
|
||||
"manual",
|
||||
),
|
||||
]
|
||||
|
||||
# -- Options for manual page output ---------------------------------------
|
||||
|
||||
# One entry per manual page. List of tuples
|
||||
# (source start file, name, description, authors, manual section).
|
||||
man_pages = [
|
||||
(
|
||||
master_doc,
|
||||
"AdafruitPortalBaselibrary",
|
||||
"Adafruit PortalBase Library Documentation",
|
||||
[author],
|
||||
1,
|
||||
),
|
||||
]
|
||||
|
||||
# -- Options for Texinfo output -------------------------------------------
|
||||
|
||||
# Grouping the document tree into Texinfo files. List of tuples
|
||||
# (source start file, target name, title, author,
|
||||
# dir menu entry, description, category)
|
||||
texinfo_documents = [
|
||||
(
|
||||
master_doc,
|
||||
"AdafruitPortalBaseLibrary",
|
||||
"Adafruit PortalBase Library Documentation",
|
||||
author,
|
||||
"AdafruitPortalBaseLibrary",
|
||||
"One line description of project.",
|
||||
"Miscellaneous",
|
||||
),
|
||||
]
|
||||
40
docs/index.rst
Normal file
40
docs/index.rst
Normal file
|
|
@ -0,0 +1,40 @@
|
|||
.. include:: ../README.rst
|
||||
|
||||
Table of Contents
|
||||
=================
|
||||
|
||||
.. toctree::
|
||||
:maxdepth: 4
|
||||
:hidden:
|
||||
|
||||
self
|
||||
|
||||
.. toctree::
|
||||
:caption: API Reference
|
||||
:maxdepth: 3
|
||||
|
||||
api
|
||||
|
||||
.. toctree::
|
||||
:caption: Tutorials
|
||||
|
||||
.. toctree::
|
||||
:caption: Related Products
|
||||
|
||||
.. toctree::
|
||||
:caption: Other Links
|
||||
|
||||
Download <https://github.com/adafruit/Adafruit_CircuitPython_PortalBase/releases/latest>
|
||||
CircuitPython Reference Documentation <https://circuitpython.readthedocs.io>
|
||||
CircuitPython Support Forum <https://forums.adafruit.com/viewforum.php?f=60>
|
||||
Discord Chat <https://adafru.it/discord>
|
||||
Adafruit Learning System <https://learn.adafruit.com>
|
||||
Adafruit Blog <https://blog.adafruit.com>
|
||||
Adafruit Store <https://www.adafruit.com>
|
||||
|
||||
Indices and tables
|
||||
==================
|
||||
|
||||
* :ref:`genindex`
|
||||
* :ref:`modindex`
|
||||
* :ref:`search`
|
||||
3
docs/index.rst.license
Normal file
3
docs/index.rst.license
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
|
||||
SPDX-License-Identifier: MIT
|
||||
6
pyproject.toml
Normal file
6
pyproject.toml
Normal file
|
|
@ -0,0 +1,6 @@
|
|||
# SPDX-FileCopyrightText: 2020 Diego Elio Pettenò
|
||||
#
|
||||
# SPDX-License-Identifier: Unlicense
|
||||
|
||||
[tool.black]
|
||||
target-version = ['py35']
|
||||
14
requirements.txt
Normal file
14
requirements.txt
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
# SPDX-FileCopyrightText: Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
Adafruit-Blinka
|
||||
adafruit-blinka-displayio
|
||||
adafruit-circuitpython-bitmap-font
|
||||
adafruit-circuitpython-display-text
|
||||
adafruit-circuitpython-neopixel
|
||||
adafruit-circuitpython-requests
|
||||
adafruit-circuitpython-adafruitio
|
||||
adafruit-circuitpython-simpleio
|
||||
adafruit-circuitpython-fakerequests
|
||||
67
setup.py
Normal file
67
setup.py
Normal file
|
|
@ -0,0 +1,67 @@
|
|||
# SPDX-FileCopyrightText: 2017 Scott Shawcroft, written for Adafruit Industries
|
||||
# SPDX-FileCopyrightText: Copyright (c) 2020 Melissa LeBlanc-Williams for Adafruit Industries
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
"""A setuptools based setup module.
|
||||
|
||||
See:
|
||||
https://packaging.python.org/en/latest/distributing.html
|
||||
https://github.com/pypa/sampleproject
|
||||
"""
|
||||
|
||||
from setuptools import setup, find_packages
|
||||
|
||||
# To use a consistent encoding
|
||||
from codecs import open
|
||||
from os import path
|
||||
|
||||
here = path.abspath(path.dirname(__file__))
|
||||
|
||||
# Get the long description from the README file
|
||||
with open(path.join(here, "README.rst"), encoding="utf-8") as f:
|
||||
long_description = f.read()
|
||||
|
||||
setup(
|
||||
name="adafruit-circuitpython-portalbase",
|
||||
use_scm_version=True,
|
||||
setup_requires=["setuptools_scm"],
|
||||
description="Base Library for the Portal-style libraries.",
|
||||
long_description=long_description,
|
||||
long_description_content_type="text/x-rst",
|
||||
# The project's main homepage.
|
||||
url="https://github.com/adafruit/Adafruit_CircuitPython_PortalBase",
|
||||
# Author details
|
||||
author="Adafruit Industries",
|
||||
author_email="circuitpython@adafruit.com",
|
||||
install_requires=[
|
||||
"Adafruit-Blinka",
|
||||
"adafruit-blinka-displayio",
|
||||
"adafruit-circuitpython-bitmap-font",
|
||||
"adafruit-circuitpython-display-text",
|
||||
"adafruit-circuitpython-neopixel",
|
||||
"adafruit-circuitpython-requests",
|
||||
"adafruit-circuitpython-adafruitio",
|
||||
"adafruit-circuitpython-simpleio",
|
||||
"adafruit-circuitpython-fakerequests",
|
||||
],
|
||||
# Choose your license
|
||||
license="MIT",
|
||||
# See https://pypi.python.org/pypi?%3Aaction=list_classifiers
|
||||
classifiers=[
|
||||
"Development Status :: 3 - Alpha",
|
||||
"Intended Audience :: Developers",
|
||||
"Topic :: Software Development :: Libraries",
|
||||
"Topic :: System :: Hardware",
|
||||
"License :: OSI Approved :: MIT License",
|
||||
"Programming Language :: Python :: 3",
|
||||
"Programming Language :: Python :: 3.4",
|
||||
"Programming Language :: Python :: 3.5",
|
||||
],
|
||||
# What does your project relate to?
|
||||
keywords="adafruit blinka circuitpython micropython portalbase PyPortal MatrixPortal "
|
||||
"MagTag Portal base",
|
||||
# You can just specify the packages manually here if your project is
|
||||
# simple. Or you can use find_packages().
|
||||
packages=["adafruit_portalbase"],
|
||||
)
|
||||
Loading…
Reference in a new issue