graphql-engine/docs/graphql/manual/guides/data-modelling/soft-deletes.rst
2020-05-05 09:22:08 +05:30

133 lines
4.0 KiB
ReStructuredText

.. meta::
:description: Set up soft deletes for data with Hasura
:keywords: hasura, docs, guide, soft deletes
.. _soft_deletes:
Setting up soft deletes for data
================================
.. contents:: Table of contents
:backlinks: none
:depth: 1
:local:
Overview
--------
For some projects you may require records to be "soft deleted", i.e. items should not actually be removed from the
database, but should be marked with a timestamp to indicate when they were deleted.
A common approach is to add a column such as ``deleted_at`` to the table which is a nullable timestamp. When
there is a timestamp value present, the record should be treated as deleted.
**For example:** Let's imagine that we have a simple Todo application, our ``todos`` table would resemble
the following:
.. thumbnail:: /img/graphql/manual/guides/soft-deletes-example-data.png
:alt: Todo table of a todo application
In this example we only have two todos, one has ``deleted_at`` with a timestamp value and the other contains a
``null`` value. The todo with the timestamp value in ``deleted_at`` represents a deleted todo and was deleted at
the set timestamp.
Follow the below steps to set up soft deletes for the ``todos`` table:
Step 1: Add a **deleted_at** column
-----------------------------------
Add a column with the following definition to the ``todos`` table:
``deleted_at, type: timestamp, nullable, default: null``
Step 2: Use **update** instead of **delete** mutations
------------------------------------------------------
After the previous step, we have a ``deleted_at`` column whose value will be ``null`` by default unless
set explicitly.
Now in our application logic, instead of ``delete_todos`` mutations, use ``update_todos`` mutations to set
the ``deleted_at`` field to the current timestamp:
.. code-block:: graphql
:emphasize-lines: 7
# Replace the delete mutations with this update mutation
mutation {
update_todos(
where: {
name: {_eq: "Stuff already done!"}
},
_set: {deleted_at: "now()"}
) {
returning {
id
name
deleted_at
}
}
}
Step 3: Set up appropriate insert/update/delete permissions
-----------------------------------------------------------
Now, we need to ensure that appropriate :ref:`permissions <authorization>` are set to avoid
actual deletes from happening and allowing update of the ``deleted_at`` field.
Here are some typical rules we should set:
**Delete permissions** - remove all access
.. thumbnail:: /img/graphql/manual/guides/soft-deletes-delete-perms.png
:alt: Delete permissions for role user
**Insert permissions** - remove access for inserting into ``deleted_at`` column
.. thumbnail:: /img/graphql/manual/guides/soft-deletes-insert-perms.png
:alt: Insert permissions for role user
**Update permissions** - allow access for updating ``deleted_at`` column
.. thumbnail:: /img/graphql/manual/guides/soft-deletes-update-perms.png
:alt: Update permissions for role user
Step 4: Restrict access to soft-deleted records
-----------------------------------------------
Now that we have set up the soft deleting pattern for records, we need to ensure that we restrict the "deleted"
records from being accessed.
We can achieve this by setting appropriate :ref:`permissions <authorization>` for roles which have
access to the ``todos`` table.
For example, let's say that a role ``user`` can only access non-deleted todos, we need to add the following
permission rule to ensure this:
.. thumbnail:: /img/graphql/manual/guides/soft-deletes-select-perms.png
:alt: Restrict access to soft-deleted records
Now the role ``user`` can only access non-deleted ``todos``:
.. graphiql::
:view_only:
:query:
query {
todos {
id
name
deleted_at
}
}
:response:
{
"data": {
"todos": [
{
"id": "34c4e2f9-c3e2-4147-9138-f67b6e7e2947",
"name": "Get stuff done",
"deleted_at": null
}
]
}
}