PyRQA


NamePyRQA JSON
Version 8.1.0 PyPI version JSON
download
home_page
SummaryRecurrence analysis in a massively parallel manner using the OpenCL framework.
upload_time2024-02-17 20:26:07
maintainer
docs_urlNone
authorTobias Rawald
requires_python
licenseApache License 2.0
keywords nonlinear time series analysis dynamical system recurrence quantification analysis rqa cross recurrence quantification analysis crqa joint recurrence quantification analysis jrqa recurrence plot rp cross recurrence plot crp joint recurrence plot jrp
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            PyRQA
=====

Highlights
----------

-  Perform recurrence analysis on long time series in a time efficient
   manner using the OpenCL framework.
-  Conduct recurrence quantification analysis (*RQA*) and cross
   recurrence quantification analysis (*CRQA*).
-  Compute recurrence plots (*RP*) and cross recurrence plots (*CRP*).
-  Compute unthresholded recurrence plots (*URP*) and unthresholded
   cross recurrence plots (*UCRP*).
-  Conduct joint recurrence quantification analysis (*JRQA*) and compute
   joint recurrence plots (*JRP*).
-  Employ the euclidean, maximum or taxicab metric for determining state
   similarity.
-  Choose the fixed radius or radius corridor neighbourhood condition.
-  Use univariate time series or pre-embedded time series as input data.
-  Select either the half, single or double floating point precision for
   conducting the analytical computations.
-  Leverage machine learning techniques that automatically choose the
   fastest from a set of implementations.
-  Apply the computing capabilities of GPUs, CPUs and other platforms
   that support OpenCL.
-  Use multiple computing devices of the same or different type in
   parallel.

Table of Contents
-----------------

1.  `General Information <#general-information>`__
2.  `Recommended Citation <#recommended-citation>`__
3.  `Installation <#installation>`__
4.  `OpenCL Setup <#opencl-setup>`__
5.  `Usage <#usage>`__
6.  `Performance Tuning <#performance-tuning>`__
7.  `Testing <#testing>`__
8.  `Origin <#origin>`__
9.  `Acknowledgements <#acknowledgements>`__
10. `Publications <#publications>`__
11. `Development Status <#development-status>`__
12. `Release Notes <#release-notes>`__

General Information
-------------------

PyRQA is a tool to conduct recurrence analysis in a massively parallel
manner using the OpenCL framework. It is designed to efficiently process
time series consisting of hundreds of thousands of data points.

PyRQA supports the computation of the following quantitative measures:

-  Recurrence rate (*RR*)
-  Determinism (*DET*)
-  Average diagonal line length (*L*)
-  Longest diagonal line length (*L_max*)
-  Divergence (*DIV*)
-  Entropy diagonal lines (*L_entr*)
-  Laminarity (*LAM*)
-  Trapping time (*TT*)
-  Longest vertical line length (*V_max*)
-  Entropy vertical lines (*V_entr*)
-  Average white vertical line length (*W*)
-  Longest white vertical line length (*W_max*)
-  Longest white vertical line length divergence (*W_div*)
-  Entropy white vertical lines (*W_entr*)

PyRQA additionally allows to create the corresponding recurrence plot,
which can be exported as an image file.

Recommended Citation
--------------------

Please acknowledge the use of PyRQA by citing the following publication.

   Rawald, T., Sips, M., Marwan, N. (2017): PyRQA - Conducting
   Recurrence Quantification Analysis on Very Long Time Series
   Efficiently. - Computers and Geosciences, 104, pp. 101-108.

Installation
------------

PyRQA and all of its dependencies can be installed via the following
command.

.. code:: bash

   pip install PyRQA

OpenCL Setup
------------

The analytical implementations provided by PyRQA rely on features that
are part of *OpenCL 1.1*, which is a fairly mature standard and
supported by a large number of platforms. The OpenCL computing devices
employed need to support at least this version to being able to use
PyRQA.

It may be required to install additional software, e.g., runtimes or
drivers, to execute PyRQA on computing devices such as GPUs and CPUs.
References to vendor-specific information are presented below.

*AMD*:

-  https://www.amd.com/en/support
-  https://github.com/RadeonOpenCompute/ROCm
-  https://community.amd.com/community/devgurus/opencl
-  https://www.amd.com/en/support/kb/release-notes/amdgpu-installation

*ARM*:

-  https://developer.arm.com/docs/100614/0312

*Intel*:

-  https://software.intel.com/en-us/articles/opencl-drivers
-  https://software.intel.com/en-us/articles/sdk-for-opencl-gsg

*NVIDIA*:

-  https://developer.nvidia.com/opencl
-  https://developer.nvidia.com/cuda-downloads

*Vendor-independent*:

-  http://portablecl.org

Usage
-----

The following subsections depict the usage of PyRQA. Each subsection
focuses on specific functionality that is provided by the package. The
presentation of expected output shall ensure the reproducibility of the
analytical results.

Basic Computations
~~~~~~~~~~~~~~~~~~

RQA computations are conducted as follows.

.. code:: python

   from pyrqa.time_series import TimeSeries
   from pyrqa.settings import Settings
   from pyrqa.analysis_type import Classic
   from pyrqa.neighbourhood import FixedRadius
   from pyrqa.metric import EuclideanMetric
   from pyrqa.computation import RQAComputation
   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]
   time_series = TimeSeries(data_points,
                            embedding_dimension=2,
                            time_delay=2)
   settings = Settings(time_series,
                       analysis_type=Classic,
                       neighbourhood=FixedRadius(0.65),
                       similarity_measure=EuclideanMetric,
                       theiler_corrector=1)
   computation = RQAComputation.create(settings,
                                       verbose=True)
   result = computation.run()
   result.min_diagonal_line_length = 2
   result.min_vertical_line_length = 2
   result.min_white_vertical_line_length = 2
   print(result)

The following output is expected.

::

   RQA Result:
   ===========

   Minimum diagonal line length (L_min): 2
   Minimum vertical line length (V_min): 2
   Minimum white vertical line length (W_min): 2

   Recurrence rate (RR): 0.371901
   Determinism (DET): 0.411765
   Average diagonal line length (L): 2.333333
   Longest diagonal line length (L_max): 3
   Divergence (DIV): 0.333333
   Entropy diagonal lines (L_entr): 0.636514
   Laminarity (LAM): 0.400000
   Trapping time (TT): 2.571429
   Longest vertical line length (V_max): 4
   Entropy vertical lines (V_entr): 0.955700
   Average white vertical line length (W): 2.538462
   Longest white vertical line length (W_max): 6
   Longest white vertical line length inverse (W_div): 0.166667
   Entropy white vertical lines (W_entr): 0.839796

   Ratio determinism / recurrence rate (DET/RR): 1.107190
   Ratio laminarity / determinism (LAM/DET): 0.971429

The corresponding recurrence plot is created likewise. Note that the
parameter ``theiler_corrector`` is ignored regarding the creation of the
plot.

.. code:: python

   from pyrqa.computation import RPComputation
   from pyrqa.image_generator import ImageGenerator
   computation = RPComputation.create(settings)
   result = computation.run()
   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,
                                       'recurrence_plot.png')

Cross Recurrence Analysis
~~~~~~~~~~~~~~~~~~~~~~~~~

PyRQA further offers the opportunity to conduct cross recurrence
analysis (*CRQA* and *CRP*), in addition to the classic recurrence
analysis (*RQA* and *RP*). For this purpose, two time series of
potentially different length are provided as input. Note that the
corresponding computations require to set the same value regarding the
embedding dimension. Two different time delay values may be used
regarding the first and the second time series. To enable cross
recurrence analysis, the parameter ``analysis_type`` has to be changed
from ``Classic`` to ``Cross``, when creating the ``Settings`` object. A
*CRQA* example is given below.

.. code:: python

   from pyrqa.analysis_type import Cross
   data_points_x = [0.9, 0.1, 0.2, 0.3, 0.5, 1.7, 0.4, 0.8, 1.5]
   time_series_x = TimeSeries(data_points_x,
                              embedding_dimension=2,
                              time_delay=1)
   data_points_y = [0.3, 1.3, 0.6, 0.2, 1.1, 1.9, 1.3, 0.4, 0.7, 0.9, 1.6]
   time_series_y = TimeSeries(data_points_y,
                              embedding_dimension=2,
                              time_delay=2)
   time_series = (time_series_x,
                  time_series_y)
   settings = Settings(time_series,
                       analysis_type=Cross,
                       neighbourhood=FixedRadius(0.73),
                       similarity_measure=EuclideanMetric,
                       theiler_corrector=0)
   computation = RQAComputation.create(settings,
                                       verbose=True)
   result = computation.run()
   result.min_diagonal_line_length = 2
   result.min_vertical_line_length = 2
   result.min_white_vertical_line_length = 2
   print(result)

The following output is expected.

