ethyca-fides 2.62.1b1__py2.py3-none-any.whl → 2.62.1b3__py2.py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (109) hide show
  1. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/METADATA +1 -1
  2. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/RECORD +109 -107
  3. fides/_version.py +3 -3
  4. fides/api/alembic/migrations/versions/440a5b9a3493_add_fides_user_respondent_email_.py +80 -0
  5. fides/api/db/base.py +3 -0
  6. fides/api/models/audit_log.py +0 -3
  7. fides/api/models/fides_user.py +51 -4
  8. fides/api/models/fides_user_permissions.py +25 -7
  9. fides/api/models/fides_user_respondent_email_verification.py +112 -0
  10. fides/api/oauth/roles.py +19 -0
  11. fides/api/service/connectors/query_configs/saas_query_config.py +3 -8
  12. fides/api/service/connectors/saas_connector.py +7 -4
  13. fides/api/service/privacy_request/request_runner_service.py +47 -18
  14. fides/api/tasks/storage.py +4 -3
  15. fides/common/api/scope_registry.py +6 -0
  16. fides/config/config_proxy.py +0 -1
  17. fides/config/execution_settings.py +0 -4
  18. fides/ui-build/static/admin/404.html +1 -1
  19. fides/ui-build/static/admin/add-systems/manual.html +1 -1
  20. fides/ui-build/static/admin/add-systems/multiple.html +1 -1
  21. fides/ui-build/static/admin/add-systems.html +1 -1
  22. fides/ui-build/static/admin/consent/configure/add-vendors.html +1 -1
  23. fides/ui-build/static/admin/consent/configure.html +1 -1
  24. fides/ui-build/static/admin/consent/privacy-experience/[id].html +1 -1
  25. fides/ui-build/static/admin/consent/privacy-experience/new.html +1 -1
  26. fides/ui-build/static/admin/consent/privacy-experience.html +1 -1
  27. fides/ui-build/static/admin/consent/privacy-notices/[id].html +1 -1
  28. fides/ui-build/static/admin/consent/privacy-notices/new.html +1 -1
  29. fides/ui-build/static/admin/consent/privacy-notices.html +1 -1
  30. fides/ui-build/static/admin/consent/properties.html +1 -1
  31. fides/ui-build/static/admin/consent/reporting.html +1 -1
  32. fides/ui-build/static/admin/consent.html +1 -1
  33. fides/ui-build/static/admin/data-catalog/[systemId]/projects/[projectUrn]/[resourceUrn].html +1 -1
  34. fides/ui-build/static/admin/data-catalog/[systemId]/projects/[projectUrn].html +1 -1
  35. fides/ui-build/static/admin/data-catalog/[systemId]/projects.html +1 -1
  36. fides/ui-build/static/admin/data-catalog/[systemId]/resources/[resourceUrn].html +1 -1
  37. fides/ui-build/static/admin/data-catalog/[systemId]/resources.html +1 -1
  38. fides/ui-build/static/admin/data-catalog.html +1 -1
  39. fides/ui-build/static/admin/data-discovery/action-center/[monitorId]/[systemId].html +1 -1
  40. fides/ui-build/static/admin/data-discovery/action-center/[monitorId].html +1 -1
  41. fides/ui-build/static/admin/data-discovery/action-center.html +1 -1
  42. fides/ui-build/static/admin/data-discovery/activity.html +1 -1
  43. fides/ui-build/static/admin/data-discovery/detection/[resourceUrn].html +1 -1
  44. fides/ui-build/static/admin/data-discovery/detection.html +1 -1
  45. fides/ui-build/static/admin/data-discovery/discovery/[resourceUrn].html +1 -1
  46. fides/ui-build/static/admin/data-discovery/discovery.html +1 -1
  47. fides/ui-build/static/admin/datamap.html +1 -1
  48. fides/ui-build/static/admin/dataset/[datasetId]/[collectionName]/[...subfieldNames].html +1 -1
  49. fides/ui-build/static/admin/dataset/[datasetId]/[collectionName].html +1 -1
  50. fides/ui-build/static/admin/dataset/[datasetId].html +1 -1
  51. fides/ui-build/static/admin/dataset/new.html +1 -1
  52. fides/ui-build/static/admin/dataset.html +1 -1
  53. fides/ui-build/static/admin/datastore-connection/[id].html +1 -1
  54. fides/ui-build/static/admin/datastore-connection/new.html +1 -1
  55. fides/ui-build/static/admin/datastore-connection.html +1 -1
  56. fides/ui-build/static/admin/index.html +1 -1
  57. fides/ui-build/static/admin/integrations/[id].html +1 -1
  58. fides/ui-build/static/admin/integrations.html +1 -1
  59. fides/ui-build/static/admin/lib/fides-ext-gpp.js +1 -1
  60. fides/ui-build/static/admin/lib/fides-headless.js +1 -1
  61. fides/ui-build/static/admin/lib/fides-preview.js +1 -1
  62. fides/ui-build/static/admin/lib/fides-tcf.js +2 -2
  63. fides/ui-build/static/admin/lib/fides.js +2 -2
  64. fides/ui-build/static/admin/login/[provider].html +1 -1
  65. fides/ui-build/static/admin/login.html +1 -1
  66. fides/ui-build/static/admin/messaging/[id].html +1 -1
  67. fides/ui-build/static/admin/messaging/add-template.html +1 -1
  68. fides/ui-build/static/admin/messaging.html +1 -1
  69. fides/ui-build/static/admin/poc/ant-components.html +1 -1
  70. fides/ui-build/static/admin/poc/form-experiments/AntForm.html +1 -1
  71. fides/ui-build/static/admin/poc/form-experiments/FormikAntFormItem.html +1 -1
  72. fides/ui-build/static/admin/poc/form-experiments/FormikControlled.html +1 -1
  73. fides/ui-build/static/admin/poc/form-experiments/FormikField.html +1 -1
  74. fides/ui-build/static/admin/poc/form-experiments/FormikSpreadField.html +1 -1
  75. fides/ui-build/static/admin/poc/forms.html +1 -1
  76. fides/ui-build/static/admin/poc/table-migration.html +1 -1
  77. fides/ui-build/static/admin/privacy-requests/[id].html +1 -1
  78. fides/ui-build/static/admin/privacy-requests/configure/messaging.html +1 -1
  79. fides/ui-build/static/admin/privacy-requests/configure/storage.html +1 -1
  80. fides/ui-build/static/admin/privacy-requests/configure.html +1 -1
  81. fides/ui-build/static/admin/privacy-requests.html +1 -1
  82. fides/ui-build/static/admin/properties/[id].html +1 -1
  83. fides/ui-build/static/admin/properties/add-property.html +1 -1
  84. fides/ui-build/static/admin/properties.html +1 -1
  85. fides/ui-build/static/admin/reporting/datamap.html +1 -1
  86. fides/ui-build/static/admin/settings/about.html +1 -1
  87. fides/ui-build/static/admin/settings/consent/[configuration_id]/[purpose_id].html +1 -1
  88. fides/ui-build/static/admin/settings/consent.html +1 -1
  89. fides/ui-build/static/admin/settings/custom-fields.html +1 -1
  90. fides/ui-build/static/admin/settings/domain-records.html +1 -1
  91. fides/ui-build/static/admin/settings/domains.html +1 -1
  92. fides/ui-build/static/admin/settings/email-templates.html +1 -1
  93. fides/ui-build/static/admin/settings/locations.html +1 -1
  94. fides/ui-build/static/admin/settings/organization.html +1 -1
  95. fides/ui-build/static/admin/settings/regulations.html +1 -1
  96. fides/ui-build/static/admin/systems/configure/[id]/test-datasets.html +1 -1
  97. fides/ui-build/static/admin/systems/configure/[id].html +1 -1
  98. fides/ui-build/static/admin/systems.html +1 -1
  99. fides/ui-build/static/admin/taxonomy.html +1 -1
  100. fides/ui-build/static/admin/user-management/new.html +1 -1
  101. fides/ui-build/static/admin/user-management/profile/[id].html +1 -1
  102. fides/ui-build/static/admin/user-management.html +1 -1
  103. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/WHEEL +0 -0
  104. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/entry_points.txt +0 -0
  105. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/licenses/LICENSE +0 -0
  106. {ethyca_fides-2.62.1b1.dist-info → ethyca_fides-2.62.1b3.dist-info}/top_level.txt +0 -0
  107. /fides/ui-build/static/admin/_next/static/{Sez6kqW5dI0lK60IMwICO → D-B-G-FV05_6iwXWBGS0J}/_buildManifest.js +0 -0
  108. /fides/ui-build/static/admin/_next/static/{Sez6kqW5dI0lK60IMwICO → D-B-G-FV05_6iwXWBGS0J}/_ssgManifest.js +0 -0
  109. /fides/ui-build/static/admin/_next/static/chunks/pages/{_app-94950df9c97cd079.js → _app-f1add6386651ca3d.js} +0 -0
