-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathartefact_enumerator.py
692 lines (599 loc) · 24.4 KB
/
artefact_enumerator.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
'''
This module reads all Kubernetes CRDs of type "ScanConfiguration" and starts a
separate thread for each of these configurations. Then it retrieves all artefacts
of the specified components and types and updates their respective compliance
snapshot in the delivery database. If certain changes apply to the compliance
snapshots, a corresponding backlog item is created as a result.
'''
import argparse
import atexit
import collections
import collections.abc
import datetime
import hashlib
import logging
import os
import threading
import ci.log
import cnudie.iter
import cnudie.retrieve
import delivery.client
import dso.model
import ocm
import config
import ctx_util
import k8s.backlog
import k8s.logging
import k8s.model
import k8s.runtime_artefacts
import k8s.util
import lookups
logger = logging.getLogger(__name__)
ci.log.configure_default_logging()
k8s.logging.configure_kubernetes_logging()
own_dir = os.path.abspath(os.path.dirname(__file__))
default_cache_dir = os.path.join(own_dir, '.cache')
def deserialise_scan_configurations(
namespace: str,
kubernetes_api: k8s.util.KubernetesApi,
) -> collections.abc.Generator[tuple[str, config.ScanConfiguration, None, None]]:
scan_cfg_crds = kubernetes_api.custom_kubernetes_api.list_namespaced_custom_object(
group=k8s.model.ScanConfigurationCrd.DOMAIN,
version=k8s.model.ScanConfigurationCrd.VERSION,
plural=k8s.model.ScanConfigurationCrd.PLURAL_NAME,
namespace=namespace,
).get('items')
for scan_cfg_crd in scan_cfg_crds:
if not (spec := scan_cfg_crd.get('spec')):
continue
cfg_name = scan_cfg_crd.get('metadata').get('name')
yield cfg_name, config.deserialise_scan_configuration(
spec_config=spec,
included_services=(
config.Services.ARTEFACT_ENUMERATOR,
config.Services.BDBA,
config.Services.ISSUE_REPLICATOR,
config.Services.CLAMAV,
),
)
def sprint_dates(
delivery_client: delivery.client.DeliveryServiceClient,
date_name: str='release_decision',
) -> tuple[datetime.date]:
sprints = delivery_client.sprints()
sprint_dates = tuple(
sprint.find_sprint_date(name=date_name).value.date()
for sprint in sprints
)
if not sprint_dates:
raise ValueError('no sprints found')
return sprint_dates
def correlation_id(
artefact: dso.model.ComponentArtefactId,
latest_processing_date: datetime.date,
version: str='v1',
) -> str:
'''
the correlation id neither contains the `component_version` nor the
`artefact_version` to group compliance snapshots of the same artefact
across different versions (e.g. necessary for GitHub tracking issues).
Also, a version prefix is added to be able to differentiate correlation
ids in case their calculation changed
'''
digest_str = (
artefact.component_name + artefact.artefact_kind +
artefact.artefact.artefact_name + artefact.artefact.artefact_type +
latest_processing_date.isoformat()
)
digest = hashlib.shake_128(digest_str.encode('utf-8')).hexdigest(
length=23,
)
return f'{version}/{digest}'
def create_compliance_snapshot(
cfg_name: str,
artefact: dso.model.ComponentArtefactId,
latest_processing_date: datetime.date,
now: datetime.datetime=datetime.datetime.now(),
today: datetime.date=datetime.date.today(),
) -> dso.model.ArtefactMetadata:
meta = dso.model.Metadata(
datasource=dso.model.Datasource.ARTEFACT_ENUMERATOR,
type=dso.model.Datatype.COMPLIANCE_SNAPSHOTS,
creation_date=now,
last_update=now,
)
data = dso.model.ComplianceSnapshot(
cfg_name=cfg_name,
latest_processing_date=latest_processing_date,
correlation_id=correlation_id(
artefact=artefact,
latest_processing_date=latest_processing_date,
),
state=[dso.model.ComplianceSnapshotState(
timestamp=now,
status=dso.model.ComplianceSnapshotStatuses.ACTIVE,
)],
)
return dso.model.ArtefactMetadata(
artefact=artefact,
meta=meta,
data=data,
discovery_date=today,
)
def _iter_ocm_artefacts(
components: tuple[config.Component],
artefact_types: tuple[str],
node_filter: collections.abc.Callable[[cnudie.iter.Node], bool],
delivery_client: delivery.client.DeliveryServiceClient,
component_descriptor_lookup: cnudie.retrieve.ComponentDescriptorLookupById,
) -> collections.abc.Generator[dso.model.ComponentArtefactId, None, None]:
for component in components:
versions = delivery_client.greatest_component_versions(
component_name=component.component_name,
max_versions=component.max_versions_limit,
greatest_version=component.version,
ocm_repo=component.ocm_repo,
version_filter=component.version_filter,
)
for version in versions:
component_id = ocm.ComponentIdentity(
name=component.component_name,
version=version,
)
if ocm_repo := component.ocm_repo:
component = component_descriptor_lookup(
component_id,
ocm_repository_lookup=cnudie.retrieve.ocm_repository_lookup(ocm_repo),
).component
else:
component = component_descriptor_lookup(component_id).component
# note: adjust node filter here once other artefacts become processable as well
for artefact_node in cnudie.iter.iter(
component=component,
lookup=component_descriptor_lookup,
node_filter=lambda node: (
cnudie.iter.Filter.resources(node) and
node.artefact.type in artefact_types and
node_filter(node)
),
):
yield dso.model.component_artefact_id_from_ocm(
component=artefact_node.component,
artefact=artefact_node.artefact,
)
def _create_and_update_compliance_snapshots_of_artefact(
cfg_name: str,
artefact: dso.model.ComponentArtefactId,
compliance_snapshots: list[dso.model.ArtefactMetadata],
sprints: tuple[datetime.date],
now: datetime.datetime=datetime.datetime.now(),
today: datetime.date=datetime.date.today(),
) -> tuple[list[dso.model.ArtefactMetadata], bool]:
update_is_required = False
for sprint_date in sprints:
if any(
compliance_snapshot for compliance_snapshot in compliance_snapshots
if compliance_snapshot.data.latest_processing_date == sprint_date
):
# compliance snapshot already exists for this artefact for this sprint
continue
compliance_snapshots.append(create_compliance_snapshot(
cfg_name=cfg_name,
artefact=artefact,
latest_processing_date=sprint_date,
now=now,
today=today,
))
update_is_required = True
if update_is_required:
logger.info(f'created compliance snapshots for {artefact=}')
for compliance_snapshot in compliance_snapshots:
if (
compliance_snapshot.data.current_state().status !=
dso.model.ComplianceSnapshotStatuses.ACTIVE
):
compliance_snapshot.data.state.append(dso.model.ComplianceSnapshotState(
timestamp=now,
status=dso.model.ComplianceSnapshotStatuses.ACTIVE,
))
compliance_snapshot.data.purge_old_states()
update_is_required = True
return compliance_snapshots, update_is_required
def _calculate_backlog_item_priority(
service: config.Services,
compliance_snapshots: list[dso.model.ArtefactMetadata],
interval: int,
status: dso.model.ComplianceSnapshotStatuses | str | int | None=None,
now: datetime.datetime=datetime.datetime.now(),
) -> k8s.backlog.BacklogPriorities:
'''
- interval has passed -> priority LOW
- compliance snapshot was just created -> priority HIGH
- compliance snapshot status has changed -> priority HIGH
'''
priority = k8s.backlog.BacklogPriorities.NONE
for compliance_snapshot in compliance_snapshots:
current_state = compliance_snapshot.data.current_state(
service=service,
)
if not current_state or (status and status != current_state.status):
priority = max(priority, k8s.backlog.BacklogPriorities.HIGH)
# the priority won't change anymore in this loop -> early exit
break
elif now - current_state.timestamp >= datetime.timedelta(
seconds=interval,
):
priority = max(priority, k8s.backlog.BacklogPriorities.LOW)
return priority
def _create_backlog_item(
cfg_name: str,
namespace: str,
kubernetes_api: k8s.util.KubernetesApi,
artefact: dso.model.ComponentArtefactId,
compliance_snapshots: list[dso.model.ArtefactMetadata],
service: config.Services,
interval_seconds: int,
status: dso.model.ComplianceSnapshotStatuses | str | int | None=None,
now: datetime.datetime=datetime.datetime.now(),
) -> tuple[list[dso.model.ArtefactMetadata], bool]:
priority = _calculate_backlog_item_priority(
service=service,
compliance_snapshots=compliance_snapshots,
interval=interval_seconds,
now=now,
status=status,
)
if not priority:
# no need to create a backlog item for this artefact
return compliance_snapshots, False
# there is a need to create a new backlog item, thus update issue replicator state for
# every compliance snapshot of this artefact so that the configured replication interval
# can be acknowledged correctly; otherwise, the replication might happen to often because
# the state of some compliance snapshots for the artefact might not have changed
for compliance_snapshot in compliance_snapshots:
compliance_snapshot.data.state.append(dso.model.ComplianceSnapshotState(
timestamp=now,
status=status,
service=service,
))
compliance_snapshot.data.purge_old_states(
service=service
)
was_created = k8s.backlog.create_unique_backlog_item(
service=service,
cfg_name=cfg_name,
namespace=namespace,
kubernetes_api=kubernetes_api,
artefact=artefact,
priority=priority,
)
if was_created:
logger.info(f'created {service} backlog item with {priority=} for {artefact=}')
return compliance_snapshots, True
def _process_compliance_snapshots_of_artefact(
cfg_name: str,
scan_config: config.ScanConfiguration,
namespace: str,
kubernetes_api: k8s.util.KubernetesApi,
delivery_client: delivery.client.DeliveryServiceClient,
artefact: dso.model.ComponentArtefactId,
compliance_snapshots: list[dso.model.ArtefactMetadata],
sprints: tuple[datetime.date],
types: tuple[dso.model.Datatype],
now: datetime.datetime=datetime.datetime.now(),
today: datetime.date=datetime.date.today(),
):
compliance_snapshots, update_is_required = _create_and_update_compliance_snapshots_of_artefact(
cfg_name=cfg_name,
artefact=artefact,
compliance_snapshots=compliance_snapshots,
sprints=sprints,
now=now,
today=today,
)
if scan_config.bdba_config:
compliance_snapshots, bdba_update_is_required = _create_backlog_item(
cfg_name=cfg_name,
namespace=namespace,
kubernetes_api=kubernetes_api,
artefact=artefact,
compliance_snapshots=compliance_snapshots,
service=config.Services.BDBA,
interval_seconds=scan_config.bdba_config.rescan_interval,
now=now,
)
update_is_required |= bdba_update_is_required
if scan_config.issue_replicator_config:
findings = delivery_client.query_metadata(
artefacts=(artefact,),
type=types,
)
compliance_snapshots, issue_update_is_required = _create_backlog_item(
cfg_name=cfg_name,
namespace=namespace,
kubernetes_api=kubernetes_api,
artefact=artefact,
compliance_snapshots=compliance_snapshots,
service=config.Services.ISSUE_REPLICATOR,
interval_seconds=scan_config.issue_replicator_config.replication_interval,
status=len(findings),
now=now,
)
update_is_required |= issue_update_is_required
if scan_config.clamav_config:
interval = scan_config.clamav_config.rescan_interval
compliance_snapshots, malware_update_is_required = _create_backlog_item(
cfg_name=cfg_name,
namespace=namespace,
kubernetes_api=kubernetes_api,
artefact=artefact,
compliance_snapshots=compliance_snapshots,
service=config.Services.CLAMAV,
interval_seconds=interval,
now=now,
)
update_is_required |= malware_update_is_required
if not update_is_required:
logger.info(
f'{len(compliance_snapshots)} compliance snapshots did not change, '
f'no need to update in delivery-db ({artefact=})'
)
return
delivery_client.update_metadata(data=compliance_snapshots)
logger.info(
f'updated {len(compliance_snapshots)} compliance snapshots in delivery-db ({artefact=})'
)
def _process_inactive_compliance_snapshots(
cfg_name: str,
scan_config: config.ScanConfiguration,
namespace: str,
kubernetes_api: k8s.util.KubernetesApi,
delivery_client: delivery.client.DeliveryServiceClient,
compliance_snapshots: list[dso.model.ArtefactMetadata],
now: datetime.datetime=datetime.datetime.now(),
):
cs_by_artefact = collections.defaultdict(list)
for compliance_snapshot in compliance_snapshots:
cs_by_artefact[compliance_snapshot.artefact].append(compliance_snapshot)
for compliance_snapshots in cs_by_artefact.values():
artefact = compliance_snapshots[0].artefact
update_is_required = False
deletable_compliance_snapshots: list[dso.model.ArtefactMetadata] = []
for compliance_snapshot in compliance_snapshots:
current_general_state = compliance_snapshot.data.current_state()
if current_general_state.status != dso.model.ComplianceSnapshotStatuses.INACTIVE:
compliance_snapshot.data.state.append(dso.model.ComplianceSnapshotState(
timestamp=now,
status=dso.model.ComplianceSnapshotStatuses.INACTIVE,
))
compliance_snapshot.data.purge_old_states()
current_general_state = compliance_snapshot.data.current_state()
update_is_required = True
if scan_config.issue_replicator_config:
compliance_snapshot.data.state.append(dso.model.ComplianceSnapshotState(
timestamp=now,
status=0,
service=config.Services.ISSUE_REPLICATOR,
))
compliance_snapshot.data.purge_old_states(
service=config.Services.ISSUE_REPLICATOR,
)
if now - current_general_state.timestamp >= datetime.timedelta(
seconds=scan_config.artefact_enumerator_config.compliance_snapshot_grace_period,
):
deletable_compliance_snapshots.append(compliance_snapshot)
if update_is_required:
delivery_client.update_metadata(data=compliance_snapshots)
logger.info(
f'updated {len(compliance_snapshots)} inactive compliance snapshots in delivery-db '
f'({artefact=})'
)
if scan_config.issue_replicator_config:
priority = k8s.backlog.BacklogPriorities.HIGH
was_created = k8s.backlog.create_unique_backlog_item(
service=config.Services.ISSUE_REPLICATOR,
cfg_name=cfg_name,
namespace=namespace,
kubernetes_api=kubernetes_api,
artefact=artefact,
priority=priority,
)
if was_created:
logger.info(
f'created issue replicator backlog item with {priority=} for inactive '
f'{artefact=}'
)
if deletable_compliance_snapshots:
delivery_client.delete_metadata(data=deletable_compliance_snapshots)
logger.info(
f'deleted {len(deletable_compliance_snapshots)} inactive compliance snapshots in '
f'delivery-db ({artefact=})'
)
def enumerate_artefacts(
cfg_name: str,
scan_config: config.ScanConfiguration,
namespace: str,
kubernetes_api: k8s.util.KubernetesApi,
delivery_client: delivery.client.DeliveryServiceClient,
component_descriptor_lookup: cnudie.retrieve.ComponentDescriptorLookupById,
types: tuple[dso.model.Datatype],
):
'''
retrieves first of all the unique artefacts referenced by the configured components and the
available runtime artefacts from the respective custom resources as well as all compliance
snapshots belonging to the given `cfg_name`. These compliance snapshots are differentiated
between "active" (still referenced by one of the before retrieved artefacts) and "inactive"
(not referenced anymore). While iterating the artefacts, the active compliance snapshots are
being created/updated (status change) and based on this, it is evaluated if a new backlog item
must be created (and if yes, it will be created). The inactive compliance snapshots are also
being updated (status change) and if the configured grace period has passed, they are deleted
from the delivery-db. Also, for each artefact becoming inactive, a backlog item for the issue
replicator must be created.
'''
# store current date + time to ensure they are consistent for whole enumeration
now = datetime.datetime.now()
today = datetime.date.today()
time_range = scan_config.artefact_enumerator_config.sprints_time_range
logger.info(f'{time_range=}')
sprints = tuple(
date for date in sprint_dates(delivery_client=delivery_client)
if not time_range or (date >= time_range.start_date and date <= time_range.end_date)
)
logger.info(f'{len(sprints)=}')
ocm_artefacts = set(_iter_ocm_artefacts(
components=scan_config.artefact_enumerator_config.components,
artefact_types=scan_config.artefact_enumerator_config.artefact_types,
node_filter=scan_config.artefact_enumerator_config.node_filter,
delivery_client=delivery_client,
component_descriptor_lookup=component_descriptor_lookup,
))
logger.info(f'{len(ocm_artefacts)=}')
runtime_artefacts = {
runtime_artefact.artefact
for runtime_artefact in k8s.runtime_artefacts.iter_runtime_artefacts(
namespace=namespace,
kubernetes_api=kubernetes_api,
)
}
logger.info(f'{len(runtime_artefacts)=}')
artefacts = ocm_artefacts | runtime_artefacts
logger.info(f'{len(artefacts)=}')
compliance_snapshots = delivery_client.query_metadata(
type=dso.model.Datatype.COMPLIANCE_SNAPSHOTS,
)
compliance_snapshots = [
compliance_snapshot for compliance_snapshot in compliance_snapshots
if compliance_snapshot.data.cfg_name == cfg_name # TODO mv to delivery service
]
logger.info(f'{len(compliance_snapshots)=}')
active_compliance_snapshots = tuple(
compliance_snapshot for compliance_snapshot in compliance_snapshots
if compliance_snapshot.artefact in artefacts
)
logger.info(f'{len(active_compliance_snapshots)=}')
inactive_compliance_snapshots = tuple(
compliance_snapshot for compliance_snapshot in compliance_snapshots
if compliance_snapshot.artefact not in artefacts
)
logger.info(f'{len(inactive_compliance_snapshots)=}')
for artefact in artefacts:
compliance_snapshots = [
compliance_snapshot for compliance_snapshot in active_compliance_snapshots
if compliance_snapshot.artefact == artefact
]
_process_compliance_snapshots_of_artefact(
cfg_name=cfg_name,
scan_config=scan_config,
namespace=namespace,
kubernetes_api=kubernetes_api,
delivery_client=delivery_client,
artefact=artefact,
compliance_snapshots=compliance_snapshots,
sprints=sprints,
types=types,
now=now,
today=today,
)
_process_inactive_compliance_snapshots(
cfg_name=cfg_name,
scan_config=scan_config,
namespace=namespace,
kubernetes_api=kubernetes_api,
delivery_client=delivery_client,
compliance_snapshots=inactive_compliance_snapshots,
now=now,
)
def parse_args():
parser = argparse.ArgumentParser()
parser.add_argument(
'--k8s-cfg-name',
help='specify kubernetes cluster to interact with',
default=os.environ.get('K8S_CFG_NAME'),
)
parser.add_argument(
'--kubeconfig',
help='''
specify kubernetes cluster to interact with extensions (and logs); if both
`k8s-cfg-name` and `kubeconfig` are set, `k8s-cfg-name` takes precedence
''',
)
parser.add_argument(
'--k8s-namespace',
help='specify kubernetes cluster namespace to interact with',
default=os.environ.get('K8S_TARGET_NAMESPACE'),
)
parser.add_argument(
'--delivery-service-url',
help='''
specify the url of the delivery service to use instead of the one configured in the
respective scan configuration
''',
)
parser.add_argument('--cache-dir', default=default_cache_dir)
parsed_arguments = parser.parse_args()
if not parsed_arguments.k8s_namespace:
raise ValueError(
'k8s namespace must be set, either via argument "k8s-namespace" '
'or via environment variable "K8S_TARGET_NAMESPACE"'
)
return parsed_arguments
def main():
parsed_arguments = parse_args()
namespace = parsed_arguments.k8s_namespace
cfg_factory = ctx_util.cfg_factory()
if parsed_arguments.k8s_cfg_name:
kubernetes_cfg = cfg_factory.kubernetes(parsed_arguments.k8s_cfg_name)
kubernetes_api = k8s.util.kubernetes_api(kubernetes_cfg=kubernetes_cfg)
else:
kubernetes_api = k8s.util.kubernetes_api(
kubeconfig_path=parsed_arguments.kubeconfig,
)
k8s.logging.init_logging_thread(
service=config.Services.ARTEFACT_ENUMERATOR,
namespace=namespace,
kubernetes_api=kubernetes_api,
)
atexit.register(
k8s.logging.log_to_crd,
service=config.Services.ARTEFACT_ENUMERATOR,
namespace=namespace,
kubernetes_api=kubernetes_api,
)
scan_configs_for_cfg_name = deserialise_scan_configurations(
namespace=namespace,
kubernetes_api=kubernetes_api,
)
types = (
dso.model.Datatype.VULNERABILITY,
dso.model.Datatype.LICENSE,
dso.model.Datatype.MALWARE_FINDING,
)
for cfg_name, scan_config in scan_configs_for_cfg_name:
if not (delivery_service_url := parsed_arguments.delivery_service_url):
delivery_service_url = scan_config.artefact_enumerator_config.delivery_service_url
delivery_client = delivery.client.DeliveryServiceClient(
routes=delivery.client.DeliveryServiceRoutes(
base_url=delivery_service_url,
),
auth_token_lookup=lookups.github_auth_token_lookup,
)
component_descriptor_lookup = lookups.init_component_descriptor_lookup(
cache_dir=parsed_arguments.cache_dir,
delivery_client=delivery_client,
)
thread = threading.Thread(
target=enumerate_artefacts,
args=(
cfg_name,
scan_config,
namespace,
kubernetes_api,
delivery_client,
component_descriptor_lookup,
types,
),
)
thread.name = cfg_name
thread.start()
if __name__ == '__main__':
main()