pyxtension


Namepyxtension JSON
Version 1.16.11 PyPI version JSON
download
home_pagehttps://github.com/asuiu/pyxtension
SummaryExtension library for Python
upload_time2024-08-26 12:49:22
maintainerNone
docs_urlhttps://pythonhosted.org/pyxtension/
authorAndrei Suiu
requires_python<4,>=3.8
licenseNone
keywords
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI
coveralls test coverage No coveralls.
            # pyxtension
[![build Status](https://travis-ci.org/asuiu/pyxtension.svg?branch=master)](https://travis-ci.org/asuiu/pyxtension)
[![Coverage Status](https://coveralls.io/repos/asuiu/pyxtension/badge.svg?branch=master&service=github)](https://coveralls.io/github/asuiu/pyxtension?branch=master)

[pyxtension](https://github.com/asuiu/pyxtension) is a pure Python MIT-licensed library that includes Scala-like streams (using [Fluent Interface pattern](https://en.wikipedia.org/wiki/Fluent_interface)), Json with attribute access syntax, and other common-use stuff.

###### Note:

**Drop support & maintenance for Python 2.x version, due to [Py2 death](https://www.python.org/doc/sunset-python-2/).**
 
 Although Py2 version will remain in the repository, I won't update PyPi package, so the last Py2 version of the `pyxtension` available at [PyPi](https://pypi.org/project/pyxtension/) will remain [`1.12.7`](https://pypi.org/project/pyxtension/1.12.7/)

Starting with [`1.13.0`](https://pypi.org/project/pyxtension/1.13.0/) I've migrated the packaging & distributing method to [Wheel](https://pythonwheels.com/).
      
## Installation
```
pip install pyxtension
```
or from Github:
```
git clone https://github.com/asuiu/pyxtension.git
cd pyxtension
python setup.py install
```
or
```
git submodule add https://github.com/asuiu/pyxtension.git
```

## Modules overview
### Json.py
##### Json
A `dict` subclass to represent a Json object. You should be able to use this
absolutely anywhere you can use a `dict`. While this is probably the class you
want to use, there are a few caveats that follow from this being a `dict` under
the hood.

**Never again will you have to write code like this**:
```python
body = {
    'query': {
        'filtered': {
            'query': {
                'match': {'description': 'addictive'}
            },
            'filter': {
                'term': {'created_by': 'ASU'}
            }
        }
    }
}
```

From now on, you may simply write the following three lines:
```python
body = Json()
body.query.filtered.query.match.description = 'addictive'
body.query.filtered.filter.term.created_by = 'ASU'
```
### streams.py
#### stream
`stream` subclasses `collections.Iterable`. It's the same Python iterable, but with more added methods, suitable for multithreading and multiprocess processings.
Used to create stream processing pipelines, similar to those used in [Scala](http://www.scala-lang.org/) and [MapReduce](https://en.wikipedia.org/wiki/MapReduce) programming model.
Those who used [Apache Spark](http://spark.apache.org/) [RDD](http://spark.apache.org/docs/latest/programming-guide.html#rdd-operations) functions will find this model of processing very easy to use.

### [streams](https://github.com/asuiu/pyxtension/blob/master/streams.py)
**Never again will you have to write code like this**:
```python
> lst = xrange(1,6)
> reduce(lambda x, y: x * y, map(lambda _: _ * _, filter(lambda _: _ % 2 == 0, lst)))
64
```
From now on, you may simply write the following lines:
```python
> the_stream = stream( xrange(1,6) )
> the_stream.\
    filter(lambda _: _ % 2 == 0).\
    map(lambda _: _ * _).\
    reduce(lambda x, y: x * y)
64
```

#### A Word Count [Map-Reduce](https://en.wikipedia.org/wiki/MapReduce) naive example using multiprocessing map
```python
corpus = [
    "MapReduce is a programming model and an associated implementation for processing and generating large data sets with a parallel, distributed algorithm on a cluster.",
    "At Google, MapReduce was used to completely regenerate Google's index of the World Wide Web",
    "Conceptually similar approaches have been very well known since 1995 with the Message Passing Interface standard having reduce and scatter operations."]

def reduceMaps(m1, m2):
    for k, v in m2.iteritems():
        m1[k] = m1.get(k, 0) + v
    return m1

word_counts = stream(corpus).\
    mpmap(lambda line: stream(line.lower().split(' ')).countByValue()).\
    reduce(reduceMaps)
```

#### Basic methods
###### **map(f)**
Identic with builtin `map` but returns a stream


###### **mpmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**
Parallel ordered map using `multiprocessing.Pool.imap()`.

It can replace the `map` when need to split computations to multiple cores, and order of results matters.

It spawns at most `poolSize` processes and applies the `f` function.

It won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.

The elements in the result stream appears in the same order they appear in the initial iterable.

```
:type f: (T) -> V
:rtype: `stream`
```


###### **mpfastmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**
Parallel ordered map using `multiprocessing.Pool.imap_unordered()`.

It can replace the `map` when the ordered of results doesn't matter.

It spawns at most `poolSize` processes and applies the `f` function.

It won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.

The elements in the result stream appears in the unpredicted order.

```
:type f: (T) -> V
:rtype: `stream`
```


###### **fastmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**
Parallel unordered map using multithreaded pool.
It can replace the `map` when the ordered of results doesn't matter.

It spawns at most `poolSize` threads and applies the `f` function.

The elements in the result stream appears in the **unpredicted** order.

It won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.

Because of CPython [GIL](https://wiki.python.org/moin/GlobalInterpreterLock) it's most usefull for I/O or CPU intensive consuming native functions, or on Jython or IronPython interpreters.

:type f: (T) -> V

:rtype: `stream`

###### **mtmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**
Parallel ordered map using multithreaded pool. 
It can replace the `map` and the order of output stream will be the same as of the input.

It spawns at most `poolSize` threads and applies the `f` function.

The elements in the result stream appears in the **predicted** order.

It won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.  

Because of CPython [GIL](https://wiki.python.org/moin/GlobalInterpreterLock) it's most usefull for I/O or CPU intensive consuming native functions, or on Jython or IronPython interpreters.

:type f: (T) -> V

:rtype: `stream`


###### **flatMap(predicate=_IDENTITY_FUNC)**
:param predicate: is a function that will receive elements of self collection and return an iterable

By default predicate is an identity function

:type predicate: (V)-> collections.Iterable[T]

:return: will return stream of objects of the same type of elements from the stream returned by predicate()

Example:
```python
stream([[1, 2], [3, 4], [4, 5]]).flatMap().toList() == [1, 2, 3, 4, 4, 5]
```


###### **filter(predicate)**
identic with builtin filter, but returns stream


###### **reversed()**
returns reversed stream


###### **exists(predicate)**
Tests whether a predicate holds for some of the elements of this sequence.

:rtype: bool

Example:
```python
stream([1, 2, 3]).exists(0) -> False
stream([1, 2, 3]).exists(1) -> True
```


###### **keyBy(keyfunc = _IDENTITY_FUNC)**
Transforms stream of values to a stream of tuples (key, value)

:param keyfunc: function to map values to keys

:type keyfunc: (V) -> T

:return: stream of Key, Value pairs

:rtype: stream[( T, V )]

Example:
```python
stream([1, 2, 3, 4]).keyBy(lambda _:_ % 2) -> [(1, 1), (0, 2), (1, 3), (0, 4)]
```

###### **groupBy()**
groupBy([keyfunc]) -> Make an iterator that returns consecutive keys and groups from the iterable.

The iterable needs not to be sorted on the same key function, but the keyfunction need to return hasable objects.

:param keyfunc: [Optional] The key is a function computing a key value for each element.

:type keyfunc: (T) -> (V)

:return: (key, sub-iterator) grouped by each value of key(value).

:rtype: stream[ ( V, slist[T] ) ]

Example:
```python
stream([1, 2, 3, 4]).groupBy(lambda _: _ % 2) -> [(0, [2, 4]), (1, [1, 3])]
```

###### **countByValue()**
Returns a collections.Counter of values

Example
```python
stream(['a', 'b', 'a', 'b', 'c', 'd']).countByValue() == {'a': 2, 'b': 2, 'c': 1, 'd': 1}
```

###### **distinct()**
Returns stream of distinct values. Values must be hashable.
```python
stream(['a', 'b', 'a', 'b', 'c', 'd']).distinct() == {'a', 'b', 'c', 'd'}
```


###### **reduce(f, init=None)**
same arguments with builtin reduce() function


###### **toSet()**
returns sset() instance


###### **toList()**
returns slist() instance


###### **toMap()**
returns sdict() instance


###### **sorted(key=None, cmp=None, reverse=False)**
same arguments with builtin sorted()


###### **size()**
returns length of stream. Use carefully on infinite streams.


###### **join(f)**
Returns a string joined by f. Proivides same functionality as str.join() builtin method.

if f is basestring, uses it to join the stream, else f should be a callable that returns a string to be used for join


###### **mkString(f)**
identic with join(f)


###### **take(n)**
    returns first n elements from stream


###### **head()**
    returns first element from stream


###### **zip()**
    the same behavior with itertools.izip()

###### **unique(predicate=_IDENTITY_FUNC)**
    Returns a stream of unique (according to predicate) elements appearing in the same order as in original stream

    The items returned by predicate should be hashable and comparable.


#### Statistics related methods
###### **entropy()**
calculates the Shannon entropy of the values from stream


###### **pstddev()**
Calculates the population standard deviation.


###### **mean()**
returns the arithmetical mean of the values


###### **sum()**
returns the sum of elements from stream


###### **min(key=_IDENTITY_FUNC)**
same functionality with builtin min() funcion


###### **min_default(default, key=_IDENTITY_FUNC)**
same functionality with min() but returns :default: when called on empty streams


###### **max()**
same functionality with builtin max()


###### **maxes(key=_IDENTITY_FUNC)**
returns a stream of max values from stream


###### **mins(key=_IDENTITY_FUNC)**
returns a stream of min values from stream


### Other classes
##### slist
Inherits `streams.stream` and built-in `list` classes, and keeps in memory a list allowing faster index access
##### sset
Inherits `streams.stream` and built-in `set` classes, and keeps in memory the whole set of values
##### sdict
Inherits `streams.stream` and built-in `dict`, and keeps in memory the dict object.
##### defaultstreamdict
Inherits `streams.sdict` and adds functionality  of `collections.defaultdict` from stdlib


### [Json](https://github.com/asuiu/pyxtension/blob/master/Json.py)

[Json](https://github.com/asuiu/pyxtension/blob/master/Json.py) is a module that provides mapping objects that allow their elements to be accessed both as keys and as attributes:

```python
    > from pyxtension import Json
> a = Json({'foo': 'bar'})
> a.foo
'bar'
> a['foo']
'bar'
```

Attribute access makes it easy to create convenient, hierarchical settings objects:
```python
    with open('settings.yaml') as fileobj:
        settings = Json(yaml.safe_load(fileobj))

    cursor = connect(**settings.db.credentials).cursor()

    cursor.execute("SELECT column FROM table;")
```

### Basic Usage

Json comes with two different classes, `Json`, and `JsonList`.
Json is fairly similar to native `dict` as it extends it an is a mutable mapping that allow creating, accessing, and deleting key-value pairs as attributes.
`JsonList` is similar to native `list` as it extends it and offers a way to transform the `dict` objects from inside also in `Json` instances.

#### Construction
###### Directly from a JSON string
```python
> Json('{"key1": "val1", "lst1": [1,2] }')
{u'key1': u'val1', u'lst1': [1, 2]}
```
###### From `tuple`s:
```python
> Json( ('key1','val1'), ('lst1', [1,2]) )
{'key1': 'val1', 'lst1': [1, 2]}
# keep in mind that you should provide at least two tuples with key-value pairs
```
###### As a built-in `dict`
```python
> Json( [('key1','val1'), ('lst1', [1,2])] )
{'key1': 'val1', 'lst1': [1, 2]}

Json({'key1': 'val1', 'lst1': [1, 2]})
{'key1': 'val1', 'lst1': [1, 2]}
```
#### Convert to a `dict`
```python
> json = Json({'key1': 'val1', 'lst1': [1, 2]})
> json.toOrig()
{'key1': 'val1', 'lst1': [1, 2]}
```

#### Valid Names

Any key can be used as an attribute as long as:

1. The key represents a valid attribute (i.e., it is a string comprised only of
   alphanumeric characters and underscores that doesn't start with a number)
2. The key does not shadow a class attribute (e.g., get).

#### Attributes vs. Keys
There is a minor difference between accessing a value as an attribute vs.
accessing it as a key, is that when a dict is accessed as an attribute, it will
automatically be converted to a `Json` object. This allows you to recursively
access keys::
```python
    > attr = Json({'foo': {'bar': 'baz'}})
    > attr.foo.bar
    'baz'
```
Relatedly, by default, sequence types that aren't `bytes`, `str`, or `unicode`
(e.g., `list`s, `tuple`s) will automatically be converted to `tuple`s, with any
mappings converted to `Json`:
```python
    > attr = Json({'foo': [{'bar': 'baz'}, {'bar': 'qux'}]})
    > for sub_attr in attr.foo:
    >     print(sub_attr.bar)
    'baz'
    'qux'
```
To get this recursive functionality for keys that cannot be used as attributes,
you can replicate the behavior by using dict syntax on `Json` object::
```python
> json = Json({1: {'two': 3}})
> json[1].two
3
```
`JsonList` usage examples:
```
> json = Json('{"lst":[1,2,3]}')
> type(json.lst)
<class 'pyxtension.Json.JsonList'>

> json = Json('{"1":[1,2]}')
> json["1"][1]
2
```


Assignment as keys will still work::
```python
> json = Json({'foo': {'bar': 'baz'}})
> json['foo']['bar'] = 'baz'
> json.foo
{'bar': 'baz'}
```

### frozendict
`frozendict` is a simple immutable dictionary, where you can't change the internal variables of the class, and they are all immutable objects. Reinvoking `__init__` also doesn't alter the object.

The API is the same as `dict`, without methods that can change the immutability. 

`frozendict` is also hashable and can be used as keys for other dictionaries, of course with the condition that all values of the frozendict are also hashable.

```python
>>> from pyxtension import frozendict

>>> fd = frozendict({"A": "B", "C": "D"})
>>> print(fd)
{'A': 'B', 'C': 'D'}

>>> fd["A"] = "C"
TypeError: object is immutable

>>> hash(fd)
-5063792767678978828
```

### License
pyxtension is released under a GNU Public license.
The idea for [Json](https://github.com/asuiu/pyxtension/blob/master/Json.py) module was inspired from [addict](https://github.com/mewwts/addict) and [AttrDict](https://github.com/bcj/AttrDict),
but it has a better performance with lower memory consumption.

### Alternatives
There are other libraries that support Fluent Interface streams as alternatives to Pyxtension, but being much more poor in features for streaming:
- https://pypi.org/project/lazy-streams/
- https://pypi.org/project/pystreams/
- https://pypi.org/project/fluentpy/
- https://github.com/matthagy/scalaps
- https://pypi.org/project/infixpy/ mentioned [here](https://stackoverflow.com/questions/49001986/left-to-right-application-of-operations-on-a-list-in-python3/62585964?noredirect=1#comment111806251_62585964)
- https://github.com/sspipe/sspipe


and something quite different from Fluent patterm, that makes kind of Piping: https://github.com/sspipe/sspipe and https://github.com/JulienPalard/Pipe

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/asuiu/pyxtension",
    "name": "pyxtension",
    "maintainer": null,
    "docs_url": "https://pythonhosted.org/pyxtension/",
    "requires_python": "<4,>=3.8",
    "maintainer_email": null,
    "keywords": null,
    "author": "Andrei Suiu",
    "author_email": "andrei.suiu@gmail.com",
    "download_url": null,
    "platform": null,
    "description": "# pyxtension\r\n[![build Status](https://travis-ci.org/asuiu/pyxtension.svg?branch=master)](https://travis-ci.org/asuiu/pyxtension)\r\n[![Coverage Status](https://coveralls.io/repos/asuiu/pyxtension/badge.svg?branch=master&service=github)](https://coveralls.io/github/asuiu/pyxtension?branch=master)\r\n\r\n[pyxtension](https://github.com/asuiu/pyxtension) is a pure Python MIT-licensed library that includes Scala-like streams (using [Fluent Interface pattern](https://en.wikipedia.org/wiki/Fluent_interface)), Json with attribute access syntax, and other common-use stuff.\r\n\r\n###### Note:\r\n\r\n**Drop support & maintenance for Python 2.x version, due to [Py2 death](https://www.python.org/doc/sunset-python-2/).**\r\n \r\n Although Py2 version will remain in the repository, I won't update PyPi package, so the last Py2 version of the `pyxtension` available at [PyPi](https://pypi.org/project/pyxtension/) will remain [`1.12.7`](https://pypi.org/project/pyxtension/1.12.7/)\r\n\r\nStarting with [`1.13.0`](https://pypi.org/project/pyxtension/1.13.0/) I've migrated the packaging & distributing method to [Wheel](https://pythonwheels.com/).\r\n      \r\n## Installation\r\n```\r\npip install pyxtension\r\n```\r\nor from Github:\r\n```\r\ngit clone https://github.com/asuiu/pyxtension.git\r\ncd pyxtension\r\npython setup.py install\r\n```\r\nor\r\n```\r\ngit submodule add https://github.com/asuiu/pyxtension.git\r\n```\r\n\r\n## Modules overview\r\n### Json.py\r\n##### Json\r\nA `dict` subclass to represent a Json object. You should be able to use this\r\nabsolutely anywhere you can use a `dict`. While this is probably the class you\r\nwant to use, there are a few caveats that follow from this being a `dict` under\r\nthe hood.\r\n\r\n**Never again will you have to write code like this**:\r\n```python\r\nbody = {\r\n    'query': {\r\n        'filtered': {\r\n            'query': {\r\n                'match': {'description': 'addictive'}\r\n            },\r\n            'filter': {\r\n                'term': {'created_by': 'ASU'}\r\n            }\r\n        }\r\n    }\r\n}\r\n```\r\n\r\nFrom now on, you may simply write the following three lines:\r\n```python\r\nbody = Json()\r\nbody.query.filtered.query.match.description = 'addictive'\r\nbody.query.filtered.filter.term.created_by = 'ASU'\r\n```\r\n### streams.py\r\n#### stream\r\n`stream` subclasses `collections.Iterable`. It's the same Python iterable, but with more added methods, suitable for multithreading and multiprocess processings.\r\nUsed to create stream processing pipelines, similar to those used in [Scala](http://www.scala-lang.org/) and [MapReduce](https://en.wikipedia.org/wiki/MapReduce) programming model.\r\nThose who used [Apache Spark](http://spark.apache.org/) [RDD](http://spark.apache.org/docs/latest/programming-guide.html#rdd-operations) functions will find this model of processing very easy to use.\r\n\r\n### [streams](https://github.com/asuiu/pyxtension/blob/master/streams.py)\r\n**Never again will you have to write code like this**:\r\n```python\r\n> lst = xrange(1,6)\r\n> reduce(lambda x, y: x * y, map(lambda _: _ * _, filter(lambda _: _ % 2 == 0, lst)))\r\n64\r\n```\r\nFrom now on, you may simply write the following lines:\r\n```python\r\n> the_stream = stream( xrange(1,6) )\r\n> the_stream.\\\r\n    filter(lambda _: _ % 2 == 0).\\\r\n    map(lambda _: _ * _).\\\r\n    reduce(lambda x, y: x * y)\r\n64\r\n```\r\n\r\n#### A Word Count [Map-Reduce](https://en.wikipedia.org/wiki/MapReduce) naive example using multiprocessing map\r\n```python\r\ncorpus = [\r\n    \"MapReduce is a programming model and an associated implementation for processing and generating large data sets with a parallel, distributed algorithm on a cluster.\",\r\n    \"At Google, MapReduce was used to completely regenerate Google's index of the World Wide Web\",\r\n    \"Conceptually similar approaches have been very well known since 1995 with the Message Passing Interface standard having reduce and scatter operations.\"]\r\n\r\ndef reduceMaps(m1, m2):\r\n    for k, v in m2.iteritems():\r\n        m1[k] = m1.get(k, 0) + v\r\n    return m1\r\n\r\nword_counts = stream(corpus).\\\r\n    mpmap(lambda line: stream(line.lower().split(' ')).countByValue()).\\\r\n    reduce(reduceMaps)\r\n```\r\n\r\n#### Basic methods\r\n###### **map(f)**\r\nIdentic with builtin `map` but returns a stream\r\n\r\n\r\n###### **mpmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**\r\nParallel ordered map using `multiprocessing.Pool.imap()`.\r\n\r\nIt can replace the `map` when need to split computations to multiple cores, and order of results matters.\r\n\r\nIt spawns at most `poolSize` processes and applies the `f` function.\r\n\r\nIt won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.\r\n\r\nThe elements in the result stream appears in the same order they appear in the initial iterable.\r\n\r\n```\r\n:type f: (T) -> V\r\n:rtype: `stream`\r\n```\r\n\r\n\r\n###### **mpfastmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**\r\nParallel ordered map using `multiprocessing.Pool.imap_unordered()`.\r\n\r\nIt can replace the `map` when the ordered of results doesn't matter.\r\n\r\nIt spawns at most `poolSize` processes and applies the `f` function.\r\n\r\nIt won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.\r\n\r\nThe elements in the result stream appears in the unpredicted order.\r\n\r\n```\r\n:type f: (T) -> V\r\n:rtype: `stream`\r\n```\r\n\r\n\r\n###### **fastmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**\r\nParallel unordered map using multithreaded pool.\r\nIt can replace the `map` when the ordered of results doesn't matter.\r\n\r\nIt spawns at most `poolSize` threads and applies the `f` function.\r\n\r\nThe elements in the result stream appears in the **unpredicted** order.\r\n\r\nIt won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.\r\n\r\nBecause of CPython [GIL](https://wiki.python.org/moin/GlobalInterpreterLock) it's most usefull for I/O or CPU intensive consuming native functions, or on Jython or IronPython interpreters.\r\n\r\n:type f: (T) -> V\r\n\r\n:rtype: `stream`\r\n\r\n###### **mtmap(self, f: Callable[[_K], _V], poolSize: int = cpu_count(), bufferSize: Optional[int] = None)**\r\nParallel ordered map using multithreaded pool. \r\nIt can replace the `map` and the order of output stream will be the same as of the input.\r\n\r\nIt spawns at most `poolSize` threads and applies the `f` function.\r\n\r\nThe elements in the result stream appears in the **predicted** order.\r\n\r\nIt won't take more than `bufferSize` elements from the input unless it was already required by output, so you can use it with `takeWhile` on infinite streams and not be afraid that it will continue work in background.  \r\n\r\nBecause of CPython [GIL](https://wiki.python.org/moin/GlobalInterpreterLock) it's most usefull for I/O or CPU intensive consuming native functions, or on Jython or IronPython interpreters.\r\n\r\n:type f: (T) -> V\r\n\r\n:rtype: `stream`\r\n\r\n\r\n###### **flatMap(predicate=_IDENTITY_FUNC)**\r\n:param predicate: is a function that will receive elements of self collection and return an iterable\r\n\r\nBy default predicate is an identity function\r\n\r\n:type predicate: (V)-> collections.Iterable[T]\r\n\r\n:return: will return stream of objects of the same type of elements from the stream returned by predicate()\r\n\r\nExample:\r\n```python\r\nstream([[1, 2], [3, 4], [4, 5]]).flatMap().toList() == [1, 2, 3, 4, 4, 5]\r\n```\r\n\r\n\r\n###### **filter(predicate)**\r\nidentic with builtin filter, but returns stream\r\n\r\n\r\n###### **reversed()**\r\nreturns reversed stream\r\n\r\n\r\n###### **exists(predicate)**\r\nTests whether a predicate holds for some of the elements of this sequence.\r\n\r\n:rtype: bool\r\n\r\nExample:\r\n```python\r\nstream([1, 2, 3]).exists(0) -> False\r\nstream([1, 2, 3]).exists(1) -> True\r\n```\r\n\r\n\r\n###### **keyBy(keyfunc = _IDENTITY_FUNC)**\r\nTransforms stream of values to a stream of tuples (key, value)\r\n\r\n:param keyfunc: function to map values to keys\r\n\r\n:type keyfunc: (V) -> T\r\n\r\n:return: stream of Key, Value pairs\r\n\r\n:rtype: stream[( T, V )]\r\n\r\nExample:\r\n```python\r\nstream([1, 2, 3, 4]).keyBy(lambda _:_ % 2) -> [(1, 1), (0, 2), (1, 3), (0, 4)]\r\n```\r\n\r\n###### **groupBy()**\r\ngroupBy([keyfunc]) -> Make an iterator that returns consecutive keys and groups from the iterable.\r\n\r\nThe iterable needs not to be sorted on the same key function, but the keyfunction need to return hasable objects.\r\n\r\n:param keyfunc: [Optional] The key is a function computing a key value for each element.\r\n\r\n:type keyfunc: (T) -> (V)\r\n\r\n:return: (key, sub-iterator) grouped by each value of key(value).\r\n\r\n:rtype: stream[ ( V, slist[T] ) ]\r\n\r\nExample:\r\n```python\r\nstream([1, 2, 3, 4]).groupBy(lambda _: _ % 2) -> [(0, [2, 4]), (1, [1, 3])]\r\n```\r\n\r\n###### **countByValue()**\r\nReturns a collections.Counter of values\r\n\r\nExample\r\n```python\r\nstream(['a', 'b', 'a', 'b', 'c', 'd']).countByValue() == {'a': 2, 'b': 2, 'c': 1, 'd': 1}\r\n```\r\n\r\n###### **distinct()**\r\nReturns stream of distinct values. Values must be hashable.\r\n```python\r\nstream(['a', 'b', 'a', 'b', 'c', 'd']).distinct() == {'a', 'b', 'c', 'd'}\r\n```\r\n\r\n\r\n###### **reduce(f, init=None)**\r\nsame arguments with builtin reduce() function\r\n\r\n\r\n###### **toSet()**\r\nreturns sset() instance\r\n\r\n\r\n###### **toList()**\r\nreturns slist() instance\r\n\r\n\r\n###### **toMap()**\r\nreturns sdict() instance\r\n\r\n\r\n###### **sorted(key=None, cmp=None, reverse=False)**\r\nsame arguments with builtin sorted()\r\n\r\n\r\n###### **size()**\r\nreturns length of stream. Use carefully on infinite streams.\r\n\r\n\r\n###### **join(f)**\r\nReturns a string joined by f. Proivides same functionality as str.join() builtin method.\r\n\r\nif f is basestring, uses it to join the stream, else f should be a callable that returns a string to be used for join\r\n\r\n\r\n###### **mkString(f)**\r\nidentic with join(f)\r\n\r\n\r\n###### **take(n)**\r\n    returns first n elements from stream\r\n\r\n\r\n###### **head()**\r\n    returns first element from stream\r\n\r\n\r\n###### **zip()**\r\n    the same behavior with itertools.izip()\r\n\r\n###### **unique(predicate=_IDENTITY_FUNC)**\r\n    Returns a stream of unique (according to predicate) elements appearing in the same order as in original stream\r\n\r\n    The items returned by predicate should be hashable and comparable.\r\n\r\n\r\n#### Statistics related methods\r\n###### **entropy()**\r\ncalculates the Shannon entropy of the values from stream\r\n\r\n\r\n###### **pstddev()**\r\nCalculates the population standard deviation.\r\n\r\n\r\n###### **mean()**\r\nreturns the arithmetical mean of the values\r\n\r\n\r\n###### **sum()**\r\nreturns the sum of elements from stream\r\n\r\n\r\n###### **min(key=_IDENTITY_FUNC)**\r\nsame functionality with builtin min() funcion\r\n\r\n\r\n###### **min_default(default, key=_IDENTITY_FUNC)**\r\nsame functionality with min() but returns :default: when called on empty streams\r\n\r\n\r\n###### **max()**\r\nsame functionality with builtin max()\r\n\r\n\r\n###### **maxes(key=_IDENTITY_FUNC)**\r\nreturns a stream of max values from stream\r\n\r\n\r\n###### **mins(key=_IDENTITY_FUNC)**\r\nreturns a stream of min values from stream\r\n\r\n\r\n### Other classes\r\n##### slist\r\nInherits `streams.stream` and built-in `list` classes, and keeps in memory a list allowing faster index access\r\n##### sset\r\nInherits `streams.stream` and built-in `set` classes, and keeps in memory the whole set of values\r\n##### sdict\r\nInherits `streams.stream` and built-in `dict`, and keeps in memory the dict object.\r\n##### defaultstreamdict\r\nInherits `streams.sdict` and adds functionality  of `collections.defaultdict` from stdlib\r\n\r\n\r\n### [Json](https://github.com/asuiu/pyxtension/blob/master/Json.py)\r\n\r\n[Json](https://github.com/asuiu/pyxtension/blob/master/Json.py) is a module that provides mapping objects that allow their elements to be accessed both as keys and as attributes:\r\n\r\n```python\r\n    > from pyxtension import Json\r\n> a = Json({'foo': 'bar'})\r\n> a.foo\r\n'bar'\r\n> a['foo']\r\n'bar'\r\n```\r\n\r\nAttribute access makes it easy to create convenient, hierarchical settings objects:\r\n```python\r\n    with open('settings.yaml') as fileobj:\r\n        settings = Json(yaml.safe_load(fileobj))\r\n\r\n    cursor = connect(**settings.db.credentials).cursor()\r\n\r\n    cursor.execute(\"SELECT column FROM table;\")\r\n```\r\n\r\n### Basic Usage\r\n\r\nJson comes with two different classes, `Json`, and `JsonList`.\r\nJson is fairly similar to native `dict` as it extends it an is a mutable mapping that allow creating, accessing, and deleting key-value pairs as attributes.\r\n`JsonList` is similar to native `list` as it extends it and offers a way to transform the `dict` objects from inside also in `Json` instances.\r\n\r\n#### Construction\r\n###### Directly from a JSON string\r\n```python\r\n> Json('{\"key1\": \"val1\", \"lst1\": [1,2] }')\r\n{u'key1': u'val1', u'lst1': [1, 2]}\r\n```\r\n###### From `tuple`s:\r\n```python\r\n> Json( ('key1','val1'), ('lst1', [1,2]) )\r\n{'key1': 'val1', 'lst1': [1, 2]}\r\n# keep in mind that you should provide at least two tuples with key-value pairs\r\n```\r\n###### As a built-in `dict`\r\n```python\r\n> Json( [('key1','val1'), ('lst1', [1,2])] )\r\n{'key1': 'val1', 'lst1': [1, 2]}\r\n\r\nJson({'key1': 'val1', 'lst1': [1, 2]})\r\n{'key1': 'val1', 'lst1': [1, 2]}\r\n```\r\n#### Convert to a `dict`\r\n```python\r\n> json = Json({'key1': 'val1', 'lst1': [1, 2]})\r\n> json.toOrig()\r\n{'key1': 'val1', 'lst1': [1, 2]}\r\n```\r\n\r\n#### Valid Names\r\n\r\nAny key can be used as an attribute as long as:\r\n\r\n1. The key represents a valid attribute (i.e., it is a string comprised only of\r\n   alphanumeric characters and underscores that doesn't start with a number)\r\n2. The key does not shadow a class attribute (e.g., get).\r\n\r\n#### Attributes vs. Keys\r\nThere is a minor difference between accessing a value as an attribute vs.\r\naccessing it as a key, is that when a dict is accessed as an attribute, it will\r\nautomatically be converted to a `Json` object. This allows you to recursively\r\naccess keys::\r\n```python\r\n    > attr = Json({'foo': {'bar': 'baz'}})\r\n    > attr.foo.bar\r\n    'baz'\r\n```\r\nRelatedly, by default, sequence types that aren't `bytes`, `str`, or `unicode`\r\n(e.g., `list`s, `tuple`s) will automatically be converted to `tuple`s, with any\r\nmappings converted to `Json`:\r\n```python\r\n    > attr = Json({'foo': [{'bar': 'baz'}, {'bar': 'qux'}]})\r\n    > for sub_attr in attr.foo:\r\n    >     print(sub_attr.bar)\r\n    'baz'\r\n    'qux'\r\n```\r\nTo get this recursive functionality for keys that cannot be used as attributes,\r\nyou can replicate the behavior by using dict syntax on `Json` object::\r\n```python\r\n> json = Json({1: {'two': 3}})\r\n> json[1].two\r\n3\r\n```\r\n`JsonList` usage examples:\r\n```\r\n> json = Json('{\"lst\":[1,2,3]}')\r\n> type(json.lst)\r\n<class 'pyxtension.Json.JsonList'>\r\n\r\n> json = Json('{\"1\":[1,2]}')\r\n> json[\"1\"][1]\r\n2\r\n```\r\n\r\n\r\nAssignment as keys will still work::\r\n```python\r\n> json = Json({'foo': {'bar': 'baz'}})\r\n> json['foo']['bar'] = 'baz'\r\n> json.foo\r\n{'bar': 'baz'}\r\n```\r\n\r\n### frozendict\r\n`frozendict` is a simple immutable dictionary, where you can't change the internal variables of the class, and they are all immutable objects. Reinvoking `__init__` also doesn't alter the object.\r\n\r\nThe API is the same as `dict`, without methods that can change the immutability. \r\n\r\n`frozendict` is also hashable and can be used as keys for other dictionaries, of course with the condition that all values of the frozendict are also hashable.\r\n\r\n```python\r\n>>> from pyxtension import frozendict\r\n\r\n>>> fd = frozendict({\"A\": \"B\", \"C\": \"D\"})\r\n>>> print(fd)\r\n{'A': 'B', 'C': 'D'}\r\n\r\n>>> fd[\"A\"] = \"C\"\r\nTypeError: object is immutable\r\n\r\n>>> hash(fd)\r\n-5063792767678978828\r\n```\r\n\r\n### License\r\npyxtension is released under a GNU Public license.\r\nThe idea for [Json](https://github.com/asuiu/pyxtension/blob/master/Json.py) module was inspired from [addict](https://github.com/mewwts/addict) and [AttrDict](https://github.com/bcj/AttrDict),\r\nbut it has a better performance with lower memory consumption.\r\n\r\n### Alternatives\r\nThere are other libraries that support Fluent Interface streams as alternatives to Pyxtension, but being much more poor in features for streaming:\r\n- https://pypi.org/project/lazy-streams/\r\n- https://pypi.org/project/pystreams/\r\n- https://pypi.org/project/fluentpy/\r\n- https://github.com/matthagy/scalaps\r\n- https://pypi.org/project/infixpy/ mentioned [here](https://stackoverflow.com/questions/49001986/left-to-right-application-of-operations-on-a-list-in-python3/62585964?noredirect=1#comment111806251_62585964)\r\n- https://github.com/sspipe/sspipe\r\n\r\n\r\nand something quite different from Fluent patterm, that makes kind of Piping: https://github.com/sspipe/sspipe and https://github.com/JulienPalard/Pipe\r\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "Extension library for Python",
    "version": "1.16.11",
    "project_urls": {
        "Homepage": "https://github.com/asuiu/pyxtension"
    },
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "28fcb5f3ae6090fd56a8f71d9b7a451295ca6cbc9889bf79e3855c16db5e3d49",
                "md5": "70f1ab4ed8623ae43a17c65317644c4d",
                "sha256": "4be4b48433c5a53054816391656539d4901c61fab4983b30b706ead93dd93ab3"
            },
            "downloads": -1,
            "filename": "pyxtension-1.16.11-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "70f1ab4ed8623ae43a17c65317644c4d",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": "<4,>=3.8",
            "size": 22044,
            "upload_time": "2024-08-26T12:49:22",
            "upload_time_iso_8601": "2024-08-26T12:49:22.537257Z",
            "url": "https://files.pythonhosted.org/packages/28/fc/b5f3ae6090fd56a8f71d9b7a451295ca6cbc9889bf79e3855c16db5e3d49/pyxtension-1.16.11-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-08-26 12:49:22",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "asuiu",
    "github_project": "pyxtension",
    "travis_ci": true,
    "coveralls": false,
    "github_actions": false,
    "requirements": [],
    "lcname": "pyxtension"
}
        
Elapsed time: 1.04776s