::

   CRQA Result:
   ============

   Minimum diagonal line length (L_min): 2
   Minimum vertical line length (V_min): 2
   Minimum white vertical line length (W_min): 2

   Recurrence rate (RR): 0.319444
   Determinism (DET): 0.521739
   Average diagonal line length (L): 2.400000
   Longest diagonal line length (L_max): 3
   Divergence (DIV): 0.333333
   Entropy diagonal lines (L_entr): 0.673012
   Laminarity (LAM): 0.434783
   Trapping time (TT): 2.500000
   Longest vertical line length (V_max): 3
   Entropy vertical lines (V_entr): 0.693147
   Average white vertical line length (W): 3.500000
   Longest white vertical line length (W_max): 8
   Longest white vertical line length inverse (W_div): 0.125000
   Entropy white vertical lines (W_entr): 1.424130

   Ratio determinism / recurrence rate (DET/RR): 1.633270
   Ratio laminarity / determinism (LAM/DET): 0.833333

The corresponding cross recurrence plot is created likewise.

.. code:: python

   from pyrqa.computation import RPComputation
   from pyrqa.image_generator import ImageGenerator
   computation = RPComputation.create(settings)
   result = computation.run()
   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,
                                       'cross_recurrence_plot.png')

Neighbourhood Condition Selection
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

PyRQA currently supports the fixed radius as well as the radius corridor
neighbourhood condition. While the first refers to a single radius, the
latter requires the assignment of an inner and outer radius. The
specific condition is passed using the parameter ``neighbourhood`` to
the constructor of a ``Settings`` object. The creation of a fixed radius
and a radius corridor neighbourhood is presented below.

.. code:: python

   from pyrqa.neighbourhood import FixedRadius, RadiusCorridor
   fixed_radius = FixedRadius(radius=0.43)
   radius_corridor = RadiusCorridor(inner_radius=0.32, 
                                    outer_radius=0.86)          

Unthresholded Recurrence Plots
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

PyRQA allows to create unthresholded *RP*\ s and *CRP*\ s by selecting
the ``Unthresholded`` neighbourhood condition. This results in a
non-binary matrix, containing the mutual distances between the system
states, based on the similarity measure selected. PyRQA provides
functionality to normalize these distances to values between ``0`` and
``1``. Additionally, the normalized matrix can be represented as a
grayscale image. Darker shades of grey indicate smaller distances
whereas lighter shades of grey indicate larger distances. An example on
how to create an unthresholded cross recurrence plot is given below.

.. code:: python

   from pyrqa.neighbourhood import Unthresholded
   settings = Settings(time_series,
                       analysis_type=Cross,
                       neighbourhood=Unthresholded(),
                       similarity_measure=EuclideanMetric)
   computation = RPComputation.create(settings)
   result = computation.run()
   ImageGenerator.save_unthresholded_recurrence_plot(result.recurrence_matrix_reverse_normalized,
                                                     'unthresholded_cross_recurrence_plot.png')

Joint Recurrence Analysis
~~~~~~~~~~~~~~~~~~~~~~~~~

In addition to classic and cross recurrence analysis, PyRQA provides
functionality to conduct joint recurrence analysis. This includes in
particular joint recurrence quantification analysis (*JRQA*) as well as
joint recurrence plot (*JRP*). On an abstract level, a joint recurrence
plot is a combination of two individual plots, both having the same
extent regarding the *X* and *Y* axis. Regarding PyRQA, each of those
two plots may either be of the analysis type ``Classic`` or ``Cross``,
potentially having different characteristics regarding:

-  Time series data,
-  Embedding dimension,
-  Time delay,
-  Neighbourhood condition, and
-  Similarity measure.

In contrast, the same value for ``theiler_corrector`` is expected
regarding the quantitative analysis. Note that a joint recurrence plot
by definition relies on thresholded input plots, eliminating the
application of the ``Unthresholded`` neighbourhood condition.

The settings of the two individual plots are encapsulated in a
``JointSettings`` object. The quantification of joint recurrence plots
is based on the same measures as for recurrence plots and cross
recurrence plots. An example on how to conduct *JRQA* is given below.

.. code:: python

   from pyrqa.computation import JRQAComputation
   from pyrqa.metric import MaximumMetric, TaxicabMetric
   from pyrqa.settings import JointSettings
   data_points_1 = [1.0, 0.7, 0.5, 0.1, 1.7, 1.5, 1.2, 0.4, 0.6, 1.5, 0.8, 0.3]
   time_series_1 = TimeSeries(data_points,
                              embedding_dimension=3,
                              time_delay=1)
   settings_1 = Settings(time_series_1,
                         analysis_type=Classic,
                         neighbourhood=RadiusCorridor(inner_radius=0.14,
                                                      outer_radius=0.97),
                         similarity_measure=MaximumMetric,
                         theiler_corrector=1)
   data_points_2_x = [0.7, 0.1, 1.1, 1.4, 1.0, 0.5, 1.0, 1.9, 1.7, 0.9, 1.5, 0.6]
   time_series_2_x = TimeSeries(data_points_2_x,
                                embedding_dimension=2,
                                time_delay=1)
   data_points_2_y = [0.4, 0.7, 0.9, 0.3, 1.9, 1.3, 1.2, 0.2, 1.1, 0.6, 0.8, 0.1, 0.5]
   time_series_2_y = TimeSeries(data_points_2_y,
                                embedding_dimension=2,
                                time_delay=2)
   time_series_2 = (time_series_2_x,
                    time_series_2_y)
   settings_2 = Settings(time_series_2,
                         analysis_type=Cross,
                         neighbourhood=FixedRadius(0.83),
                         similarity_measure=TaxicabMetric,
                         theiler_corrector=1)
   joint_settings = JointSettings(settings_1,
                                  settings_2)
   computation = JRQAComputation.create(joint_settings,
                                        verbose=True)
   result = computation.run()
   result.min_diagonal_line_length = 2
   result.min_vertical_line_length = 1
   result.min_white_vertical_line_length = 2
   print(result)

The following output is expected.

::

   JRQA Result:
   ============

   Minimum diagonal line length (L_min): 2
   Minimum vertical line length (V_min): 1
   Minimum white vertical line length (W_min): 2

   Recurrence rate (RR): 0.157025
   Determinism (DET): 0.263158
   Average diagonal line length (L): 2.500000
   Longest diagonal line length (L_max): 3
   Divergence (DIV): 0.333333
   Entropy diagonal lines (L_entr): 0.693147
   Laminarity (LAM): 1.000000
   Trapping time (TT): 1.000000
   Longest vertical line length (V_max): 1
   Entropy vertical lines (V_entr): 0.000000
   Average white vertical line length (W): 3.960000
   Longest white vertical line length (W_max): 11
   Longest white vertical line length inverse (W_div): 0.090909
   Entropy white vertical lines (W_entr): 1.588760

   Ratio determinism / recurrence rate (DET/RR): 1.675900
   Ratio laminarity / determinism (LAM/DET): 3.800000

The corresponding joint recurrence plot is created likewise.

.. code:: python

   from pyrqa.computation import JRPComputation
   computation = JRPComputation.create(joint_settings)
   result = computation.run()
   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,
                                       'joint_recurrence_plot.png')

Time Series Embedding
~~~~~~~~~~~~~~~~~~~~~

The previous examples refer to one or multiple univariate time series as
input data. These ``TimeSeries`` are embedded in multi-dimensional space
using the time delay method. In addition, PyRQA allows to analyze time
series that are already embedded in multi-dimensional space. These
``EmbeddedSeries`` are represented as a list of *m*-dimensional arrays,
where *m* refers to the embedding dimension. An example of using a
pre-embedded series considering recurrence quantification analysis is
given below.

.. code:: python

   from pyrqa.time_series import EmbeddedSeries
   data_points = [[0.1, 1.3], [0.5, 0.7], [1.3, 0.8], [0.7, 1.4], [0.8, 1.6], [1.4, 1.2], [1.6, 0.4], [1.2, 1.1], [0.4, 0.8], [1.1, 0.2], [0.8, 1.3]]
   embedded_series = EmbeddedSeries(data_points)
   settings = Settings(embedded_series,
                       analysis_type=Classic,
                       neighbourhood=FixedRadius(0.65),
                       similarity_measure=EuclideanMetric,
                       theiler_corrector=1)
   computation = RQAComputation.create(settings,
                                       verbose=True)
   result = computation.run()
   result.min_diagonal_line_length = 2
   result.min_vertical_line_length = 2
   result.min_white_vertical_line_length = 2
   print(result)

The following output is expected.

::

   RQA Result:
   ===========

   Minimum diagonal line length (L_min): 2
   Minimum vertical line length (V_min): 2
   Minimum white vertical line length (W_min): 2

   Recurrence rate (RR): 0.371901
   Determinism (DET): 0.411765
   Average diagonal line length (L): 2.333333
   Longest diagonal line length (L_max): 3
   Divergence (DIV): 0.333333
   Entropy diagonal lines (L_entr): 0.636514
   Laminarity (LAM): 0.400000
   Trapping time (TT): 2.571429
   Longest vertical line length (V_max): 4
   Entropy vertical lines (V_entr): 0.955700
   Average white vertical line length (W): 2.538462
   Longest white vertical line length (W_max): 6
   Longest white vertical line length inverse (W_div): 0.166667
   Entropy white vertical lines (W_entr): 0.839796

   Ratio determinism / recurrence rate (DET/RR): 1.107190
   Ratio laminarity / determinism (LAM/DET): 0.971429

