Metadata-Version: 2.1
Name: default-mutable
Version: 0.1.0
Summary: Prevent mutable default arguments behavior
Home-page: https://github.com/roodrepo/default_mutable
License: MIT
Keywords: default arguments,default list,default dict,decorator
Author: Rood Repo
Author-email: roodrepo@gmail.com
Requires-Python: >=3,<4
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.4
Classifier: Programming Language :: Python :: 3.5
Classifier: Programming Language :: Python :: 3.6
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Project-URL: Repository, https://github.com/roodrepo/default_mutable
Description-Content-Type: text/markdown

Default Mutable
==========

### *Prevent mutable default value behavior*

Default Mutable is a Python decorator that overrides default mutable values and prevents object mutation from one call to another. More info on https://docs.python-guide.org/writing/gotchas/

## Features

- Easy to use
- Preset default value

## Advantages

- Highly flexible
- Lightweight and independent
- Open-source
- Real use cases
- Support & documentation

## Authors

- Rudy Fernandez

## Install
The easiest way to install default_mutable using pip:
`pip install default_mutable`

## [Examples](https://github.com/roodrepo/default_mutable/blob/v0-dev/examples/example1.py)


Python’s default arguments are evaluated once when the function is defined, not each time the function is called (like it is in say, Ruby). This means that if you use a mutable default argument and mutate it, you will and have mutated that object for all future calls to the function as well.
```python
def function1(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function1:')
function1(idx= 1)   # 1 {'a1': 'a1'} ['a']
function1(idx= 2)   # 2 {'a1': 'a1', 'a2': 'a2'} ['a', 'a']
```

---

Setting all the arguments with mutable default arguments to an **EMPTY** dictionary or list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable
def function2(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function2:')
function2(1)    # 1 {'a1': 'a1'} ['a']
function2(2)    # 2 {'a2': 'a2'} ['a']
```

---

Setting all the arguments with mutable default arguments to an **EMPTY** dictionary or list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable
def function2_2(idx, dict1 = {}, list1 = ['list_init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function2_2:')
function2_2(1)    # 1 {'a1': 'a1'} ['a']
function2_2(2)    # 2 {'a2': 'a2'} ['a']
```

---

When the decorator is executed with not arguments, we get the default behavior (same as "function1")
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable()
def function3(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function3:')
function3(idx= 1)   # 1 {'a1': 'a1'} ['a']
function3(idx= 2)   # 2 {'a1': 'a1', 'a2': 'a2'} ['a', 'a']
```

---

As no value is passed, the decorator set the default value for "dict1" and "list1" an empty dictionary and empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', 'list1')
def function4(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function4:')
function4(idx= 1)   # 1 {'a1': 'a1'} ['a']
function4(idx= 2)   # 2 {'a2': 'a2'} ['a']
```

---

As no value is passed, the decorator set the default value for "dict1" and "list1" an empty dictionary and empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', 'list1')
def function4_2(idx, dict1 = {}, list1 = ['init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function4_2:')
function4_2(idx= 1)   # 1 {'a1': 'a1'} ['a']
function4_2(idx= 2)   # 2 {'a2': 'a2'} ['a']
```

---
As no value is passed, the decorator set the default value for "list1" an empty list. "dict1" has the default behavior
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('list1')
def function4_3(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function4_3:')
function4_3(idx= 1) # 1 {'a1': 'a1'} ['a']
function4_3(idx= 2) # 2 {'a1': 'a1', 'a2': 'a2'} ['a']
```

---
The default values are mutable and "list1" get incremented from one function to the other
```python
def function5(idx, dict1 = {}, list1 = ['init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function5:')
function5(idx= 1)   # 1 {'a1': 'a1'} ['init', 'a']
function5(idx= 2)   # 2 {'a1': 'a1', 'a2': 'a2'} ['init', 'a', 'a']
```

---
Default value for "list1" is not an empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', list1= ['init'])
def function6(idx, dict1 = {}, list1 = ['init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function6:')
function6(idx= 1)   # 1 {'a1': 'a1'} ['init', 'a']
function6(idx= 2)   # 2 {'a2': 'a2'} ['init', 'a']
```

---
Default value for "list1" is not an empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', list1= ['init'])
def function7(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function7:')
function7(idx= 1)   # 1 {'a1': 'a1'} ['init', 'a']
function7(idx= 2)   # 2 {'a2': 'a2'} ['init', 'a']
```

---
Default value are empty dictionary and empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', list1= [])
def function7_2(idx, dict1 = {}, list1 = ['init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function7_2:')
function7_2(idx= 1) # 1 {'a1': 'a1'} ['a']
function7_2(idx= 2) # 2 {'a2': 'a2'} ['a']
```
---
Default value are empty dictionary and empty list
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable('dict1', 'list1')
def function7_3(idx, dict1 = {}, list1 = ['init'], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function7_3:')
function7_3(idx= 1) # 1 {'a1': 'a1'} ['a']
function7_3(idx= 2) # 2 {'a2': 'a2'} ['a']
```

---
Default value for "dict1" and "list1" are preset
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable(dict1= {'dict_init': 'dict_init'}, list1= ['init'])
def function8(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function8:')
function8(idx= 1)   # 1 {'dict_init': 'dict_init', 'a1': 'a1'} ['init', 'a']
function8(idx= 2)   # 2 {'dict_init': 'dict_init', 'a2': 'a2'} ['init', 'a']
```

---
Passed values are not overridden
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable(dict1= {'dict_init': 'dict_init'}, list1= ['init'])
def function9(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function9:')
function9(idx= 1, dict1= {'dict_value': 'dict_value'}, list1= ['list_value'])   # 1 {'dict_value': 'dict_value', 'a1': 'a1'} ['list_value', 'a']
function9(idx= 2, dict1= {'dict_value': 'dict_value'}, list1= ['list_value'])   # 2 {'dict_value': 'dict_value', 'a2': 'a2'} ['list_value', 'a']

```

---
Passed values are not overridden
```python
from default_mutable.DefaultMutable import defaultMutable

@defaultMutable(dict1= {'dict_init': 'dict_init'}, list1= ['init'])
def function10(idx, dict1 = {}, list1 = [], element = 'a'):
    list1.append(element)
    dict1[f'{element}{idx}'] = f'{element}{idx}'
    print('', idx, dict1, list1)

print('')
print('function10:')
function10(1, {'dict_value': 'dict_value'}, ['list_value']) # 1 {'dict_value': 'dict_value', 'a1': 'a1'} ['list_value', 'a']
function10(2, {'dict_value': 'dict_value'}, ['list_value']) # 2 {'dict_value': 'dict_value', 'a2': 'a2'} ['list_value', 'a']
```


