cartography 0.102.0rc2__py3-none-any.whl → 0.103.0rc1__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.

Files changed (251) hide show
  1. cartography/__main__.py +1 -2
  2. cartography/_version.py +2 -2
  3. cartography/cli.py +302 -253
  4. cartography/client/core/tx.py +39 -18
  5. cartography/config.py +4 -0
  6. cartography/driftdetect/__main__.py +1 -2
  7. cartography/driftdetect/add_shortcut.py +10 -2
  8. cartography/driftdetect/cli.py +71 -75
  9. cartography/driftdetect/detect_deviations.py +7 -3
  10. cartography/driftdetect/get_states.py +20 -8
  11. cartography/driftdetect/model.py +5 -5
  12. cartography/driftdetect/serializers.py +8 -6
  13. cartography/driftdetect/storage.py +2 -2
  14. cartography/graph/cleanupbuilder.py +35 -15
  15. cartography/graph/job.py +46 -17
  16. cartography/graph/querybuilder.py +165 -80
  17. cartography/graph/statement.py +35 -26
  18. cartography/intel/analysis.py +4 -1
  19. cartography/intel/aws/__init__.py +114 -55
  20. cartography/intel/aws/apigateway.py +134 -63
  21. cartography/intel/aws/cloudtrail.py +127 -0
  22. cartography/intel/aws/config.py +56 -20
  23. cartography/intel/aws/dynamodb.py +108 -40
  24. cartography/intel/aws/ec2/__init__.py +2 -2
  25. cartography/intel/aws/ec2/auto_scaling_groups.py +181 -78
  26. cartography/intel/aws/ec2/elastic_ip_addresses.py +41 -13
  27. cartography/intel/aws/ec2/images.py +49 -20
  28. cartography/intel/aws/ec2/instances.py +234 -136
  29. cartography/intel/aws/ec2/internet_gateways.py +40 -11
  30. cartography/intel/aws/ec2/key_pairs.py +44 -20
  31. cartography/intel/aws/ec2/launch_templates.py +101 -59
  32. cartography/intel/aws/ec2/load_balancer_v2s.py +104 -39
  33. cartography/intel/aws/ec2/load_balancers.py +82 -42
  34. cartography/intel/aws/ec2/network_acls.py +89 -65
  35. cartography/intel/aws/ec2/network_interfaces.py +146 -87
  36. cartography/intel/aws/ec2/reserved_instances.py +45 -16
  37. cartography/intel/aws/ec2/route_tables.py +138 -98
  38. cartography/intel/aws/ec2/security_groups.py +71 -21
  39. cartography/intel/aws/ec2/snapshots.py +61 -22
  40. cartography/intel/aws/ec2/subnets.py +54 -18
  41. cartography/intel/aws/ec2/tgw.py +100 -34
  42. cartography/intel/aws/ec2/util.py +1 -1
  43. cartography/intel/aws/ec2/volumes.py +69 -41
  44. cartography/intel/aws/ec2/vpc.py +37 -12
  45. cartography/intel/aws/ec2/vpc_peerings.py +83 -24
  46. cartography/intel/aws/ecr.py +88 -32
  47. cartography/intel/aws/ecs.py +83 -47
  48. cartography/intel/aws/eks.py +55 -29
  49. cartography/intel/aws/elasticache.py +42 -18
  50. cartography/intel/aws/elasticsearch.py +57 -20
  51. cartography/intel/aws/emr.py +61 -23
  52. cartography/intel/aws/iam.py +401 -145
  53. cartography/intel/aws/iam_instance_profiles.py +22 -22
  54. cartography/intel/aws/identitycenter.py +71 -37
  55. cartography/intel/aws/inspector.py +159 -89
  56. cartography/intel/aws/kms.py +92 -38
  57. cartography/intel/aws/lambda_function.py +103 -34
  58. cartography/intel/aws/organizations.py +30 -10
  59. cartography/intel/aws/permission_relationships.py +133 -51
  60. cartography/intel/aws/rds.py +249 -85
  61. cartography/intel/aws/redshift.py +107 -46
  62. cartography/intel/aws/resourcegroupstaggingapi.py +120 -66
  63. cartography/intel/aws/resources.py +53 -46
  64. cartography/intel/aws/route53.py +108 -61
  65. cartography/intel/aws/s3.py +168 -83
  66. cartography/intel/aws/s3accountpublicaccessblock.py +157 -0
  67. cartography/intel/aws/secretsmanager.py +24 -12
  68. cartography/intel/aws/securityhub.py +20 -9
  69. cartography/intel/aws/sns.py +166 -0
  70. cartography/intel/aws/sqs.py +60 -28
  71. cartography/intel/aws/ssm.py +70 -30
  72. cartography/intel/aws/util/arns.py +7 -7
  73. cartography/intel/aws/util/common.py +31 -4
  74. cartography/intel/azure/__init__.py +78 -19
  75. cartography/intel/azure/compute.py +101 -27
  76. cartography/intel/azure/cosmosdb.py +496 -170
  77. cartography/intel/azure/sql.py +296 -105
  78. cartography/intel/azure/storage.py +322 -113
  79. cartography/intel/azure/subscription.py +39 -23
  80. cartography/intel/azure/tenant.py +13 -4
  81. cartography/intel/azure/util/credentials.py +95 -55
  82. cartography/intel/bigfix/__init__.py +2 -2
  83. cartography/intel/bigfix/computers.py +93 -65
  84. cartography/intel/create_indexes.py +3 -2
  85. cartography/intel/crowdstrike/__init__.py +11 -9
  86. cartography/intel/crowdstrike/endpoints.py +5 -1
  87. cartography/intel/crowdstrike/spotlight.py +8 -3
  88. cartography/intel/cve/__init__.py +46 -13
  89. cartography/intel/cve/feed.py +48 -12
  90. cartography/intel/digitalocean/__init__.py +22 -13
  91. cartography/intel/digitalocean/compute.py +75 -108
  92. cartography/intel/digitalocean/management.py +44 -80
  93. cartography/intel/digitalocean/platform.py +48 -43
  94. cartography/intel/dns.py +36 -10
  95. cartography/intel/duo/__init__.py +21 -16
  96. cartography/intel/duo/api_host.py +14 -9
  97. cartography/intel/duo/endpoints.py +50 -45
  98. cartography/intel/duo/groups.py +18 -14
  99. cartography/intel/duo/phones.py +37 -34
  100. cartography/intel/duo/tokens.py +26 -23
  101. cartography/intel/duo/users.py +54 -50
  102. cartography/intel/duo/web_authn_credentials.py +30 -25
  103. cartography/intel/entra/__init__.py +25 -7
  104. cartography/intel/entra/ou.py +112 -0
  105. cartography/intel/entra/users.py +69 -63
  106. cartography/intel/gcp/__init__.py +185 -49
  107. cartography/intel/gcp/compute.py +418 -231
  108. cartography/intel/gcp/crm.py +96 -43
  109. cartography/intel/gcp/dns.py +60 -19
  110. cartography/intel/gcp/gke.py +72 -38
  111. cartography/intel/gcp/iam.py +61 -41
  112. cartography/intel/gcp/storage.py +84 -55
  113. cartography/intel/github/__init__.py +13 -11
  114. cartography/intel/github/repos.py +270 -137
  115. cartography/intel/github/teams.py +170 -88
  116. cartography/intel/github/users.py +70 -39
  117. cartography/intel/github/util.py +36 -34
  118. cartography/intel/gsuite/__init__.py +47 -26
  119. cartography/intel/gsuite/api.py +73 -30
  120. cartography/intel/jamf/__init__.py +19 -1
  121. cartography/intel/jamf/computers.py +30 -7
  122. cartography/intel/jamf/util.py +7 -2
  123. cartography/intel/kandji/__init__.py +6 -3
  124. cartography/intel/kandji/devices.py +14 -8
  125. cartography/intel/kubernetes/namespaces.py +7 -4
  126. cartography/intel/kubernetes/pods.py +7 -4
  127. cartography/intel/kubernetes/services.py +8 -4
  128. cartography/intel/lastpass/__init__.py +2 -2
  129. cartography/intel/lastpass/users.py +23 -12
  130. cartography/intel/oci/__init__.py +44 -11
  131. cartography/intel/oci/iam.py +134 -38
  132. cartography/intel/oci/organizations.py +13 -6
  133. cartography/intel/oci/utils.py +43 -20
  134. cartography/intel/okta/__init__.py +66 -15
  135. cartography/intel/okta/applications.py +42 -20
  136. cartography/intel/okta/awssaml.py +93 -33
  137. cartography/intel/okta/factors.py +16 -4
  138. cartography/intel/okta/groups.py +56 -29
  139. cartography/intel/okta/organization.py +5 -1
  140. cartography/intel/okta/origins.py +6 -2
  141. cartography/intel/okta/roles.py +15 -5
  142. cartography/intel/okta/users.py +20 -8
  143. cartography/intel/okta/utils.py +6 -4
  144. cartography/intel/pagerduty/__init__.py +8 -7
  145. cartography/intel/pagerduty/escalation_policies.py +18 -6
  146. cartography/intel/pagerduty/schedules.py +12 -4
  147. cartography/intel/pagerduty/services.py +11 -4
  148. cartography/intel/pagerduty/teams.py +8 -3
  149. cartography/intel/pagerduty/users.py +3 -1
  150. cartography/intel/pagerduty/vendors.py +3 -1
  151. cartography/intel/semgrep/__init__.py +24 -6
  152. cartography/intel/semgrep/dependencies.py +50 -28
  153. cartography/intel/semgrep/deployment.py +3 -1
  154. cartography/intel/semgrep/findings.py +42 -18
  155. cartography/intel/snipeit/__init__.py +17 -3
  156. cartography/intel/snipeit/asset.py +12 -6
  157. cartography/intel/snipeit/user.py +8 -5
  158. cartography/intel/snipeit/util.py +9 -4
  159. cartography/models/aws/apigateway.py +21 -17
  160. cartography/models/aws/apigatewaycertificate.py +28 -22
  161. cartography/models/aws/apigatewayresource.py +28 -20
  162. cartography/models/aws/apigatewaystage.py +33 -25
  163. cartography/models/aws/cloudtrail/__init__.py +0 -0
  164. cartography/models/aws/cloudtrail/trail.py +61 -0
  165. cartography/models/aws/dynamodb/gsi.py +30 -22
  166. cartography/models/aws/dynamodb/tables.py +25 -17
  167. cartography/models/aws/ec2/auto_scaling_groups.py +102 -82
  168. cartography/models/aws/ec2/images.py +36 -34
  169. cartography/models/aws/ec2/instances.py +51 -45
  170. cartography/models/aws/ec2/keypair.py +21 -16
  171. cartography/models/aws/ec2/keypair_instance.py +28 -21
  172. cartography/models/aws/ec2/launch_configurations.py +30 -26
  173. cartography/models/aws/ec2/launch_template_versions.py +48 -38
  174. cartography/models/aws/ec2/launch_templates.py +21 -17
  175. cartography/models/aws/ec2/load_balancer_listeners.py +27 -23
  176. cartography/models/aws/ec2/load_balancers.py +47 -37
  177. cartography/models/aws/ec2/network_acl_rules.py +38 -30
  178. cartography/models/aws/ec2/network_acls.py +38 -29
  179. cartography/models/aws/ec2/networkinterface_instance.py +52 -39
  180. cartography/models/aws/ec2/networkinterfaces.py +53 -37
  181. cartography/models/aws/ec2/privateip_networkinterface.py +32 -22
  182. cartography/models/aws/ec2/reservations.py +18 -14
  183. cartography/models/aws/ec2/route_table_associations.py +44 -34
  184. cartography/models/aws/ec2/route_tables.py +50 -43
  185. cartography/models/aws/ec2/routes.py +45 -37
  186. cartography/models/aws/ec2/securitygroup_instance.py +29 -20
  187. cartography/models/aws/ec2/securitygroup_networkinterface.py +24 -15
  188. cartography/models/aws/ec2/subnet_instance.py +24 -19
  189. cartography/models/aws/ec2/subnet_networkinterface.py +40 -31
  190. cartography/models/aws/ec2/volumes.py +47 -40
  191. cartography/models/aws/eks/clusters.py +23 -21
  192. cartography/models/aws/emr.py +32 -30
  193. cartography/models/aws/iam/instanceprofile.py +33 -24
  194. cartography/models/aws/identitycenter/awsidentitycenter.py +18 -14
  195. cartography/models/aws/identitycenter/awspermissionset.py +37 -29
  196. cartography/models/aws/identitycenter/awsssouser.py +23 -21
  197. cartography/models/aws/inspector/findings.py +77 -65
  198. cartography/models/aws/inspector/packages.py +35 -29
  199. cartography/models/aws/s3/__init__.py +0 -0
  200. cartography/models/aws/s3/account_public_access_block.py +51 -0
  201. cartography/models/aws/sns/__init__.py +0 -0
  202. cartography/models/aws/sns/topic.py +50 -0
  203. cartography/models/aws/ssm/instance_information.py +51 -39
  204. cartography/models/aws/ssm/instance_patch.py +32 -26
  205. cartography/models/bigfix/bigfix_computer.py +42 -38
  206. cartography/models/bigfix/bigfix_root.py +3 -3
  207. cartography/models/core/common.py +12 -10
  208. cartography/models/core/nodes.py +5 -2
  209. cartography/models/core/relationships.py +14 -6
  210. cartography/models/crowdstrike/hosts.py +37 -35
  211. cartography/models/cve/cve.py +34 -32
  212. cartography/models/cve/cve_feed.py +6 -6
  213. cartography/models/digitalocean/__init__.py +0 -0
  214. cartography/models/digitalocean/account.py +21 -0
  215. cartography/models/digitalocean/droplet.py +56 -0
  216. cartography/models/digitalocean/project.py +48 -0
  217. cartography/models/duo/api_host.py +3 -3
  218. cartography/models/duo/endpoint.py +43 -41
  219. cartography/models/duo/group.py +14 -14
  220. cartography/models/duo/phone.py +27 -27
  221. cartography/models/duo/token.py +16 -16
  222. cartography/models/duo/user.py +46 -44
  223. cartography/models/duo/web_authn_credential.py +27 -19
  224. cartography/models/entra/ou.py +48 -0
  225. cartography/models/entra/tenant.py +24 -18
  226. cartography/models/entra/user.py +64 -48
  227. cartography/models/gcp/iam.py +23 -23
  228. cartography/models/github/orgs.py +5 -4
  229. cartography/models/github/teams.py +37 -31
  230. cartography/models/github/users.py +34 -23
  231. cartography/models/kandji/device.py +22 -16
  232. cartography/models/kandji/tenant.py +6 -4
  233. cartography/models/lastpass/tenant.py +3 -3
  234. cartography/models/lastpass/user.py +32 -28
  235. cartography/models/semgrep/dependencies.py +36 -24
  236. cartography/models/semgrep/deployment.py +5 -5
  237. cartography/models/semgrep/findings.py +58 -42
  238. cartography/models/semgrep/locations.py +27 -21
  239. cartography/models/snipeit/asset.py +30 -21
  240. cartography/models/snipeit/tenant.py +6 -4
  241. cartography/models/snipeit/user.py +19 -12
  242. cartography/stats.py +3 -3
  243. cartography/sync.py +107 -31
  244. cartography/util.py +84 -62
  245. {cartography-0.102.0rc2.dist-info → cartography-0.103.0rc1.dist-info}/METADATA +3 -14
  246. cartography-0.103.0rc1.dist-info/RECORD +396 -0
  247. {cartography-0.102.0rc2.dist-info → cartography-0.103.0rc1.dist-info}/WHEEL +1 -1
  248. cartography-0.102.0rc2.dist-info/RECORD +0 -381
  249. {cartography-0.102.0rc2.dist-info → cartography-0.103.0rc1.dist-info}/entry_points.txt +0 -0
  250. {cartography-0.102.0rc2.dist-info → cartography-0.103.0rc1.dist-info}/licenses/LICENSE +0 -0
  251. {cartography-0.102.0rc2.dist-info → cartography-0.103.0rc1.dist-info}/top_level.txt +0 -0
