dissect.target 3.19.dev58__py3-none-any.whl → 3.20__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- dissect/target/container.py +1 -1
- dissect/target/exceptions.py +6 -5
- dissect/target/filesystem.py +2 -2
- dissect/target/filesystems/btrfs.py +14 -5
- dissect/target/filesystems/config.py +5 -1
- dissect/target/filesystems/extfs.py +5 -4
- dissect/target/filesystems/fat.py +22 -16
- dissect/target/filesystems/ffs.py +11 -4
- dissect/target/filesystems/jffs.py +12 -7
- dissect/target/filesystems/ntfs.py +22 -6
- dissect/target/filesystems/overlay.py +14 -4
- dissect/target/filesystems/smb.py +3 -3
- dissect/target/filesystems/squashfs.py +4 -4
- dissect/target/filesystems/vmfs.py +4 -4
- dissect/target/filesystems/xfs.py +15 -8
- dissect/target/helpers/compat/path_common.py +5 -5
- dissect/target/helpers/configutil.py +128 -32
- dissect/target/helpers/cyber.py +2 -0
- dissect/target/helpers/data/windowsZones.xml +19 -23
- dissect/target/helpers/docs.py +1 -1
- dissect/target/helpers/keychain.py +2 -0
- dissect/target/helpers/mount.py +2 -1
- dissect/target/helpers/record.py +29 -2
- dissect/target/helpers/record_modifier.py +5 -1
- dissect/target/helpers/regutil.py +56 -26
- dissect/target/loader.py +1 -1
- dissect/target/loaders/mqtt.py +104 -9
- dissect/target/loaders/proxmox.py +68 -0
- dissect/target/loaders/vma.py +1 -1
- dissect/target/loaders/xva.py +1 -1
- dissect/target/plugin.py +24 -21
- dissect/target/plugins/apps/av/mcafee.py +2 -0
- dissect/target/plugins/apps/av/sophos.py +2 -0
- dissect/target/plugins/apps/av/trendmicro.py +2 -0
- dissect/target/plugins/apps/browser/chromium.py +27 -6
- dissect/target/plugins/apps/container/docker.py +48 -32
- dissect/target/plugins/apps/editor/__init__.py +0 -0
- dissect/target/plugins/apps/editor/editor.py +23 -0
- dissect/target/plugins/apps/{texteditor → editor}/windowsnotepad.py +40 -31
- dissect/target/plugins/apps/other/__init__.py +0 -0
- dissect/target/plugins/apps/other/env.py +56 -0
- dissect/target/plugins/apps/shell/powershell.py +6 -2
- dissect/target/plugins/apps/shell/wget.py +91 -0
- dissect/target/plugins/apps/ssh/openssh.py +2 -0
- dissect/target/plugins/apps/ssh/opensshd.py +2 -0
- dissect/target/plugins/apps/virtualization/__init__.py +0 -0
- dissect/target/plugins/apps/virtualization/vmware_workstation.py +61 -0
- dissect/target/plugins/apps/vpn/wireguard.py +9 -9
- dissect/target/plugins/apps/webhosting/cpanel.py +2 -0
- dissect/target/plugins/apps/webserver/caddy.py +2 -0
- dissect/target/plugins/apps/webserver/nginx.py +2 -0
- dissect/target/plugins/child/esxi.py +3 -1
- dissect/target/plugins/child/parallels.py +68 -0
- dissect/target/plugins/child/proxmox.py +23 -0
- dissect/target/plugins/child/virtuozzo.py +12 -8
- dissect/target/plugins/child/vmware_workstation.py +23 -8
- dissect/target/plugins/filesystem/acquire_hash.py +2 -1
- dissect/target/plugins/filesystem/icat.py +15 -11
- dissect/target/plugins/filesystem/ntfs/mft.py +10 -6
- dissect/target/plugins/filesystem/ntfs/mft_timeline.py +3 -1
- dissect/target/plugins/filesystem/ntfs/usnjrnl.py +2 -0
- dissect/target/plugins/filesystem/ntfs/utils.py +3 -1
- dissect/target/plugins/filesystem/unix/suid.py +4 -1
- dissect/target/plugins/filesystem/walkfs.py +2 -0
- dissect/target/plugins/general/example.py +2 -2
- dissect/target/plugins/general/loaders.py +18 -5
- dissect/target/plugins/general/network.py +20 -5
- dissect/target/plugins/general/osinfo.py +1 -0
- dissect/target/plugins/general/plugins.py +53 -10
- dissect/target/plugins/os/unix/_os.py +70 -44
- dissect/target/plugins/os/unix/applications.py +78 -0
- dissect/target/plugins/os/unix/bsd/citrix/history.py +2 -0
- dissect/target/plugins/os/unix/bsd/osx/_os.py +4 -21
- dissect/target/plugins/os/unix/bsd/osx/network.py +92 -0
- dissect/target/plugins/os/unix/bsd/osx/user.py +4 -0
- dissect/target/plugins/os/unix/cronjobs.py +8 -4
- dissect/target/plugins/os/unix/etc/etc.py +4 -0
- dissect/target/plugins/os/unix/generic.py +2 -0
- dissect/target/plugins/os/unix/history.py +27 -25
- dissect/target/plugins/os/unix/linux/_os.py +8 -10
- dissect/target/plugins/os/unix/linux/cmdline.py +2 -0
- dissect/target/plugins/os/unix/linux/debian/apt.py +4 -1
- dissect/target/plugins/os/unix/linux/debian/dpkg.py +3 -3
- dissect/target/plugins/os/unix/linux/debian/proxmox/__init__.py +0 -0
- dissect/target/plugins/os/unix/linux/debian/proxmox/_os.py +141 -0
- dissect/target/plugins/os/unix/linux/debian/proxmox/vm.py +29 -0
- dissect/target/plugins/os/unix/linux/debian/snap.py +79 -0
- dissect/target/plugins/os/unix/linux/environ.py +2 -0
- dissect/target/plugins/os/unix/linux/fortios/_os.py +74 -63
- dissect/target/plugins/os/unix/linux/fortios/generic.py +2 -0
- dissect/target/plugins/os/unix/linux/fortios/locale.py +2 -0
- dissect/target/plugins/os/unix/linux/modules.py +2 -0
- dissect/target/plugins/os/unix/linux/netstat.py +2 -0
- dissect/target/{helpers → plugins/os/unix/linux}/network_managers.py +11 -9
- dissect/target/plugins/os/unix/linux/processes.py +2 -0
- dissect/target/plugins/os/unix/linux/redhat/yum.py +4 -1
- dissect/target/plugins/os/unix/linux/services.py +5 -3
- dissect/target/plugins/os/unix/linux/sockets.py +2 -0
- dissect/target/plugins/os/unix/linux/suse/zypper.py +4 -1
- dissect/target/plugins/os/unix/locale.py +2 -0
- dissect/target/plugins/os/unix/locate/gnulocate.py +4 -2
- dissect/target/plugins/os/unix/locate/mlocate.py +2 -0
- dissect/target/plugins/os/unix/locate/plocate.py +3 -1
- dissect/target/plugins/os/unix/log/atop.py +2 -0
- dissect/target/plugins/os/unix/log/audit.py +3 -1
- dissect/target/plugins/os/unix/log/auth.py +351 -38
- dissect/target/plugins/os/unix/log/journal.py +123 -101
- dissect/target/plugins/os/unix/log/lastlog.py +5 -3
- dissect/target/plugins/os/unix/log/messages.py +51 -27
- dissect/target/plugins/os/unix/log/utmp.py +52 -71
- dissect/target/plugins/os/unix/packagemanager.py +5 -38
- dissect/target/plugins/os/unix/shadow.py +3 -1
- dissect/target/plugins/os/unix/trash.py +132 -0
- dissect/target/plugins/os/windows/_os.py +22 -41
- dissect/target/plugins/os/windows/activitiescache.py +9 -4
- dissect/target/plugins/os/windows/adpolicy.py +2 -1
- dissect/target/plugins/os/windows/amcache.py +16 -13
- dissect/target/plugins/os/windows/defender.py +4 -3
- dissect/target/plugins/os/windows/dpapi/keyprovider/credhist.py +3 -0
- dissect/target/plugins/os/windows/dpapi/keyprovider/empty.py +3 -0
- dissect/target/plugins/os/windows/dpapi/keyprovider/keychain.py +3 -0
- dissect/target/plugins/os/windows/dpapi/keyprovider/lsa.py +3 -0
- dissect/target/plugins/os/windows/env.py +1 -2
- dissect/target/plugins/os/windows/exchange/exchange.py +6 -4
- dissect/target/plugins/os/windows/generic.py +68 -19
- dissect/target/plugins/os/windows/lnk.py +2 -0
- dissect/target/plugins/os/windows/locale.py +9 -3
- dissect/target/plugins/os/windows/log/etl.py +5 -4
- dissect/target/plugins/os/windows/log/evt.py +12 -8
- dissect/target/plugins/os/windows/log/evtx.py +9 -7
- dissect/target/plugins/os/windows/log/mssql.py +103 -0
- dissect/target/plugins/os/windows/log/pfro.py +2 -1
- dissect/target/plugins/os/windows/network.py +380 -0
- dissect/target/plugins/os/windows/notifications.py +6 -4
- dissect/target/plugins/os/windows/prefetch.py +7 -2
- dissect/target/plugins/os/windows/regf/7zip.py +9 -1
- dissect/target/plugins/os/windows/regf/applications.py +62 -0
- dissect/target/plugins/os/windows/regf/auditpol.py +2 -1
- dissect/target/plugins/os/windows/regf/bam.py +3 -1
- dissect/target/plugins/os/windows/regf/cit.py +14 -12
- dissect/target/plugins/os/windows/regf/clsid.py +6 -3
- dissect/target/plugins/os/windows/regf/firewall.py +2 -1
- dissect/target/plugins/os/windows/regf/mru.py +9 -8
- dissect/target/plugins/os/windows/regf/nethist.py +6 -3
- dissect/target/plugins/os/windows/regf/recentfilecache.py +3 -1
- dissect/target/plugins/os/windows/regf/regf.py +5 -1
- dissect/target/plugins/os/windows/regf/shellbags.py +351 -345
- dissect/target/plugins/os/windows/regf/shimcache.py +1 -1
- dissect/target/plugins/os/windows/regf/usb.py +2 -1
- dissect/target/plugins/os/windows/regf/userassist.py +2 -1
- dissect/target/plugins/os/windows/registry.py +11 -0
- dissect/target/plugins/os/windows/services.py +3 -2
- dissect/target/plugins/os/windows/startupinfo.py +7 -2
- dissect/target/plugins/os/windows/syscache.py +5 -2
- dissect/target/plugins/os/windows/tasks.py +1 -1
- dissect/target/plugins/os/windows/thumbcache.py +11 -5
- dissect/target/plugins/os/windows/ual.py +12 -9
- dissect/target/plugins/os/windows/wer.py +21 -6
- dissect/target/plugins/os/windows/wua_history.py +0 -1
- dissect/target/target.py +13 -8
- dissect/target/tools/dump/utils.py +4 -0
- dissect/target/tools/fsutils.py +1 -1
- dissect/target/tools/info.py +1 -1
- dissect/target/tools/mount.py +15 -5
- dissect/target/tools/query.py +15 -9
- dissect/target/tools/shell.py +98 -9
- dissect/target/tools/utils.py +7 -7
- dissect/target/volume.py +4 -4
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/METADATA +6 -2
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/RECORD +176 -160
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/WHEEL +1 -1
- dissect/target/helpers/targetd.py +0 -58
- dissect/target/loaders/targetd.py +0 -223
- dissect/target/plugins/apps/texteditor/texteditor.py +0 -13
- dissect/target/plugins/os/unix/etc.py +0 -9
- /dissect/target/plugins/apps/{texteditor → database}/__init__.py +0 -0
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/COPYRIGHT +0 -0
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/LICENSE +0 -0
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/entry_points.txt +0 -0
- {dissect.target-3.19.dev58.dist-info → dissect.target-3.20.dist-info}/top_level.txt +0 -0
@@ -2,6 +2,7 @@ from __future__ import annotations
|
|
2
2
|
|
3
3
|
import io
|
4
4
|
import json
|
5
|
+
import logging
|
5
6
|
import re
|
6
7
|
import sys
|
7
8
|
from collections import deque
|
@@ -47,6 +48,9 @@ except ImportError:
|
|
47
48
|
HAS_TOML = False
|
48
49
|
|
49
50
|
|
51
|
+
log = logging.getLogger(__name__)
|
52
|
+
|
53
|
+
|
50
54
|
def _update_dictionary(current: dict[str, Any], key: str, value: Any) -> None:
|
51
55
|
if prev_value := current.get(key):
|
52
56
|
if isinstance(prev_value, dict):
|
@@ -65,9 +69,7 @@ def _update_dictionary(current: dict[str, Any], key: str, value: Any) -> None:
|
|
65
69
|
|
66
70
|
|
67
71
|
class PeekableIterator:
|
68
|
-
|
69
|
-
https://more-itertools.readthedocs.io/en/stable/_modules/more_itertools/more.html#peekable
|
70
|
-
"""
|
72
|
+
# https://more-itertools.readthedocs.io/en/stable/_modules/more_itertools/more.html#peekable
|
71
73
|
|
72
74
|
def __init__(self, iterable):
|
73
75
|
self._iterator = iter(iterable)
|
@@ -94,9 +96,6 @@ class PeekableIterator:
|
|
94
96
|
class ConfigurationParser:
|
95
97
|
"""A configuration parser where you can configure certain aspects of the parsing mechanism.
|
96
98
|
|
97
|
-
Attributes:
|
98
|
-
parsed_data: The resulting dictionary after parsing.
|
99
|
-
|
100
99
|
Args:
|
101
100
|
collapse: A ``bool`` or an ``Iterator``:
|
102
101
|
If ``True``: it will collapse all the resulting dictionary values.
|
@@ -161,7 +160,7 @@ class ConfigurationParser:
|
|
161
160
|
def get(self, item: str, default: Optional[Any] = None) -> Any:
|
162
161
|
return self.parsed_data.get(item, default)
|
163
162
|
|
164
|
-
def read_file(self, fh: TextIO) -> None:
|
163
|
+
def read_file(self, fh: TextIO | io.BytesIO) -> None:
|
165
164
|
"""Parse a configuration file.
|
166
165
|
|
167
166
|
Raises:
|
@@ -191,6 +190,8 @@ class Default(ConfigurationParser):
|
|
191
190
|
|
192
191
|
This parser splits only on the first ``separator`` it finds:
|
193
192
|
|
193
|
+
.. code-block::
|
194
|
+
|
194
195
|
key<separator>value -> {"key": "value"}
|
195
196
|
|
196
197
|
key<separator>value\n
|
@@ -303,8 +304,16 @@ class Txt(ConfigurationParser):
|
|
303
304
|
self.parsed_data = {"content": fh.read(), "size": str(fh.tell())}
|
304
305
|
|
305
306
|
|
307
|
+
class Bin(ConfigurationParser):
|
308
|
+
|
309
|
+
"""Read the file into ``binary`` and show the number of bytes read"""
|
310
|
+
|
311
|
+
def parse_file(self, fh: io.BytesIO) -> None:
|
312
|
+
self.parsed_data = {"binary": fh.read(), "size": str(fh.tell())}
|
313
|
+
|
314
|
+
|
306
315
|
class Xml(ConfigurationParser):
|
307
|
-
"""Parses an XML file. Ignores any constructor parameters passed from ``ConfigurationParser
|
316
|
+
"""Parses an XML file. Ignores any constructor parameters passed from ``ConfigurationParser``."""
|
308
317
|
|
309
318
|
def _tree(self, tree: ElementTree, root: bool = False) -> dict:
|
310
319
|
"""Very simple but robust xml -> dict implementation, see comments."""
|
@@ -383,8 +392,9 @@ class ListUnwrapper:
|
|
383
392
|
def unwrap(data: Union[dict, list]) -> Union[dict, list]:
|
384
393
|
"""Transforms a list with dictionaries to a dictionary.
|
385
394
|
|
386
|
-
The order of the list is preserved. If no dictionary is found,
|
387
|
-
|
395
|
+
The order of the list is preserved. If no dictionary is found, the list remains untouched:
|
396
|
+
|
397
|
+
.. code-block::
|
388
398
|
|
389
399
|
["value1", "value2"] -> ["value1", "value2"]
|
390
400
|
|
@@ -457,6 +467,76 @@ class Toml(ConfigurationParser):
|
|
457
467
|
raise ConfigurationParsingError("Failed to parse file, please install tomli.")
|
458
468
|
|
459
469
|
|
470
|
+
class Env(ConfigurationParser):
|
471
|
+
"""Parses ``.env`` file contents according to Docker and bash specification.
|
472
|
+
|
473
|
+
Does not apply interpolation of substituted values, eg. ``foo=${bar}`` and does not attempt
|
474
|
+
to parse list or dict strings. Does not support dynamic env files, eg. `` foo=`bar` ``. Also
|
475
|
+
does not support multi-line key/value assignments (yet).
|
476
|
+
|
477
|
+
Resources:
|
478
|
+
- https://docs.docker.com/compose/environment-variables/variable-interpolation/#env-file-syntax
|
479
|
+
- https://github.com/theskumar/python-dotenv/blob/main/src/dotenv/parser.py
|
480
|
+
"""
|
481
|
+
|
482
|
+
RE_KV = re.compile(r"^(?P<key>.+?)=(?P<value>(\".+?\")|(\'.+?\')|(.*?))?(?P<comment> \#.+?)?$")
|
483
|
+
|
484
|
+
def __init__(self, comments: bool = True, *args, **kwargs) -> None:
|
485
|
+
super().__init__(*args, **kwargs)
|
486
|
+
self.comments = comments
|
487
|
+
self.parsed_data: dict | tuple[dict, str | None] = {}
|
488
|
+
|
489
|
+
def parse_file(self, fh: TextIO) -> None:
|
490
|
+
for line in fh.readlines():
|
491
|
+
# Blank lines are ignored.
|
492
|
+
# Lines beginning with ``#`` are processed as comments and ignored.
|
493
|
+
if not line or line[0] == "#" or "=" not in line:
|
494
|
+
continue
|
495
|
+
|
496
|
+
# Each line represents a key-value pair. Values can optionally be quoted.
|
497
|
+
# Inline comments for unquoted values must be preceded with a space.
|
498
|
+
# Value may be empty.
|
499
|
+
match = self.RE_KV.match(line)
|
500
|
+
|
501
|
+
# Line could be invalid
|
502
|
+
if not match:
|
503
|
+
log.warning("Could not parse line in %s: '%s'", fh, line)
|
504
|
+
continue
|
505
|
+
|
506
|
+
key = match.groupdict()["key"]
|
507
|
+
value = match.groupdict().get("value") or ""
|
508
|
+
value = value.strip()
|
509
|
+
comment = match.groupdict().get("comment")
|
510
|
+
comment = comment.replace(" # ", "", 1) if comment else None
|
511
|
+
|
512
|
+
# Surrounding whitespace characters are removed, unless quoted.
|
513
|
+
if value and ((value[0] == '"' and value[-1] == '"') or (value[0] == "'" and value[-1] == "'")):
|
514
|
+
is_quoted = True
|
515
|
+
value = value.strip("\"'")
|
516
|
+
else:
|
517
|
+
is_quoted = False
|
518
|
+
value = value.strip()
|
519
|
+
|
520
|
+
# Unquoted values may start with a quote if they are properly escaped.
|
521
|
+
if not is_quoted and value[:2] in ["\\'", '\\"']:
|
522
|
+
value = value[1:]
|
523
|
+
|
524
|
+
# Interpret boolean values
|
525
|
+
if value.lower() in ["1", "true"]:
|
526
|
+
value = True
|
527
|
+
elif value.lower() in ["0", "false"]:
|
528
|
+
value = False
|
529
|
+
|
530
|
+
# Interpret integer values
|
531
|
+
if isinstance(value, str) and re.match(r"^[0-9]{1,}$", value):
|
532
|
+
value = int(value)
|
533
|
+
|
534
|
+
if key.strip() in self.parsed_data:
|
535
|
+
log.warning("Duplicate environment key '%s' in file %s", key.strip(), fh)
|
536
|
+
|
537
|
+
self.parsed_data[key.strip()] = (value, comment) if self.comments else value
|
538
|
+
|
539
|
+
|
460
540
|
class ScopeManager:
|
461
541
|
"""A (context)manager for dictionary scoping.
|
462
542
|
|
@@ -540,6 +620,8 @@ class Indentation(Default):
|
|
540
620
|
|
541
621
|
The parser parses this as the following:
|
542
622
|
|
623
|
+
.. code-block::
|
624
|
+
|
543
625
|
key value
|
544
626
|
key2 value2
|
545
627
|
-> {"key value": {"key2": "value2"}}
|
@@ -562,7 +644,7 @@ class Indentation(Default):
|
|
562
644
|
Args:
|
563
645
|
manager: A :class:`ScopeManager` that contains the logic to ``push`` and ``pop`` scopes. And keeps state.
|
564
646
|
line: The line to be parsed.
|
565
|
-
key: The key that should be updated during a :method:`ScopeManager.push
|
647
|
+
key: The key that should be updated during a :method:`ScopeManager.push`.
|
566
648
|
next_line: The next line to be parsed.
|
567
649
|
|
568
650
|
Returns:
|
@@ -612,26 +694,28 @@ class SystemD(Indentation):
|
|
612
694
|
"""A :class:`ConfigurationParser` that specifically parses systemd configuration files.
|
613
695
|
|
614
696
|
Examples:
|
615
|
-
>>> systemd_data = textwrap.dedent(
|
616
|
-
'''
|
617
|
-
[Section1]
|
618
|
-
Key=Value
|
619
|
-
[Section2]
|
620
|
-
Key2=Value 2\\
|
621
|
-
Value 2 continued
|
622
|
-
'''
|
623
|
-
)
|
624
|
-
>>> parser = SystemD(io.StringIO(systemd_data))
|
625
|
-
>>> parser.parser_items
|
626
|
-
{
|
627
|
-
"Section1": {
|
628
|
-
"Key": "Value
|
629
|
-
},
|
630
|
-
"Section2": {
|
631
|
-
"Key2": "Value2 Value 2 continued
|
632
|
-
}
|
633
|
-
}
|
634
697
|
|
698
|
+
.. code-block::
|
699
|
+
|
700
|
+
>>> systemd_data = textwrap.dedent(
|
701
|
+
'''
|
702
|
+
[Section1]
|
703
|
+
Key=Value
|
704
|
+
[Section2]
|
705
|
+
Key2=Value 2\\
|
706
|
+
Value 2 continued
|
707
|
+
'''
|
708
|
+
)
|
709
|
+
>>> parser = SystemD(io.StringIO(systemd_data))
|
710
|
+
>>> parser.parser_items
|
711
|
+
{
|
712
|
+
"Section1": {
|
713
|
+
"Key": "Value
|
714
|
+
},
|
715
|
+
"Section2": {
|
716
|
+
"Key2": "Value2 Value 2 continued
|
717
|
+
}
|
718
|
+
}
|
635
719
|
"""
|
636
720
|
|
637
721
|
def _change_scope(
|
@@ -733,6 +817,8 @@ MATCH_MAP: dict[str, ParserConfig] = {
|
|
733
817
|
"*/sysconfig/network-scripts/ifcfg-*": ParserConfig(Default),
|
734
818
|
"*/sysctl.d/*.conf": ParserConfig(Default),
|
735
819
|
"*/xml/*": ParserConfig(Xml),
|
820
|
+
"*.bashrc": ParserConfig(Txt),
|
821
|
+
"*/vim/vimrc*": ParserConfig(Txt),
|
736
822
|
}
|
737
823
|
|
738
824
|
CONFIG_MAP: dict[tuple[str, ...], ParserConfig] = {
|
@@ -744,6 +830,13 @@ CONFIG_MAP: dict[tuple[str, ...], ParserConfig] = {
|
|
744
830
|
"cnf": ParserConfig(Default),
|
745
831
|
"conf": ParserConfig(Default, separator=(r"\s",)),
|
746
832
|
"sample": ParserConfig(Txt),
|
833
|
+
"sh": ParserConfig(Txt),
|
834
|
+
"key": ParserConfig(Txt),
|
835
|
+
"crt": ParserConfig(Txt),
|
836
|
+
"pem": ParserConfig(Txt),
|
837
|
+
"pl": ParserConfig(Txt), # various admin panels
|
838
|
+
"lua": ParserConfig(Txt), # wireshark etc.
|
839
|
+
"txt": ParserConfig(Txt),
|
747
840
|
"systemd": ParserConfig(SystemD),
|
748
841
|
"template": ParserConfig(Txt),
|
749
842
|
"toml": ParserConfig(Toml),
|
@@ -759,6 +852,7 @@ KNOWN_FILES: dict[str, type[ConfigurationParser]] = {
|
|
759
852
|
"nsswitch.conf": ParserConfig(Default, separator=(":",)),
|
760
853
|
"lsb-release": ParserConfig(Default),
|
761
854
|
"catalog": ParserConfig(Xml),
|
855
|
+
"ld.so.cache": ParserConfig(Bin),
|
762
856
|
"fstab": ParserConfig(
|
763
857
|
CSVish,
|
764
858
|
separator=(r"\s",),
|
@@ -832,9 +926,11 @@ def parse_config(
|
|
832
926
|
parser_type = _select_parser(entry, hint)
|
833
927
|
|
834
928
|
parser = parser_type.create_parser(options)
|
835
|
-
|
836
929
|
with entry.open() as fh:
|
837
|
-
|
930
|
+
if not isinstance(parser, Bin):
|
931
|
+
open_file = io.TextIOWrapper(fh, encoding="utf-8")
|
932
|
+
else:
|
933
|
+
open_file = io.BytesIO(fh.read())
|
838
934
|
parser.read_file(open_file)
|
839
935
|
|
840
936
|
return parser
|
dissect/target/helpers/cyber.py
CHANGED
@@ -5,8 +5,9 @@ Copyright © 1991-2013 Unicode, Inc.
|
|
5
5
|
CLDR data files are interpreted according to the LDML specification (http://unicode.org/reports/tr35/)
|
6
6
|
For terms of use, see http://www.unicode.org/copyright.html
|
7
7
|
|
8
|
-
NOTE:
|
9
|
-
|
8
|
+
NOTE: This file should be updated every ~6 months.
|
9
|
+
Updated at: 2024-10-28
|
10
|
+
Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/windowsZones.xml
|
10
11
|
-->
|
11
12
|
|
12
13
|
<supplementalData>
|
@@ -33,7 +34,6 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
33
34
|
<mapZone other="Hawaiian Standard Time" territory="001" type="Pacific/Honolulu"/>
|
34
35
|
<mapZone other="Hawaiian Standard Time" territory="CK" type="Pacific/Rarotonga"/>
|
35
36
|
<mapZone other="Hawaiian Standard Time" territory="PF" type="Pacific/Tahiti"/>
|
36
|
-
<mapZone other="Hawaiian Standard Time" territory="UM" type="Pacific/Johnston"/>
|
37
37
|
<mapZone other="Hawaiian Standard Time" territory="US" type="Pacific/Honolulu"/>
|
38
38
|
<mapZone other="Hawaiian Standard Time" territory="ZZ" type="Etc/GMT+10"/>
|
39
39
|
|
@@ -52,7 +52,7 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
52
52
|
|
53
53
|
<!-- (UTC-08:00) Baja California -->
|
54
54
|
<mapZone other="Pacific Standard Time (Mexico)" territory="001" type="America/Tijuana"/>
|
55
|
-
<mapZone other="Pacific Standard Time (Mexico)" territory="MX" type="America/Tijuana
|
55
|
+
<mapZone other="Pacific Standard Time (Mexico)" territory="MX" type="America/Tijuana"/>
|
56
56
|
|
57
57
|
<!-- (UTC-08:00) Coordinated Universal Time-08 -->
|
58
58
|
<mapZone other="UTC-08" territory="001" type="Etc/GMT+8"/>
|
@@ -63,7 +63,6 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
63
63
|
<mapZone other="Pacific Standard Time" territory="001" type="America/Los_Angeles"/>
|
64
64
|
<mapZone other="Pacific Standard Time" territory="CA" type="America/Vancouver"/>
|
65
65
|
<mapZone other="Pacific Standard Time" territory="US" type="America/Los_Angeles"/>
|
66
|
-
<mapZone other="Pacific Standard Time" territory="ZZ" type="PST8PDT"/>
|
67
66
|
|
68
67
|
<!-- (UTC-07:00) Arizona -->
|
69
68
|
<mapZone other="US Mountain Standard Time" territory="001" type="America/Phoenix"/>
|
@@ -73,15 +72,14 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
73
72
|
<mapZone other="US Mountain Standard Time" territory="ZZ" type="Etc/GMT+7"/>
|
74
73
|
|
75
74
|
<!-- (UTC-07:00) Chihuahua, La Paz, Mazatlan -->
|
76
|
-
<mapZone other="Mountain Standard Time (Mexico)" territory="001" type="America/
|
77
|
-
<mapZone other="Mountain Standard Time (Mexico)" territory="MX" type="America/
|
75
|
+
<mapZone other="Mountain Standard Time (Mexico)" territory="001" type="America/Mazatlan"/>
|
76
|
+
<mapZone other="Mountain Standard Time (Mexico)" territory="MX" type="America/Mazatlan"/>
|
78
77
|
|
79
78
|
<!-- (UTC-07:00) Mountain Time (US & Canada) -->
|
80
79
|
<mapZone other="Mountain Standard Time" territory="001" type="America/Denver"/>
|
81
|
-
<mapZone other="Mountain Standard Time" territory="CA" type="America/Edmonton America/Cambridge_Bay America/Inuvik
|
82
|
-
<mapZone other="Mountain Standard Time" territory="MX" type="America/
|
80
|
+
<mapZone other="Mountain Standard Time" territory="CA" type="America/Edmonton America/Cambridge_Bay America/Inuvik"/>
|
81
|
+
<mapZone other="Mountain Standard Time" territory="MX" type="America/Ciudad_Juarez"/>
|
83
82
|
<mapZone other="Mountain Standard Time" territory="US" type="America/Denver America/Boise"/>
|
84
|
-
<mapZone other="Mountain Standard Time" territory="ZZ" type="MST7MDT"/>
|
85
83
|
|
86
84
|
<!-- (UTC-07:00) Yukon -->
|
87
85
|
<mapZone other="Yukon Standard Time" territory="001" type="America/Whitehorse"/>
|
@@ -100,10 +98,9 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
100
98
|
|
101
99
|
<!-- (UTC-06:00) Central Time (US & Canada) -->
|
102
100
|
<mapZone other="Central Standard Time" territory="001" type="America/Chicago"/>
|
103
|
-
<mapZone other="Central Standard Time" territory="CA" type="America/Winnipeg America/
|
104
|
-
<mapZone other="Central Standard Time" territory="MX" type="America/Matamoros"/>
|
101
|
+
<mapZone other="Central Standard Time" territory="CA" type="America/Winnipeg America/Rankin_Inlet America/Resolute"/>
|
102
|
+
<mapZone other="Central Standard Time" territory="MX" type="America/Matamoros America/Ojinaga"/>
|
105
103
|
<mapZone other="Central Standard Time" territory="US" type="America/Chicago America/Indiana/Knox America/Indiana/Tell_City America/Menominee America/North_Dakota/Beulah America/North_Dakota/Center America/North_Dakota/New_Salem"/>
|
106
|
-
<mapZone other="Central Standard Time" territory="ZZ" type="CST6CDT"/>
|
107
104
|
|
108
105
|
<!-- (UTC-06:00) Easter Island -->
|
109
106
|
<mapZone other="Easter Island Standard Time" territory="001" type="Pacific/Easter"/>
|
@@ -111,7 +108,7 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
111
108
|
|
112
109
|
<!-- (UTC-06:00) Guadalajara, Mexico City, Monterrey -->
|
113
110
|
<mapZone other="Central Standard Time (Mexico)" territory="001" type="America/Mexico_City"/>
|
114
|
-
<mapZone other="Central Standard Time (Mexico)" territory="MX" type="America/Mexico_City America/Bahia_Banderas America/Merida America/Monterrey"/>
|
111
|
+
<mapZone other="Central Standard Time (Mexico)" territory="MX" type="America/Mexico_City America/Bahia_Banderas America/Merida America/Monterrey America/Chihuahua "/>
|
115
112
|
|
116
113
|
<!-- (UTC-06:00) Saskatchewan -->
|
117
114
|
<mapZone other="Canada Central Standard Time" territory="001" type="America/Regina"/>
|
@@ -136,9 +133,8 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
136
133
|
<!-- (UTC-05:00) Eastern Time (US & Canada) -->
|
137
134
|
<mapZone other="Eastern Standard Time" territory="001" type="America/New_York"/>
|
138
135
|
<mapZone other="Eastern Standard Time" territory="BS" type="America/Nassau"/>
|
139
|
-
<mapZone other="Eastern Standard Time" territory="CA" type="America/Toronto America/Iqaluit
|
136
|
+
<mapZone other="Eastern Standard Time" territory="CA" type="America/Toronto America/Iqaluit"/>
|
140
137
|
<mapZone other="Eastern Standard Time" territory="US" type="America/New_York America/Detroit America/Indiana/Petersburg America/Indiana/Vincennes America/Indiana/Winamac America/Kentucky/Monticello America/Louisville"/>
|
141
|
-
<mapZone other="Eastern Standard Time" territory="ZZ" type="EST5EDT"/>
|
142
138
|
|
143
139
|
<!-- (UTC-05:00) Haiti -->
|
144
140
|
<mapZone other="Haiti Standard Time" territory="001" type="America/Port-au-Prince"/>
|
@@ -424,7 +420,7 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
424
420
|
<mapZone other="FLE Standard Time" territory="FI" type="Europe/Helsinki"/>
|
425
421
|
<mapZone other="FLE Standard Time" territory="LT" type="Europe/Vilnius"/>
|
426
422
|
<mapZone other="FLE Standard Time" territory="LV" type="Europe/Riga"/>
|
427
|
-
<mapZone other="FLE Standard Time" territory="UA" type="Europe/Kiev
|
423
|
+
<mapZone other="FLE Standard Time" territory="UA" type="Europe/Kiev"/>
|
428
424
|
|
429
425
|
<!-- (UTC+02:00) Jerusalem -->
|
430
426
|
<mapZone other="Israel Standard Time" territory="001" type="Asia/Jerusalem"/>
|
@@ -541,7 +537,8 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
541
537
|
<!-- (UTC+05:00) Ashgabat, Tashkent -->
|
542
538
|
<mapZone other="West Asia Standard Time" territory="001" type="Asia/Tashkent"/>
|
543
539
|
<mapZone other="West Asia Standard Time" territory="AQ" type="Antarctica/Mawson"/>
|
544
|
-
|
540
|
+
<!-- Microsoft may create a new zone dedicated for Almaty and Qostanay. -->
|
541
|
+
<mapZone other="West Asia Standard Time" territory="KZ" type="Asia/Oral Asia/Almaty Asia/Aqtau Asia/Aqtobe Asia/Atyrau Asia/Qostanay"/>
|
545
542
|
<mapZone other="West Asia Standard Time" territory="MV" type="Indian/Maldives"/>
|
546
543
|
<mapZone other="West Asia Standard Time" territory="TF" type="Indian/Kerguelen"/>
|
547
544
|
<mapZone other="West Asia Standard Time" territory="TJ" type="Asia/Dushanbe"/>
|
@@ -573,13 +570,12 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
573
570
|
<mapZone other="Nepal Standard Time" territory="001" type="Asia/Katmandu"/>
|
574
571
|
<mapZone other="Nepal Standard Time" territory="NP" type="Asia/Katmandu"/>
|
575
572
|
|
576
|
-
<!-- (UTC+06:00) Astana -->
|
577
|
-
<mapZone other="Central Asia Standard Time" territory="001" type="Asia/
|
573
|
+
<!-- (UTC+06:00) Astana --> <!-- Microsoft probably keeps Central Asia Standard Time, but change Astana to something else. -->
|
574
|
+
<mapZone other="Central Asia Standard Time" territory="001" type="Asia/Bishkek"/>
|
578
575
|
<mapZone other="Central Asia Standard Time" territory="AQ" type="Antarctica/Vostok"/>
|
579
576
|
<mapZone other="Central Asia Standard Time" territory="CN" type="Asia/Urumqi"/>
|
580
577
|
<mapZone other="Central Asia Standard Time" territory="IO" type="Indian/Chagos"/>
|
581
578
|
<mapZone other="Central Asia Standard Time" territory="KG" type="Asia/Bishkek"/>
|
582
|
-
<mapZone other="Central Asia Standard Time" territory="KZ" type="Asia/Almaty Asia/Qostanay"/>
|
583
579
|
<mapZone other="Central Asia Standard Time" territory="ZZ" type="Etc/GMT-6"/>
|
584
580
|
|
585
581
|
<!-- (UTC+06:00) Dhaka -->
|
@@ -656,7 +652,7 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
656
652
|
|
657
653
|
<!-- (UTC+08:00) Ulaanbaatar -->
|
658
654
|
<mapZone other="Ulaanbaatar Standard Time" territory="001" type="Asia/Ulaanbaatar"/>
|
659
|
-
<mapZone other="Ulaanbaatar Standard Time" territory="MN" type="Asia/Ulaanbaatar
|
655
|
+
<mapZone other="Ulaanbaatar Standard Time" territory="MN" type="Asia/Ulaanbaatar"/>
|
660
656
|
|
661
657
|
<!-- (UTC+08:45) Eucla -->
|
662
658
|
<mapZone other="Aus Central W. Standard Time" territory="001" type="Australia/Eucla"/>
|
@@ -713,7 +709,7 @@ Source: https://github.com/unicode-org/cldr/blob/main/common/supplemental/window
|
|
713
709
|
|
714
710
|
<!-- (UTC+10:00) Hobart -->
|
715
711
|
<mapZone other="Tasmania Standard Time" territory="001" type="Australia/Hobart"/>
|
716
|
-
<mapZone other="Tasmania Standard Time" territory="AU" type="Australia/Hobart
|
712
|
+
<mapZone other="Tasmania Standard Time" territory="AU" type="Australia/Hobart Antarctica/Macquarie"/>
|
717
713
|
|
718
714
|
<!-- (UTC+10:00) Vladivostok -->
|
719
715
|
<mapZone other="Vladivostok Standard Time" territory="001" type="Asia/Vladivostok"/>
|
dissect/target/helpers/docs.py
CHANGED
@@ -46,7 +46,7 @@ def get_real_func_obj(func: Callable) -> Tuple[Type, Callable]:
|
|
46
46
|
return (klass, func)
|
47
47
|
|
48
48
|
|
49
|
-
def get_docstring(obj: Any, placeholder=NO_DOCS) -> str:
|
49
|
+
def get_docstring(obj: Any, placeholder: str = NO_DOCS) -> str:
|
50
50
|
"""Get object's docstring or a placeholder if no docstring found"""
|
51
51
|
# Use of `inspect.cleandoc()` is preferred to `textwrap.dedent()` here
|
52
52
|
# because many multi-line docstrings in the codebase
|
dissect/target/helpers/mount.py
CHANGED
@@ -8,7 +8,6 @@ from dissect.util.feature import Feature, feature_enabled
|
|
8
8
|
|
9
9
|
from dissect.target.filesystem import Filesystem, FilesystemEntry
|
10
10
|
|
11
|
-
HAS_FUSE3 = False
|
12
11
|
if feature_enabled(Feature.BETA):
|
13
12
|
from fuse3 import FuseOSError, Operations
|
14
13
|
from fuse3.c_fuse import fuse_config_p, fuse_conn_info_p
|
@@ -20,6 +19,8 @@ else:
|
|
20
19
|
fuse_config_p = c_void_p
|
21
20
|
fuse_conn_info_p = c_void_p
|
22
21
|
|
22
|
+
HAS_FUSE3 = False
|
23
|
+
|
23
24
|
|
24
25
|
log = logging.getLogger(__name__)
|
25
26
|
|
dissect/target/helpers/record.py
CHANGED
@@ -165,8 +165,13 @@ WindowsInterfaceRecord = TargetRecordDescriptor(
|
|
165
165
|
[
|
166
166
|
*COMMON_INTERFACE_ELEMENTS,
|
167
167
|
("varint", "vlan"),
|
168
|
-
("
|
168
|
+
("net.ipnetwork[]", "network"),
|
169
|
+
("varint", "metric"),
|
170
|
+
("stringlist", "search_domain"),
|
171
|
+
("datetime", "first_connected"),
|
169
172
|
("datetime", "last_connected"),
|
173
|
+
("net.ipaddress[]", "subnetmask"),
|
174
|
+
("boolean", "dhcp"),
|
170
175
|
],
|
171
176
|
)
|
172
177
|
|
@@ -175,7 +180,29 @@ MacInterfaceRecord = TargetRecordDescriptor(
|
|
175
180
|
[
|
176
181
|
*COMMON_INTERFACE_ELEMENTS,
|
177
182
|
("varint", "vlan"),
|
178
|
-
("
|
183
|
+
("net.ipnetwork[]", "network"),
|
179
184
|
("varint", "interface_service_order"),
|
185
|
+
("boolean", "dhcp"),
|
180
186
|
],
|
181
187
|
)
|
188
|
+
|
189
|
+
|
190
|
+
COMMON_APPLICATION_FIELDS = [
|
191
|
+
("datetime", "ts_modified"),
|
192
|
+
("datetime", "ts_installed"),
|
193
|
+
("string", "name"),
|
194
|
+
("string", "version"),
|
195
|
+
("string", "author"),
|
196
|
+
("string", "type"),
|
197
|
+
("path", "path"),
|
198
|
+
]
|
199
|
+
|
200
|
+
UnixApplicationRecord = TargetRecordDescriptor(
|
201
|
+
"unix/application",
|
202
|
+
COMMON_APPLICATION_FIELDS,
|
203
|
+
)
|
204
|
+
|
205
|
+
WindowsApplicationRecord = TargetRecordDescriptor(
|
206
|
+
"windows/application",
|
207
|
+
COMMON_APPLICATION_FIELDS,
|
208
|
+
)
|
@@ -1,3 +1,4 @@
|
|
1
|
+
import logging
|
1
2
|
from functools import partial
|
2
3
|
from typing import Callable, Iterable, Iterator
|
3
4
|
|
@@ -95,13 +96,16 @@ def modify_record(target: Target, record: Record, modifier_function: ModifierFun
|
|
95
96
|
try:
|
96
97
|
_record = modifier_function(field_name, resolved_path)
|
97
98
|
except FilesystemError as e:
|
98
|
-
|
99
|
+
level = logging.INFO if isinstance(e, FileNotFoundError) else logging.WARNING
|
100
|
+
target.log.log(
|
101
|
+
level,
|
99
102
|
"Unable to modify record '%s' with function '%s': %s",
|
100
103
|
record._desc.name,
|
101
104
|
modifier_function.__name__,
|
102
105
|
e,
|
103
106
|
)
|
104
107
|
target.log.debug("", exc_info=e)
|
108
|
+
|
105
109
|
else:
|
106
110
|
additional_records.append(_record)
|
107
111
|
|