cartography 0.117.0__py3-none-any.whl → 0.119.0__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.
Potentially problematic release.
This version of cartography might be problematic. Click here for more details.
- cartography/_version.py +2 -2
- cartography/cli.py +31 -0
- cartography/client/core/tx.py +19 -3
- cartography/config.py +14 -0
- cartography/data/indexes.cypher +0 -6
- cartography/graph/job.py +13 -7
- cartography/graph/statement.py +4 -0
- cartography/intel/aws/__init__.py +22 -9
- cartography/intel/aws/apigateway.py +18 -5
- cartography/intel/aws/ec2/elastic_ip_addresses.py +3 -1
- cartography/intel/aws/ec2/internet_gateways.py +4 -2
- cartography/intel/aws/ec2/load_balancer_v2s.py +11 -5
- cartography/intel/aws/ec2/network_interfaces.py +4 -0
- cartography/intel/aws/ec2/reserved_instances.py +3 -1
- cartography/intel/aws/ec2/tgw.py +11 -5
- cartography/intel/aws/ec2/volumes.py +1 -1
- cartography/intel/aws/ecr.py +209 -26
- cartography/intel/aws/ecr_image_layers.py +143 -42
- cartography/intel/aws/elasticsearch.py +13 -4
- cartography/intel/aws/identitycenter.py +93 -54
- cartography/intel/aws/inspector.py +90 -46
- cartography/intel/aws/permission_relationships.py +3 -3
- cartography/intel/aws/resourcegroupstaggingapi.py +1 -1
- cartography/intel/aws/s3.py +26 -13
- cartography/intel/aws/ssm.py +3 -5
- cartography/intel/azure/compute.py +9 -4
- cartography/intel/azure/cosmosdb.py +31 -15
- cartography/intel/azure/sql.py +25 -12
- cartography/intel/azure/storage.py +19 -9
- cartography/intel/azure/subscription.py +3 -1
- cartography/intel/crowdstrike/spotlight.py +5 -2
- cartography/intel/entra/app_role_assignments.py +9 -2
- cartography/intel/gcp/__init__.py +26 -9
- cartography/intel/gcp/clients.py +8 -4
- cartography/intel/gcp/compute.py +42 -21
- cartography/intel/gcp/crm/folders.py +9 -3
- cartography/intel/gcp/crm/orgs.py +8 -3
- cartography/intel/gcp/crm/projects.py +14 -3
- cartography/intel/github/repos.py +23 -5
- cartography/intel/gsuite/__init__.py +12 -8
- cartography/intel/gsuite/groups.py +291 -0
- cartography/intel/gsuite/users.py +142 -0
- cartography/intel/jamf/computers.py +7 -1
- cartography/intel/oci/iam.py +23 -9
- cartography/intel/oci/organizations.py +3 -1
- cartography/intel/oci/utils.py +28 -5
- cartography/intel/okta/awssaml.py +9 -8
- cartography/intel/okta/users.py +1 -1
- cartography/intel/ontology/__init__.py +44 -0
- cartography/intel/ontology/devices.py +54 -0
- cartography/intel/ontology/users.py +54 -0
- cartography/intel/ontology/utils.py +121 -0
- cartography/intel/pagerduty/escalation_policies.py +13 -6
- cartography/intel/pagerduty/schedules.py +9 -4
- cartography/intel/pagerduty/services.py +7 -3
- cartography/intel/pagerduty/teams.py +5 -2
- cartography/intel/pagerduty/users.py +3 -1
- cartography/intel/pagerduty/vendors.py +3 -1
- cartography/intel/trivy/__init__.py +109 -58
- cartography/models/airbyte/user.py +4 -0
- cartography/models/anthropic/user.py +4 -0
- cartography/models/aws/ec2/networkinterfaces.py +2 -0
- cartography/models/aws/ecr/image.py +55 -0
- cartography/models/aws/ecr/repository_image.py +1 -1
- cartography/models/aws/iam/group_membership.py +3 -2
- cartography/models/aws/identitycenter/awsssouser.py +3 -1
- cartography/models/bigfix/bigfix_computer.py +1 -1
- cartography/models/cloudflare/member.py +4 -0
- cartography/models/crowdstrike/hosts.py +1 -1
- cartography/models/duo/endpoint.py +1 -1
- cartography/models/duo/phone.py +2 -2
- cartography/models/duo/user.py +4 -0
- cartography/models/entra/user.py +2 -1
- cartography/models/github/users.py +4 -0
- cartography/models/gsuite/__init__.py +0 -0
- cartography/models/gsuite/group.py +218 -0
- cartography/models/gsuite/tenant.py +29 -0
- cartography/models/gsuite/user.py +107 -0
- cartography/models/kandji/device.py +1 -2
- cartography/models/keycloak/user.py +4 -0
- cartography/models/lastpass/user.py +4 -0
- cartography/models/ontology/__init__.py +0 -0
- cartography/models/ontology/device.py +125 -0
- cartography/models/ontology/mapping/__init__.py +16 -0
- cartography/models/ontology/mapping/data/__init__.py +1 -0
- cartography/models/ontology/mapping/data/devices.py +160 -0
- cartography/models/ontology/mapping/data/users.py +239 -0
- cartography/models/ontology/mapping/specs.py +65 -0
- cartography/models/ontology/user.py +52 -0
- cartography/models/openai/user.py +4 -0
- cartography/models/scaleway/iam/user.py +4 -0
- cartography/models/snipeit/asset.py +1 -0
- cartography/models/snipeit/user.py +4 -0
- cartography/models/tailscale/device.py +1 -1
- cartography/models/tailscale/user.py +6 -1
- cartography/rules/data/frameworks/mitre_attack/requirements/t1098_account_manipulation/__init__.py +176 -89
- cartography/sync.py +4 -1
- cartography/util.py +49 -18
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/METADATA +3 -3
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/RECORD +104 -89
- cartography/data/jobs/cleanup/gsuite_ingest_groups_cleanup.json +0 -23
- cartography/data/jobs/cleanup/gsuite_ingest_users_cleanup.json +0 -11
- cartography/intel/gsuite/api.py +0 -355
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/WHEEL +0 -0
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/entry_points.txt +0 -0
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/licenses/LICENSE +0 -0
- {cartography-0.117.0.dist-info → cartography-0.119.0.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
from cartography.models.ontology.mapping.specs import OntologyFieldMapping
|
|
2
|
+
from cartography.models.ontology.mapping.specs import OntologyMapping
|
|
3
|
+
from cartography.models.ontology.mapping.specs import OntologyNodeMapping
|
|
4
|
+
from cartography.models.ontology.mapping.specs import OntologyRelMapping
|
|
5
|
+
|
|
6
|
+
bigfix_mapping = OntologyMapping(
|
|
7
|
+
module_name="bigfix",
|
|
8
|
+
nodes=[
|
|
9
|
+
OntologyNodeMapping(
|
|
10
|
+
node_label="BigfixComputer",
|
|
11
|
+
fields=[
|
|
12
|
+
OntologyFieldMapping(
|
|
13
|
+
ontology_field="hostname", node_field="computername", required=True
|
|
14
|
+
),
|
|
15
|
+
OntologyFieldMapping(ontology_field="os", node_field="os"),
|
|
16
|
+
],
|
|
17
|
+
),
|
|
18
|
+
],
|
|
19
|
+
)
|
|
20
|
+
crowdstrike_mapping = OntologyMapping(
|
|
21
|
+
module_name="crowdstrike",
|
|
22
|
+
nodes=[
|
|
23
|
+
OntologyNodeMapping(
|
|
24
|
+
node_label="CrowdstrikeHost",
|
|
25
|
+
fields=[
|
|
26
|
+
OntologyFieldMapping(
|
|
27
|
+
ontology_field="hostname", node_field="hostname", required=True
|
|
28
|
+
),
|
|
29
|
+
OntologyFieldMapping(
|
|
30
|
+
ontology_field="os_version", node_field="os_version"
|
|
31
|
+
),
|
|
32
|
+
OntologyFieldMapping(
|
|
33
|
+
ontology_field="platform", node_field="platform_name"
|
|
34
|
+
),
|
|
35
|
+
OntologyFieldMapping(
|
|
36
|
+
ontology_field="serial_number", node_field="serial_number"
|
|
37
|
+
),
|
|
38
|
+
OntologyFieldMapping(
|
|
39
|
+
ontology_field="instance_id", node_field="instance_id"
|
|
40
|
+
),
|
|
41
|
+
],
|
|
42
|
+
),
|
|
43
|
+
],
|
|
44
|
+
)
|
|
45
|
+
duo_mapping = OntologyMapping(
|
|
46
|
+
module_name="duo",
|
|
47
|
+
nodes=[
|
|
48
|
+
OntologyNodeMapping(
|
|
49
|
+
node_label="DuoEndpoint",
|
|
50
|
+
fields=[
|
|
51
|
+
OntologyFieldMapping(
|
|
52
|
+
ontology_field="hostname", node_field="device_name", required=True
|
|
53
|
+
),
|
|
54
|
+
OntologyFieldMapping(ontology_field="os", node_field="os_family"),
|
|
55
|
+
OntologyFieldMapping(
|
|
56
|
+
ontology_field="os_version", node_field="os_version"
|
|
57
|
+
),
|
|
58
|
+
OntologyFieldMapping(ontology_field="model", node_field="model"),
|
|
59
|
+
],
|
|
60
|
+
),
|
|
61
|
+
OntologyNodeMapping(
|
|
62
|
+
node_label="DuoPhone",
|
|
63
|
+
fields=[
|
|
64
|
+
OntologyFieldMapping(
|
|
65
|
+
ontology_field="hostname", node_field="name", required=True
|
|
66
|
+
),
|
|
67
|
+
OntologyFieldMapping(ontology_field="model", node_field="model"),
|
|
68
|
+
OntologyFieldMapping(ontology_field="platform", node_field="platform"),
|
|
69
|
+
],
|
|
70
|
+
),
|
|
71
|
+
],
|
|
72
|
+
rels=[
|
|
73
|
+
OntologyRelMapping(
|
|
74
|
+
__comment__="Link Device to User based on DuoUser-DuoPhone relationship",
|
|
75
|
+
query="MATCH (u:User)-[:HAS_ACCOUNT]->(:DuoUser)-[:HAS_DUO_PHONE]-(:DuoPhone)<-[:OBSERVED_AS]-(d:Device) MERGE (u)-[r:OWNS]->(d) ON CREATE SET r.firstseen = timestamp() SET r.lastupdated = $UPDATE_TAG",
|
|
76
|
+
iterative=False,
|
|
77
|
+
),
|
|
78
|
+
OntologyRelMapping(
|
|
79
|
+
__comment__="Link Device to User based on DuoUser-DuoEndpoint relationship",
|
|
80
|
+
query="MATCH (u:User)-[:HAS_ACCOUNT]->(:DuoUser)-[:HAS_DUO_ENDPOINT]-(:DuoEndpoint)<-[:OBSERVED_AS]-(d:Device) MERGE (u)-[r:OWNS]->(d) ON CREATE SET r.firstseen = timestamp() SET r.lastupdated = $UPDATE_TAG",
|
|
81
|
+
iterative=False,
|
|
82
|
+
),
|
|
83
|
+
],
|
|
84
|
+
)
|
|
85
|
+
kandji_mapping = OntologyMapping(
|
|
86
|
+
module_name="kandji",
|
|
87
|
+
nodes=[
|
|
88
|
+
OntologyNodeMapping(
|
|
89
|
+
node_label="KandjiDevice",
|
|
90
|
+
fields=[
|
|
91
|
+
OntologyFieldMapping(
|
|
92
|
+
ontology_field="hostname", node_field="device_name", required=True
|
|
93
|
+
),
|
|
94
|
+
OntologyFieldMapping(
|
|
95
|
+
ontology_field="serial_number", node_field="serial_number"
|
|
96
|
+
),
|
|
97
|
+
OntologyFieldMapping(
|
|
98
|
+
ontology_field="os_version", node_field="os_version"
|
|
99
|
+
),
|
|
100
|
+
OntologyFieldMapping(ontology_field="model", node_field="model"),
|
|
101
|
+
OntologyFieldMapping(ontology_field="platform", node_field="platform"),
|
|
102
|
+
],
|
|
103
|
+
),
|
|
104
|
+
],
|
|
105
|
+
)
|
|
106
|
+
snipeit_mapping = OntologyMapping(
|
|
107
|
+
module_name="snipeit",
|
|
108
|
+
nodes=[
|
|
109
|
+
OntologyNodeMapping(
|
|
110
|
+
node_label="SnipeitAsset",
|
|
111
|
+
fields=[
|
|
112
|
+
OntologyFieldMapping(
|
|
113
|
+
ontology_field="hostname", node_field="name", required=True
|
|
114
|
+
),
|
|
115
|
+
OntologyFieldMapping(
|
|
116
|
+
ontology_field="serial_number", node_field="serial"
|
|
117
|
+
),
|
|
118
|
+
OntologyFieldMapping(ontology_field="model", node_field="model"),
|
|
119
|
+
],
|
|
120
|
+
),
|
|
121
|
+
],
|
|
122
|
+
rels=[
|
|
123
|
+
OntologyRelMapping(
|
|
124
|
+
__comment__="Link Device to User based on SnipeitUser-SnipeitAsset relationship",
|
|
125
|
+
query="MATCH (u:User)-[:HAS_ACCOUNT]->(:SnipeitUser)-[:HAS_CHECKED_OUT]-(:SnipeitAsset)<-[:OBSERVED_AS]-(d:Device) MERGE (u)-[r:OWNS]->(d) ON CREATE SET r.firstseen = timestamp() SET r.lastupdated = $UPDATE_TAG",
|
|
126
|
+
iterative=False,
|
|
127
|
+
)
|
|
128
|
+
],
|
|
129
|
+
)
|
|
130
|
+
tailscale_mapping = OntologyMapping(
|
|
131
|
+
module_name="tailscale",
|
|
132
|
+
nodes=[
|
|
133
|
+
OntologyNodeMapping(
|
|
134
|
+
node_label="TailscaleDevice",
|
|
135
|
+
fields=[
|
|
136
|
+
OntologyFieldMapping(
|
|
137
|
+
ontology_field="hostname", node_field="hostname", required=True
|
|
138
|
+
),
|
|
139
|
+
OntologyFieldMapping(ontology_field="os", node_field="os"),
|
|
140
|
+
],
|
|
141
|
+
),
|
|
142
|
+
],
|
|
143
|
+
rels=[
|
|
144
|
+
OntologyRelMapping(
|
|
145
|
+
__comment__="Link Device to User based on TailscaleUser-TailscaleDevice relationship",
|
|
146
|
+
query="MATCH (u:User)-[:HAS_ACCOUNT]->(:TailscaleUser)-[:OWNS]-(:TailscaleDevice)<-[:OBSERVED_AS]-(d:Device) MERGE (u)-[r:OWNS]->(d) ON CREATE SET r.firstseen = timestamp() SET r.lastupdated = $UPDATE_TAG",
|
|
147
|
+
iterative=False,
|
|
148
|
+
)
|
|
149
|
+
],
|
|
150
|
+
)
|
|
151
|
+
|
|
152
|
+
|
|
153
|
+
DEVICES_ONTOLOGY_MAPPING: dict[str, OntologyMapping] = {
|
|
154
|
+
"bigfix": bigfix_mapping,
|
|
155
|
+
"crowdstrike": crowdstrike_mapping,
|
|
156
|
+
"duo": duo_mapping,
|
|
157
|
+
"kandji": kandji_mapping,
|
|
158
|
+
"snipeit": snipeit_mapping,
|
|
159
|
+
"tailscale": tailscale_mapping,
|
|
160
|
+
}
|
|
@@ -0,0 +1,239 @@
|
|
|
1
|
+
from cartography.models.ontology.mapping.specs import OntologyFieldMapping
|
|
2
|
+
from cartography.models.ontology.mapping.specs import OntologyMapping
|
|
3
|
+
from cartography.models.ontology.mapping.specs import OntologyNodeMapping
|
|
4
|
+
|
|
5
|
+
anthropic_mapping = OntologyMapping(
|
|
6
|
+
module_name="anthropic",
|
|
7
|
+
nodes=[
|
|
8
|
+
OntologyNodeMapping(
|
|
9
|
+
node_label="AnthropicUser",
|
|
10
|
+
fields=[
|
|
11
|
+
OntologyFieldMapping(
|
|
12
|
+
ontology_field="email", node_field="email", required=True
|
|
13
|
+
),
|
|
14
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="name"),
|
|
15
|
+
],
|
|
16
|
+
),
|
|
17
|
+
],
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
airbyte_mapping = OntologyMapping(
|
|
21
|
+
module_name="airbyte",
|
|
22
|
+
nodes=[
|
|
23
|
+
OntologyNodeMapping(
|
|
24
|
+
node_label="AirbyteUser",
|
|
25
|
+
fields=[
|
|
26
|
+
OntologyFieldMapping(
|
|
27
|
+
ontology_field="email", node_field="email", required=True
|
|
28
|
+
),
|
|
29
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="name"),
|
|
30
|
+
],
|
|
31
|
+
),
|
|
32
|
+
],
|
|
33
|
+
)
|
|
34
|
+
|
|
35
|
+
cloudflare_mapping = OntologyMapping(
|
|
36
|
+
module_name="cloudflare",
|
|
37
|
+
nodes=[
|
|
38
|
+
OntologyNodeMapping(
|
|
39
|
+
node_label="CloudflareMember",
|
|
40
|
+
fields=[
|
|
41
|
+
OntologyFieldMapping(
|
|
42
|
+
ontology_field="email", node_field="email", required=True
|
|
43
|
+
),
|
|
44
|
+
OntologyFieldMapping(
|
|
45
|
+
ontology_field="firstname", node_field="firstname"
|
|
46
|
+
),
|
|
47
|
+
OntologyFieldMapping(ontology_field="lastname", node_field="lastname"),
|
|
48
|
+
],
|
|
49
|
+
),
|
|
50
|
+
],
|
|
51
|
+
)
|
|
52
|
+
|
|
53
|
+
duo_mapping = OntologyMapping(
|
|
54
|
+
module_name="duo",
|
|
55
|
+
nodes=[
|
|
56
|
+
OntologyNodeMapping(
|
|
57
|
+
node_label="DuoUser",
|
|
58
|
+
fields=[
|
|
59
|
+
OntologyFieldMapping(
|
|
60
|
+
ontology_field="email", node_field="email", required=True
|
|
61
|
+
),
|
|
62
|
+
OntologyFieldMapping(
|
|
63
|
+
ontology_field="firstname", node_field="firstname"
|
|
64
|
+
),
|
|
65
|
+
OntologyFieldMapping(ontology_field="lastname", node_field="lastname"),
|
|
66
|
+
OntologyFieldMapping(ontology_field="username", node_field="username"),
|
|
67
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="realname"),
|
|
68
|
+
],
|
|
69
|
+
),
|
|
70
|
+
],
|
|
71
|
+
)
|
|
72
|
+
entra_mapping = OntologyMapping(
|
|
73
|
+
module_name="entra",
|
|
74
|
+
nodes=[
|
|
75
|
+
OntologyNodeMapping(
|
|
76
|
+
node_label="EntraUser",
|
|
77
|
+
fields=[
|
|
78
|
+
OntologyFieldMapping(
|
|
79
|
+
ontology_field="email", node_field="email", required=True
|
|
80
|
+
),
|
|
81
|
+
OntologyFieldMapping(
|
|
82
|
+
ontology_field="firstname", node_field="given_name"
|
|
83
|
+
),
|
|
84
|
+
OntologyFieldMapping(ontology_field="lastname", node_field="surname"),
|
|
85
|
+
OntologyFieldMapping(
|
|
86
|
+
ontology_field="fullname", node_field="display_name"
|
|
87
|
+
),
|
|
88
|
+
],
|
|
89
|
+
),
|
|
90
|
+
],
|
|
91
|
+
)
|
|
92
|
+
github_mapping = OntologyMapping(
|
|
93
|
+
module_name="github",
|
|
94
|
+
nodes=[
|
|
95
|
+
OntologyNodeMapping(
|
|
96
|
+
node_label="GitHubUser",
|
|
97
|
+
fields=[
|
|
98
|
+
OntologyFieldMapping(
|
|
99
|
+
ontology_field="email", node_field="email", required=True
|
|
100
|
+
),
|
|
101
|
+
OntologyFieldMapping(ontology_field="username", node_field="username"),
|
|
102
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="fullname"),
|
|
103
|
+
],
|
|
104
|
+
),
|
|
105
|
+
],
|
|
106
|
+
)
|
|
107
|
+
gsuite_mapping = OntologyMapping(
|
|
108
|
+
module_name="gsuite",
|
|
109
|
+
nodes=[
|
|
110
|
+
OntologyNodeMapping(
|
|
111
|
+
node_label="GSuiteUser",
|
|
112
|
+
fields=[
|
|
113
|
+
OntologyFieldMapping(
|
|
114
|
+
ontology_field="email", node_field="email", required=True
|
|
115
|
+
),
|
|
116
|
+
OntologyFieldMapping(
|
|
117
|
+
ontology_field="firstname", node_field="given_name"
|
|
118
|
+
),
|
|
119
|
+
OntologyFieldMapping(
|
|
120
|
+
ontology_field="lastname", node_field="family_name"
|
|
121
|
+
),
|
|
122
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="name"),
|
|
123
|
+
],
|
|
124
|
+
),
|
|
125
|
+
],
|
|
126
|
+
)
|
|
127
|
+
keycloak_mapping = OntologyMapping(
|
|
128
|
+
module_name="keycloak",
|
|
129
|
+
nodes=[
|
|
130
|
+
OntologyNodeMapping(
|
|
131
|
+
node_label="KeycloakUser",
|
|
132
|
+
fields=[
|
|
133
|
+
OntologyFieldMapping(
|
|
134
|
+
ontology_field="email", node_field="email", required=True
|
|
135
|
+
),
|
|
136
|
+
OntologyFieldMapping(ontology_field="username", node_field="username"),
|
|
137
|
+
OntologyFieldMapping(
|
|
138
|
+
ontology_field="firstname", node_field="first_name"
|
|
139
|
+
),
|
|
140
|
+
OntologyFieldMapping(ontology_field="lastname", node_field="last_name"),
|
|
141
|
+
],
|
|
142
|
+
),
|
|
143
|
+
],
|
|
144
|
+
)
|
|
145
|
+
lastpass_mapping = OntologyMapping(
|
|
146
|
+
module_name="lastpass",
|
|
147
|
+
nodes=[
|
|
148
|
+
OntologyNodeMapping(
|
|
149
|
+
node_label="LastpassUser",
|
|
150
|
+
fields=[
|
|
151
|
+
OntologyFieldMapping(
|
|
152
|
+
ontology_field="email", node_field="email", required=True
|
|
153
|
+
),
|
|
154
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="name"),
|
|
155
|
+
],
|
|
156
|
+
),
|
|
157
|
+
],
|
|
158
|
+
)
|
|
159
|
+
openai_mapping = OntologyMapping(
|
|
160
|
+
module_name="openai",
|
|
161
|
+
nodes=[
|
|
162
|
+
OntologyNodeMapping(
|
|
163
|
+
node_label="OpenAIUser",
|
|
164
|
+
fields=[
|
|
165
|
+
OntologyFieldMapping(
|
|
166
|
+
ontology_field="email", node_field="email", required=True
|
|
167
|
+
),
|
|
168
|
+
OntologyFieldMapping(ontology_field="fullname", node_field="name"),
|
|
169
|
+
],
|
|
170
|
+
),
|
|
171
|
+
],
|
|
172
|
+
)
|
|
173
|
+
scaleway_mapping = OntologyMapping(
|
|
174
|
+
module_name="scaleway",
|
|
175
|
+
nodes=[
|
|
176
|
+
OntologyNodeMapping(
|
|
177
|
+
node_label="ScalewayUser",
|
|
178
|
+
fields=[
|
|
179
|
+
OntologyFieldMapping(
|
|
180
|
+
ontology_field="email", node_field="email", required=True
|
|
181
|
+
),
|
|
182
|
+
OntologyFieldMapping(
|
|
183
|
+
ontology_field="first_name", node_field="first_name"
|
|
184
|
+
),
|
|
185
|
+
OntologyFieldMapping(
|
|
186
|
+
ontology_field="last_name", node_field="last_name"
|
|
187
|
+
),
|
|
188
|
+
OntologyFieldMapping(ontology_field="username", node_field="username"),
|
|
189
|
+
],
|
|
190
|
+
),
|
|
191
|
+
],
|
|
192
|
+
)
|
|
193
|
+
snipeit_mapping = OntologyMapping(
|
|
194
|
+
module_name="snipeit",
|
|
195
|
+
nodes=[
|
|
196
|
+
OntologyNodeMapping(
|
|
197
|
+
node_label="SnipeitUser",
|
|
198
|
+
fields=[
|
|
199
|
+
OntologyFieldMapping(
|
|
200
|
+
ontology_field="email", node_field="email", required=True
|
|
201
|
+
),
|
|
202
|
+
OntologyFieldMapping(ontology_field="username", node_field="username"),
|
|
203
|
+
],
|
|
204
|
+
),
|
|
205
|
+
],
|
|
206
|
+
)
|
|
207
|
+
tailscale_mapping = OntologyMapping(
|
|
208
|
+
module_name="tailscale",
|
|
209
|
+
nodes=[
|
|
210
|
+
OntologyNodeMapping(
|
|
211
|
+
node_label="TailscaleUser",
|
|
212
|
+
fields=[
|
|
213
|
+
OntologyFieldMapping(
|
|
214
|
+
ontology_field="email", node_field="email", required=True
|
|
215
|
+
),
|
|
216
|
+
OntologyFieldMapping(
|
|
217
|
+
ontology_field="fullname", node_field="display_name"
|
|
218
|
+
),
|
|
219
|
+
],
|
|
220
|
+
),
|
|
221
|
+
],
|
|
222
|
+
)
|
|
223
|
+
|
|
224
|
+
|
|
225
|
+
USERS_ONTOLOGY_MAPPING: dict[str, OntologyMapping] = {
|
|
226
|
+
"anthropic": anthropic_mapping,
|
|
227
|
+
"airbyte": airbyte_mapping,
|
|
228
|
+
"cloudflare": cloudflare_mapping,
|
|
229
|
+
"duo": duo_mapping,
|
|
230
|
+
"entra": entra_mapping,
|
|
231
|
+
"github": github_mapping,
|
|
232
|
+
"gsuite": gsuite_mapping,
|
|
233
|
+
"keycloak": keycloak_mapping,
|
|
234
|
+
"lastpass": lastpass_mapping,
|
|
235
|
+
"openai": openai_mapping,
|
|
236
|
+
"scaleway": scaleway_mapping,
|
|
237
|
+
"snipeit": snipeit_mapping,
|
|
238
|
+
"tailscale": tailscale_mapping,
|
|
239
|
+
}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
from dataclasses import dataclass
|
|
2
|
+
from dataclasses import field
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
@dataclass(frozen=True)
|
|
6
|
+
class OntologyFieldMapping:
|
|
7
|
+
"""Mapping between an ontology field and a module's node field.
|
|
8
|
+
|
|
9
|
+
Attributes:
|
|
10
|
+
ontology_field: The field name in the ontology.
|
|
11
|
+
node_field: The corresponding field name in the module's node.
|
|
12
|
+
required: Whether this field is required to create an ontology node or not.
|
|
13
|
+
|
|
14
|
+
Example:
|
|
15
|
+
OntologyFieldMapping(ontology_field="email", node_field="email_address", required=True)
|
|
16
|
+
This mapping indicates that the 'email' field in the ontology corresponds to the 'email_address' field in the module's node and is required.
|
|
17
|
+
So {"id": "123", "email_address": "<email_value>"} can be mapped to an ontology node, but {"id": "123"} cannot.
|
|
18
|
+
"""
|
|
19
|
+
|
|
20
|
+
ontology_field: str
|
|
21
|
+
node_field: str
|
|
22
|
+
required: bool = False
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
@dataclass(frozen=True)
|
|
26
|
+
class OntologyNodeMapping:
|
|
27
|
+
"""Mapping for a node in the ontology.
|
|
28
|
+
|
|
29
|
+
Attributes:
|
|
30
|
+
node_label: The label of the ontology node.
|
|
31
|
+
fields: A list of OntologyFieldMapping defining the field mappings for this node.
|
|
32
|
+
"""
|
|
33
|
+
|
|
34
|
+
node_label: str
|
|
35
|
+
fields: list[OntologyFieldMapping]
|
|
36
|
+
|
|
37
|
+
|
|
38
|
+
@dataclass(frozen=True)
|
|
39
|
+
class OntologyRelMapping:
|
|
40
|
+
"""Mapping for a relationship in the ontology.
|
|
41
|
+
|
|
42
|
+
Attributes:
|
|
43
|
+
query: The query used to retrieve this relationship.
|
|
44
|
+
iterative: Whether this relationship requires batch processing (iterative) or can be created in a single query.
|
|
45
|
+
__comment__: An optional comment about this relationship.
|
|
46
|
+
"""
|
|
47
|
+
|
|
48
|
+
query: str
|
|
49
|
+
iterative: bool = False
|
|
50
|
+
__comment__: str | None = None
|
|
51
|
+
|
|
52
|
+
|
|
53
|
+
@dataclass(frozen=True)
|
|
54
|
+
class OntologyMapping:
|
|
55
|
+
"""Ontology mapping for a specific module.
|
|
56
|
+
|
|
57
|
+
Attributes:
|
|
58
|
+
module_name: The name of the module.
|
|
59
|
+
nodes: A list of OntologyNodeMapping defining the nodes for this module.
|
|
60
|
+
rels: A list of OntologyRelMapping defining the relationships for this module.
|
|
61
|
+
"""
|
|
62
|
+
|
|
63
|
+
module_name: str
|
|
64
|
+
nodes: list[OntologyNodeMapping]
|
|
65
|
+
rels: list[OntologyRelMapping] = field(default_factory=list)
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
# See https://d3fend.mitre.org/dao/artifact/d3f:User/
|
|
2
|
+
from dataclasses import dataclass
|
|
3
|
+
|
|
4
|
+
from cartography.models.core.common import PropertyRef
|
|
5
|
+
from cartography.models.core.nodes import CartographyNodeProperties
|
|
6
|
+
from cartography.models.core.nodes import CartographyNodeSchema
|
|
7
|
+
from cartography.models.core.nodes import ExtraNodeLabels
|
|
8
|
+
from cartography.models.core.relationships import CartographyRelProperties
|
|
9
|
+
from cartography.models.core.relationships import CartographyRelSchema
|
|
10
|
+
from cartography.models.core.relationships import LinkDirection
|
|
11
|
+
from cartography.models.core.relationships import make_target_node_matcher
|
|
12
|
+
from cartography.models.core.relationships import OtherRelationships
|
|
13
|
+
from cartography.models.core.relationships import TargetNodeMatcher
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
@dataclass(frozen=True)
|
|
17
|
+
class UserNodeProperties(CartographyNodeProperties):
|
|
18
|
+
id: PropertyRef = PropertyRef("email")
|
|
19
|
+
lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
|
|
20
|
+
email: PropertyRef = PropertyRef("email", extra_index=True)
|
|
21
|
+
username: PropertyRef = PropertyRef("username")
|
|
22
|
+
fullname: PropertyRef = PropertyRef("fullname")
|
|
23
|
+
firstname: PropertyRef = PropertyRef("firstname")
|
|
24
|
+
lastname: PropertyRef = PropertyRef("lastname")
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
@dataclass(frozen=True)
|
|
28
|
+
class UserToUserAccountRelProperties(CartographyRelProperties):
|
|
29
|
+
lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
# (:User)-[:HAS_ACCOUNT]->(:UserAccount)
|
|
33
|
+
# This is a relationship to a sementic label used by modules' users nodes
|
|
34
|
+
class UserToUserAccountRel(CartographyRelSchema):
|
|
35
|
+
target_node_label: str = "UserAccount"
|
|
36
|
+
target_node_matcher: TargetNodeMatcher = make_target_node_matcher(
|
|
37
|
+
{"email": PropertyRef("email")},
|
|
38
|
+
)
|
|
39
|
+
direction: LinkDirection = LinkDirection.OUTWARD
|
|
40
|
+
rel_label: str = "HAS_ACCOUNT"
|
|
41
|
+
properties: UserToUserAccountRelProperties = UserToUserAccountRelProperties()
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
@dataclass(frozen=True)
|
|
45
|
+
class UserSchema(CartographyNodeSchema):
|
|
46
|
+
label: str = "User"
|
|
47
|
+
extra_node_labels: ExtraNodeLabels = ExtraNodeLabels(["Ontology"])
|
|
48
|
+
properties: UserNodeProperties = UserNodeProperties()
|
|
49
|
+
scoped_cleanup: bool = False
|
|
50
|
+
other_relationships: OtherRelationships = OtherRelationships(
|
|
51
|
+
rels=[UserToUserAccountRel()],
|
|
52
|
+
)
|
|
@@ -3,6 +3,7 @@ from dataclasses import dataclass
|
|
|
3
3
|
from cartography.models.core.common import PropertyRef
|
|
4
4
|
from cartography.models.core.nodes import CartographyNodeProperties
|
|
5
5
|
from cartography.models.core.nodes import CartographyNodeSchema
|
|
6
|
+
from cartography.models.core.nodes import ExtraNodeLabels
|
|
6
7
|
from cartography.models.core.relationships import CartographyRelProperties
|
|
7
8
|
from cartography.models.core.relationships import CartographyRelSchema
|
|
8
9
|
from cartography.models.core.relationships import LinkDirection
|
|
@@ -43,6 +44,9 @@ class OpenAIUserToOrganizationRel(CartographyRelSchema):
|
|
|
43
44
|
@dataclass(frozen=True)
|
|
44
45
|
class OpenAIUserSchema(CartographyNodeSchema):
|
|
45
46
|
label: str = "OpenAIUser"
|
|
47
|
+
extra_node_labels: ExtraNodeLabels = ExtraNodeLabels(
|
|
48
|
+
["UserAccount"]
|
|
49
|
+
) # UserAccount label is used for ontology mapping
|
|
46
50
|
properties: OpenAIUserNodeProperties = OpenAIUserNodeProperties()
|
|
47
51
|
sub_resource_relationship: OpenAIUserToOrganizationRel = (
|
|
48
52
|
OpenAIUserToOrganizationRel()
|
|
@@ -3,6 +3,7 @@ from dataclasses import dataclass
|
|
|
3
3
|
from cartography.models.core.common import PropertyRef
|
|
4
4
|
from cartography.models.core.nodes import CartographyNodeProperties
|
|
5
5
|
from cartography.models.core.nodes import CartographyNodeSchema
|
|
6
|
+
from cartography.models.core.nodes import ExtraNodeLabels
|
|
6
7
|
from cartography.models.core.relationships import CartographyRelProperties
|
|
7
8
|
from cartography.models.core.relationships import CartographyRelSchema
|
|
8
9
|
from cartography.models.core.relationships import LinkDirection
|
|
@@ -54,6 +55,9 @@ class ScalewayUserToOrganizationRel(CartographyRelSchema):
|
|
|
54
55
|
@dataclass(frozen=True)
|
|
55
56
|
class ScalewayUserSchema(CartographyNodeSchema):
|
|
56
57
|
label: str = "ScalewayUser"
|
|
58
|
+
extra_node_labels: ExtraNodeLabels = ExtraNodeLabels(
|
|
59
|
+
["UserAccount"]
|
|
60
|
+
) # UserAccount label is used for ontology mapping
|
|
57
61
|
properties: ScalewayUserNodeProperties = ScalewayUserNodeProperties()
|
|
58
62
|
sub_resource_relationship: ScalewayUserToOrganizationRel = (
|
|
59
63
|
ScalewayUserToOrganizationRel()
|
|
@@ -22,6 +22,7 @@ class SnipeitAssetNodeProperties(CartographyNodeProperties):
|
|
|
22
22
|
lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
|
|
23
23
|
|
|
24
24
|
# SnipeIT specific properties
|
|
25
|
+
name: PropertyRef = PropertyRef("name", extra_index=True)
|
|
25
26
|
asset_tag: PropertyRef = PropertyRef("asset_tag")
|
|
26
27
|
assigned_to: PropertyRef = PropertyRef("assigned_to.email")
|
|
27
28
|
category: PropertyRef = PropertyRef("category.name")
|
|
@@ -3,6 +3,7 @@ from dataclasses import dataclass
|
|
|
3
3
|
from cartography.models.core.common import PropertyRef
|
|
4
4
|
from cartography.models.core.nodes import CartographyNodeProperties
|
|
5
5
|
from cartography.models.core.nodes import CartographyNodeSchema
|
|
6
|
+
from cartography.models.core.nodes import ExtraNodeLabels
|
|
6
7
|
from cartography.models.core.relationships import CartographyRelProperties
|
|
7
8
|
from cartography.models.core.relationships import CartographyRelSchema
|
|
8
9
|
from cartography.models.core.relationships import LinkDirection
|
|
@@ -48,6 +49,9 @@ class SnipeitTenantToSnipeitUserRel(CartographyRelSchema):
|
|
|
48
49
|
@dataclass(frozen=True)
|
|
49
50
|
class SnipeitUserSchema(CartographyNodeSchema):
|
|
50
51
|
label: str = "SnipeitUser" # The label of the node
|
|
52
|
+
extra_node_labels: ExtraNodeLabels = ExtraNodeLabels(
|
|
53
|
+
["UserAccount"]
|
|
54
|
+
) # UserAccount label is used for ontology mapping
|
|
51
55
|
properties: SnipeitUserNodeProperties = (
|
|
52
56
|
SnipeitUserNodeProperties()
|
|
53
57
|
) # An object representing all properties
|
|
@@ -17,7 +17,7 @@ class TailscaleDeviceNodeProperties(CartographyNodeProperties):
|
|
|
17
17
|
id: PropertyRef = PropertyRef("nodeId")
|
|
18
18
|
lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
|
|
19
19
|
name: PropertyRef = PropertyRef("name")
|
|
20
|
-
hostname: PropertyRef = PropertyRef("hostname")
|
|
20
|
+
hostname: PropertyRef = PropertyRef("hostname", extra_index=True)
|
|
21
21
|
client_version: PropertyRef = PropertyRef("clientVersion")
|
|
22
22
|
update_available: PropertyRef = PropertyRef("updateAvailable")
|
|
23
23
|
os: PropertyRef = PropertyRef("os")
|
|
@@ -3,6 +3,7 @@ from dataclasses import dataclass
|
|
|
3
3
|
from cartography.models.core.common import PropertyRef
|
|
4
4
|
from cartography.models.core.nodes import CartographyNodeProperties
|
|
5
5
|
from cartography.models.core.nodes import CartographyNodeSchema
|
|
6
|
+
from cartography.models.core.nodes import ExtraNodeLabels
|
|
6
7
|
from cartography.models.core.relationships import CartographyRelProperties
|
|
7
8
|
from cartography.models.core.relationships import CartographyRelSchema
|
|
8
9
|
from cartography.models.core.relationships import LinkDirection
|
|
@@ -15,7 +16,8 @@ class TailscaleUserNodeProperties(CartographyNodeProperties):
|
|
|
15
16
|
id: PropertyRef = PropertyRef("id")
|
|
16
17
|
lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
|
|
17
18
|
display_name: PropertyRef = PropertyRef("displayName")
|
|
18
|
-
login_name: PropertyRef = PropertyRef("loginName"
|
|
19
|
+
login_name: PropertyRef = PropertyRef("loginName")
|
|
20
|
+
email: PropertyRef = PropertyRef("loginName", extra_index=True)
|
|
19
21
|
profile_pic_url: PropertyRef = PropertyRef("profilePicUrl")
|
|
20
22
|
created: PropertyRef = PropertyRef("created")
|
|
21
23
|
type: PropertyRef = PropertyRef("type")
|
|
@@ -48,5 +50,8 @@ class TailscaleUserToTailnetRel(CartographyRelSchema):
|
|
|
48
50
|
@dataclass(frozen=True)
|
|
49
51
|
class TailscaleUserSchema(CartographyNodeSchema):
|
|
50
52
|
label: str = "TailscaleUser"
|
|
53
|
+
extra_node_labels: ExtraNodeLabels = ExtraNodeLabels(
|
|
54
|
+
["UserAccount"]
|
|
55
|
+
) # UserAccount label is used for ontology mapping
|
|
51
56
|
properties: TailscaleUserNodeProperties = TailscaleUserNodeProperties()
|
|
52
57
|
sub_resource_relationship: TailscaleUserToTailnetRel = TailscaleUserToTailnetRel()
|