Getting and Checking Consent¶
If your application passes a user’s personal information (such as PII) to third-party services, it’s often good to get the user’s consent (and for EU users protected by the GDPR, it may be legally required). This consent may be needed for two features, or for a dozen, and if you’re making use of extensions, there may be even more things that need consent.
Djblets provides a djblets.privacy.consent module for registering
things that require consent, for checking on whether consent was given, for
providing UI for consent management, and for auditing the consent choices made
by the user over time in a safe and secure way.
Consent Requirements¶
BaseConsentRequirement is the main class that everything
else centers around. Subclasses define a requirement with a unique ID and
information about what requires consent, primarily for display purposes.
Any part of a codebase requiring consent will construct an instance of this
class (or a subclass of it) and register it in the
ConsentRequirementsRegistry (accessible through
get_consent_requirements_registry().
This looks like:
from djblets.privacy.consent import (BaseConsentRequirement,
                                     get_consent_requirements_registry)
class MyConsentRequirement(BaseConsentRequirement):
    requirement_id = 'my-requirement-id'
    name = 'My Requirement'
    intent_description = (
        'A description about the requirement, presented to the user '
        'clearly and informatively.'
    )
    data_use_description = (
        'A brief summary of what data gets sent to the data processor '
        'service.'
    )
    icons = {
        '1x': '/path/to/logo.png',
        '2x': '/path/to/logo@2x.png',
    }
my_requirement = MyConsentRequirement()
get_consent_requirements_registry().register(my_requirement)
These requirements can be checked for consent to determine if consent was granted, denied, or not yet decided upon:
from djblets.privacy.consent import Consent
...
consent = my_requirement.get_consent(user)
if consent == Consent.GRANTED:
    ...
elif consent == Consent.DENIED:
    ...
elif consent == Consent.UNSET:
    ...
Tracking Consent Decisions¶
A decision made on a consent requirement is represented as a
ConsentData instance, tracked by a
BaseConsentTracker.
ConsentData stores whether a given
BaseConsentRequirement ID has been
granted or denied, along with additional data for audit purposes: The consent
decision’s timestamp, source location (which can be a URL or some other
identifier), and custom application-provided metadata.
The consent tracker (accessible via
get_consent_tracker()) tracks that
consent, recording it for later audits. It stores the data along with an
identifier that maps to the user (defaults to a SHA256 hash of their e-mail
address).
The default consent tracker uses the database (storing in the
StoredConsentData model), but applications
can change how consent is stored and looked up by subclassing the base
tracker and setting settings.DJBLETS_PRIVACY_CONSENT_TRACKER to its full
module/class path.
If using the built-in UI, much of this happens behind the scenes. If you need
to record consent directly, you can use
record_consent_data_list().
from django.utils import timezone
from djblets.privacy.consent import get_consent_tracker
...
now = timezone.now()
get_consent_tracker().record_consent_data_list(
    user,
    [
        my_requirement_1.build_consent_data(
            granted=True,
            timestamp=now,
            source='https://example.com/accounts/consent/'),
        my_requirement_2.build_consent_data(
            granted=False,
            timestamp=now,
            source='https://example.com/accounts/consent/'),
    ])
Or to get all the consent decisions filed by a user (for display in the UI,
for example), use get_all_consent().
from django.utils import six, timezone
from djblets.privacy.consent import get_consent_tracker
...
now = timezone.now()
all_consent = get_consent_tracker().get_all_consent(user)
if my_requirement_1.requirement_id in all_consent:
    if all_consent[my_requirement_1.requirement_id] == Consent.GRANTED:
       ...
    elif all_consent[my_requirement_1.requirement_id] == Consent.DENIED:
       ...
Requiring Consent Decisions¶
If you are using consent tracking, you will likely want to gate certain views
that require the user to have made consent decisions. We provide a decorator
for functional views and a mixin for class-style views. If the viewing user is
authenticated and has any pending consent requirements, they will be redirected
to the URL specified in
settings.DJBLETS_PRIVACY_PENDING_CONSENT_REDIRECT_URL.
This setting can also be a function, in which case it accepts the current
HttpRequest.
from django.http import HttpResponse
from django.views.generic.base import View
from djblets.privacy.consent.views import (CheckPendingConsentMixin,
                                           check_pending_consent)
from djblets.views.generic.base import PrePostDispatchViewMixin
class SimpleView(CheckPendingConsentMixin, PrePostDispatchViewMixin, View):
    def get(self, request, **kwargs):
        return HttpResponse('You have no pending consent requirements.')
@check_pending_consent
def simple_view(request):
     return HttpResponse('You have no pending consent requirements.')