Note that this example corresponds to the introductory example, which
demonstrates the basic usage. It differs regarding the representation of
the multi-dimensional vectors. ``EmbeddedSeries`` objects can be used
while conducting recurrence analysis, cross recurrence analysis as well
as joint recurrence analysis. This refers to performing quantitative
analyses as well as creating recurrence plots. Regarding cross and joint
recurrence analysis ``TimeSeries`` objects may be combined with
``EmbeddedSeries`` objects.

Custom OpenCL Environment
~~~~~~~~~~~~~~~~~~~~~~~~~

The previous examples use the default OpenCL environment. A custom
environment can also be created via command line input. For this
purpose, the parameter ``command_line`` has to be set to ``True``, when
creating an ``OpenCL`` object.

.. code:: python

   from pyrqa.opencl import OpenCL
   opencl = OpenCL(command_line=True)

The OpenCL platform as well as the computing devices can also be
selected manually using their identifiers.

.. code:: python

   opencl = OpenCL(platform_id=0,
                   device_ids=(0,))

The ``OpenCL`` object generated is passed as a parameter while creating
a computation object.

.. code:: python

   computation = RPComputation.create(settings,
                                      verbose=True,
                                      opencl=opencl)

Floating Point Precision
~~~~~~~~~~~~~~~~~~~~~~~~

PyRQA allows to specify the precision of the time series data, which in
turn determines the precision of the computations conducted by the
OpenCL devices. Currently, the following precisions are supported:

-  Half precision (16-bit),
-  Single precision (32-bit), and
-  Double precision (64-bit).

By default, the single precision is applied. Note that not all
precisions may be supported by the OpenCL devices employed. Furthermore,
the selected precision influences the performance of the computations on
a particular device.

The precision is set by specifying the corresponding data type, short
``dtype``, of the time series data. The following example depicts the
usage of double precision floating point values.

.. code:: python

   import numpy as np
   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]
   time_series = TimeSeries(data_points,
                            embedding_dimension=2,
                            time_delay=2,
                            dtype=np.float64)

Performance Tuning
------------------

PyRQA offers the opportunity to reduce the runtime of analytical
computations using performance tuning. There is a distinction between
manual as well as automatic performance tuning. Both aspects are
highlighted in the following subsections. Note that every analytical
method implemented by PyRQA can be executed without conducting
performance tuning. Nonetheless, it may enable significant runtime
reductions regarding the analysis of very long time series.

Adaptive Implementation Selection
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Adaptive implementation selection allows to automatically select well
performing implementations regarding RQA and recurrence plot
computations provided by PyRQA. The approach dynamically adapts the
selection to the current computational scenario as well as the
properties of the OpenCL devices employed. The selection is performed
using one of multiple strategies, each referred to as ``selector``. They
rely on a set of customized implementation ``variants``, which may be
parameterized using a set of keyword arguments called
``variants_kwargs``. Note that the same selection strategies can be used
for *RQA* and *CRQA*, *RP* and *CRP*, *URP* and *UCRP* as well as *JRQA*
and *JRP* computations.

.. code:: python

   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_bit_no_recycling import ColumnNoOverlapMaterialisationBitNoRecycling
   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_bit_recycling import ColumnNoOverlapMaterialisationBitRecycling
   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_byte_no_recycling import ColumnNoOverlapMaterialisationByteNoRecycling
   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_byte_recycling import ColumnNoOverlapMaterialisationByteRecycling
   from pyrqa.variants.rqa.radius.column_no_overlap_no_materialisation import ColumnNoOverlapNoMaterialisation
   from pyrqa.variants.rqa.radius.column_overlap_materialisation_bit_no_recycling import ColumnOverlapMaterialisationBitNoRecycling
   from pyrqa.variants.rqa.radius.column_overlap_materialisation_bit_recycling import ColumnOverlapMaterialisationBitRecycling
   from pyrqa.variants.rqa.radius.column_overlap_materialisation_byte_no_recycling import ColumnOverlapMaterialisationByteNoRecycling
   from pyrqa.variants.rqa.radius.column_overlap_materialisation_byte_recycling import ColumnOverlapMaterialisationByteRecycling
   from pyrqa.variants.rqa.radius.column_overlap_no_materialisation import ColumnOverlapNoMaterialisation
   from pyrqa.selector import EpsilonGreedySelector
   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]
   time_series = TimeSeries(data_points,
                            embedding_dimension=2,
                            time_delay=2)
   settings = Settings(time_series,
                       analysis_type=Classic,
                       neighbourhood=FixedRadius(0.65),
                       similarity_measure=EuclideanMetric,
                       theiler_corrector=1)
   computation = RQAComputation.create(settings,
                                       selector=EpsilonGreedySelector(explore=10),
                                       variants=(ColumnNoOverlapMaterialisationBitNoRecycling,
                                                 ColumnNoOverlapMaterialisationBitRecycling,
                                                 ColumnNoOverlapMaterialisationByteNoRecycling,
                                                 ColumnNoOverlapMaterialisationByteRecycling,
                                                 ColumnNoOverlapNoMaterialisation,
                                                 ColumnOverlapMaterialisationBitNoRecycling,
                                                 ColumnOverlapMaterialisationBitRecycling,
                                                 ColumnOverlapMaterialisationByteNoRecycling,
                                                 ColumnOverlapMaterialisationByteRecycling,
                                                 ColumnOverlapNoMaterialisation))

Input Data Overlapping
~~~~~~~~~~~~~~~~~~~~~~

PyRQA comprises a set of ``variants`` for each analytical method, which
differ regarding certain implementation details. The specific
characteristics of each variant are reflected in its name. The naming
convention is applied accordingly to all analytical methods.

One criterion is the way how the input data is represented within the
memory of the computing devices. The multi-dimensional vectors may
either be stored column-wise or row-wise. Considering ``TimeSeries``
objects, to which the time delay method is applied, the columns may
additionally be stored in an overlapped fashion. This allows leveraging
data compression as well as potential runtime reductions. Using
``EmbeddedSeries`` objects as input data, the multi-dimensional vectors
may only be stored non-overlapping. As an example, variants referring to
the joint recurrence quantification analysis are mentioned below.

.. code:: python

   from pyrqa.variants.jrqa.radius.column_no_overlap_materialisation_byte_no_recycling import ColumnNoOverlapMaterialisationByteNoRecycling
   from pyrqa.variants.jrqa.radius.row_no_overlap_materialisation_byte_no_recycling import RowNoOverlapMaterialisationByteNoRecycling
   from pyrqa.variants.jrqa.radius.column_overlap_materialisation_byte_no_recycling import ColumnOverlapMaterialisationByteNoRecycling

Note that these variants have the same implementation characteristics,
except from the representation of the input data. Only those variants
that store the input data non-overlapping may be applied to
``EmbeddedSeries`` objects. By default, non-overlapping variants are
selected.

OpenCL Compiler Optimisations
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

OpenCL compiler optimisations aim at improving the performance of the
operations conducted by the computing devices. Regarding PyRQA, they are
disabled by default to ensure the comparability of the analytical
results. They can be enabled by assigning the value ``True`` to the
corresponding keyword argument ``optimisations_enabled``.

.. code:: python

   computation = RQAComputation.create(settings,
                                       selector=EpsilonGreedySelector(explore=10),
                                       variants=(ColumnNoOverlapMaterialisationBitNoRecycling,
                                                 ColumnNoOverlapMaterialisationBitRecycling,
                                                 ColumnNoOverlapMaterialisationByteNoRecycling,
                                                 ColumnNoOverlapMaterialisationByteRecycling,
                                                 ColumnNoOverlapNoMaterialisation,
                                                 ColumnOverlapMaterialisationBitNoRecycling,
                                                 ColumnOverlapMaterialisationBitRecycling,
                                                 ColumnOverlapMaterialisationByteNoRecycling,
                                                 ColumnOverlapMaterialisationByteRecycling,
                                                 ColumnOverlapNoMaterialisation),
                                       variants_kwargs={'optimisations_enabled': True})

Loop Unrolling
~~~~~~~~~~~~~~

Besides gaining drastic performance improvements using parallel
computing techniques, parts of the PyRQA kernel processing rely on
loops. OpenCL offers the opportunity to leverage additional performance
improvements by unrolling these loops. This is conducted by the OpenCL
compiler that is applied to the kernel functions. Note that loop
unrolling may not be supported by all OpenCL platforms.

PyRQA offers the opportunity to manually specify loop unrolling factors
that can be passed as a tuple when creating a selector. While creating a
computation object, each factor is combined with each implementation
variant. The resulting combinations are further used regarding adaptive
implementation selection. An example on how to specify loop unrolling
factors is given below.

.. code:: python

   from pyrqa.selector import EpsilonFirstSelector
   selector = EpsilonFirstSelector(explore=15,
                                   loop_unroll_factors=(1,2,4,8,16,32))

Submatrix Edge Length
~~~~~~~~~~~~~~~~~~~~~

During the analytical processing, PyRQA subdivides the recurrence matrix
into a number of submatrices. To get more information on the recurrence
matrix partitioning, please refer to the publications mentioned below.
The corresponding parameter ``edge_length`` can be set while creating a
computation object.

.. code:: python

   computation = RQAComputation.create(settings,
                                       edge_length=12800)