@@ -1,11 +1,15 @@
1
- from typing import List
1
+ from typing import List, cast
2
2
 
3
3
  from sqlalchemy import ARRAY, Column, ForeignKey, String
4
- from sqlalchemy.orm import backref, relationship
4
+ from sqlalchemy.orm import Session, relationship
5
5
 
6
6
  from fides.api.db.base_class import Base
7
7
  from fides.api.models.fides_user import FidesUser
8
- from fides.api.oauth.roles import ROLES_TO_SCOPES_MAPPING
8
+ from fides.api.oauth.roles import (
9
+ EXTERNAL_RESPONDENT,
10
+ RESPONDENT,
11
+ ROLES_TO_SCOPES_MAPPING,
12
+ )
9
13
 
10
14
 
11
15
  class FidesUserPermissions(Base):
@@ -13,10 +17,7 @@ class FidesUserPermissions(Base):
13
17
 
14
18
  user_id = Column(String, ForeignKey(FidesUser.id), nullable=False, unique=True)
15
19
  roles = Column(ARRAY(String), nullable=False, server_default="{}", default=dict)
16
- user = relationship(
17
- FidesUser,
18
- backref=backref("permissions", cascade="all,delete", uselist=False),
19
- )
20
+ user = relationship(FidesUser, back_populates="permissions", uselist=False)
20
21
 