@@ -16,18 +16,19 @@ class SnipeitAssetNodeProperties(CartographyNodeProperties):
16
16
  """
17
17
  https://snipe-it.readme.io/reference/hardware-list
18
18
  """
19
+
19
20
  # Common properties
20
- id: PropertyRef = PropertyRef('id')
21
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
21
+ id: PropertyRef = PropertyRef("id")
22
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
22
23
 
23
24
  # SnipeIT specific properties
24
- asset_tag: PropertyRef = PropertyRef('asset_tag')
25
- assigned_to: PropertyRef = PropertyRef('assigned_to.email')
26
- category: PropertyRef = PropertyRef('category.name')
27
- company: PropertyRef = PropertyRef('company.name')
28
- manufacturer: PropertyRef = PropertyRef('manufacturer.name')
29
- model: PropertyRef = PropertyRef('model.name')
30
- serial: PropertyRef = PropertyRef('serial', extra_index=True)
25
+ asset_tag: PropertyRef = PropertyRef("asset_tag")
26
+ assigned_to: PropertyRef = PropertyRef("assigned_to.email")
27
+ category: PropertyRef = PropertyRef("category.name")
28
+ company: PropertyRef = PropertyRef("company.name")
29
+ manufacturer: PropertyRef = PropertyRef("manufacturer.name")
30
+ model: PropertyRef = PropertyRef("model.name")
31
+ serial: PropertyRef = PropertyRef("serial", extra_index=True)
31
32
 
