Metadata-Version: 2.1
Name: django-copyist
Version: 0.1.2
Summary: Tool for precise and efficient django model copying
Home-page: https://github.com/abondar/django-copyist
License: Apache-2.0
Keywords: django,copy,clone,model
Author: abondar
Requires-Python: >=3.11,<4.0
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Requires-Dist: Django (>=3.2)
Project-URL: Documentation, https://abondar.github.io/django-copyist/
Project-URL: Repository, https://github.com/abondar/django-copyist.git
Description-Content-Type: text/x-rst

django-copyist
==========================================

Tool for precise and efficient copying of Django models instances.

Do you live in fear of the day when PM will come to you
and ask to implement a copy feature for your root model,
that has a lot of relations and you know that it will be a pain to implement it in a way that will work properly?

Well, fear no more, as `django-copyist` got you covered


Features
--------

- **Precision** - Copy only what you need in a way that you need with help of custom copy actions.
- **Readability** - With declarative style config - you can easily see what and how is copied, no need to hop between models and parse it in your head.
- **Efficiency** - Unlike some other solutions and naive approaches, copyist is copying your data without recursive hopping between model instances, which gives it a magnitudes of speedup on big data sets.
- **Flexibility** - Copyist covers all steps that are there for data copy, including validation of data, pre-copy and post-copy actions. Copyist also work good with de-normalized data, not judging you for your choices.
- **Interactive** - Copyist provides a way to interact with the copy process, allowing application to see what exactly is going to be done, and propagate that info to end user to decide if he wants to go through with it.
- **Reusability** - With copyist your copy flow is not nailed down to model, allowing you defining different approaches for same model, and at the same time reuse existing configurations.

Motivation
----------

This project was build as in-house tool for project with complex hierarchy of models,
where we needed to copy them in a very particular way.

Existing solutions like `django-clone <https://github.com/tj-django/django-clone>`_  were designed
in a way that didn't fit our needs, as they required to modify models and
didn't allow to have full control over the copying process.

This project aims to provide a more flexible and efficient way to copy Django models instances, while
not affecting existing codebase.

Quickstart
==========

This pages aims to get you going with django-copyist as quickly as possible.

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

To install with pip:

.. code-block:: console

   pip install django-copyist

To install with poetry:

.. code-block:: console

   poetry add django-copyist


Basic usage
-----------

Assuming you have following models in your Django app:

.. code-block:: python

    from django.db import models


    class Company(models.Model):
        name = models.CharField(max_length=100)
        address = models.CharField(max_length=100)


    class Project(models.Model):
        name = models.CharField(max_length=100)
        company = models.ForeignKey(
            Company, related_name="projects", on_delete=models.CASCADE
        )


    class Employee(models.Model):
        name = models.CharField(max_length=100)
        company = models.ForeignKey(
            Company, related_name="employees", on_delete=models.CASCADE
        )


    class Counterpart(models.Model):
        name = models.CharField(max_length=100)
        external_id = models.IntegerField()
        project = models.ForeignKey(
            Project, related_name="counterparts", on_delete=models.CASCADE
        )


    class Task(models.Model):
        name = models.CharField(max_length=100)
        description = models.TextField()

        assignee = models.ForeignKey(
            Employee, related_name="tasks", on_delete=models.CASCADE
        )
        project = models.ForeignKey(Project, related_name="tasks", on_delete=models.CASCADE)
        counterparts = models.ManyToManyField(Counterpart, related_name="tasks")


And you want to create full copy of company with all nested data, but also want it to be created with different name and address.
In this case you should write following ModelCopyConfig

.. code-block:: python

    from django_copyist.config import (
        ModelCopyConfig,
        TAKE_FROM_ORIGIN,
        MakeCopy,
        UpdateToCopied,
        FieldCopyConfig,
        CopyActions,
    )
    from example.demo.models import (
        Project,
        Counterpart,
        Task,
        Company,
        Employee,
    )


    config = ModelCopyConfig(
        model=Company,
        filter_field_to_input_key={"id": "company_id"},
        field_copy_actions={
            "name": FieldCopyConfig(
                action=CopyActions.TAKE_FROM_INPUT,
                input_key="new_company_name",
            ),
            "address": FieldCopyConfig(
                action=CopyActions.TAKE_FROM_INPUT,
                input_key="new_company_address",
            ),
            "projects": MakeCopy(
                ModelCopyConfig(
                    model=Project,
                    field_copy_actions={
                        "name": TAKE_FROM_ORIGIN,
                        "counterparts": MakeCopy(
                            ModelCopyConfig(
                                model=Counterpart,
                                field_copy_actions={
                                    "name": TAKE_FROM_ORIGIN,
                                    "external_id": TAKE_FROM_ORIGIN,
                                },
                            )
                        ),
                    },
                )
            ),
            "employees": MakeCopy(
                ModelCopyConfig(
                    model=Employee,
                    field_copy_actions={
                        "name": TAKE_FROM_ORIGIN,
                    },
                )
            ),
        },
        compound_copy_actions=[
            ModelCopyConfig(
                model=Task,
                field_copy_actions={
                    "name": TAKE_FROM_ORIGIN,
                    "description": TAKE_FROM_ORIGIN,
                    "counterparts": UpdateToCopied(Counterpart),
                    "project": UpdateToCopied(Project),
                    "assignee": UpdateToCopied(Employee),
                },
            )
        ],
    )