21
22
  @property
22
23
  def total_scopes(self) -> List[str]:
@@ -26,3 +27,20 @@ class FidesUserPermissions(Base):
26
27
  all_scopes += ROLES_TO_SCOPES_MAPPING.get(role, [])
27
28
 
28
29
  return sorted(list(set(all_scopes)))
30
+
31
+ def is_respondent(self) -> bool:
32
+ """Check if the user is a respondent."""
33
+ return any(role in self.roles for role in [RESPONDENT, EXTERNAL_RESPONDENT])
34
+
35
+ def update_roles(self, db: Session, new_roles: List[str]) -> None:
36
+ """Update the user's roles if allowed.
37
+ Raises ValueError if role changes are not allowed."""
38
+ if self.is_respondent():
39
+ raise ValueError("Role changes are not allowed for respondents")
40
+
41
+ self.roles = new_roles
42
+ self.save(db)
43
+
44
+ def update(self, db: Session, *, data: dict) -> "FidesUserPermissions":
45
+ """Update the user permissions with the provided data."""
46
+ return cast(FidesUserPermissions, super().update(db, data=data))
@@ -0,0 +1,112 @@
1
+ from __future__ import annotations
2
+
3
+ from datetime import datetime, timedelta, timezone
4
+ from typing import TYPE_CHECKING, Any, Optional
5
+
6
+ from sqlalchemy import Column, DateTime, ForeignKey, String
7
+ from sqlalchemy.ext.declarative import declared_attr
8
+ from sqlalchemy.orm import Session, relationship
9
+
10
+ from fides.api.cryptography.cryptographic_util import generate_secure_random_string
11
+ from fides.api.db.base_class import Base
12
+ from fides.api.util.identity_verification import IdentityVerificationMixin
13
+ from fides.config import get_config
14
+
15
+ CONFIG = get_config()
16
+
17
+
18
+ if TYPE_CHECKING:
19
+ from fides.api.models.fides_user import FidesUser
20
+
21
+ # Access links stay active for 45 days - the same as the DSR expiration. A new link is generated for each email.
22
+ # The emails are created for new DSRs which are assigned to the respondent.
23
+ ACCESS_LINK_TTL_DAYS = 45
24
+
25
+
26
+ class FidesUserRespondentEmailVerification(Base, IdentityVerificationMixin):
27
+ """Model for handling email verification for external respondents.
28
+
29
+ This handles two types of verification:
30
+ 1. Access links - long-lived (45 days) for initial access
31
+ 2. Verification codes - short-lived (1 hour) for actual verification
32
+
33
+ When an email is sent to an external respondent, a new verification is created with a new access token is created.
34
+ When a respondent clicks the link in the email, the access token is verified and a verification code is generated.
35
+ The verification code is sent to the respondent's email address and the respondent is prompted to enter the code.
36
+ Verification is handled by the `IdentityVerificationMixin` class.
37
+ """
38
+
39
+ @declared_attr
40
+ def __tablename__(self) -> str:
41
+ return "fides_user_respondent_email_verification"
42
+
43
+ user_id = Column(
44
+ String,
45
+ ForeignKey("fidesuser.id", ondelete="CASCADE"),
46
+ nullable=False,
47
+ index=True,
48
+ )
49
+
50
+ access_token = Column(
51
+ String, nullable=False, unique=True, index=True
52
+ ) # Token for the access link
53
+ access_token_expires_at = Column(DateTime(timezone=True), nullable=False)
54
+ identity_verified_at = Column(DateTime(timezone=True), nullable=True)
55
+
56
+ user = relationship(
57
+ "FidesUser",
58
+ back_populates="email_verifications",
59
+ foreign_keys=[user_id],
60
+ )
61
+
62
+ @classmethod
63
+ def create(
64
+ cls, db: Session, *, data: dict[str, Any], check_name: bool = False
65
+ ) -> FidesUserRespondentEmailVerification:
66
+ """
67
+ Create a FidesUserEmailVerification record with a new access token.
68
+ The verification code will be generated when the access link is used.
69
+ """
70
+ # Generate a secure token for the access link
71
+ access_token = generate_secure_random_string(32)
72
+ expires_at = datetime.now(timezone.utc) + timedelta(days=ACCESS_LINK_TTL_DAYS)
73
+
74
+ verification = super().create(
75
+ db,
76
+ data={
77
+ "user_id": data["user_id"],
78
+ "access_token": access_token,
79
+ "access_token_expires_at": expires_at,
80
+ },
81
+ )
82
+
83
+ return verification
84
+
85
+ def is_access_token_expired(self) -> bool:
86
+ """Check if the access token has expired."""
87
+ if not self.access_token_expires_at:
88
+ return True
89
+
90
+ current_time_utc = datetime.now(timezone.utc)
91
+ return current_time_utc > self.access_token_expires_at
92
+
93
+ def verify_access_token(self, token: str) -> bool:
94
+ """Verify the access token and generate a verification code if valid."""
95
+ if self.is_access_token_expired():
96
+ return False
97
+
98
+ return self.access_token == token
99
+
100
+ def verify_identity(
101
+ self,
102
+ db: Session,
103
+ provided_code: Optional[str] = None,
104
+ ) -> None:
105
+ """A method to call the internal identity verification method provided by the
106
+ `IdentityVerificationMixin`."""
107
+ if self.is_access_token_expired():
108
+ raise ValueError("Access token has expired.")
109
+
110
+ self._verify_identity(provided_code=provided_code)
111
+ self.identity_verified_at = datetime.now(timezone.utc)
112
+ self.save(db)
fides/api/oauth/roles.py CHANGED
@@ -27,6 +27,8 @@ from fides.common.api.scope_registry import (
27
27
  PRIVACY_NOTICE_READ,
28
28
  PRIVACY_REQUEST_CALLBACK_RESUME,
29
29
  PRIVACY_REQUEST_DELETE,
30
+ PRIVACY_REQUEST_MANUAL_STEPS_RESPOND,
31
+ PRIVACY_REQUEST_MANUAL_STEPS_REVIEW,
30
32
  PRIVACY_REQUEST_NOTIFICATIONS_CREATE_OR_UPDATE,
31
33
  PRIVACY_REQUEST_NOTIFICATIONS_READ,
32
34
  PRIVACY_REQUEST_READ,
@@ -52,6 +54,8 @@ CONTRIBUTOR = "contributor"
52
54
  OWNER = "owner"
53
55
  VIEWER = "viewer"
54
56
  VIEWER_AND_APPROVER = "viewer_and_approver"
57
+ RESPONDENT = "respondent"
58
+ EXTERNAL_RESPONDENT = "external_respondent"
55
59
 
56
60
 
57
61
  class RoleRegistryEnum(Enum):
@@ -62,6 +66,8 @@ class RoleRegistryEnum(Enum):
62
66
  Approver - Limited viewer but can approve Privacy Requests
63
67
  Viewer + Approver = Full View and can approve Privacy Requests
64
68
  Contributor - Can't configure storage and messaging
69
+ Respondent - Internal user who can respond to manual steps
70
+ External Respondent - External user who can only respond to assigned manual steps
65
71
  """
66
72
 
67
73
  owner = OWNER
@@ -69,6 +75,8 @@ class RoleRegistryEnum(Enum):
69
75
  viewer = VIEWER
70
76
  approver = APPROVER
71
77
  contributor = CONTRIBUTOR
78
+ respondent = RESPONDENT
79
+ external_respondent = EXTERNAL_RESPONDENT
72
80
 
73
81
 
74
82
  approver_scopes = [
@@ -79,6 +87,7 @@ approver_scopes = [
79
87
  PRIVACY_REQUEST_VIEW_DATA,
80
88
  PRIVACY_REQUEST_DELETE,
81
89
  USER_READ, # allows approver to view user management table and update their own password
90
+ PRIVACY_REQUEST_MANUAL_STEPS_REVIEW, # allows approvers to see all manual steps
82
91
  ]
83
92
 
84
93
 
@@ -114,6 +123,14 @@ viewer_scopes = [ # Intentionally omitted USER_PERMISSION_READ and PRIVACY_REQU
114
123
  USER_READ,
115
124
  ]
116
125
 
126
+ respondent_scopes = [
127
+ PRIVACY_REQUEST_MANUAL_STEPS_RESPOND, # allows respondents to respond to assigned manual steps
128
+ ]
129
+
130
+ external_respondent_scopes = [
131
+ PRIVACY_REQUEST_MANUAL_STEPS_RESPOND, # allows external respondents to respond to assigned manual steps
132
+ ]
133
+
117
134
  not_contributor_scopes = [
118
135
  CONNECTOR_TEMPLATE_REGISTER,
119
136
  STORAGE_CREATE_OR_UPDATE,
@@ -130,6 +147,8 @@ ROLES_TO_SCOPES_MAPPING: Dict[str, List] = {
130
147
  VIEWER: sorted(viewer_scopes),
131
148
  APPROVER: sorted(approver_scopes),
132
149
  CONTRIBUTOR: sorted(list(set(SCOPE_REGISTRY) - set(not_contributor_scopes))),
150
+ RESPONDENT: sorted(respondent_scopes),
151
+ EXTERNAL_RESPONDENT: sorted(external_respondent_scopes),
133
152
  }
134
153
 
135
154
 
@@ -52,7 +52,6 @@ from fides.api.util.saas_util import (
52
52
  get_identities,
53
53
  )
54
54
  from fides.common.api.v1.urn_registry import REQUEST_TASK_CALLBACK, V1_URL_PREFIX
55
- from fides.config.config_proxy import ConfigProxy
56
55
 
57
56
  T = TypeVar("T")
58
57
 
@@ -144,16 +143,12 @@ class SaaSQueryConfig(QueryConfig[SaaSRequestParams]):
144
143
  """
145
144
  Returns a tuple of the preferred action and SaaSRequest to use for masking.
146
145
  An update request is preferred, but we can use a gdpr delete endpoint or
147
- delete endpoint if not MASKING_STRICT.
146
+ delete endpoint.
148
147
  """
149
148
 
150
149
  update: Optional[SaaSRequest] = self.get_erasure_request_by_action("update")
151
- gdpr_delete: Optional[SaaSRequest] = None
152
- delete: Optional[SaaSRequest] = None
153
-
154
- if not ConfigProxy(db).execution.masking_strict:
155
- gdpr_delete = self.data_protection_request
156
- delete = self.get_erasure_request_by_action("delete")
150
+ gdpr_delete: Optional[SaaSRequest] = self.data_protection_request
151
+ delete: Optional[SaaSRequest] = self.get_erasure_request_by_action("delete")
157
152
 
158
153
  try:
159
154
  # Return first viable option
@@ -530,11 +530,15 @@ class SaaSConnector(BaseConnector[AuthenticatedClient], Contextualizable):
530
530
 
531
531
  session = Session.object_session(privacy_request)
532
532
  masking_request = query_config.get_masking_request(session)
533
+ rows_updated = 0
534
+
533
535
  if not masking_request:
534
- raise Exception(
535
- f"Either no masking request configured or no valid masking request for {node.address.collection}. "
536
- f"Check that MASKING_STRICT env var is appropriately set"
536
+ logger.info(
537
+ "No masking request found for the '{}' collection in {}",
538
+ self.current_collection_name,
539
+ self.saas_config.fides_key, # type: ignore
537
540
  )
541
+ return rows_updated
538
542
 
539
543
  self.set_saas_request_state(masking_request)
540
544
 
@@ -565,7 +569,6 @@ class SaaSConnector(BaseConnector[AuthenticatedClient], Contextualizable):
565
569
  cast(Optional[List[PostProcessorStrategy]], masking_request.postprocessors),
566
570
  )
567
571
 
568
- rows_updated = 0
569
572
  client = self.create_client()
570
573
  for row in rows:
571
574
  try:
@@ -1,3 +1,4 @@
1
+ import time
1
2
  from datetime import datetime, timedelta
2
3
  from typing import Any, Dict, List, Optional, Set, Tuple
3
4
 
@@ -67,6 +68,7 @@ from fides.api.tasks import DatabaseTask, celery_app
67
68
  from fides.api.tasks.scheduled.scheduler import scheduler
68
69
  from fides.api.util.collection_util import Row
69
70
  from fides.api.util.logger import Pii, _log_exception, _log_warning
71
+ from fides.api.util.logger_context_utils import LoggerContextKeys, log_context
70
72
  from fides.common.api.v1.urn_registry import (
71
73
  PRIVACY_REQUEST_TRANSFER_TO_PARENT,
72
74
  V1_URL_PREFIX,
@@ -200,6 +202,11 @@ def run_webhooks_and_report_status(
200
202
  return True
201
203
 
202
204
 
205
+ @log_context(
206
+ capture_args={
207
+ "privacy_request_id": LoggerContextKeys.privacy_request_id,
208
+ }
209
+ )
203
210
  def upload_access_results( # pylint: disable=R0912
204
211
  session: Session,
205
212
  policy: Policy,
@@ -210,6 +217,7 @@ def upload_access_results( # pylint: disable=R0912
210
217
  fides_connector_datasets: Set[str],
211
218
  ) -> List[str]:
212
219
  """Process the data uploads after the access portion of the privacy request has completed"""
220
+ start_time = time.time()
213
221
  download_urls: List[str] = []
214
222
  if not access_result:
215
223
  logger.info("No results returned for access request")
@@ -251,12 +259,30 @@ def upload_access_results( # pylint: disable=R0912
251
259
  )
252
260
  if download_url:
253
261
  download_urls.append(download_url)
262
+ privacy_request.add_success_execution_log(
263
+ session,
264
+ connection_key=None,
265
+ dataset_name="Access Package Upload",
266
+ collection_name=None,
267
+ message="Access Package Upload successful for privacy request.",
268
+ action_type=ActionType.access,
269
+ )
270
+ logger.bind(
271
+ time_taken=time.time() - start_time,
272
+ ).info("Access Package Upload successful for privacy request.")
254
273
  except common_exceptions.StorageUploadError as exc:
255
- logger.error(
256
- "Error uploading subject access data for rule {} on policy {}: {}",
257
- rule.key,
258
- policy.key,
259
- Pii(str(exc)),
274
+ logger.bind(
275
+ policy_key=policy.key,
276
+ rule_key=rule.key,
277
+ error=Pii(str(exc)),
278
+ ).error("Error uploading subject access data for rule.")
279
+ privacy_request.add_error_execution_log(
280
+ session,
281
+ connection_key=None,
282
+ dataset_name="Access Package Upload",
283
+ collection_name=None,
284
+ message="Access Package Upload failed for privacy request.",
285
+ action_type=ActionType.access,
260
286
  )
261
287
  privacy_request.status = PrivacyRequestStatus.error
262
288
  # Save the results we uploaded to the user for later retrieval
@@ -596,19 +622,22 @@ def run_privacy_request(
596
622
  # If dev mode, log traceback
597
623
  _log_exception(e, CONFIG.dev_mode)
598
624
  return
599
- privacy_request.finished_processing_at = datetime.utcnow()
600
- AuditLog.create(
601
- db=session,
602
- data={
603
- "user_id": "system",
604
- "privacy_request_id": privacy_request.id,
605
- "action": AuditLogAction.finished,
606
- "message": "",
607
- },
608
- )
609
- privacy_request.status = PrivacyRequestStatus.complete
610
- logger.info("Privacy request run completed.")
611
- privacy_request.save(db=session)
625
+
626
+ # Only mark as complete if not in error state
627
+ if privacy_request.status != PrivacyRequestStatus.error:
628
+ privacy_request.finished_processing_at = datetime.utcnow()
629
+ AuditLog.create(
630
+ db=session,
631
+ data={
632
+ "user_id": "system",
633
+ "privacy_request_id": privacy_request.id,
634
+ "action": AuditLogAction.finished,
635
+ "message": "",
636
+ },
637
+ )
638
+ privacy_request.status = PrivacyRequestStatus.complete
639
+ logger.info("Privacy request run completed.")
640
+ privacy_request.save(db=session)
612
641
 
613
642
 
614
643
  def initiate_privacy_request_completion_email(
@@ -11,6 +11,7 @@ from botocore.exceptions import ClientError, ParamValidationError
11
11
  from fideslang.validation import AnyHttpUrlString
12
12
  from loguru import logger
13
13
 
14
+ from fides.api.common_exceptions import StorageUploadError
14
15
  from fides.api.cryptography.cryptographic_util import bytes_to_b64_str
15
16
  from fides.api.schemas.storage.storage import ResponseFormat, StorageSecrets
16
17
  from fides.api.service.privacy_request.dsr_package.dsr_report_builder import (
@@ -160,9 +161,9 @@ def upload_to_s3( # pylint: disable=R0913
160
161
  logger.error(
161
162
  "Encountered error while uploading and generating link for s3 object: {}", e
162
163
  )
163
- raise e
164
+ raise StorageUploadError(f"Error uploading to S3: {e}")
164
165
  except ParamValidationError as e:
165
- raise ValueError(f"The parameters you provided are incorrect: {e}")
166
+ raise StorageUploadError(f"The parameters you provided are incorrect: {e}")
166
167
 
167
168
 
168
169
  def upload_to_gcs(
@@ -205,7 +206,7 @@ def upload_to_gcs(
205
206
  "Encountered error while uploading and generating link for Google Cloud Storage object: {}",
206
207
  e,
207
208
  )
208
- raise e
209
+ raise StorageUploadError(f"Error uploading to Google Cloud Storage: {e}")
209
210
 
210
211
 
211
212
  def upload_to_local(
@@ -37,6 +37,7 @@ EXEC = "exec"
37
37
  FIDES_TAXONOMY = "fides_taxonomy"
38
38
  GENERATE = "generate"
39
39
  INSTANTIATE = "instantiate"
40
+ MANUAL_STEPS = "manual-steps"
40
41
  MASKING = "masking"
41
42
  MESSAGING = "messaging"
42
43
  ORGANIZATION = "organization"
@@ -51,6 +52,7 @@ PRIVACY_REQUEST_NOTIFICATIONS = "privacy-request-notifications"
51
52
  READ = "read"
52
53
  REGISTER = "register"
53
54
  RESET = "reset"
55
+ RESPOND = "respond"
54
56
  RESUME = "resume"
55
57
  REVIEW = "review"
56
58
  RULE = "rule"
@@ -174,6 +176,8 @@ PRIVACY_PREFERENCE_HISTORY_READ = f"{PRIVACY_PREFERENCE_HISTORY}:{READ}"
174
176
  PRIVACY_REQUEST_CALLBACK_RESUME = f"{PRIVACY_REQUEST}:{RESUME}" # User has permission to restart a paused privacy request
175
177
  PRIVACY_REQUEST_CREATE = f"{PRIVACY_REQUEST}:{CREATE}"
176
178
  PRIVACY_REQUEST_DELETE = f"{PRIVACY_REQUEST}:{DELETE}"
179
+ PRIVACY_REQUEST_MANUAL_STEPS_REVIEW = f"{PRIVACY_REQUEST}:{MANUAL_STEPS}:{REVIEW}"
180
+ PRIVACY_REQUEST_MANUAL_STEPS_RESPOND = f"{PRIVACY_REQUEST}:{MANUAL_STEPS}:{RESPOND}"
177
181
  PRIVACY_REQUEST_NOTIFICATIONS_CREATE_OR_UPDATE = (
178
182
  f"{PRIVACY_REQUEST_NOTIFICATIONS}:{CREATE_OR_UPDATE}"
179
183
  )
@@ -312,6 +316,8 @@ SCOPE_DOCS = {
312
316
  PRIVACY_REQUEST_CALLBACK_RESUME: "Restart paused privacy requests",
313
317
  PRIVACY_REQUEST_READ_ACCESS_RESULTS: "Download access data for the privacy request",
314
318
  PRIVACY_REQUEST_DELETE: "Remove privacy requests",
319
+ PRIVACY_REQUEST_MANUAL_STEPS_RESPOND: "Respond to manual steps for the privacy request",
320
+ PRIVACY_REQUEST_MANUAL_STEPS_REVIEW: "Review manual steps for the privacy request",
315
321
  PRIVACY_REQUEST_NOTIFICATIONS_CREATE_OR_UPDATE: "",
316
322
  PRIVACY_REQUEST_NOTIFICATIONS_READ: "",
317
323
  PRIVACY_REQUEST_READ: "View privacy requests",
@@ -106,7 +106,6 @@ class ExecutionSettingsProxy(ConfigProxyBase):
106
106
  subject_identity_verification_required: bool
107
107
  disable_consent_identity_verification: bool
108
108
  require_manual_request_approval: bool
109
- masking_strict: bool
110
109
 
111
110
  def __getattribute__(self, name: str) -> Any:
112
111
  """
@@ -11,10 +11,6 @@ ENV_PREFIX = "FIDES__EXECUTION__"
11
11
  class ExecutionSettings(FidesSettings):
12
12
  """Configuration settings for DSR execution."""
13
13
 
14
- masking_strict: bool = Field(
15
- default=True,
16
- description="If set to True, only use UPDATE requests to mask data. If False, Fides will use any defined DELETE or GDPR DELETE endpoints to remove PII, which may extend beyond the specific data categories that configured in your execution policy.",
17
- )
18
14
  privacy_request_delay_timeout: int = Field(
19
15
  default=3600,
20
16
  description="The amount of time to wait for actions which delay privacy requests (e.g., pre- and post-processing webhooks).",
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/pages/404-e868487119cd14fc.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/404","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/pages/404-e868487119cd14fc.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/404","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><link rel="preload" href="/_next/static/css/92fc10cc7ed1aba3.css" as="style"/><link rel="stylesheet" href="/_next/static/css/92fc10cc7ed1aba3.css" data-n-p=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/9278-9b1b5970f0702668.js" defer=""></script><script src="/_next/static/chunks/5277-e8a036319456127f.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/7980-041df432c4d3a68b.js" defer=""></script><script src="/_next/static/chunks/8499-06ab15acbfec037c.js" defer=""></script><script src="/_next/static/chunks/3426-77fccf2c9a5e10cd.js" defer=""></script><script src="/_next/static/chunks/9965-92f4a28823a5e623.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems/manual-1292fd216f2839c7.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems/manual","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><link rel="preload" href="/_next/static/css/92fc10cc7ed1aba3.css" as="style"/><link rel="stylesheet" href="/_next/static/css/92fc10cc7ed1aba3.css" data-n-p=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/9278-9b1b5970f0702668.js" defer=""></script><script src="/_next/static/chunks/5277-e8a036319456127f.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/7980-041df432c4d3a68b.js" defer=""></script><script src="/_next/static/chunks/8499-06ab15acbfec037c.js" defer=""></script><script src="/_next/static/chunks/3426-77fccf2c9a5e10cd.js" defer=""></script><script src="/_next/static/chunks/9965-92f4a28823a5e623.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems/manual-1292fd216f2839c7.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems/manual","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/796-7424eb6391142ccd.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems/multiple-2ffc37614e9d1fa9.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems/multiple","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/796-7424eb6391142ccd.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems/multiple-2ffc37614e9d1fa9.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems/multiple","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><link rel="preload" href="/_next/static/css/972dc7eef106ee7c.css" as="style"/><link rel="stylesheet" href="/_next/static/css/972dc7eef106ee7c.css" data-n-p=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems-0234733a90ed5127.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><link rel="preload" href="/_next/static/css/972dc7eef106ee7c.css" as="style"/><link rel="stylesheet" href="/_next/static/css/972dc7eef106ee7c.css" data-n-p=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/pages/add-systems-0234733a90ed5127.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/add-systems","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/796-7424eb6391142ccd.js" defer=""></script><script src="/_next/static/chunks/pages/consent/configure/add-vendors-c8033d6560635046.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/configure/add-vendors","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/796-7424eb6391142ccd.js" defer=""></script><script src="/_next/static/chunks/pages/consent/configure/add-vendors-c8033d6560635046.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/configure/add-vendors","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/pages/consent/configure-2ba21b95aa3bb15e.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/configure","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/c78d26b1-88a3e1bacb2a03c2.js" defer=""></script><script src="/_next/static/chunks/6060-cb1ab5be7067bf7b.js" defer=""></script><script src="/_next/static/chunks/7553-0f3e9e7eaf1a8b62.js" defer=""></script><script src="/_next/static/chunks/4481-9357e3930d35073b.js" defer=""></script><script src="/_next/static/chunks/401-a882199f6c94a4dc.js" defer=""></script><script src="/_next/static/chunks/3923-c6dcaeaa3267f254.js" defer=""></script><script src="/_next/static/chunks/pages/consent/configure-2ba21b95aa3bb15e.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/configure","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/4294-5b3aa9d605217915.js" defer=""></script><script src="/_next/static/chunks/9014-eeae6f581158e645.js" defer=""></script><script src="/_next/static/chunks/69-ef4c11c574b4e6dd.js" defer=""></script><script src="/_next/static/chunks/6277-db781b6c1dde3c84.js" defer=""></script><script src="/_next/static/chunks/1817-d2543e460cec0dcb.js" defer=""></script><script src="/_next/static/chunks/pages/consent/privacy-experience/%5Bid%5D-94391554a7607c5b.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/privacy-experience/[id]","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/4294-5b3aa9d605217915.js" defer=""></script><script src="/_next/static/chunks/9014-eeae6f581158e645.js" defer=""></script><script src="/_next/static/chunks/69-ef4c11c574b4e6dd.js" defer=""></script><script src="/_next/static/chunks/6277-db781b6c1dde3c84.js" defer=""></script><script src="/_next/static/chunks/1817-d2543e460cec0dcb.js" defer=""></script><script src="/_next/static/chunks/pages/consent/privacy-experience/%5Bid%5D-94391554a7607c5b.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/privacy-experience/[id]","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
@@ -1 +1 @@
1
- <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-94950df9c97cd079.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/4294-5b3aa9d605217915.js" defer=""></script><script src="/_next/static/chunks/9014-eeae6f581158e645.js" defer=""></script><script src="/_next/static/chunks/69-ef4c11c574b4e6dd.js" defer=""></script><script src="/_next/static/chunks/6277-db781b6c1dde3c84.js" defer=""></script><script src="/_next/static/chunks/1817-d2543e460cec0dcb.js" defer=""></script><script src="/_next/static/chunks/pages/consent/privacy-experience/new-7f38473ce267348a.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_buildManifest.js" defer=""></script><script src="/_next/static/Sez6kqW5dI0lK60IMwICO/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/privacy-experience/new","query":{},"buildId":"Sez6kqW5dI0lK60IMwICO","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>
1
+ <!DOCTYPE html><html lang="en"><head><meta charSet="utf-8"/><meta name="viewport" content="width=device-width"/><meta name="next-head-count" content="2"/><link data-next-font="" rel="preconnect" href="/" crossorigin="anonymous"/><link rel="preload" href="/_next/static/css/1b227ba7eabbfe2f.css" as="style"/><link rel="stylesheet" href="/_next/static/css/1b227ba7eabbfe2f.css" data-n-g=""/><noscript data-n-css=""></noscript><script defer="" nomodule="" src="/_next/static/chunks/polyfills-42372ed130431b0a.js"></script><script src="/_next/static/chunks/webpack-3a61b934ba2fb620.js" defer=""></script><script src="/_next/static/chunks/framework-c92fc3344e6fd165.js" defer=""></script><script src="/_next/static/chunks/main-090643377c8254e6.js" defer=""></script><script src="/_next/static/chunks/pages/_app-f1add6386651ca3d.js" defer=""></script><script src="/_next/static/chunks/2858-0b44609b6be7850b.js" defer=""></script><script src="/_next/static/chunks/2866-a73888c17a195cbe.js" defer=""></script><script src="/_next/static/chunks/4294-5b3aa9d605217915.js" defer=""></script><script src="/_next/static/chunks/9014-eeae6f581158e645.js" defer=""></script><script src="/_next/static/chunks/69-ef4c11c574b4e6dd.js" defer=""></script><script src="/_next/static/chunks/6277-db781b6c1dde3c84.js" defer=""></script><script src="/_next/static/chunks/1817-d2543e460cec0dcb.js" defer=""></script><script src="/_next/static/chunks/pages/consent/privacy-experience/new-7f38473ce267348a.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_buildManifest.js" defer=""></script><script src="/_next/static/D-B-G-FV05_6iwXWBGS0J/_ssgManifest.js" defer=""></script><style>.data-ant-cssinjs-cache-path{content:"";}</style></head><body><div id="__next"><div style="height:100%;display:flex"></div></div><script id="__NEXT_DATA__" type="application/json">{"props":{"pageProps":{}},"page":"/consent/privacy-experience/new","query":{},"buildId":"D-B-G-FV05_6iwXWBGS0J","nextExport":true,"autoExport":true,"isFallback":false,"scriptLoader":[]}</script></body></html>