32
33
 
33
34
  ###
@@ -35,45 +36,53 @@ class SnipeitAssetNodeProperties(CartographyNodeProperties):
35
36
  ###
36
37
  @dataclass(frozen=True)
37
38
  class SnipeitTenantToSnipeitAssetRelProperties(CartographyRelProperties):
38
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
39
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
39
40
 
40
41
 
41
42
  @dataclass(frozen=True)
42
43
  class SnipeitTenantToSnipeitAssetRel(CartographyRelSchema):
43
- target_node_label: str = 'SnipeitTenant'
44
+ target_node_label: str = "SnipeitTenant"
44
45
  target_node_matcher: TargetNodeMatcher = make_target_node_matcher(
45
- {'id': PropertyRef('TENANT_ID', set_in_kwargs=True)},
46
+ {"id": PropertyRef("TENANT_ID", set_in_kwargs=True)},
46
47
  )
47
48
  direction: LinkDirection = LinkDirection.INWARD
48
49
  rel_label: str = "HAS_ASSET"
49
- properties: SnipeitTenantToSnipeitAssetRelProperties = SnipeitTenantToSnipeitAssetRelProperties()
50
+ properties: SnipeitTenantToSnipeitAssetRelProperties = (
51
+ SnipeitTenantToSnipeitAssetRelProperties()
52
+ )
50
53
 