And then you can execute copy action like this:

.. code-block:: python

    from django_copyist.copy_request import CopyRequest
    from django_copyist.copyist import CopyistConfig, Copyist

    copy_request = CopyRequest(
        config=CopyistConfig([config]),
        input_data={
            "company_id": company_id,
            "new_company_name": new_company_name,
            "new_company_address": new_company_address,
        },
        confirm_write=False,
    )
    result = Copyist(copy_request).execute_copy_request()

With this, all company data should be copied.
That seems like a lot to take in, so let's break it down to what exactly happens here:

1. We define a `ModelCopyConfig` for the `Company` model.

.. code-block:: python

    config = ModelCopyConfig(
        model=Company,
        filter_field_to_input_key={"id": "company_id"},
    ...

`ModelCopyConfig` is a class that defines how to copy a model. It takes the model class as the first argument and a dictionary that maps the filter field to the input key. This is used to find the object to copy.

2. Next we define `field_copy_actions` for the `Company` model.

.. code-block:: python

    field_copy_actions={
        "name": FieldCopyConfig(
            action=CopyActions.TAKE_FROM_INPUT,
            input_key="new_company_name",
        ),
        "address": FieldCopyConfig(
            action=CopyActions.TAKE_FROM_INPUT,
            input_key="new_company_address",
        ),
        "projects": MakeCopy(
            ModelCopyConfig(
                model=Project,
                field_copy_actions={
                    "name": TAKE_FROM_ORIGIN,
                    "counterparts": MakeCopy(
                        ModelCopyConfig(
                            model=Counterpart,
                            field_copy_actions={
                                "name": TAKE_FROM_ORIGIN,
                                "external_id": TAKE_FROM_ORIGIN,
                            },
                        )
                    ),
                },
            )
        ),
        "employees": MakeCopy(
            ModelCopyConfig(
                model=Employee,
                field_copy_actions={
                    "name": TAKE_FROM_ORIGIN,
                },
            )
        ),
    ...

`field_copy_actions` is a dictionary that maps the field name to a `FieldCopyConfig` object.

The `FieldCopyConfig` object defines how to copy the field. In this case, we take the `name` and `address` fields from the input data.

`TAKE_FROM_ORIGIN` is a shortcut for creating `FieldCopyConfig` with `CopyActions.TAKE_FROM_ORIGIN` action, which takes value for new object from original object.

We also define how to copy the `projects` and `employees` fields.

We use the `MakeCopy` action to copy the related objects.
`MakeCopy` is a shortcut for creating `FieldCopyConfig` with `CopyActions.MAKE_COPY` action and reference to given model.
Nested `MakeCopy` automatically propagate parent id to child object.

3. We define `compound_copy_actions` for the `Company` model.

.. code-block:: python

    compound_copy_actions=[
        ModelCopyConfig(
            model=Task,
            field_copy_actions={
                "name": TAKE_FROM_ORIGIN,
                "description": TAKE_FROM_ORIGIN,
                "counterparts": UpdateToCopied(Counterpart),
                "project": UpdateToCopied(Project),
                "assignee": UpdateToCopied(Employee),
            },
        )
    ...

`compound_copy_actions` is a list of `ModelCopyConfig` objects that define how
to copy related objects that are not directly related to the model, or related through multiple relations that need to be created beforehand.

`compound_copy_actions` are executed after all fields are copied.

In this case, we define how to copy the `Task` model. We take the `name` and `description` fields from the original object. We also define how to copy the `counterparts`, `project`, and `assignee` fields.

`UpdateToCopied` is a shortcut for creating `FieldCopyConfig` with `CopyActions.UPDATE_TO_COPIED` action and reference to given model.
It will search mapping of previously copied objects and update reference to copied object.

4. We create a `CopyRequest` object with the `CopyistConfig` and input data.

.. code-block:: python

    copy_request = CopyRequest(
        config=CopyistConfig([config]),
        input_data={
            "company_id": company_id,
            "new_company_name": new_company_name,
            "new_company_address": new_company_address,
        },
        confirm_write=False,
    )
    ...

`CopyRequest` is a class that defines the copy request. It takes the `CopyistConfig` object, input data, and a boolean flag that indicates whether to confirm the write operation.

`CopyistConfig` is a class that defines the configuration for the copy operation. It takes a list of `ModelCopyConfig` objects.

`input_data` is a dictionary that contains the input data for the copy operation. It is later used in filtering or `TAKE_FROM_INPUT` actions.

`confirm_write` is a boolean flag that indicates whether to confirm the write operation,
even if there are issues with matching objects in origin location with objects in target destination.
It is not used in this example, but you can read more about it in overview section of this documentation.

5. We execute the copy request.
.. code-block:: python

    result = Copyist(copy_request).execute_copy_request()

`Copyist` is a class that executes the copy request. It takes the `CopyRequest` object as an argument.

`execute_copy_request` method returns `CopyResult` object that contains information about the copy operation. Read more about it in overview section.

And like this you have copied the company with all related data and can see and edit configuration in one place.

Next steps
----------

This is just a basic example of how to use django-copyist.
It can do much more granular control on how it should execute copy, and you can read more about it in the documentation.