Note that the edge length has to be chosen such that a submatrix fits in
the memory of the OpenCL computing devices. Experiments have shown that
values between ``10000`` and ``20000`` are a reasonable choice.
Nonetheless, values outside of these boundaries may lead to lower
runtimes for specific computational scenarios. Furthermore, recurrence
matrices with a smaller extent than the value of ``edge_length`` are
processed without being partitioned.

Testing
-------

PyRQA provides a single-threaded baseline implementation for each
analytical method. These implementations do not use OpenCL
functionality. They serve as a ground truth regarding the analytical
computations. The basic tests for all supported analytical methods can
be executed cumulatively.

.. code:: bash

   python -m pyrqa.test

The complete set of tests can be executed by adding the option
``--all``.

.. code:: bash

   python -m pyrqa.test --all

Note that there might occur minor deviations regarding the analytical
results. These deviations may stem from varying precisions regarding the
computing devices employed.

Origin
------

The PyRQA package was initiated by computer scientists from the
Humboldt-Universität zu Berlin (https://www.hu-berlin.de) and the GFZ
German Research Centre for Geosciences (https://www.gfz-potsdam.de).

Acknowledgements
----------------

We would like to thank Norbert Marwan from the Potsdam Institute for
Climate Impact Research (https://www.pik-potsdam.de) for his continuous
support of the project. Please visit his website
http://recurrence-plot.tk/ for further information on recurrence
analysis. Initial research and development of PyRQA was funded by the
Deutsche Forschungsgemeinschaft (https://www.dfg.de/).

Publications
------------

The underlying computational approach of PyRQA is described in detail
within the following thesis, which is openly accessible at
https://edoc.hu-berlin.de/handle/18452/19518.

   Rawald, T. (2018): Scalable and Efficient Analysis of Large
   High-Dimensional Data Sets in the Context of Recurrence Analysis, PhD
   Thesis, Berlin : Humboldt-Universität zu Berlin, 299 p.

Selected aspects of the computational approach are presented within the
following publications.

   Rawald, T., Sips, M., Marwan, N., Dransch, D. (2014): Fast
   Computation of Recurrences in Long Time Series. - In: Marwan, N.,
   Riley, M., Guiliani, A., Webber, C. (Eds.), Translational
   Recurrences. From Mathematical Theory to Real-World Applications,
   (Springer Proceedings in Mathematics and Statistics ; 103), p. 17-29.

..

   Rawald, T., Sips, M., Marwan, N., Leser, U. (2015): Massively
   Parallel Analysis of Similarity Matrices on Heterogeneous Hardware. -
   In: Fischer, P. M., Alonso, G., Arenas, M., Geerts, F. (Eds.),
   Proceedings of the Workshops of the EDBT/ICDT 2015 Joint Conference
   (EDBT/ICDT), (CEUR Workshop Proceedings ; 1330), p. 56-62.

Development Status
------------------

The development status of the PyRQA package is considered as *Beta*.
Please send feature requests and bug reports to the email address that
is listed in the project metadata.

Release Notes
-------------

8.1.0
~~~~~

-  Addition of a JRQA variant referring to non-overlapping column-wise
   storage of input data.
-  Replaced all occurrences of ``numpy.float`` with ``numpy.float32``.

.. _section-1:

8.0.0
~~~~~

-  Usage of pre-embedded time series as input data.
-  Addition of variants referring to non-overlapping column-wise storage
   of input data.
-  Refactoring of the public API.
-  Updated documentation.

.. _section-2:

7.0.1
~~~~~

-  Modification of the build configuration.
-  Updated documentation.

.. _section-3:

7.0.0
~~~~~

-  Addition of two variants regarding joint recurrence quantification
   analysis computations.
-  Removal of obsolete source code.
-  Refactoring of the public API.
-  Updated documentation.

.. _section-4:

6.0.0
~~~~~

-  Addition of the joint recurrence quantification analysis (*JRQA*) and
   joint recurrence plot (*JRP*) computations.
-  Refactoring of the test implementation.
-  Refactoring of the public API.
-  Updated documentation.

.. _section-5:

5.1.0
~~~~~

-  Addition of the unthresholded recurrence plot (*URP*) and
   unthresholded cross recurrence plot (*UCRP*) computations.
-  Updated documentation.

.. _section-6:

5.0.0
~~~~~

-  Refactoring of the public API.
-  Updated documentation.

.. _section-7:

4.1.0
~~~~~

-  Usage of two different time delay values regarding the cross
   recurrence plot (*CRP*) and cross recurrence quantification analysis
   (*CRQA*).
-  Updated documentation.

.. _section-8:

4.0.0
~~~~~

-  Addition of the cross recurrence plot (*CRP*) and cross recurrence
   quantification analysis (*CRQA*) computations.
-  Addition of the radius corridor neighbourhood condition for
   determining state similarity.
-  Addition of an additional variant regarding recurrence plot
   computations.
-  Renaming of directories and classes referring to recurrence plot
   computations.
-  Removal of obsolete source code.
-  Updated documentation.

.. _section-9:

3.0.0
~~~~~

-  Source code cleanup.
-  Renaming of the implementation variants regarding RQA and recurrence
   plot processing.
-  Removal of the module ``file_reader.py``. Please refer for example to
   ``numpy.genfromtxt`` to read data from files (see
   https://docs.scipy.org/doc/numpy/reference/generated/numpy.genfromtxt.html).
-  Updated documentation.

.. _section-10:

2.0.1
~~~~~

-  Updated documentation.

.. _section-11:

2.0.0
~~~~~

-  Major refactoring.
-  Removal of operator and variant implementations that do not refer to
   OpenCL brute force computing.
-  Time series data may be represented using half, single and double
   precision floating point values, which is reflected in the
   computations on the OpenCL devices.
-  Several changes to the public API.

.. _section-12:

1.0.6
~~~~~

-  Changes to the public API have been made, e.g., to the definition of
   the settings. This leads to an increase in the major version number
   (see https://semver.org/).
-  Time series objects either consist of one or multiple series. The
   former requires to specify a value for the embedding delay as well as
   the time delay parameter.
-  Regarding the RQA computations, minimum line lengths are now
   specified on the result object. This allows to compute quantitative
   results using different lengths without having to inspect the matrix
   using the same parametrisation multiple times.
-  Modules for selecting well-performing implementations based on greedy
   selection strategies have been added. By default, the selection pool
   consists of a single pre-defined implementation.
-  Operators and implementation variants based on multidimensional
   search trees and grid data structures have been added.
-  The diagonal line based quantitative measures are modified regarding
   the semantics of the Theiler corrector.
-  The creation of the OpenCL environment now supports device fission.

.. _section-13:

0.1.0
~~~~~

-  Initial release.
            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "PyRQA",
    "maintainer": "",
    "docs_url": null,
    "requires_python": "",
    "maintainer_email": "",
    "keywords": "nonlinear,time series analysis,dynamical system,recurrence quantification analysis,RQA,cross recurrence quantification analysis,CRQA,joint recurrence quantification analysis,JRQA,recurrence plot,RP,cross recurrence plot,CRP,joint recurrence plot,JRP",
    "author": "Tobias Rawald",
    "author_email": "pyrqa@gmx.net",
    "download_url": "https://files.pythonhosted.org/packages/37/b1/7177332b933b8be56a8adb8828563fa49de81339b9f35660b37a2f338609/PyRQA-8.1.0.tar.gz",
    "platform": null,
    "description": "PyRQA\n=====\n\nHighlights\n----------\n\n-  Perform recurrence analysis on long time series in a time efficient\n   manner using the OpenCL framework.\n-  Conduct recurrence quantification analysis (*RQA*) and cross\n   recurrence quantification analysis (*CRQA*).\n-  Compute recurrence plots (*RP*) and cross recurrence plots (*CRP*).\n-  Compute unthresholded recurrence plots (*URP*) and unthresholded\n   cross recurrence plots (*UCRP*).\n-  Conduct joint recurrence quantification analysis (*JRQA*) and compute\n   joint recurrence plots (*JRP*).\n-  Employ the euclidean, maximum or taxicab metric for determining state\n   similarity.\n-  Choose the fixed radius or radius corridor neighbourhood condition.\n-  Use univariate time series or pre-embedded time series as input data.\n-  Select either the half, single or double floating point precision for\n   conducting the analytical computations.\n-  Leverage machine learning techniques that automatically choose the\n   fastest from a set of implementations.\n-  Apply the computing capabilities of GPUs, CPUs and other platforms\n   that support OpenCL.\n-  Use multiple computing devices of the same or different type in\n   parallel.\n\nTable of Contents\n-----------------\n\n1.  `General Information <#general-information>`__\n2.  `Recommended Citation <#recommended-citation>`__\n3.  `Installation <#installation>`__\n4.  `OpenCL Setup <#opencl-setup>`__\n5.  `Usage <#usage>`__\n6.  `Performance Tuning <#performance-tuning>`__\n7.  `Testing <#testing>`__\n8.  `Origin <#origin>`__\n9.  `Acknowledgements <#acknowledgements>`__\n10. `Publications <#publications>`__\n11. `Development Status <#development-status>`__\n12. `Release Notes <#release-notes>`__\n\nGeneral Information\n-------------------\n\nPyRQA is a tool to conduct recurrence analysis in a massively parallel\nmanner using the OpenCL framework. It is designed to efficiently process\ntime series consisting of hundreds of thousands of data points.\n\nPyRQA supports the computation of the following quantitative measures:\n\n-  Recurrence rate (*RR*)\n-  Determinism (*DET*)\n-  Average diagonal line length (*L*)\n-  Longest diagonal line length (*L_max*)\n-  Divergence (*DIV*)\n-  Entropy diagonal lines (*L_entr*)\n-  Laminarity (*LAM*)\n-  Trapping time (*TT*)\n-  Longest vertical line length (*V_max*)\n-  Entropy vertical lines (*V_entr*)\n-  Average white vertical line length (*W*)\n-  Longest white vertical line length (*W_max*)\n-  Longest white vertical line length divergence (*W_div*)\n-  Entropy white vertical lines (*W_entr*)\n\nPyRQA additionally allows to create the corresponding recurrence plot,\nwhich can be exported as an image file.\n\nRecommended Citation\n--------------------\n\nPlease acknowledge the use of PyRQA by citing the following publication.\n\n   Rawald, T., Sips, M., Marwan, N. (2017): PyRQA - Conducting\n   Recurrence Quantification Analysis on Very Long Time Series\n   Efficiently. - Computers and Geosciences, 104, pp.\u00a0101-108.\n\nInstallation\n------------\n\nPyRQA and all of its dependencies can be installed via the following\ncommand.\n\n.. code:: bash\n\n   pip install PyRQA\n\nOpenCL Setup\n------------\n\nThe analytical implementations provided by PyRQA rely on features that\nare part of *OpenCL 1.1*, which is a fairly mature standard and\nsupported by a large number of platforms. The OpenCL computing devices\nemployed need to support at least this version to being able to use\nPyRQA.\n\nIt may be required to install additional software, e.g., runtimes or\ndrivers, to execute PyRQA on computing devices such as GPUs and CPUs.\nReferences to vendor-specific information are presented below.\n\n*AMD*:\n\n-  https://www.amd.com/en/support\n-  https://github.com/RadeonOpenCompute/ROCm\n-  https://community.amd.com/community/devgurus/opencl\n-  https://www.amd.com/en/support/kb/release-notes/amdgpu-installation\n\n*ARM*:\n\n-  https://developer.arm.com/docs/100614/0312\n\n*Intel*:\n\n-  https://software.intel.com/en-us/articles/opencl-drivers\n-  https://software.intel.com/en-us/articles/sdk-for-opencl-gsg\n\n*NVIDIA*:\n\n-  https://developer.nvidia.com/opencl\n-  https://developer.nvidia.com/cuda-downloads\n\n*Vendor-independent*:\n\n-  http://portablecl.org\n\nUsage\n-----\n\nThe following subsections depict the usage of PyRQA. Each subsection\nfocuses on specific functionality that is provided by the package. The\npresentation of expected output shall ensure the reproducibility of the\nanalytical results.\n\nBasic Computations\n~~~~~~~~~~~~~~~~~~\n\nRQA computations are conducted as follows.\n\n.. code:: python\n\n   from pyrqa.time_series import TimeSeries\n   from pyrqa.settings import Settings\n   from pyrqa.analysis_type import Classic\n   from pyrqa.neighbourhood import FixedRadius\n   from pyrqa.metric import EuclideanMetric\n   from pyrqa.computation import RQAComputation\n   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]\n   time_series = TimeSeries(data_points,\n                            embedding_dimension=2,\n                            time_delay=2)\n   settings = Settings(time_series,\n                       analysis_type=Classic,\n                       neighbourhood=FixedRadius(0.65),\n                       similarity_measure=EuclideanMetric,\n                       theiler_corrector=1)\n   computation = RQAComputation.create(settings,\n                                       verbose=True)\n   result = computation.run()\n   result.min_diagonal_line_length = 2\n   result.min_vertical_line_length = 2\n   result.min_white_vertical_line_length = 2\n   print(result)\n\nThe following output is expected.\n\n::\n\n   RQA Result:\n   ===========\n\n   Minimum diagonal line length (L_min): 2\n   Minimum vertical line length (V_min): 2\n   Minimum white vertical line length (W_min): 2\n\n   Recurrence rate (RR): 0.371901\n   Determinism (DET): 0.411765\n   Average diagonal line length (L): 2.333333\n   Longest diagonal line length (L_max): 3\n   Divergence (DIV): 0.333333\n   Entropy diagonal lines (L_entr): 0.636514\n   Laminarity (LAM): 0.400000\n   Trapping time (TT): 2.571429\n   Longest vertical line length (V_max): 4\n   Entropy vertical lines (V_entr): 0.955700\n   Average white vertical line length (W): 2.538462\n   Longest white vertical line length (W_max): 6\n   Longest white vertical line length inverse (W_div): 0.166667\n   Entropy white vertical lines (W_entr): 0.839796\n\n   Ratio determinism / recurrence rate (DET/RR): 1.107190\n   Ratio laminarity / determinism (LAM/DET): 0.971429\n\nThe corresponding recurrence plot is created likewise. Note that the\nparameter ``theiler_corrector`` is ignored regarding the creation of the\nplot.\n\n.. code:: python\n\n   from pyrqa.computation import RPComputation\n   from pyrqa.image_generator import ImageGenerator\n   computation = RPComputation.create(settings)\n   result = computation.run()\n   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,\n                                       'recurrence_plot.png')\n\nCross Recurrence Analysis\n~~~~~~~~~~~~~~~~~~~~~~~~~\n\nPyRQA further offers the opportunity to conduct cross recurrence\nanalysis (*CRQA* and *CRP*), in addition to the classic recurrence\nanalysis (*RQA* and *RP*). For this purpose, two time series of\npotentially different length are provided as input. Note that the\ncorresponding computations require to set the same value regarding the\nembedding dimension. Two different time delay values may be used\nregarding the first and the second time series. To enable cross\nrecurrence analysis, the parameter ``analysis_type`` has to be changed\nfrom ``Classic`` to ``Cross``, when creating the ``Settings`` object. A\n*CRQA* example is given below.\n\n.. code:: python\n\n   from pyrqa.analysis_type import Cross\n   data_points_x = [0.9, 0.1, 0.2, 0.3, 0.5, 1.7, 0.4, 0.8, 1.5]\n   time_series_x = TimeSeries(data_points_x,\n                              embedding_dimension=2,\n                              time_delay=1)\n   data_points_y = [0.3, 1.3, 0.6, 0.2, 1.1, 1.9, 1.3, 0.4, 0.7, 0.9, 1.6]\n   time_series_y = TimeSeries(data_points_y,\n                              embedding_dimension=2,\n                              time_delay=2)\n   time_series = (time_series_x,\n                  time_series_y)\n   settings = Settings(time_series,\n                       analysis_type=Cross,\n                       neighbourhood=FixedRadius(0.73),\n                       similarity_measure=EuclideanMetric,\n                       theiler_corrector=0)\n   computation = RQAComputation.create(settings,\n                                       verbose=True)\n   result = computation.run()\n   result.min_diagonal_line_length = 2\n   result.min_vertical_line_length = 2\n   result.min_white_vertical_line_length = 2\n   print(result)\n\nThe following output is expected.\n\n::\n\n   CRQA Result:\n   ============\n\n   Minimum diagonal line length (L_min): 2\n   Minimum vertical line length (V_min): 2\n   Minimum white vertical line length (W_min): 2\n\n   Recurrence rate (RR): 0.319444\n   Determinism (DET): 0.521739\n   Average diagonal line length (L): 2.400000\n   Longest diagonal line length (L_max): 3\n   Divergence (DIV): 0.333333\n   Entropy diagonal lines (L_entr): 0.673012\n   Laminarity (LAM): 0.434783\n   Trapping time (TT): 2.500000\n   Longest vertical line length (V_max): 3\n   Entropy vertical lines (V_entr): 0.693147\n   Average white vertical line length (W): 3.500000\n   Longest white vertical line length (W_max): 8\n   Longest white vertical line length inverse (W_div): 0.125000\n   Entropy white vertical lines (W_entr): 1.424130\n\n   Ratio determinism / recurrence rate (DET/RR): 1.633270\n   Ratio laminarity / determinism (LAM/DET): 0.833333\n\nThe corresponding cross recurrence plot is created likewise.\n\n.. code:: python\n\n   from pyrqa.computation import RPComputation\n   from pyrqa.image_generator import ImageGenerator\n   computation = RPComputation.create(settings)\n   result = computation.run()\n   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,\n                                       'cross_recurrence_plot.png')\n\nNeighbourhood Condition Selection\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n\nPyRQA currently supports the fixed radius as well as the radius corridor\nneighbourhood condition. While the first refers to a single radius, the\nlatter requires the assignment of an inner and outer radius. The\nspecific condition is passed using the parameter ``neighbourhood`` to\nthe constructor of a ``Settings`` object. The creation of a fixed radius\nand a radius corridor neighbourhood is presented below.\n\n.. code:: python\n\n   from pyrqa.neighbourhood import FixedRadius, RadiusCorridor\n   fixed_radius = FixedRadius(radius=0.43)\n   radius_corridor = RadiusCorridor(inner_radius=0.32, \n                                    outer_radius=0.86)          \n\nUnthresholded Recurrence Plots\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n\nPyRQA allows to create unthresholded *RP*\\ s and *CRP*\\ s by selecting\nthe ``Unthresholded`` neighbourhood condition. This results in a\nnon-binary matrix, containing the mutual distances between the system\nstates, based on the similarity measure selected. PyRQA provides\nfunctionality to normalize these distances to values between ``0`` and\n``1``. Additionally, the normalized matrix can be represented as a\ngrayscale image. Darker shades of grey indicate smaller distances\nwhereas lighter shades of grey indicate larger distances. An example on\nhow to create an unthresholded cross recurrence plot is given below.\n\n.. code:: python\n\n   from pyrqa.neighbourhood import Unthresholded\n   settings = Settings(time_series,\n                       analysis_type=Cross,\n                       neighbourhood=Unthresholded(),\n                       similarity_measure=EuclideanMetric)\n   computation = RPComputation.create(settings)\n   result = computation.run()\n   ImageGenerator.save_unthresholded_recurrence_plot(result.recurrence_matrix_reverse_normalized,\n                                                     'unthresholded_cross_recurrence_plot.png')\n\nJoint Recurrence Analysis\n~~~~~~~~~~~~~~~~~~~~~~~~~\n\nIn addition to classic and cross recurrence analysis, PyRQA provides\nfunctionality to conduct joint recurrence analysis. This includes in\nparticular joint recurrence quantification analysis (*JRQA*) as well as\njoint recurrence plot (*JRP*). On an abstract level, a joint recurrence\nplot is a combination of two individual plots, both having the same\nextent regarding the *X* and *Y* axis. Regarding PyRQA, each of those\ntwo plots may either be of the analysis type ``Classic`` or ``Cross``,\npotentially having different characteristics regarding:\n\n-  Time series data,\n-  Embedding dimension,\n-  Time delay,\n-  Neighbourhood condition, and\n-  Similarity measure.\n\nIn contrast, the same value for ``theiler_corrector`` is expected\nregarding the quantitative analysis. Note that a joint recurrence plot\nby definition relies on thresholded input plots, eliminating the\napplication of the ``Unthresholded`` neighbourhood condition.\n\nThe settings of the two individual plots are encapsulated in a\n``JointSettings`` object. The quantification of joint recurrence plots\nis based on the same measures as for recurrence plots and cross\nrecurrence plots. An example on how to conduct *JRQA* is given below.\n\n.. code:: python\n\n   from pyrqa.computation import JRQAComputation\n   from pyrqa.metric import MaximumMetric, TaxicabMetric\n   from pyrqa.settings import JointSettings\n   data_points_1 = [1.0, 0.7, 0.5, 0.1, 1.7, 1.5, 1.2, 0.4, 0.6, 1.5, 0.8, 0.3]\n   time_series_1 = TimeSeries(data_points,\n                              embedding_dimension=3,\n                              time_delay=1)\n   settings_1 = Settings(time_series_1,\n                         analysis_type=Classic,\n                         neighbourhood=RadiusCorridor(inner_radius=0.14,\n                                                      outer_radius=0.97),\n                         similarity_measure=MaximumMetric,\n                         theiler_corrector=1)\n   data_points_2_x = [0.7, 0.1, 1.1, 1.4, 1.0, 0.5, 1.0, 1.9, 1.7, 0.9, 1.5, 0.6]\n   time_series_2_x = TimeSeries(data_points_2_x,\n                                embedding_dimension=2,\n                                time_delay=1)\n   data_points_2_y = [0.4, 0.7, 0.9, 0.3, 1.9, 1.3, 1.2, 0.2, 1.1, 0.6, 0.8, 0.1, 0.5]\n   time_series_2_y = TimeSeries(data_points_2_y,\n                                embedding_dimension=2,\n                                time_delay=2)\n   time_series_2 = (time_series_2_x,\n                    time_series_2_y)\n   settings_2 = Settings(time_series_2,\n                         analysis_type=Cross,\n                         neighbourhood=FixedRadius(0.83),\n                         similarity_measure=TaxicabMetric,\n                         theiler_corrector=1)\n   joint_settings = JointSettings(settings_1,\n                                  settings_2)\n   computation = JRQAComputation.create(joint_settings,\n                                        verbose=True)\n   result = computation.run()\n   result.min_diagonal_line_length = 2\n   result.min_vertical_line_length = 1\n   result.min_white_vertical_line_length = 2\n   print(result)\n\nThe following output is expected.\n\n::\n\n   JRQA Result:\n   ============\n\n   Minimum diagonal line length (L_min): 2\n   Minimum vertical line length (V_min): 1\n   Minimum white vertical line length (W_min): 2\n\n   Recurrence rate (RR): 0.157025\n   Determinism (DET): 0.263158\n   Average diagonal line length (L): 2.500000\n   Longest diagonal line length (L_max): 3\n   Divergence (DIV): 0.333333\n   Entropy diagonal lines (L_entr): 0.693147\n   Laminarity (LAM): 1.000000\n   Trapping time (TT): 1.000000\n   Longest vertical line length (V_max): 1\n   Entropy vertical lines (V_entr): 0.000000\n   Average white vertical line length (W): 3.960000\n   Longest white vertical line length (W_max): 11\n   Longest white vertical line length inverse (W_div): 0.090909\n   Entropy white vertical lines (W_entr): 1.588760\n\n   Ratio determinism / recurrence rate (DET/RR): 1.675900\n   Ratio laminarity / determinism (LAM/DET): 3.800000\n\nThe corresponding joint recurrence plot is created likewise.\n\n.. code:: python\n\n   from pyrqa.computation import JRPComputation\n   computation = JRPComputation.create(joint_settings)\n   result = computation.run()\n   ImageGenerator.save_recurrence_plot(result.recurrence_matrix_reverse,\n                                       'joint_recurrence_plot.png')\n\nTime Series Embedding\n~~~~~~~~~~~~~~~~~~~~~\n\nThe previous examples refer to one or multiple univariate time series as\ninput data. These ``TimeSeries`` are embedded in multi-dimensional space\nusing the time delay method. In addition, PyRQA allows to analyze time\nseries that are already embedded in multi-dimensional space. These\n``EmbeddedSeries`` are represented as a list of *m*-dimensional arrays,\nwhere *m* refers to the embedding dimension. An example of using a\npre-embedded series considering recurrence quantification analysis is\ngiven below.\n\n.. code:: python\n\n   from pyrqa.time_series import EmbeddedSeries\n   data_points = [[0.1, 1.3], [0.5, 0.7], [1.3, 0.8], [0.7, 1.4], [0.8, 1.6], [1.4, 1.2], [1.6, 0.4], [1.2, 1.1], [0.4, 0.8], [1.1, 0.2], [0.8, 1.3]]\n   embedded_series = EmbeddedSeries(data_points)\n   settings = Settings(embedded_series,\n                       analysis_type=Classic,\n                       neighbourhood=FixedRadius(0.65),\n                       similarity_measure=EuclideanMetric,\n                       theiler_corrector=1)\n   computation = RQAComputation.create(settings,\n                                       verbose=True)\n   result = computation.run()\n   result.min_diagonal_line_length = 2\n   result.min_vertical_line_length = 2\n   result.min_white_vertical_line_length = 2\n   print(result)\n\nThe following output is expected.\n\n::\n\n   RQA Result:\n   ===========\n\n   Minimum diagonal line length (L_min): 2\n   Minimum vertical line length (V_min): 2\n   Minimum white vertical line length (W_min): 2\n\n   Recurrence rate (RR): 0.371901\n   Determinism (DET): 0.411765\n   Average diagonal line length (L): 2.333333\n   Longest diagonal line length (L_max): 3\n   Divergence (DIV): 0.333333\n   Entropy diagonal lines (L_entr): 0.636514\n   Laminarity (LAM): 0.400000\n   Trapping time (TT): 2.571429\n   Longest vertical line length (V_max): 4\n   Entropy vertical lines (V_entr): 0.955700\n   Average white vertical line length (W): 2.538462\n   Longest white vertical line length (W_max): 6\n   Longest white vertical line length inverse (W_div): 0.166667\n   Entropy white vertical lines (W_entr): 0.839796\n\n   Ratio determinism / recurrence rate (DET/RR): 1.107190\n   Ratio laminarity / determinism (LAM/DET): 0.971429\n\nNote that this example corresponds to the introductory example, which\ndemonstrates the basic usage. It differs regarding the representation of\nthe multi-dimensional vectors. ``EmbeddedSeries`` objects can be used\nwhile conducting recurrence analysis, cross recurrence analysis as well\nas joint recurrence analysis. This refers to performing quantitative\nanalyses as well as creating recurrence plots. Regarding cross and joint\nrecurrence analysis ``TimeSeries`` objects may be combined with\n``EmbeddedSeries`` objects.\n\nCustom OpenCL Environment\n~~~~~~~~~~~~~~~~~~~~~~~~~\n\nThe previous examples use the default OpenCL environment. A custom\nenvironment can also be created via command line input. For this\npurpose, the parameter ``command_line`` has to be set to ``True``, when\ncreating an ``OpenCL`` object.\n\n.. code:: python\n\n   from pyrqa.opencl import OpenCL\n   opencl = OpenCL(command_line=True)\n\nThe OpenCL platform as well as the computing devices can also be\nselected manually using their identifiers.\n\n.. code:: python\n\n   opencl = OpenCL(platform_id=0,\n                   device_ids=(0,))\n\nThe ``OpenCL`` object generated is passed as a parameter while creating\na computation object.\n\n.. code:: python\n\n   computation = RPComputation.create(settings,\n                                      verbose=True,\n                                      opencl=opencl)\n\nFloating Point Precision\n~~~~~~~~~~~~~~~~~~~~~~~~\n\nPyRQA allows to specify the precision of the time series data, which in\nturn determines the precision of the computations conducted by the\nOpenCL devices. Currently, the following precisions are supported:\n\n-  Half precision (16-bit),\n-  Single precision (32-bit), and\n-  Double precision (64-bit).\n\nBy default, the single precision is applied. Note that not all\nprecisions may be supported by the OpenCL devices employed. Furthermore,\nthe selected precision influences the performance of the computations on\na particular device.\n\nThe precision is set by specifying the corresponding data type, short\n``dtype``, of the time series data. The following example depicts the\nusage of double precision floating point values.\n\n.. code:: python\n\n   import numpy as np\n   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]\n   time_series = TimeSeries(data_points,\n                            embedding_dimension=2,\n                            time_delay=2,\n                            dtype=np.float64)\n\nPerformance Tuning\n------------------\n\nPyRQA offers the opportunity to reduce the runtime of analytical\ncomputations using performance tuning. There is a distinction between\nmanual as well as automatic performance tuning. Both aspects are\nhighlighted in the following subsections. Note that every analytical\nmethod implemented by PyRQA can be executed without conducting\nperformance tuning. Nonetheless, it may enable significant runtime\nreductions regarding the analysis of very long time series.\n\nAdaptive Implementation Selection\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n\nAdaptive implementation selection allows to automatically select well\nperforming implementations regarding RQA and recurrence plot\ncomputations provided by PyRQA. The approach dynamically adapts the\nselection to the current computational scenario as well as the\nproperties of the OpenCL devices employed. The selection is performed\nusing one of multiple strategies, each referred to as ``selector``. They\nrely on a set of customized implementation ``variants``, which may be\nparameterized using a set of keyword arguments called\n``variants_kwargs``. Note that the same selection strategies can be used\nfor *RQA* and *CRQA*, *RP* and *CRP*, *URP* and *UCRP* as well as *JRQA*\nand *JRP* computations.\n\n.. code:: python\n\n   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_bit_no_recycling import ColumnNoOverlapMaterialisationBitNoRecycling\n   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_bit_recycling import ColumnNoOverlapMaterialisationBitRecycling\n   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_byte_no_recycling import ColumnNoOverlapMaterialisationByteNoRecycling\n   from pyrqa.variants.rqa.radius.column_no_overlap_materialisation_byte_recycling import ColumnNoOverlapMaterialisationByteRecycling\n   from pyrqa.variants.rqa.radius.column_no_overlap_no_materialisation import ColumnNoOverlapNoMaterialisation\n   from pyrqa.variants.rqa.radius.column_overlap_materialisation_bit_no_recycling import ColumnOverlapMaterialisationBitNoRecycling\n   from pyrqa.variants.rqa.radius.column_overlap_materialisation_bit_recycling import ColumnOverlapMaterialisationBitRecycling\n   from pyrqa.variants.rqa.radius.column_overlap_materialisation_byte_no_recycling import ColumnOverlapMaterialisationByteNoRecycling\n   from pyrqa.variants.rqa.radius.column_overlap_materialisation_byte_recycling import ColumnOverlapMaterialisationByteRecycling\n   from pyrqa.variants.rqa.radius.column_overlap_no_materialisation import ColumnOverlapNoMaterialisation\n   from pyrqa.selector import EpsilonGreedySelector\n   data_points = [0.1, 0.5, 1.3, 0.7, 0.8, 1.4, 1.6, 1.2, 0.4, 1.1, 0.8, 0.2, 1.3]\n   time_series = TimeSeries(data_points,\n                            embedding_dimension=2,\n                            time_delay=2)\n   settings = Settings(time_series,\n                       analysis_type=Classic,\n                       neighbourhood=FixedRadius(0.65),\n                       similarity_measure=EuclideanMetric,\n                       theiler_corrector=1)\n   computation = RQAComputation.create(settings,\n                                       selector=EpsilonGreedySelector(explore=10),\n                                       variants=(ColumnNoOverlapMaterialisationBitNoRecycling,\n                                                 ColumnNoOverlapMaterialisationBitRecycling,\n                                                 ColumnNoOverlapMaterialisationByteNoRecycling,\n                                                 ColumnNoOverlapMaterialisationByteRecycling,\n                                                 ColumnNoOverlapNoMaterialisation,\n                                                 ColumnOverlapMaterialisationBitNoRecycling,\n                                                 ColumnOverlapMaterialisationBitRecycling,\n                                                 ColumnOverlapMaterialisationByteNoRecycling,\n                                                 ColumnOverlapMaterialisationByteRecycling,\n                                                 ColumnOverlapNoMaterialisation))\n\nInput Data Overlapping\n~~~~~~~~~~~~~~~~~~~~~~\n\nPyRQA comprises a set of ``variants`` for each analytical method, which\ndiffer regarding certain implementation details. The specific\ncharacteristics of each variant are reflected in its name. The naming\nconvention is applied accordingly to all analytical methods.\n\nOne criterion is the way how the input data is represented within the\nmemory of the computing devices. The multi-dimensional vectors may\neither be stored column-wise or row-wise. Considering ``TimeSeries``\nobjects, to which the time delay method is applied, the columns may\nadditionally be stored in an overlapped fashion. This allows leveraging\ndata compression as well as potential runtime reductions. Using\n``EmbeddedSeries`` objects as input data, the multi-dimensional vectors\nmay only be stored non-overlapping. As an example, variants referring to\nthe joint recurrence quantification analysis are mentioned below.\n\n.. code:: python\n\n   from pyrqa.variants.jrqa.radius.column_no_overlap_materialisation_byte_no_recycling import ColumnNoOverlapMaterialisationByteNoRecycling\n   from pyrqa.variants.jrqa.radius.row_no_overlap_materialisation_byte_no_recycling import RowNoOverlapMaterialisationByteNoRecycling\n   from pyrqa.variants.jrqa.radius.column_overlap_materialisation_byte_no_recycling import ColumnOverlapMaterialisationByteNoRecycling\n\nNote that these variants have the same implementation characteristics,\nexcept from the representation of the input data. Only those variants\nthat store the input data non-overlapping may be applied to\n``EmbeddedSeries`` objects. By default, non-overlapping variants are\nselected.\n\nOpenCL Compiler Optimisations\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n\nOpenCL compiler optimisations aim at improving the performance of the\noperations conducted by the computing devices. Regarding PyRQA, they are\ndisabled by default to ensure the comparability of the analytical\nresults. They can be enabled by assigning the value ``True`` to the\ncorresponding keyword argument ``optimisations_enabled``.\n\n.. code:: python\n\n   computation = RQAComputation.create(settings,\n                                       selector=EpsilonGreedySelector(explore=10),\n                                       variants=(ColumnNoOverlapMaterialisationBitNoRecycling,\n                                                 ColumnNoOverlapMaterialisationBitRecycling,\n                                                 ColumnNoOverlapMaterialisationByteNoRecycling,\n                                                 ColumnNoOverlapMaterialisationByteRecycling,\n                                                 ColumnNoOverlapNoMaterialisation,\n                                                 ColumnOverlapMaterialisationBitNoRecycling,\n                                                 ColumnOverlapMaterialisationBitRecycling,\n                                                 ColumnOverlapMaterialisationByteNoRecycling,\n                                                 ColumnOverlapMaterialisationByteRecycling,\n                                                 ColumnOverlapNoMaterialisation),\n                                       variants_kwargs={'optimisations_enabled': True})\n\nLoop Unrolling\n~~~~~~~~~~~~~~\n\nBesides gaining drastic performance improvements using parallel\ncomputing techniques, parts of the PyRQA kernel processing rely on\nloops. OpenCL offers the opportunity to leverage additional performance\nimprovements by unrolling these loops. This is conducted by the OpenCL\ncompiler that is applied to the kernel functions. Note that loop\nunrolling may not be supported by all OpenCL platforms.\n\nPyRQA offers the opportunity to manually specify loop unrolling factors\nthat can be passed as a tuple when creating a selector. While creating a\ncomputation object, each factor is combined with each implementation\nvariant. The resulting combinations are further used regarding adaptive\nimplementation selection. An example on how to specify loop unrolling\nfactors is given below.\n\n.. code:: python\n\n   from pyrqa.selector import EpsilonFirstSelector\n   selector = EpsilonFirstSelector(explore=15,\n                                   loop_unroll_factors=(1,2,4,8,16,32))\n\nSubmatrix Edge Length\n~~~~~~~~~~~~~~~~~~~~~\n\nDuring the analytical processing, PyRQA subdivides the recurrence matrix\ninto a number of submatrices. To get more information on the recurrence\nmatrix partitioning, please refer to the publications mentioned below.\nThe corresponding parameter ``edge_length`` can be set while creating a\ncomputation object.\n\n.. code:: python\n\n   computation = RQAComputation.create(settings,\n                                       edge_length=12800)\n\nNote that the edge length has to be chosen such that a submatrix fits in\nthe memory of the OpenCL computing devices. Experiments have shown that\nvalues between ``10000`` and ``20000`` are a reasonable choice.\nNonetheless, values outside of these boundaries may lead to lower\nruntimes for specific computational scenarios. Furthermore, recurrence\nmatrices with a smaller extent than the value of ``edge_length`` are\nprocessed without being partitioned.\n\nTesting\n-------\n\nPyRQA provides a single-threaded baseline implementation for each\nanalytical method. These implementations do not use OpenCL\nfunctionality. They serve as a ground truth regarding the analytical\ncomputations. The basic tests for all supported analytical methods can\nbe executed cumulatively.\n\n.. code:: bash\n\n   python -m pyrqa.test\n\nThe complete set of tests can be executed by adding the option\n``--all``.\n\n.. code:: bash\n\n   python -m pyrqa.test --all\n\nNote that there might occur minor deviations regarding the analytical\nresults. These deviations may stem from varying precisions regarding the\ncomputing devices employed.\n\nOrigin\n------\n\nThe PyRQA package was initiated by computer scientists from the\nHumboldt-Universit\u00e4t zu Berlin (https://www.hu-berlin.de) and the GFZ\nGerman Research Centre for Geosciences (https://www.gfz-potsdam.de).\n\nAcknowledgements\n----------------\n\nWe would like to thank Norbert Marwan from the Potsdam Institute for\nClimate Impact Research (https://www.pik-potsdam.de) for his continuous\nsupport of the project. Please visit his website\nhttp://recurrence-plot.tk/ for further information on recurrence\nanalysis. Initial research and development of PyRQA was funded by the\nDeutsche Forschungsgemeinschaft (https://www.dfg.de/).\n\nPublications\n------------\n\nThe underlying computational approach of PyRQA is described in detail\nwithin the following thesis, which is openly accessible at\nhttps://edoc.hu-berlin.de/handle/18452/19518.\n\n   Rawald, T. (2018): Scalable and Efficient Analysis of Large\n   High-Dimensional Data Sets in the Context of Recurrence Analysis, PhD\n   Thesis, Berlin : Humboldt-Universit\u00e4t zu Berlin, 299 p.\n\nSelected aspects of the computational approach are presented within the\nfollowing publications.\n\n   Rawald, T., Sips, M., Marwan, N., Dransch, D. (2014): Fast\n   Computation of Recurrences in Long Time Series. - In: Marwan, N.,\n   Riley, M., Guiliani, A., Webber, C. (Eds.), Translational\n   Recurrences. From Mathematical Theory to Real-World Applications,\n   (Springer Proceedings in Mathematics and Statistics ; 103), p.\u00a017-29.\n\n..\n\n   Rawald, T., Sips, M., Marwan, N., Leser, U. (2015): Massively\n   Parallel Analysis of Similarity Matrices on Heterogeneous Hardware. -\n   In: Fischer, P. M., Alonso, G., Arenas, M., Geerts, F. (Eds.),\n   Proceedings of the Workshops of the EDBT/ICDT 2015 Joint Conference\n   (EDBT/ICDT), (CEUR Workshop Proceedings ; 1330), p.\u00a056-62.\n\nDevelopment Status\n------------------\n\nThe development status of the PyRQA package is considered as *Beta*.\nPlease send feature requests and bug reports to the email address that\nis listed in the project metadata.\n\nRelease Notes\n-------------\n\n8.1.0\n~~~~~\n\n-  Addition of a JRQA variant referring to non-overlapping column-wise\n   storage of input data.\n-  Replaced all occurrences of ``numpy.float`` with ``numpy.float32``.\n\n.. _section-1:\n\n8.0.0\n~~~~~\n\n-  Usage of pre-embedded time series as input data.\n-  Addition of variants referring to non-overlapping column-wise storage\n   of input data.\n-  Refactoring of the public API.\n-  Updated documentation.\n\n.. _section-2:\n\n7.0.1\n~~~~~\n\n-  Modification of the build configuration.\n-  Updated documentation.\n\n.. _section-3:\n\n7.0.0\n~~~~~\n\n-  Addition of two variants regarding joint recurrence quantification\n   analysis computations.\n-  Removal of obsolete source code.\n-  Refactoring of the public API.\n-  Updated documentation.\n\n.. _section-4:\n\n6.0.0\n~~~~~\n\n-  Addition of the joint recurrence quantification analysis (*JRQA*) and\n   joint recurrence plot (*JRP*) computations.\n-  Refactoring of the test implementation.\n-  Refactoring of the public API.\n-  Updated documentation.\n\n.. _section-5:\n\n5.1.0\n~~~~~\n\n-  Addition of the unthresholded recurrence plot (*URP*) and\n   unthresholded cross recurrence plot (*UCRP*) computations.\n-  Updated documentation.\n\n.. _section-6:\n\n5.0.0\n~~~~~\n\n-  Refactoring of the public API.\n-  Updated documentation.\n\n.. _section-7:\n\n4.1.0\n~~~~~\n\n-  Usage of two different time delay values regarding the cross\n   recurrence plot (*CRP*) and cross recurrence quantification analysis\n   (*CRQA*).\n-  Updated documentation.\n\n.. _section-8:\n\n4.0.0\n~~~~~\n\n-  Addition of the cross recurrence plot (*CRP*) and cross recurrence\n   quantification analysis (*CRQA*) computations.\n-  Addition of the radius corridor neighbourhood condition for\n   determining state similarity.\n-  Addition of an additional variant regarding recurrence plot\n   computations.\n-  Renaming of directories and classes referring to recurrence plot\n   computations.\n-  Removal of obsolete source code.\n-  Updated documentation.\n\n.. _section-9:\n\n3.0.0\n~~~~~\n\n-  Source code cleanup.\n-  Renaming of the implementation variants regarding RQA and recurrence\n   plot processing.\n-  Removal of the module ``file_reader.py``. Please refer for example to\n   ``numpy.genfromtxt`` to read data from files (see\n   https://docs.scipy.org/doc/numpy/reference/generated/numpy.genfromtxt.html).\n-  Updated documentation.\n\n.. _section-10:\n\n2.0.1\n~~~~~\n\n-  Updated documentation.\n\n.. _section-11:\n\n2.0.0\n~~~~~\n\n-  Major refactoring.\n-  Removal of operator and variant implementations that do not refer to\n   OpenCL brute force computing.\n-  Time series data may be represented using half, single and double\n   precision floating point values, which is reflected in the\n   computations on the OpenCL devices.\n-  Several changes to the public API.\n\n.. _section-12:\n\n1.0.6\n~~~~~\n\n-  Changes to the public API have been made, e.g., to the definition of\n   the settings. This leads to an increase in the major version number\n   (see https://semver.org/).\n-  Time series objects either consist of one or multiple series. The\n   former requires to specify a value for the embedding delay as well as\n   the time delay parameter.\n-  Regarding the RQA computations, minimum line lengths are now\n   specified on the result object. This allows to compute quantitative\n   results using different lengths without having to inspect the matrix\n   using the same parametrisation multiple times.\n-  Modules for selecting well-performing implementations based on greedy\n   selection strategies have been added. By default, the selection pool\n   consists of a single pre-defined implementation.\n-  Operators and implementation variants based on multidimensional\n   search trees and grid data structures have been added.\n-  The diagonal line based quantitative measures are modified regarding\n   the semantics of the Theiler corrector.\n-  The creation of the OpenCL environment now supports device fission.\n\n.. _section-13:\n\n0.1.0\n~~~~~\n\n-  Initial release.",
    "bugtrack_url": null,
    "license": "Apache License 2.0",
    "summary": "Recurrence analysis in a massively parallel manner using the OpenCL framework.",
    "version": "8.1.0",
    "project_urls": null,
    "split_keywords": [
        "nonlinear",
        "time series analysis",
        "dynamical system",
        "recurrence quantification analysis",
        "rqa",
        "cross recurrence quantification analysis",
        "crqa",
        "joint recurrence quantification analysis",
        "jrqa",
        "recurrence plot",
        "rp",
        "cross recurrence plot",
        "crp",
        "joint recurrence plot",
        "jrp"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "37b17177332b933b8be56a8adb8828563fa49de81339b9f35660b37a2f338609",
                "md5": "0027929233fc497827221bb2a8c9a923",
                "sha256": "86881e2bf60e086fc963af525956978655b896233b9fd3b5820cceeb4cd9952e"
            },
            "downloads": -1,
            "filename": "PyRQA-8.1.0.tar.gz",
            "has_sig": false,
            "md5_digest": "0027929233fc497827221bb2a8c9a923",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 97712,
            "upload_time": "2024-02-17T20:26:07",
            "upload_time_iso_8601": "2024-02-17T20:26:07.215953Z",
            "url": "https://files.pythonhosted.org/packages/37/b1/7177332b933b8be56a8adb8828563fa49de81339b9f35660b37a2f338609/PyRQA-8.1.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-02-17 20:26:07",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "pyrqa"
}
        
Elapsed time: 0.62808s