51
54
 
52
55
  ###
53
56
  # (:SnipeitUser)-[:HAS_CHECKED_OUT]->(:SnipeitAsset)
54
57
  ###
55
58
  @dataclass(frozen=True)
56
- class SnipeitUserToSnipeitAssetProperties(CartographyRelProperties):
57
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
59
+ class SnipeitUserToSnipeitAssetRelProperties(CartographyRelProperties):
60
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
58
61
 
59
62
 
60
63
  @dataclass(frozen=True)
61
64
  class SnipeitUserToSnipeitAssetRel(CartographyRelSchema):
62
- target_node_label: str = 'SnipeitUser'
65
+ target_node_label: str = "SnipeitUser"
63
66
  target_node_matcher: TargetNodeMatcher = make_target_node_matcher(
64
- {'email': PropertyRef('assigned_to.email')},
67
+ {"email": PropertyRef("assigned_to.email")},
65
68
  )
66
69
  direction: LinkDirection = LinkDirection.INWARD
67
70
  rel_label: str = "HAS_CHECKED_OUT"
68
- properties: SnipeitUserToSnipeitAssetProperties = SnipeitUserToSnipeitAssetProperties()
71
+ properties: SnipeitUserToSnipeitAssetRelProperties = (
72
+ SnipeitUserToSnipeitAssetRelProperties()
73
+ )
69
74
 
