%global _empty_manifest_terminate_build 0
Name: python-django-clone
Version: 5.3.1
Release: 1
Summary: Create a clone of a django model instance.
License: MIT/Apache-2.0
URL: https://github.com/tj-django/django-clone.git
Source0: https://mirrors.nju.edu.cn/pypi/web/packages/c4/f0/d5e399c401dfaf7a46451132ed393892929a884909859ae5bdb68620c873/django-clone-5.3.1.tar.gz
BuildArch: noarch
Requires: python3-django
Requires: python3-conditional
Requires: python3-six
Requires: python3-bump2version
Requires: python3-readme-renderer[md]
Requires: python3-git-changelog
Requires: python3-pip-tools
Requires: python3-check-manifest
Requires: python3-django
Requires: python3-conditional
Requires: python3-six
Requires: python3-tox
Requires: python3-tox-gh-actions
Requires: python3-pluggy
Requires: python3-mock
Requires: python3-unittest-xml-reporting
Requires: python3-codacy-coverage
Requires: python3-django-migration-fixer
Requires: python3-bump2version
Requires: python3-readme-renderer[md]
Requires: python3-git-changelog
Requires: python3-tox
Requires: python3-tox-gh-actions
Requires: python3-pluggy
Requires: python3-mock
Requires: python3-unittest-xml-reporting
Requires: python3-codacy-coverage
Requires: python3-django-migration-fixer
%description
| Python | Django | Downloads | Code Style |
|:---------:|:-------:|:-----------:|:--------------:|
| [data:image/s3,"s3://crabby-images/41513/415134e01461860b527eda09785023dc55c5f478" alt="PyPI - Python Version"](https://pypi.org/project/django-clone) | [data:image/s3,"s3://crabby-images/eaecb/eaecb11df616e39bc6f6ec81d9bef150f59bc6e0" alt="PyPI - Django Version"](https://docs.djangoproject.com/en/3.2/releases/) | [data:image/s3,"s3://crabby-images/d53c5/d53c5f76049b40150dcc1d24e1eb7de5ac0432cb" alt="Downloads"](https://pepy.tech/project/django-clone) | [data:image/s3,"s3://crabby-images/98647/986475842f2907062b79c4bb27fdd075d638e5b9" alt="Code style: black"](https://github.com/psf/black) |
| PyPI | Test | Vulnerabilities | Coverage | Code Quality | Pre-Commit |
|:---------------:|:----:|:---------------:|:--------:|:-------------:|:-------------:|
| [data:image/s3,"s3://crabby-images/c4fba/c4fbab60b5d53b9776a750b49cdda5edd360acd0" alt="PyPI version"](https://badge.fury.io/py/django-clone) | [data:image/s3,"s3://crabby-images/627a0/627a03208f8c045f597c111965ea7a882d804c45" alt="Test"](https://github.com/tj-django/django-clone/actions?query=workflow%3ATest) | [data:image/s3,"s3://crabby-images/3e97e/3e97e3804379a4f70df67b5bf835ec6a312fffc7" alt="Known Vulnerabilities"](https://snyk.io/test/github/tj-django/django-clone?targetFile=requirements.txt) | [data:image/s3,"s3://crabby-images/7c74d/7c74dd4e779f215c7a50464ee84ff7c155af86ab" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Coverage)
[data:image/s3,"s3://crabby-images/98695/986959e1f606ecbc2ac5fd26971b517339e1a975" alt="codecov"](https://codecov.io/gh/tj-django/django-clone)| [data:image/s3,"s3://crabby-images/16cd8/16cd800099bcc76fe80572c7d70b5b2ccf7d0435" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Grade) | [data:image/s3,"s3://crabby-images/e63dd/e63dd438b75fb26a6bda7b8c6ccdf0a0f6f248cd" alt="pre-commit.ci status"](https://results.pre-commit.ci/latest/github/tj-django/django-clone/main) |
## django-clone
Create copies of a model instance with explicit control on how the instance should be duplicated (limiting fields or related objects copied) with unique field detection.
This solves the problem introduced by using `instance.pk = None` and `instance.save()` which results in copying more object state than required.
## Features
* 100% test coverage.
* More control over how a model instance should be duplicated
* Multi Database support i.e Create duplicates on one or more databases.
* Restrict fields used for creating a duplicate instance.
* Detects unique fields and naively adds a suffix `copy {count}` to each duplicate instance (for supported fields only).
* Optionally differentiate between a duplicate instance and the original by appending a **copy** suffix to non unique fields (for supported fields only).
## Table of Contents
* [Installation](#installation)
* [Usage](#usage)
* [Subclassing the `CloneModel`](#subclassing-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin)
* [Duplicating a model instance](#duplicating-a-model-instance)
* [Bulk cloning a model](#bulk-cloning-a-model)
* [Creating clones without subclassing `CloneMixin`.](#creating-clones-without-subclassing-clonemixin)
* [CloneMixin attributes](#clonemixin-attributes)
* [Explicit (include only these fields)](#explicit-include-only-these-fields)
* [Implicit (include all except these fields)](#implicit-include-all-except-these-fields)
* [Django Admin](#django-admin)
* [Duplicating Models from the Django Admin view.](#duplicating-models-from-the-django-admin-view)
* [List View](#list-view)
* [Change View](#change-view)
* [CloneModelAdmin class attributes](#clonemodeladmin-class-attributes)
* [Advanced Usage](#advanced-usage)
* [Signals](#signals)
* [pre\_clone\_save, post\_clone\_save](#pre_clone_save-post_clone_save)
* [Clone Many to Many fields](#clone-many-to-many-fields)
* [Using the `CloneModel`](#using-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin-1)
* [Multi database support](#multi-database-support)
* [Compatibility](#compatibility)
* [Running locally](#running-locally)
* [Found a Bug?](#found-a-bug)
* [Contributors β¨](#contributors-)
## Installation
data:image/s3,"s3://crabby-images/74157/741570a79b1d0924b85f07742aabc30e00b3da23" alt=""
## Usage
### Subclassing the `CloneModel`
data:image/s3,"s3://crabby-images/cb9e9/cb9e9a5beb4a92111decf68a57751268cb0c5261" alt=""
### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/c001a/c001ab7826dd88bac6f9522230c6cbc6c24f6900" alt=""
### Duplicating a model instance
data:image/s3,"s3://crabby-images/fcb96/fcb96fad12614b72d5787fd034f48febeb81e379" alt=""
### Bulk cloning a model
data:image/s3,"s3://crabby-images/6231e/6231e8c8de4a41cbe64a8b0fd43a306a3a81f6be" alt=""
### Creating clones without subclassing `CloneMixin`.
> **NOTE:** :warning:
>
> * This method won't copy over related objects like Many to Many/One to Many relationships.
> * Ensure that required fields skipped from being cloned are passed in using the `attrs` kwargs.
data:image/s3,"s3://crabby-images/d0002/d00025dcb0e5807988b85be1af0461a96f6006db" alt=""
### CloneMixin attributes
| Attribute | Description |
|:------------------------------:|:------------:|
| `DUPLICATE_SUFFIX` | Suffix to append to duplicates
(NOTE: This requires `USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS`
to be enabled and supports string fields). |
`USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS` | Enable appending the `DUPLICATE_SUFFIX` to new cloned instances. |
`UNIQUE_DUPLICATE_SUFFIX` | Suffix to append to unique fields |
`USE_UNIQUE_DUPLICATE_SUFFIX` | Enable appending the `UNIQUE_DUPLICATE_SUFFIX` to new cloned instances. |
`MAX_UNIQUE_DUPLICATE_QUERY_ATTEMPTS` | The max query attempt while generating unique values for a case of unique conflicts. |
#### Explicit (include only these fields)
| Attribute | Description |
|:------------------------------:|:------------:|
| `_clone_fields` | Restrict the list of fields to copy from the instance (By default: Copies all fields excluding auto-created/non editable model fields) |
| `_clone_m2m_fields` | Restricted Many to many fields (i.e Test.tags) |
| `_clone_m2o_or_o2m_fields` | Restricted Many to One/One to Many fields |
| `_clone_o2o_fields` | Restricted One to One fields |
| `_clone_linked_m2m_fields` | Restricted Many to Many fields that should be linked to the new instance |
#### Implicit (include all except these fields)
| Attribute | Description |
|:--------------------:|:-----------:|
| `_clone_excluded_fields` | Excluded model fields. |
`_clone_excluded_m2m_fields` | Excluded many to many fields. |
`_clone_excluded_m2o_or_o2m_fields` | Excluded Many to One/One to Many fields. |
`_clone_excluded_o2o_fields` | Excluded one to one fields. |
> **NOTE:** :warning:
>
> * Ensure to either set `_clone_excluded_*` or `_clone_*`. Using both would raise errors.
### Django Admin
#### Duplicating Models from the Django Admin view.
data:image/s3,"s3://crabby-images/1e421/1e421a58bed40618e6722d71699148745e3721a4" alt=""
##### List View
data:image/s3,"s3://crabby-images/5f511/5f5113927feb54031c4b8548243e2d24465ed87e" alt="Screenshot"
##### Change View
data:image/s3,"s3://crabby-images/40371/40371e5fcc0aa4740a64a5208219417395de8020" alt="Screenshot"
#### CloneModelAdmin class attributes
data:image/s3,"s3://crabby-images/2a9a4/2a9a471b0c56d5e226d85d948b16bc5953e87cfb" alt=""
> **NOTE:** :warning:
>
> * Ensure that `model_clone` is placed before `django.contrib.admin`
```python
INSTALLED_APPS = [
'model_clone',
'django.contrib.admin',
'...',
]
```
## Advanced Usage
### Signals
#### pre\_clone\_save, post\_clone\_save
data:image/s3,"s3://crabby-images/782a7/782a777f2ca4a99101ba12d97f9a342e6595fadc" alt=""
### Clone Many to Many fields
#### Using the `CloneModel`
data:image/s3,"s3://crabby-images/94f29/94f29fa36017d7d553c668e322eba552cdaf28eb" alt=""
#### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/ff943/ff943e725fa57be8846f519c52f877c3204d98e0" alt="carbon (37)"
data:image/s3,"s3://crabby-images/ed057/ed057d94ebbfe84b23a3f13c9bbd9d2bc62fb4dd" alt=""
### Multi database support
data:image/s3,"s3://crabby-images/e7048/e7048248224fdd2404d9310fba7a7633bc8689b5" alt=""
## Compatibility
| Python | Supported version |
|--------------|--------------------|
| Python2.x | `<=2.5.3` |
| Python3.5 | `<=2.9.6` |
| Python3.6+ | All versions |
| Django | Supported version |
|--------------|---------------------|
| 1.11 | `<=2.7.2` |
| 2.x | All versions |
| 3.x | All versions |
## Running locally
```shell
$ git clone git@github.com:tj-django/django-clone.git
$ make default-user
$ make run
```
Spins up a django server running the demo app.
Visit http://127.0.0.1:8000
## Found a Bug?
To file a bug or submit a patch, please head over to [django-clone on github](https://github.com/tj-django/django-clone/issues).
If you feel generous and want to show some extra appreciation:
Support me with a :star:
[![Buy me a coffee][buymeacoffee-shield]][buymeacoffee]
[buymeacoffee]: https://www.buymeacoffee.com/jackton1
[buymeacoffee-shield]: https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png
## Contributors β¨
Thanks goes to these wonderful people:
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
%package -n python3-django-clone
Summary: Create a clone of a django model instance.
Provides: python-django-clone
BuildRequires: python3-devel
BuildRequires: python3-setuptools
BuildRequires: python3-pip
%description -n python3-django-clone
| Python | Django | Downloads | Code Style |
|:---------:|:-------:|:-----------:|:--------------:|
| [data:image/s3,"s3://crabby-images/41513/415134e01461860b527eda09785023dc55c5f478" alt="PyPI - Python Version"](https://pypi.org/project/django-clone) | [data:image/s3,"s3://crabby-images/eaecb/eaecb11df616e39bc6f6ec81d9bef150f59bc6e0" alt="PyPI - Django Version"](https://docs.djangoproject.com/en/3.2/releases/) | [data:image/s3,"s3://crabby-images/d53c5/d53c5f76049b40150dcc1d24e1eb7de5ac0432cb" alt="Downloads"](https://pepy.tech/project/django-clone) | [data:image/s3,"s3://crabby-images/98647/986475842f2907062b79c4bb27fdd075d638e5b9" alt="Code style: black"](https://github.com/psf/black) |
| PyPI | Test | Vulnerabilities | Coverage | Code Quality | Pre-Commit |
|:---------------:|:----:|:---------------:|:--------:|:-------------:|:-------------:|
| [data:image/s3,"s3://crabby-images/c4fba/c4fbab60b5d53b9776a750b49cdda5edd360acd0" alt="PyPI version"](https://badge.fury.io/py/django-clone) | [data:image/s3,"s3://crabby-images/627a0/627a03208f8c045f597c111965ea7a882d804c45" alt="Test"](https://github.com/tj-django/django-clone/actions?query=workflow%3ATest) | [data:image/s3,"s3://crabby-images/3e97e/3e97e3804379a4f70df67b5bf835ec6a312fffc7" alt="Known Vulnerabilities"](https://snyk.io/test/github/tj-django/django-clone?targetFile=requirements.txt) | [data:image/s3,"s3://crabby-images/7c74d/7c74dd4e779f215c7a50464ee84ff7c155af86ab" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Coverage)
[data:image/s3,"s3://crabby-images/98695/986959e1f606ecbc2ac5fd26971b517339e1a975" alt="codecov"](https://codecov.io/gh/tj-django/django-clone)| [data:image/s3,"s3://crabby-images/16cd8/16cd800099bcc76fe80572c7d70b5b2ccf7d0435" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Grade) | [data:image/s3,"s3://crabby-images/e63dd/e63dd438b75fb26a6bda7b8c6ccdf0a0f6f248cd" alt="pre-commit.ci status"](https://results.pre-commit.ci/latest/github/tj-django/django-clone/main) |
## django-clone
Create copies of a model instance with explicit control on how the instance should be duplicated (limiting fields or related objects copied) with unique field detection.
This solves the problem introduced by using `instance.pk = None` and `instance.save()` which results in copying more object state than required.
## Features
* 100% test coverage.
* More control over how a model instance should be duplicated
* Multi Database support i.e Create duplicates on one or more databases.
* Restrict fields used for creating a duplicate instance.
* Detects unique fields and naively adds a suffix `copy {count}` to each duplicate instance (for supported fields only).
* Optionally differentiate between a duplicate instance and the original by appending a **copy** suffix to non unique fields (for supported fields only).
## Table of Contents
* [Installation](#installation)
* [Usage](#usage)
* [Subclassing the `CloneModel`](#subclassing-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin)
* [Duplicating a model instance](#duplicating-a-model-instance)
* [Bulk cloning a model](#bulk-cloning-a-model)
* [Creating clones without subclassing `CloneMixin`.](#creating-clones-without-subclassing-clonemixin)
* [CloneMixin attributes](#clonemixin-attributes)
* [Explicit (include only these fields)](#explicit-include-only-these-fields)
* [Implicit (include all except these fields)](#implicit-include-all-except-these-fields)
* [Django Admin](#django-admin)
* [Duplicating Models from the Django Admin view.](#duplicating-models-from-the-django-admin-view)
* [List View](#list-view)
* [Change View](#change-view)
* [CloneModelAdmin class attributes](#clonemodeladmin-class-attributes)
* [Advanced Usage](#advanced-usage)
* [Signals](#signals)
* [pre\_clone\_save, post\_clone\_save](#pre_clone_save-post_clone_save)
* [Clone Many to Many fields](#clone-many-to-many-fields)
* [Using the `CloneModel`](#using-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin-1)
* [Multi database support](#multi-database-support)
* [Compatibility](#compatibility)
* [Running locally](#running-locally)
* [Found a Bug?](#found-a-bug)
* [Contributors β¨](#contributors-)
## Installation
data:image/s3,"s3://crabby-images/74157/741570a79b1d0924b85f07742aabc30e00b3da23" alt=""
## Usage
### Subclassing the `CloneModel`
data:image/s3,"s3://crabby-images/cb9e9/cb9e9a5beb4a92111decf68a57751268cb0c5261" alt=""
### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/c001a/c001ab7826dd88bac6f9522230c6cbc6c24f6900" alt=""
### Duplicating a model instance
data:image/s3,"s3://crabby-images/fcb96/fcb96fad12614b72d5787fd034f48febeb81e379" alt=""
### Bulk cloning a model
data:image/s3,"s3://crabby-images/6231e/6231e8c8de4a41cbe64a8b0fd43a306a3a81f6be" alt=""
### Creating clones without subclassing `CloneMixin`.
> **NOTE:** :warning:
>
> * This method won't copy over related objects like Many to Many/One to Many relationships.
> * Ensure that required fields skipped from being cloned are passed in using the `attrs` kwargs.
data:image/s3,"s3://crabby-images/d0002/d00025dcb0e5807988b85be1af0461a96f6006db" alt=""
### CloneMixin attributes
| Attribute | Description |
|:------------------------------:|:------------:|
| `DUPLICATE_SUFFIX` | Suffix to append to duplicates
(NOTE: This requires `USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS`
to be enabled and supports string fields). |
`USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS` | Enable appending the `DUPLICATE_SUFFIX` to new cloned instances. |
`UNIQUE_DUPLICATE_SUFFIX` | Suffix to append to unique fields |
`USE_UNIQUE_DUPLICATE_SUFFIX` | Enable appending the `UNIQUE_DUPLICATE_SUFFIX` to new cloned instances. |
`MAX_UNIQUE_DUPLICATE_QUERY_ATTEMPTS` | The max query attempt while generating unique values for a case of unique conflicts. |
#### Explicit (include only these fields)
| Attribute | Description |
|:------------------------------:|:------------:|
| `_clone_fields` | Restrict the list of fields to copy from the instance (By default: Copies all fields excluding auto-created/non editable model fields) |
| `_clone_m2m_fields` | Restricted Many to many fields (i.e Test.tags) |
| `_clone_m2o_or_o2m_fields` | Restricted Many to One/One to Many fields |
| `_clone_o2o_fields` | Restricted One to One fields |
| `_clone_linked_m2m_fields` | Restricted Many to Many fields that should be linked to the new instance |
#### Implicit (include all except these fields)
| Attribute | Description |
|:--------------------:|:-----------:|
| `_clone_excluded_fields` | Excluded model fields. |
`_clone_excluded_m2m_fields` | Excluded many to many fields. |
`_clone_excluded_m2o_or_o2m_fields` | Excluded Many to One/One to Many fields. |
`_clone_excluded_o2o_fields` | Excluded one to one fields. |
> **NOTE:** :warning:
>
> * Ensure to either set `_clone_excluded_*` or `_clone_*`. Using both would raise errors.
### Django Admin
#### Duplicating Models from the Django Admin view.
data:image/s3,"s3://crabby-images/1e421/1e421a58bed40618e6722d71699148745e3721a4" alt=""
##### List View
data:image/s3,"s3://crabby-images/5f511/5f5113927feb54031c4b8548243e2d24465ed87e" alt="Screenshot"
##### Change View
data:image/s3,"s3://crabby-images/40371/40371e5fcc0aa4740a64a5208219417395de8020" alt="Screenshot"
#### CloneModelAdmin class attributes
data:image/s3,"s3://crabby-images/2a9a4/2a9a471b0c56d5e226d85d948b16bc5953e87cfb" alt=""
> **NOTE:** :warning:
>
> * Ensure that `model_clone` is placed before `django.contrib.admin`
```python
INSTALLED_APPS = [
'model_clone',
'django.contrib.admin',
'...',
]
```
## Advanced Usage
### Signals
#### pre\_clone\_save, post\_clone\_save
data:image/s3,"s3://crabby-images/782a7/782a777f2ca4a99101ba12d97f9a342e6595fadc" alt=""
### Clone Many to Many fields
#### Using the `CloneModel`
data:image/s3,"s3://crabby-images/94f29/94f29fa36017d7d553c668e322eba552cdaf28eb" alt=""
#### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/ff943/ff943e725fa57be8846f519c52f877c3204d98e0" alt="carbon (37)"
data:image/s3,"s3://crabby-images/ed057/ed057d94ebbfe84b23a3f13c9bbd9d2bc62fb4dd" alt=""
### Multi database support
data:image/s3,"s3://crabby-images/e7048/e7048248224fdd2404d9310fba7a7633bc8689b5" alt=""
## Compatibility
| Python | Supported version |
|--------------|--------------------|
| Python2.x | `<=2.5.3` |
| Python3.5 | `<=2.9.6` |
| Python3.6+ | All versions |
| Django | Supported version |
|--------------|---------------------|
| 1.11 | `<=2.7.2` |
| 2.x | All versions |
| 3.x | All versions |
## Running locally
```shell
$ git clone git@github.com:tj-django/django-clone.git
$ make default-user
$ make run
```
Spins up a django server running the demo app.
Visit http://127.0.0.1:8000
## Found a Bug?
To file a bug or submit a patch, please head over to [django-clone on github](https://github.com/tj-django/django-clone/issues).
If you feel generous and want to show some extra appreciation:
Support me with a :star:
[![Buy me a coffee][buymeacoffee-shield]][buymeacoffee]
[buymeacoffee]: https://www.buymeacoffee.com/jackton1
[buymeacoffee-shield]: https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png
## Contributors β¨
Thanks goes to these wonderful people:
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
%package help
Summary: Development documents and examples for django-clone
Provides: python3-django-clone-doc
%description help
| Python | Django | Downloads | Code Style |
|:---------:|:-------:|:-----------:|:--------------:|
| [data:image/s3,"s3://crabby-images/41513/415134e01461860b527eda09785023dc55c5f478" alt="PyPI - Python Version"](https://pypi.org/project/django-clone) | [data:image/s3,"s3://crabby-images/eaecb/eaecb11df616e39bc6f6ec81d9bef150f59bc6e0" alt="PyPI - Django Version"](https://docs.djangoproject.com/en/3.2/releases/) | [data:image/s3,"s3://crabby-images/d53c5/d53c5f76049b40150dcc1d24e1eb7de5ac0432cb" alt="Downloads"](https://pepy.tech/project/django-clone) | [data:image/s3,"s3://crabby-images/98647/986475842f2907062b79c4bb27fdd075d638e5b9" alt="Code style: black"](https://github.com/psf/black) |
| PyPI | Test | Vulnerabilities | Coverage | Code Quality | Pre-Commit |
|:---------------:|:----:|:---------------:|:--------:|:-------------:|:-------------:|
| [data:image/s3,"s3://crabby-images/c4fba/c4fbab60b5d53b9776a750b49cdda5edd360acd0" alt="PyPI version"](https://badge.fury.io/py/django-clone) | [data:image/s3,"s3://crabby-images/627a0/627a03208f8c045f597c111965ea7a882d804c45" alt="Test"](https://github.com/tj-django/django-clone/actions?query=workflow%3ATest) | [data:image/s3,"s3://crabby-images/3e97e/3e97e3804379a4f70df67b5bf835ec6a312fffc7" alt="Known Vulnerabilities"](https://snyk.io/test/github/tj-django/django-clone?targetFile=requirements.txt) | [data:image/s3,"s3://crabby-images/7c74d/7c74dd4e779f215c7a50464ee84ff7c155af86ab" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Coverage)
[data:image/s3,"s3://crabby-images/98695/986959e1f606ecbc2ac5fd26971b517339e1a975" alt="codecov"](https://codecov.io/gh/tj-django/django-clone)| [data:image/s3,"s3://crabby-images/16cd8/16cd800099bcc76fe80572c7d70b5b2ccf7d0435" alt="Codacy Badge"](https://www.codacy.com/gh/tj-django/django-clone?utm_source=github.com\&utm_medium=referral\&utm_content=tj-django/django-clone\&utm_campaign=Badge_Grade) | [data:image/s3,"s3://crabby-images/e63dd/e63dd438b75fb26a6bda7b8c6ccdf0a0f6f248cd" alt="pre-commit.ci status"](https://results.pre-commit.ci/latest/github/tj-django/django-clone/main) |
## django-clone
Create copies of a model instance with explicit control on how the instance should be duplicated (limiting fields or related objects copied) with unique field detection.
This solves the problem introduced by using `instance.pk = None` and `instance.save()` which results in copying more object state than required.
## Features
* 100% test coverage.
* More control over how a model instance should be duplicated
* Multi Database support i.e Create duplicates on one or more databases.
* Restrict fields used for creating a duplicate instance.
* Detects unique fields and naively adds a suffix `copy {count}` to each duplicate instance (for supported fields only).
* Optionally differentiate between a duplicate instance and the original by appending a **copy** suffix to non unique fields (for supported fields only).
## Table of Contents
* [Installation](#installation)
* [Usage](#usage)
* [Subclassing the `CloneModel`](#subclassing-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin)
* [Duplicating a model instance](#duplicating-a-model-instance)
* [Bulk cloning a model](#bulk-cloning-a-model)
* [Creating clones without subclassing `CloneMixin`.](#creating-clones-without-subclassing-clonemixin)
* [CloneMixin attributes](#clonemixin-attributes)
* [Explicit (include only these fields)](#explicit-include-only-these-fields)
* [Implicit (include all except these fields)](#implicit-include-all-except-these-fields)
* [Django Admin](#django-admin)
* [Duplicating Models from the Django Admin view.](#duplicating-models-from-the-django-admin-view)
* [List View](#list-view)
* [Change View](#change-view)
* [CloneModelAdmin class attributes](#clonemodeladmin-class-attributes)
* [Advanced Usage](#advanced-usage)
* [Signals](#signals)
* [pre\_clone\_save, post\_clone\_save](#pre_clone_save-post_clone_save)
* [Clone Many to Many fields](#clone-many-to-many-fields)
* [Using the `CloneModel`](#using-the-clonemodel)
* [Using the `CloneMixin`](#using-the-clonemixin-1)
* [Multi database support](#multi-database-support)
* [Compatibility](#compatibility)
* [Running locally](#running-locally)
* [Found a Bug?](#found-a-bug)
* [Contributors β¨](#contributors-)
## Installation
data:image/s3,"s3://crabby-images/74157/741570a79b1d0924b85f07742aabc30e00b3da23" alt=""
## Usage
### Subclassing the `CloneModel`
data:image/s3,"s3://crabby-images/cb9e9/cb9e9a5beb4a92111decf68a57751268cb0c5261" alt=""
### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/c001a/c001ab7826dd88bac6f9522230c6cbc6c24f6900" alt=""
### Duplicating a model instance
data:image/s3,"s3://crabby-images/fcb96/fcb96fad12614b72d5787fd034f48febeb81e379" alt=""
### Bulk cloning a model
data:image/s3,"s3://crabby-images/6231e/6231e8c8de4a41cbe64a8b0fd43a306a3a81f6be" alt=""
### Creating clones without subclassing `CloneMixin`.
> **NOTE:** :warning:
>
> * This method won't copy over related objects like Many to Many/One to Many relationships.
> * Ensure that required fields skipped from being cloned are passed in using the `attrs` kwargs.
data:image/s3,"s3://crabby-images/d0002/d00025dcb0e5807988b85be1af0461a96f6006db" alt=""
### CloneMixin attributes
| Attribute | Description |
|:------------------------------:|:------------:|
| `DUPLICATE_SUFFIX` | Suffix to append to duplicates
(NOTE: This requires `USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS`
to be enabled and supports string fields). |
`USE_DUPLICATE_SUFFIX_FOR_NON_UNIQUE_FIELDS` | Enable appending the `DUPLICATE_SUFFIX` to new cloned instances. |
`UNIQUE_DUPLICATE_SUFFIX` | Suffix to append to unique fields |
`USE_UNIQUE_DUPLICATE_SUFFIX` | Enable appending the `UNIQUE_DUPLICATE_SUFFIX` to new cloned instances. |
`MAX_UNIQUE_DUPLICATE_QUERY_ATTEMPTS` | The max query attempt while generating unique values for a case of unique conflicts. |
#### Explicit (include only these fields)
| Attribute | Description |
|:------------------------------:|:------------:|
| `_clone_fields` | Restrict the list of fields to copy from the instance (By default: Copies all fields excluding auto-created/non editable model fields) |
| `_clone_m2m_fields` | Restricted Many to many fields (i.e Test.tags) |
| `_clone_m2o_or_o2m_fields` | Restricted Many to One/One to Many fields |
| `_clone_o2o_fields` | Restricted One to One fields |
| `_clone_linked_m2m_fields` | Restricted Many to Many fields that should be linked to the new instance |
#### Implicit (include all except these fields)
| Attribute | Description |
|:--------------------:|:-----------:|
| `_clone_excluded_fields` | Excluded model fields. |
`_clone_excluded_m2m_fields` | Excluded many to many fields. |
`_clone_excluded_m2o_or_o2m_fields` | Excluded Many to One/One to Many fields. |
`_clone_excluded_o2o_fields` | Excluded one to one fields. |
> **NOTE:** :warning:
>
> * Ensure to either set `_clone_excluded_*` or `_clone_*`. Using both would raise errors.
### Django Admin
#### Duplicating Models from the Django Admin view.
data:image/s3,"s3://crabby-images/1e421/1e421a58bed40618e6722d71699148745e3721a4" alt=""
##### List View
data:image/s3,"s3://crabby-images/5f511/5f5113927feb54031c4b8548243e2d24465ed87e" alt="Screenshot"
##### Change View
data:image/s3,"s3://crabby-images/40371/40371e5fcc0aa4740a64a5208219417395de8020" alt="Screenshot"
#### CloneModelAdmin class attributes
data:image/s3,"s3://crabby-images/2a9a4/2a9a471b0c56d5e226d85d948b16bc5953e87cfb" alt=""
> **NOTE:** :warning:
>
> * Ensure that `model_clone` is placed before `django.contrib.admin`
```python
INSTALLED_APPS = [
'model_clone',
'django.contrib.admin',
'...',
]
```
## Advanced Usage
### Signals
#### pre\_clone\_save, post\_clone\_save
data:image/s3,"s3://crabby-images/782a7/782a777f2ca4a99101ba12d97f9a342e6595fadc" alt=""
### Clone Many to Many fields
#### Using the `CloneModel`
data:image/s3,"s3://crabby-images/94f29/94f29fa36017d7d553c668e322eba552cdaf28eb" alt=""
#### Using the `CloneMixin`
data:image/s3,"s3://crabby-images/ff943/ff943e725fa57be8846f519c52f877c3204d98e0" alt="carbon (37)"
data:image/s3,"s3://crabby-images/ed057/ed057d94ebbfe84b23a3f13c9bbd9d2bc62fb4dd" alt=""
### Multi database support
data:image/s3,"s3://crabby-images/e7048/e7048248224fdd2404d9310fba7a7633bc8689b5" alt=""
## Compatibility
| Python | Supported version |
|--------------|--------------------|
| Python2.x | `<=2.5.3` |
| Python3.5 | `<=2.9.6` |
| Python3.6+ | All versions |
| Django | Supported version |
|--------------|---------------------|
| 1.11 | `<=2.7.2` |
| 2.x | All versions |
| 3.x | All versions |
## Running locally
```shell
$ git clone git@github.com:tj-django/django-clone.git
$ make default-user
$ make run
```
Spins up a django server running the demo app.
Visit http://127.0.0.1:8000
## Found a Bug?
To file a bug or submit a patch, please head over to [django-clone on github](https://github.com/tj-django/django-clone/issues).
If you feel generous and want to show some extra appreciation:
Support me with a :star:
[![Buy me a coffee][buymeacoffee-shield]][buymeacoffee]
[buymeacoffee]: https://www.buymeacoffee.com/jackton1
[buymeacoffee-shield]: https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png
## Contributors β¨
Thanks goes to these wonderful people:
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
%prep
%autosetup -n django-clone-5.3.1
%build
%py3_build
%install
%py3_install
install -d -m755 %{buildroot}/%{_pkgdocdir}
if [ -d doc ]; then cp -arf doc %{buildroot}/%{_pkgdocdir}; fi
if [ -d docs ]; then cp -arf docs %{buildroot}/%{_pkgdocdir}; fi
if [ -d example ]; then cp -arf example %{buildroot}/%{_pkgdocdir}; fi
if [ -d examples ]; then cp -arf examples %{buildroot}/%{_pkgdocdir}; fi
pushd %{buildroot}
if [ -d usr/lib ]; then
find usr/lib -type f -printf "/%h/%f\n" >> filelist.lst
fi
if [ -d usr/lib64 ]; then
find usr/lib64 -type f -printf "/%h/%f\n" >> filelist.lst
fi
if [ -d usr/bin ]; then
find usr/bin -type f -printf "/%h/%f\n" >> filelist.lst
fi
if [ -d usr/sbin ]; then
find usr/sbin -type f -printf "/%h/%f\n" >> filelist.lst
fi
touch doclist.lst
if [ -d usr/share/man ]; then
find usr/share/man -type f -printf "/%h/%f.gz\n" >> doclist.lst
fi
popd
mv %{buildroot}/filelist.lst .
mv %{buildroot}/doclist.lst .
%files -n python3-django-clone -f filelist.lst
%dir %{python3_sitelib}/*
%files help -f doclist.lst
%{_docdir}/*
%changelog
* Tue Apr 25 2023 Python_Bot - 5.3.1-1
- Package Spec generated