Qwiic_Ultrasonic_Py
Python for SFE Qwiic Ultrasonic
|
This is a template repository and associated documentation that outlines how to publish and maintain a python package for the SparkFun Qwiic ecosystem.
This repository defines the general structure of a repository and details the role of each file/location in the repository. Additionally, the use of ReadTheDocs and PyPi are outlined.
The general structure implemented follows the guidelines set forth in the python packaging structure and tools. While this document provides a high-level overview of SparkFun's implementation of a python project, details of this process and structure can be found in the Packaging Python Projects document provided by the Python Foundation.
For details and example use of the described structure, please review the existing python projects for the Qwiic system. Examples Include:
The general structure of a Qwiic Python repository is as follows
To keep the implementation simple and minimize resource needs, there are few design requirements when implementing a new Qwiic I2C driver. Each driver implements a class that encapsulates all interations with the underlying I2C device. This class implements a simple interface that enables the higher-level functionality provided by the overarching Qwiic package.
The specific implementation requirements are as follows
Each development driver package implements a class that encapsulates all interactions with this device.
The class name should be a CamelCase version of the package name. This naming schema is used by future automation functionality and follows common python methodologies
An example of a class declaration (note the use and location of the class docstring):
To support the dynamic discovery and enumeration of Qwiic boards by the Qwiic package, each object implements a set of class variables. This allows the Qwiic package to inspect these values at runtime without having to instantiate an actual object.
These variables are:
Class Variable Name | Description |
---|---|
device_name | - Set to the human-readable name of the device |
available_addresses | - Set to an array of the I2C addresses this device supports. The first address is the default |
These values are set outside of any class method, by convention they are placed right after the class declaration statement.
Example:
The Qwiic package expects the constructor of the class to implement the following signature:
The method supports the following parameters:
Parameter | Description |
---|---|
address | The I2C address to use for the device. If not provided, the default address is used |
i2c_driver | An existing Qwiic I2C device object. If not provided, the class should create an instance of driver |
The initial body of the constructor handles these parameters - setting the I2C address and constructing a I2C driver if needed. The following object constructor provides a boilerplate implementation for this functionality.
Note - the docstring for the constructor is actually the docstring for the class.
While not strictly required, the following conventions and patterns are used for qwiic driver implementations
A standard methodology for I2C device implementations is to define constants (#defines in C/C++) for I2C interaction values for a device. For Qwiic python modules these values are defined as capitalized attributes and either placed as file attributes or class attributes on the driver class.
The convention is to implement any attributes required for user interaction as class attributes. Any internal values are created as file/modules attributes.
Each class implements an is_connected()
method that returns True the specific Qwiic device is connect to the system.
This is a standard method, that often uses the following implementation pattern.
Additionally, the method is exposed as a read-only attribute on the object.
Following the pattern set by the Qwiic Arduino libraries, a begin() method is used to perform the actual initialization of the underlying I2C device.
While each device implements device specific initialization logic, the signature of this method is as follows:
There are two patterns of implementation for a package - a python module or a python package.
To the end-user a package or module looks the same, but the implementation within the repository is different.
A python module is nothing more than a single file that makes up the overall implementaiton for the package. This file has the same name as the package being imported by the user.
For example, if a user imports a module named qwiic_module
The file name would be qwiic_module.py
and reside in the root of the repository.
A python package is a folder that contains the implementation of the package. The folder can contain python source files, as well as any other resource needed for the property operation of the package.
The package directory is name is the name of the package. A file named __init__.py
in the root directory of the package defines its entry/operation and lets python know that the directory implements a package.
For example, if the user imports a package named qwiic_package
The structure of this implementation would be under a directory called qwiic_package
in the repository
Note: The souce of the implementation must be contain the license attribution statement.
The file named LICENSE contains the license for the repository. The name of the file, LICENSE, is used by other systems to identify which license the repository implements.
For example, in GitHub, when the contents of the file is viewed, the system will display details about the license and clearly indicate to the user what the license covers.
The SparkFun Qwiic python module implementations fall under the MIT license.
Each source file distributed with one of our python packags/modules must include the proper license attribution in the entry comment section of the code.
The Qwiic Python packages are licensed using the MIT license and as such should include the following statement in the top/entry section of the code:
The Qwiic Python components are packaged using standard python package/install tools, and hosted within the Python Package Index (pypy.org).
Within the repository, the files that makeup the package are the following:
The file DESCRIPTION.rst is a RST (reStructured Text) file that has a simple, high-level description of the package. When setup.py is executed, it reads the contents of this file and sets it as the description of the package. _(***Note:** The title should read Qwiic <Package Name> and the number of =
must be greater than the character length of the title.)_
The file setup.cfg contains options that the packaging tools use when creating the specific package. For the most part, the file in this template repo can be used.
The file setup.py
is a python script that is used to describe the package and build an install package. The file is used by the python package setuptools
, which is a collection of utilities that make it simple to build and distribute Python distributions.
This template repository contains an example setup.py
file for review and an overview of the file contents are below. For details on the structure of the file, please review the setup.py section in the Packaging Python Projects document.
___Description Section___
One of the first sections in setup.py
is reading in the contents of DESCRIPTION.rst.
This reads the contents of the description fine and places the resulting string into the variable long_description
. This variable is passed into the call to setup()
using the *long_description
* keyword parameter.
Note: The io.open
method is used to support uft-8 file encoding in Python versions 2.7 and 3.*.
___setuptools.setup( name=)___
This keyword is set to the name to publish the package under in PyPi.org and the name passed to the pip
command for installing the package. _(***Note:** The package name should be in the form of sparkfun_qwiic_<package_name>
.)_
The following command shows this value for the qwiic bme280:
NOTE: For PyPi/Pip, underscores _
and dashes -
are interchangeable.
___setuptools.setup( version=)___
This controls the package's release version on PyPI. _(***Note:** Start off with the lowest release value until the package is finalized; then, the version can get "bumped up" to 1.0.0
. When uploading a package to PyPI, the version number needs to be "bumped up" for any package changes to take into effect.)_
___setuptools.setup( description= and url=)___
Modify the description with the package's name and include the url for the associated product page.
___setuptools.setup( install_requires=)___
The *install_requires
* keyword arguement to setuptools.setup()
is used to specify what other python packages this package depends on.
An example of this is the sparkfun-qwiic-i2c
package, which all Qwiic board python packages use. An example of this from the Qwiic Proximity package setup()
is as follows:
For the overall Qwiic package, which depends on all driver packages, this parameter has the following form:
___setuptools.setup(classifiers=[])___
The classifiers argument to setup()
are attrbitues that describe the package and are used details specifics to the PyPi respository and users of the project. While a detailed list of of valid classifier values is available at pypy.org, the key values are the project maturity (is it Alpha, Beta, Production?) and what python versions are supported.
The example script has the following classifiers:
You can see these detailed out on the SparkFun Qwiic package (sparkfun-qwiic) on the PyPi.org repository.
___setuptools.setup(packages=[])___
If your repository defines one or more packages (directories), the names of these packages are provided to the packages
keyboard argument to setuptools. Note: this is the directory/package name the user references in python code, not the package name used by PyPi - which can also contain additional keywords.
For the Qwiic package, this is just the Qwiic directory:
or for the qwiic_micro_oled package, which includes a font subpackage:
___setuptools.setup(package_data={})___
The packaging system will include python files (.py
) files by default. If the package includes non-python files, these are specified via the pacakge_data
keyword argument, which takes a dictionary.
The provided dictionary key values are a specific location, and the value is the data files to include in the package. The data filenames can be specific names, or include wildcards.
An example of this is used in the qwiic_micro_oled package, which includes font data files, named using a .bin
file extension. These data files are located in the ./fonts
subdirectory of the package repository.
___setuptools.setup(py_modules=[])___
If the install package implements a module (source file) and not a python package (directory), the modules are specific to the setup()
method call using the py_modules=[]
keyword argument. The value of this keyword is an array that contains the names of the modules to include in the package. Note, the file suffix is not included in the provided names. _(***Note:** The module name should be in the form of "qwiic_<package_name>"
.)_
For the Qwiic BME280 package, which is implemented in a single file, the module is specified as follows:
When ready to build and upload a package to pypi.org, the following setups are performed.
You'll need an account on PyPi.org - it's a simple sign up procedure.
To publish a new package, you can use this account. If you are updating or modifying an existing package, you'll need to be added as a Maintainer of the package by the package owner.
To build and upload the packages, make sure the required python packages are installed - setuptools, twine, and wheel.
Build the distribution packages using the following commands (executing in the package root directory). First create a standard distribution:
Then a distribution in the wheel
format.
These commands will create distribution package files and place them in the ./dist
subdirectory.
The twine
command is used to upload the install packages to pypi.org. To upload the packages, use the following command:
This command will prompt for the username and password for the pypi account to use for the upload.
Once the upload is completed, the packages are now available for use via the pip installer.
NOTE: Your PyPi.org username and password can be specified in the file ~/.pypirc
instead of entering with each call to twine. The format of this file:
Details of the documentation generation process are contained in the file DOCUMENTATION.md
Adding the Module dependency to the main Qwiic package, Qwiic_Py
The overall Qwiic package, which is hosted in the Qwiic_Py repository, defines dependencies to all the SparkFun Qwiic python packages. This is accomplished by adding modules to the repo as git submodules.
New drivers are added as git submodules in the Qwiic_Py/qwiic/drivers directory.
Naming of the driver directory is important – it should map to the package name in PyPi, minus the initial ‘sparkfun-‘ name.
So for the BME280 package, which is defined in PyPi as ‘sparkfun-qwiic-bme280’, would be placed in a directory named ‘qwiic_bme280’ in the drivers folder.
To add a driver/package to the Qwiic repository, do the following steps:
Note, if you get a failure due to permissions, you may need to use the complete URL for the <repo to add>. (***Note:** Don't forget to include the submodule name (`qwiic<package_name>`) after the link for the repo.)_
Example for the Titan GPS driver (with full URL) git submodule add https://github.com/sparkfun/Qwiic_Titan_Gps_Py qwiic_titan_gps
Once completed, the Qwiic_Py package must be updated and uploaded to PyPi.
Once completed, an update/install of the sparkfun-qwiic package will include the new submodule