70
75
 
71
76
  ###
72
77
  @dataclass(frozen=True)
73
78
  class SnipeitAssetSchema(CartographyNodeSchema):
74
- label: str = 'SnipeitAsset' # The label of the node
75
- properties: SnipeitAssetNodeProperties = SnipeitAssetNodeProperties() # An object representing all properties
76
- sub_resource_relationship: SnipeitTenantToSnipeitAssetRel = SnipeitTenantToSnipeitAssetRel()
79
+ label: str = "SnipeitAsset" # The label of the node
80
+ properties: SnipeitAssetNodeProperties = (
81
+ SnipeitAssetNodeProperties()
82
+ ) # An object representing all properties
83
+ sub_resource_relationship: SnipeitTenantToSnipeitAssetRel = (
84
+ SnipeitTenantToSnipeitAssetRel()
85
+ )
77
86
  other_relationships: OtherRelationships = OtherRelationships(
78
87
  [
79
88
  SnipeitUserToSnipeitAssetRel(),
@@ -7,11 +7,13 @@ from cartography.models.core.nodes import CartographyNodeSchema
7
7
 
8
8
  @dataclass(frozen=True)
9
9
  class SnipeitTenantNodeProperties(CartographyNodeProperties):
10
- id: PropertyRef = PropertyRef('id')
11
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
10
+ id: PropertyRef = PropertyRef("id")
11
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
12
12
 
13
13
 
14
14
  @dataclass(frozen=True)
15
15
  class SnipeitTenantSchema(CartographyNodeSchema):
16
- label: str = 'SnipeitTenant' # The label of the node
17
- properties: SnipeitTenantNodeProperties = SnipeitTenantNodeProperties() # An object representing all properties
16
+ label: str = "SnipeitTenant" # The label of the node
17
+ properties: SnipeitTenantNodeProperties = (
18
+ SnipeitTenantNodeProperties()
19
+ ) # An object representing all properties
@@ -15,35 +15,42 @@ class SnipeitUserNodeProperties(CartographyNodeProperties):
15
15
  """
16
16
  Ref: https://snipe-it.readme.io/reference/users
17
17
  """
18
+
18
19
  # Common properties
19
- id: PropertyRef = PropertyRef('id')
20
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
20
+ id: PropertyRef = PropertyRef("id")
21
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
21
22
 
22
23
  # SnipeIT specific properties
23
- company: PropertyRef = PropertyRef('company_id.name', extra_index=True)
24
- email: PropertyRef = PropertyRef('email', extra_index=True)
25
- username: PropertyRef = PropertyRef('username')
24
+ company: PropertyRef = PropertyRef("company_id.name", extra_index=True)
25
+ email: PropertyRef = PropertyRef("email", extra_index=True)
26
+ username: PropertyRef = PropertyRef("username")
26
27
 
27
28
 
28
29
  @dataclass(frozen=True)
29
30
  class SnipeitTenantToSnipeitUserRelProperties(CartographyRelProperties):
30
- lastupdated: PropertyRef = PropertyRef('lastupdated', set_in_kwargs=True)
31
+ lastupdated: PropertyRef = PropertyRef("lastupdated", set_in_kwargs=True)
31
32
 
32
33
 
33
34
  @dataclass(frozen=True)
34
35
  # (:SnipeitTenant)-[:HAS_USER]->(:SnipeitUser)
35
36
  class SnipeitTenantToSnipeitUserRel(CartographyRelSchema):
36
- target_node_label: str = 'SnipeitTenant'
37
+ target_node_label: str = "SnipeitTenant"
37
38
  target_node_matcher: TargetNodeMatcher = make_target_node_matcher(
38
- {'id': PropertyRef('TENANT_ID', set_in_kwargs=True)},
39
+ {"id": PropertyRef("TENANT_ID", set_in_kwargs=True)},
39
40
  )
40
41
  direction: LinkDirection = LinkDirection.INWARD
41
42
  rel_label: str = "HAS_USER"
42
- properties: SnipeitTenantToSnipeitUserRelProperties = SnipeitTenantToSnipeitUserRelProperties()
43
+ properties: SnipeitTenantToSnipeitUserRelProperties = (
44
+ SnipeitTenantToSnipeitUserRelProperties()
45
+ )
43
46
 
44
47
 
45
48
  @dataclass(frozen=True)
46
49
  class SnipeitUserSchema(CartographyNodeSchema):
47
- label: str = 'SnipeitUser' # The label of the node
48
- properties: SnipeitUserNodeProperties = SnipeitUserNodeProperties() # An object representing all properties
49
- sub_resource_relationship: SnipeitTenantToSnipeitUserRel = SnipeitTenantToSnipeitUserRel()
50
+ label: str = "SnipeitUser" # The label of the node
51
+ properties: SnipeitUserNodeProperties = (
52
+ SnipeitUserNodeProperties()
53
+ ) # An object representing all properties
54
+ sub_resource_relationship: SnipeitTenantToSnipeitUserRel = (
55
+ SnipeitTenantToSnipeitUserRel()
56
+ )
cartography/stats.py CHANGED
@@ -17,11 +17,11 @@ class ScopedStatsClient:
17
17
 
18
18
  _client: StatsClient = None
19
19
 
20
- def __init__(self, prefix: Optional[str], root: 'ScopedStatsClient'):
20
+ def __init__(self, prefix: Optional[str], root: "ScopedStatsClient"):
21
21
  self._scope_prefix = prefix
22
22
  self._root = root
23
23
 
24
- def get_stats_client(self, scope: str) -> 'ScopedStatsClient':
24
+ def get_stats_client(self, scope: str) -> "ScopedStatsClient":
25
25
  """
26
26
  This method returns a new proxy to the same client
27
27
  which will prefix all calls to underlying methods with the scoped prefix
@@ -35,7 +35,7 @@ class ScopedStatsClient:
35
35
  return scoped_stats_client
36
36
 
37
37
  @staticmethod
38
- def get_root_client() -> 'ScopedStatsClient':
38
+ def get_root_client() -> "ScopedStatsClient":
39
39
  client = ScopedStatsClient(prefix=None, root=None) # type: ignore
40
40
  client._root = client
41
41
  return client
cartography/sync.py CHANGED
@@ -1,7 +1,9 @@
1
1
  import argparse
2
2
  import logging
3
+ import re
3
4
  import time
4
5
  from collections import OrderedDict
6
+ from pkgutil import iter_modules
5
7
  from typing import Callable
6
8
  from typing import List
7
9
  from typing import Tuple
@@ -39,28 +41,30 @@ from cartography.util import STATUS_SUCCESS
39
41
  logger = logging.getLogger(__name__)
40
42
 
41
43
 
42
- TOP_LEVEL_MODULES = OrderedDict({ # preserve order so that the default sync always runs `analysis` at the very end
43
- 'create-indexes': cartography.intel.create_indexes.run,
44
- 'aws': cartography.intel.aws.start_aws_ingestion,
45
- 'azure': cartography.intel.azure.start_azure_ingestion,
46
- 'entra': cartography.intel.entra.start_entra_ingestion,
47
- 'crowdstrike': cartography.intel.crowdstrike.start_crowdstrike_ingestion,
48
- 'gcp': cartography.intel.gcp.start_gcp_ingestion,
49
- 'gsuite': cartography.intel.gsuite.start_gsuite_ingestion,
50
- 'cve': cartography.intel.cve.start_cve_ingestion,
51
- 'oci': cartography.intel.oci.start_oci_ingestion,
52
- 'okta': cartography.intel.okta.start_okta_ingestion,
53
- 'github': cartography.intel.github.start_github_ingestion,
54
- 'digitalocean': cartography.intel.digitalocean.start_digitalocean_ingestion,
55
- 'kandji': cartography.intel.kandji.start_kandji_ingestion,
56
- 'kubernetes': cartography.intel.kubernetes.start_k8s_ingestion,
57
- 'lastpass': cartography.intel.lastpass.start_lastpass_ingestion,
58
- 'bigfix': cartography.intel.bigfix.start_bigfix_ingestion,
59
- 'duo': cartography.intel.duo.start_duo_ingestion,
60
- 'semgrep': cartography.intel.semgrep.start_semgrep_ingestion,
61
- 'snipeit': cartography.intel.snipeit.start_snipeit_ingestion,
62
- 'analysis': cartography.intel.analysis.run,
63
- })
44
+ TOP_LEVEL_MODULES = OrderedDict(
45
+ { # preserve order so that the default sync always runs `analysis` at the very end
46
+ "create-indexes": cartography.intel.create_indexes.run,
47
+ "aws": cartography.intel.aws.start_aws_ingestion,
48
+ "azure": cartography.intel.azure.start_azure_ingestion,
49
+ "entra": cartography.intel.entra.start_entra_ingestion,
50
+ "crowdstrike": cartography.intel.crowdstrike.start_crowdstrike_ingestion,
51
+ "gcp": cartography.intel.gcp.start_gcp_ingestion,
52
+ "gsuite": cartography.intel.gsuite.start_gsuite_ingestion,
53
+ "cve": cartography.intel.cve.start_cve_ingestion,
54
+ "oci": cartography.intel.oci.start_oci_ingestion,
55
+ "okta": cartography.intel.okta.start_okta_ingestion,
56
+ "github": cartography.intel.github.start_github_ingestion,
57
+ "digitalocean": cartography.intel.digitalocean.start_digitalocean_ingestion,
58
+ "kandji": cartography.intel.kandji.start_kandji_ingestion,
59
+ "kubernetes": cartography.intel.kubernetes.start_k8s_ingestion,
60
+ "lastpass": cartography.intel.lastpass.start_lastpass_ingestion,
61
+ "bigfix": cartography.intel.bigfix.start_bigfix_ingestion,
62
+ "duo": cartography.intel.duo.start_duo_ingestion,
63
+ "semgrep": cartography.intel.semgrep.start_semgrep_ingestion,
64
+ "snipeit": cartography.intel.snipeit.start_snipeit_ingestion,
65
+ "analysis": cartography.intel.analysis.run,
66
+ }
67
+ )
64
68
 
65
69
 
66
70
  class Sync:
@@ -98,7 +102,11 @@ class Sync:
98
102
  for name, func in stages:
99
103
  self.add_stage(name, func)
100
104
 
101
- def run(self, neo4j_driver: neo4j.Driver, config: Union[Config, argparse.Namespace]) -> int:
105
+ def run(
106
+ self,
107
+ neo4j_driver: neo4j.Driver,
108
+ config: Union[Config, argparse.Namespace],
109
+ ) -> int:
102
110
  """
103
111
  Execute all stages in the sync task in sequence.
104
112
 
@@ -117,12 +125,77 @@ class Sync:
117
125
  logger.warning("Sync interrupted during stage '%s'.", stage_name)
118
126
  raise
119
127
  except Exception:
120
- logger.exception("Unhandled exception during sync stage '%s'", stage_name)
128
+ logger.exception(
129
+ "Unhandled exception during sync stage '%s'",
130
+ stage_name,
131
+ )
121
132
  raise # TODO this should be configurable
122
133
  logger.info("Finishing sync stage '%s'", stage_name)
123
134
  logger.info("Finishing sync with update tag '%d'", config.update_tag)
124
135
  return STATUS_SUCCESS
125
136
 
137
+ @classmethod
138
+ def list_intel_modules(cls) -> OrderedDict:
139
+ """
140
+ List all available intel modules.
141
+
142
+ This method will load all modules in the cartography.intel package and return a dictionary of their names and
143
+ their callable functions. The keys of the dictionary are the module names, and the values are the callable
144
+ functions (with `start_{module}_ingestion` pattern) that should be executed during the sync process.
145
+ analysis and create_indexes are loaded separately to ensure they are always available and run first
146
+ (for create-index) and last (for analysis).
147
+
148
+ :rtype: OrderedDict
149
+ :return: A dictionary of available intel modules.
150
+ """
151
+ available_modules = OrderedDict({})
152
+ available_modules["create-indexes"] = cartography.intel.create_indexes.run
153
+ callable_regex = re.compile(r"^start_(.+)_ingestion$")
154
+ # Load built-in modules
155
+ for intel_module_info in iter_modules(cartography.intel.__path__):
156
+ if intel_module_info.name in ("analysis", "create_indexes"):
157
+ continue
158
+ try:
159
+ logger.debug("Loading module: %s", intel_module_info.name)
160
+ intel_module = __import__(
161
+ f"cartography.intel.{intel_module_info.name}",
162
+ fromlist=[""],
163
+ )
164
+ except ImportError as e:
165
+ logger.error(
166
+ "Failed to import module '%s'. Error: %s",
167
+ intel_module_info.name,
168
+ e,
169
+ )
170
+ continue
171
+ logger.debug("Loading module: %s", intel_module_info.name)
172
+ intel_module = __import__(
173
+ f"cartography.intel.{intel_module_info.name}",
174
+ fromlist=[""],
175
+ )
176
+ for k, v in intel_module.__dict__.items():
177
+ if not callable(v):
178
+ continue
179
+ match_callable_name = callable_regex.match(k)
180
+ if not match_callable_name:
181
+ continue
182
+ callable_module_name = (
183
+ match_callable_name.group(1) if match_callable_name else None
184
+ )
185
+ if callable_module_name != intel_module_info.name:
186
+ logger.debug(
187
+ "Module name '%s' does not match intel module name '%s'.",
188
+ callable_module_name,
189
+ intel_module_info.name,
190
+ )
191
+ available_modules[intel_module_info.name] = v
192
+ available_modules["analysis"] = cartography.intel.analysis.run
193
+ return available_modules
194
+
195
+
196
+ # Used to avoid repeatedly calling Sync.list_intel_modules()
197
+ TOP_LEVEL_MODULES = Sync.list_intel_modules()
198
+
126
199
 
127
200
  def run_with_config(sync: Sync, config: Union[Config, argparse.Namespace]) -> int:
128
201
  """
@@ -201,9 +274,12 @@ def build_default_sync() -> Sync:
201
274
  :return: The default cartography sync object.
202
275
  """
203
276
  sync = Sync()
204
- sync.add_stages([
205
- (stage_name, stage_func) for stage_name, stage_func in TOP_LEVEL_MODULES.items()
206
- ])
277
+ sync.add_stages(
278
+ [
279
+ (stage_name, stage_func)
280
+ for stage_name, stage_func in TOP_LEVEL_MODULES.items()
281
+ ],
282
+ )
207
283
  return sync
208
284
 
209
285
 
@@ -214,18 +290,18 @@ def parse_and_validate_selected_modules(selected_modules: str) -> List[str]:
214
290
  :return: A validated list of module names that we will run
215
291
  """
216
292
  validated_modules: List[str] = []
217
- for module in selected_modules.split(','):
293
+ for module in selected_modules.split(","):
218
294
  module = module.strip()
219
295
 
220
296
  if module in TOP_LEVEL_MODULES.keys():
221
297
  validated_modules.append(module)
222
298
  else:
223
- valid_modules = ', '.join(TOP_LEVEL_MODULES.keys())
299
+ valid_modules = ", ".join(TOP_LEVEL_MODULES.keys())
224
300
  raise ValueError(
225
301
  f'Error parsing `selected_modules`. You specified "{selected_modules}". '
226
- f'Please check that your string is formatted properly. '
302
+ f"Please check that your string is formatted properly. "
227
303
  f'Example valid input looks like "aws,gcp,analysis" or "azure, oci, crowdstrike". '
228
- f'Our full list of valid values is: {valid_modules}.',
304
+ f"Our full list of valid values is: {valid_modules}.",
229
305
  )
230
306
  return validated_modules
231
307