mirror of
https://github.com/overcuriousity/trace.git
synced 2025-12-21 13:32:20 +00:00
Compare commits
21 Commits
v0.2.2-alp
...
v0.2.4-alp
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
37b6503b29 | ||
|
|
f54e2560f3 | ||
|
|
a2e7798a2d | ||
|
|
15bc00a195 | ||
|
|
053369df78 | ||
|
|
eca56c0d54 | ||
|
|
06b7680982 | ||
|
|
bfefb42761 | ||
|
|
070e76467c | ||
|
|
b80dd10901 | ||
|
|
fe3c0710c6 | ||
|
|
809a4a498f | ||
|
|
931e5debc8 | ||
|
|
f91f434f7f | ||
|
|
85ca483a1d | ||
|
|
f50fd1800d | ||
|
|
b830d15d85 | ||
|
|
4a4e1e7c06 | ||
|
|
2a7d00d221 | ||
|
|
c68fc66de6 | ||
|
|
f68c8389da |
@@ -145,6 +145,14 @@ The codebase is organized into focused, single-responsibility modules to make it
|
|||||||
|
|
||||||
**Integrity System**: Every note automatically gets:
|
**Integrity System**: Every note automatically gets:
|
||||||
1. SHA256 hash of `timestamp:content` (via `Note.calculate_hash()`)
|
1. SHA256 hash of `timestamp:content` (via `Note.calculate_hash()`)
|
||||||
|
- **Timestamp Format**: Unix epoch timestamp as float (seconds since 1970-01-01 00:00:00 UTC)
|
||||||
|
- **Hash Input Format**: `"{timestamp}:{content}"` where timestamp is converted to string using Python's default str() conversion
|
||||||
|
- **Example**: For content "Suspicious process detected" with timestamp 1702345678.123456, the hash input is:
|
||||||
|
```
|
||||||
|
1702345678.123456:Suspicious process detected
|
||||||
|
```
|
||||||
|
- This ensures integrity of both WHAT was said (content) and WHEN it was said (timestamp)
|
||||||
|
- The exact float precision is preserved in the hash, making timestamps forensically tamper-evident
|
||||||
2. Optional GPG clearsign signature (if `pgp_enabled` in settings and GPG available)
|
2. Optional GPG clearsign signature (if `pgp_enabled` in settings and GPG available)
|
||||||
|
|
||||||
**Tag System**: Regex-based hashtag extraction (`#word`)
|
**Tag System**: Regex-based hashtag extraction (`#word`)
|
||||||
|
|||||||
43
README.md
43
README.md
@@ -20,7 +20,7 @@ trace "IR team gained shell access. Initial persistence checks running."
|
|||||||
trace "Observed outbound connection to 192.168.1.55 on port 80. #suspicious #network"
|
trace "Observed outbound connection to 192.168.1.55 on port 80. #suspicious #network"
|
||||||
```
|
```
|
||||||
|
|
||||||
**System Integrity Chain:** Each command-line note is immediately stamped, concatenated with its content, and hashed using SHA256 before storage. This ensures a non-repudiable log entry.
|
**System Integrity Chain:** Each command-line note is immediately stamped with a Unix epoch timestamp (seconds since 1970-01-01 00:00:00 UTC as float, e.g., `1702345678.123456`), concatenated with its content in the format `"{timestamp}:{content}"`, and hashed using SHA256 before storage. This ensures a non-repudiable log entry with forensically tamper-evident timestamps.
|
||||||
|
|
||||||
## CLI Command Reference
|
## CLI Command Reference
|
||||||
|
|
||||||
@@ -195,7 +195,7 @@ After this, you can log with just: `t "Your note here"`
|
|||||||
|
|
||||||
| Feature | Description | Operational Impact |
|
| Feature | Description | Operational Impact |
|
||||||
| :--- | :--- | :--- |
|
| :--- | :--- | :--- |
|
||||||
| **Integrity Hashing** | SHA256 applied to every log entry (content + timestamp). | **Guaranteed log integrity.** No modification possible post-entry. |
|
| **Integrity Hashing** | SHA256 applied to every log entry using format `"{unix_timestamp}:{content}"`. Timestamp is Unix epoch as float (e.g., `1702345678.123456`). | **Guaranteed log integrity.** No modification possible post-entry. Timestamps are forensically tamper-evident with full float precision. |
|
||||||
| **GPG Signing** | Optional PGP/GPG signature applied to notes. | **Non-repudiation** for formal evidence handling. |
|
| **GPG Signing** | Optional PGP/GPG signature applied to notes. | **Non-repudiation** for formal evidence handling. |
|
||||||
| **IOC Extraction** | Automatic parsing of IPv4, FQDNs, URLs, hashes, and email addresses. | **Immediate intelligence gathering** from raw text. |
|
| **IOC Extraction** | Automatic parsing of IPv4, FQDNs, URLs, hashes, and email addresses. | **Immediate intelligence gathering** from raw text. |
|
||||||
| **Tag System** | Supports `#hashtags` for classification and filtering. | **Efficient triage** of large log sets. |
|
| **Tag System** | Supports `#hashtags` for classification and filtering. | **Efficient triage** of large log sets. |
|
||||||
@@ -208,20 +208,33 @@ After this, you can log with just: `t "Your note here"`
|
|||||||
### Layer 1: Note-Level Integrity (Always Active)
|
### Layer 1: Note-Level Integrity (Always Active)
|
||||||
|
|
||||||
**Process:**
|
**Process:**
|
||||||
1. **Timestamp Generation** - Precise Unix timestamp captured at note creation
|
1. **Timestamp Generation** - Precise Unix epoch timestamp (float) captured at note creation
|
||||||
2. **Content Hashing** - SHA256 hash computed from `timestamp:content`
|
- Format: Seconds since 1970-01-01 00:00:00 UTC (e.g., `1702345678.123456`)
|
||||||
|
- Full float precision preserved for forensic tamper-evidence
|
||||||
|
2. **Content Hashing** - SHA256 hash computed from `"{timestamp}:{content}"`
|
||||||
3. **Optional Signature** - Hash is signed with investigator's GPG private key
|
3. **Optional Signature** - Hash is signed with investigator's GPG private key
|
||||||
|
|
||||||
**Mathematical Representation:**
|
**Mathematical Representation:**
|
||||||
```
|
```
|
||||||
hash = SHA256(timestamp + ":" + content)
|
timestamp = Unix epoch time as float (e.g., 1702345678.123456)
|
||||||
|
hash_input = "{timestamp}:{content}"
|
||||||
|
hash = SHA256(hash_input)
|
||||||
signature = GPG_Sign(hash, private_key)
|
signature = GPG_Sign(hash, private_key)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
```
|
||||||
|
Content: "Suspicious process detected"
|
||||||
|
Timestamp: 1702345678.123456
|
||||||
|
Hash input: "1702345678.123456:Suspicious process detected"
|
||||||
|
Hash: SHA256 of above = a3f5b2c8d9e1f4a7b6c3d8e2f5a9b4c7d1e6f3a8b5c2d9e4f7a1b8c6d3e0f5a2
|
||||||
|
```
|
||||||
|
|
||||||
**Security Properties:**
|
**Security Properties:**
|
||||||
- **Temporal Integrity**: Timestamp is cryptographically bound to content (cannot backdate notes)
|
- **Temporal Integrity**: Timestamp is cryptographically bound to content (cannot backdate notes)
|
||||||
- **Tamper Detection**: Any modification to content or timestamp invalidates the hash
|
- **Tamper Detection**: Any modification to content or timestamp invalidates the hash
|
||||||
- **Non-Repudiation**: GPG signature proves who created the note (if signing enabled)
|
- **Non-Repudiation**: GPG signature proves who created the note (if signing enabled)
|
||||||
|
- **Hash Reproducibility**: Exported markdown includes Unix timestamp for independent verification
|
||||||
- **Efficient Storage**: Signing only the hash (64 hex chars) instead of full content
|
- **Efficient Storage**: Signing only the hash (64 hex chars) instead of full content
|
||||||
|
|
||||||
### Layer 2: Export-Level Integrity (On Demand)
|
### Layer 2: Export-Level Integrity (On Demand)
|
||||||
@@ -328,6 +341,26 @@ Individual note signatures are embedded in the markdown export. To verify a spec
|
|||||||
- The GPG signature proves who created that hash
|
- The GPG signature proves who created that hash
|
||||||
- Together: Proves this specific content was created by this investigator at this time
|
- Together: Proves this specific content was created by this investigator at this time
|
||||||
|
|
||||||
|
**Hash Verification (Manual):**
|
||||||
|
|
||||||
|
To independently verify a note's hash from the markdown export:
|
||||||
|
|
||||||
|
1. Locate the note in the export file and extract:
|
||||||
|
- Unix Timestamp (e.g., `1702345678.123456`)
|
||||||
|
- Content (e.g., `"Suspicious process detected"`)
|
||||||
|
- Claimed Hash (e.g., `a3f5b2c8...`)
|
||||||
|
|
||||||
|
2. Recompute the hash:
|
||||||
|
```bash
|
||||||
|
# Using Python
|
||||||
|
python3 -c "import hashlib; print(hashlib.sha256(b'1702345678.123456:Suspicious process detected').hexdigest())"
|
||||||
|
|
||||||
|
# Using command-line tools
|
||||||
|
echo -n "1702345678.123456:Suspicious process detected" | sha256sum
|
||||||
|
```
|
||||||
|
|
||||||
|
3. Compare the computed hash with the claimed hash - they must match exactly
|
||||||
|
|
||||||
### Cryptographic Trust Model
|
### Cryptographic Trust Model
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -364,9 +364,14 @@ def export_markdown(output_file: str = "export.md"):
|
|||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
def format_note_for_export(note: Note) -> str:
|
def format_note_for_export(note: Note) -> str:
|
||||||
"""Format a single note for export (returns string instead of writing to file)"""
|
"""Format a single note for export (returns string instead of writing to file)
|
||||||
|
|
||||||
|
Includes Unix timestamp for hash reproducibility - anyone can recompute the hash
|
||||||
|
using the formula: SHA256("{unix_timestamp}:{content}")
|
||||||
|
"""
|
||||||
lines = []
|
lines = []
|
||||||
lines.append(f"- **{time.ctime(note.timestamp)}**\n")
|
lines.append(f"- **{time.ctime(note.timestamp)}**\n")
|
||||||
|
lines.append(f" - Unix Timestamp: `{note.timestamp}` (for hash verification)\n")
|
||||||
lines.append(f" - Content:\n")
|
lines.append(f" - Content:\n")
|
||||||
# Properly indent multi-line content
|
# Properly indent multi-line content
|
||||||
for line in note.content.splitlines():
|
for line in note.content.splitlines():
|
||||||
|
|||||||
@@ -43,12 +43,25 @@ class Crypto:
|
|||||||
return False, "Not a GPG signed message"
|
return False, "Not a GPG signed message"
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
# Force English output for consistent parsing across locales
|
||||||
|
# Linux/macOS: LC_ALL/LANG variables control GPG's output language
|
||||||
|
# Windows: GPG may ignore these, but encoding='utf-8' + errors='replace' provides robustness
|
||||||
|
import os
|
||||||
|
env = os.environ.copy()
|
||||||
|
# Use C.UTF-8 for English messages with UTF-8 encoding support
|
||||||
|
# Falls back gracefully via errors='replace' if locale not available
|
||||||
|
env['LC_ALL'] = 'C.UTF-8'
|
||||||
|
env['LANG'] = 'C.UTF-8'
|
||||||
|
|
||||||
proc = subprocess.Popen(
|
proc = subprocess.Popen(
|
||||||
['gpg', '--verify'],
|
['gpg', '--verify'],
|
||||||
stdin=subprocess.PIPE,
|
stdin=subprocess.PIPE,
|
||||||
stdout=subprocess.PIPE,
|
stdout=subprocess.PIPE,
|
||||||
stderr=subprocess.PIPE,
|
stderr=subprocess.PIPE,
|
||||||
text=True
|
text=True,
|
||||||
|
encoding='utf-8',
|
||||||
|
errors='replace', # Handle encoding issues on any platform
|
||||||
|
env=env
|
||||||
)
|
)
|
||||||
stdout, stderr = proc.communicate(input=signed_content, timeout=10)
|
stdout, stderr = proc.communicate(input=signed_content, timeout=10)
|
||||||
|
|
||||||
@@ -171,5 +184,25 @@ class Crypto:
|
|||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def hash_content(content: str, timestamp: float) -> str:
|
def hash_content(content: str, timestamp: float) -> str:
|
||||||
|
"""Calculate SHA256 hash of timestamp:content.
|
||||||
|
|
||||||
|
Hash input format: "{timestamp}:{content}"
|
||||||
|
- timestamp: Unix epoch timestamp as float (seconds since 1970-01-01 00:00:00 UTC)
|
||||||
|
Example: 1702345678.123456
|
||||||
|
- The float is converted to string using Python's default str() conversion
|
||||||
|
- Colon (':') separator between timestamp and content
|
||||||
|
- Ensures integrity of both WHAT was said and WHEN it was said
|
||||||
|
|
||||||
|
Args:
|
||||||
|
content: The note content to hash
|
||||||
|
timestamp: Unix epoch timestamp as float
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SHA256 hash as hexadecimal string (64 characters)
|
||||||
|
|
||||||
|
Example:
|
||||||
|
>>> hash_content("Suspicious process detected", 1702345678.123456)
|
||||||
|
Computes SHA256 of: "1702345678.123456:Suspicious process detected"
|
||||||
|
"""
|
||||||
data = f"{timestamp}:{content}".encode('utf-8')
|
data = f"{timestamp}:{content}".encode('utf-8')
|
||||||
return hashlib.sha256(data).hexdigest()
|
return hashlib.sha256(data).hexdigest()
|
||||||
|
|||||||
@@ -12,6 +12,9 @@ from .extractors import TagExtractor, IOCExtractor
|
|||||||
@dataclass
|
@dataclass
|
||||||
class Note:
|
class Note:
|
||||||
content: str
|
content: str
|
||||||
|
# Unix timestamp: seconds since 1970-01-01 00:00:00 UTC as float
|
||||||
|
# Example: 1702345678.123456
|
||||||
|
# This exact float value (with full precision) is used in hash calculation
|
||||||
timestamp: float = field(default_factory=time.time)
|
timestamp: float = field(default_factory=time.time)
|
||||||
note_id: str = field(default_factory=lambda: str(uuid.uuid4()))
|
note_id: str = field(default_factory=lambda: str(uuid.uuid4()))
|
||||||
content_hash: str = ""
|
content_hash: str = ""
|
||||||
@@ -28,7 +31,16 @@ class Note:
|
|||||||
self.iocs = IOCExtractor.extract_iocs(self.content)
|
self.iocs = IOCExtractor.extract_iocs(self.content)
|
||||||
|
|
||||||
def calculate_hash(self):
|
def calculate_hash(self):
|
||||||
# We hash the content + timestamp to ensure integrity of 'when' it was said
|
"""Calculate SHA256 hash of timestamp:content.
|
||||||
|
|
||||||
|
Hash input format: "{timestamp}:{content}"
|
||||||
|
- timestamp: Unix epoch timestamp as float (e.g., "1702345678.123456")
|
||||||
|
- The float is converted to string using Python's default str() conversion
|
||||||
|
- Colon separator between timestamp and content
|
||||||
|
- Ensures integrity of both WHAT was said and WHEN it was said
|
||||||
|
|
||||||
|
Example hash input: "1702345678.123456:Suspicious process detected"
|
||||||
|
"""
|
||||||
data = f"{self.timestamp}:{self.content}".encode('utf-8')
|
data = f"{self.timestamp}:{self.content}".encode('utf-8')
|
||||||
self.content_hash = hashlib.sha256(data).hexdigest()
|
self.content_hash = hashlib.sha256(data).hexdigest()
|
||||||
|
|
||||||
|
|||||||
@@ -222,15 +222,23 @@ class ExportHandler:
|
|||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _write_note_markdown(f, note: Note):
|
def _write_note_markdown(f, note: Note):
|
||||||
"""Helper to write a note in markdown format"""
|
"""Helper to write a note in markdown format
|
||||||
|
|
||||||
|
Includes Unix timestamp for hash reproducibility - anyone can recompute the hash
|
||||||
|
using the formula: SHA256("{unix_timestamp}:{content}")
|
||||||
|
"""
|
||||||
f.write(f"- **{time.ctime(note.timestamp)}**\n")
|
f.write(f"- **{time.ctime(note.timestamp)}**\n")
|
||||||
f.write(f" - Content: {note.content}\n")
|
f.write(f" - Unix Timestamp: `{note.timestamp}` (for hash verification)\n")
|
||||||
|
f.write(f" - Content:\n")
|
||||||
|
# Properly indent multi-line content
|
||||||
|
for line in note.content.splitlines():
|
||||||
|
f.write(f" {line}\n")
|
||||||
if note.tags:
|
if note.tags:
|
||||||
tags_str = " ".join([f"#{tag}" for tag in note.tags])
|
tags_str = " ".join([f"#{tag}" for tag in note.tags])
|
||||||
f.write(f" - Tags: {tags_str}\n")
|
f.write(f" - Tags: {tags_str}\n")
|
||||||
f.write(f" - Hash: `{note.content_hash}`\n")
|
f.write(f" - SHA256 Hash (timestamp:content): `{note.content_hash}`\n")
|
||||||
if note.signature:
|
if note.signature:
|
||||||
f.write(" - **Signature Verified:**\n")
|
f.write(" - **GPG Signature of Hash:**\n")
|
||||||
f.write(" ```\n")
|
f.write(" ```\n")
|
||||||
for line in note.signature.splitlines():
|
for line in note.signature.splitlines():
|
||||||
f.write(f" {line}\n")
|
f.write(f" {line}\n")
|
||||||
|
|||||||
@@ -39,5 +39,5 @@ def init_colors():
|
|||||||
curses.init_pair(ColorPairs.TAG, curses.COLOR_MAGENTA, curses.COLOR_BLACK)
|
curses.init_pair(ColorPairs.TAG, curses.COLOR_MAGENTA, curses.COLOR_BLACK)
|
||||||
# IOCs on selected background (red on cyan)
|
# IOCs on selected background (red on cyan)
|
||||||
curses.init_pair(ColorPairs.IOC_SELECTED, curses.COLOR_RED, curses.COLOR_CYAN)
|
curses.init_pair(ColorPairs.IOC_SELECTED, curses.COLOR_RED, curses.COLOR_CYAN)
|
||||||
# Tags on selected background (yellow on cyan)
|
# Tags on selected background (magenta on cyan)
|
||||||
curses.init_pair(ColorPairs.TAG_SELECTED, curses.COLOR_YELLOW, curses.COLOR_CYAN)
|
curses.init_pair(ColorPairs.TAG_SELECTED, curses.COLOR_MAGENTA, curses.COLOR_CYAN)
|
||||||
|
|||||||
@@ -113,15 +113,15 @@ class TextRenderer:
|
|||||||
screen.addstr(y, x_pos, text)
|
screen.addstr(y, x_pos, text)
|
||||||
screen.attroff(curses.color_pair(ColorPairs.ERROR) | curses.A_BOLD)
|
screen.attroff(curses.color_pair(ColorPairs.ERROR) | curses.A_BOLD)
|
||||||
else: # tag
|
else: # tag
|
||||||
# Tag highlighting: yellow on cyan if selected, yellow on black otherwise
|
# Tag highlighting: magenta on cyan if selected, magenta on black otherwise
|
||||||
if is_selected:
|
if is_selected:
|
||||||
screen.attron(curses.color_pair(ColorPairs.TAG_SELECTED))
|
screen.attron(curses.color_pair(ColorPairs.TAG_SELECTED))
|
||||||
screen.addstr(y, x_pos, text)
|
screen.addstr(y, x_pos, text)
|
||||||
screen.attroff(curses.color_pair(ColorPairs.TAG_SELECTED))
|
screen.attroff(curses.color_pair(ColorPairs.TAG_SELECTED))
|
||||||
else:
|
else:
|
||||||
screen.attron(curses.color_pair(ColorPairs.WARNING))
|
screen.attron(curses.color_pair(ColorPairs.TAG))
|
||||||
screen.addstr(y, x_pos, text)
|
screen.addstr(y, x_pos, text)
|
||||||
screen.attroff(curses.color_pair(ColorPairs.WARNING))
|
screen.attroff(curses.color_pair(ColorPairs.TAG))
|
||||||
|
|
||||||
x_pos += len(text)
|
x_pos += len(text)
|
||||||
last_pos = end
|
last_pos = end
|
||||||
|
|||||||
88
trace/tui/visual_constants.py
Normal file
88
trace/tui/visual_constants.py
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
"""Visual constants for consistent TUI layout and styling"""
|
||||||
|
|
||||||
|
|
||||||
|
class Layout:
|
||||||
|
"""Screen layout constants"""
|
||||||
|
HEADER_Y = 0
|
||||||
|
HEADER_X = 2
|
||||||
|
CONTENT_START_Y = 2
|
||||||
|
CONTENT_INDENT = 4
|
||||||
|
FOOTER_OFFSET_FROM_BOTTOM = 3
|
||||||
|
BORDER_OFFSET_FROM_BOTTOM = 2
|
||||||
|
STATUS_LINE_OFFSET_FROM_BOTTOM = 1 # height - 1 for status bar
|
||||||
|
NOTE_DETAIL_BOTTOM_RESERVE = 6 # height - 6 for note detail view
|
||||||
|
|
||||||
|
|
||||||
|
class Spacing:
|
||||||
|
"""Spacing and padding constants"""
|
||||||
|
SECTION_VERTICAL_GAP = 2
|
||||||
|
ITEM_VERTICAL_GAP = 1
|
||||||
|
DIALOG_MARGIN = 4
|
||||||
|
HORIZONTAL_PADDING = 6 # width - 6 for truncation
|
||||||
|
HASH_DISPLAY_PADDING = 20 # width - 20
|
||||||
|
HASH_SHORT_PADDING = 12 # width - 12 for shorter hash displays
|
||||||
|
EMPTY_STATE_PADDING = 8 # width - 8 for empty state boxes
|
||||||
|
STATUS_BAR_PADDING = 2 # width - 2 for status bar
|
||||||
|
|
||||||
|
|
||||||
|
class ColumnWidths:
|
||||||
|
"""Column widths for list displays - can be percentage-based"""
|
||||||
|
TAG_COLUMN_MIN = 30
|
||||||
|
IOC_COLUMN_MIN = 50
|
||||||
|
CONTENT_PREVIEW_MIN = 50
|
||||||
|
NOTE_PREVIEW_MIN = 60
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_tag_width(terminal_width):
|
||||||
|
"""Get responsive tag column width (40% of terminal or min 30)"""
|
||||||
|
return max(ColumnWidths.TAG_COLUMN_MIN, int(terminal_width * 0.4))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_ioc_width(terminal_width):
|
||||||
|
"""Get responsive IOC column width (50% of terminal or min 50)"""
|
||||||
|
return max(ColumnWidths.IOC_COLUMN_MIN, int(terminal_width * 0.5))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_content_preview_width(terminal_width):
|
||||||
|
"""Get responsive content preview width (50% of terminal or min 50)"""
|
||||||
|
return max(ColumnWidths.CONTENT_PREVIEW_MIN, int(terminal_width * 0.5))
|
||||||
|
|
||||||
|
|
||||||
|
class DialogSize:
|
||||||
|
"""Standard dialog dimensions (width, height)"""
|
||||||
|
SMALL = (40, 8) # Confirm dialogs
|
||||||
|
MEDIUM = (60, 15) # Settings, single input
|
||||||
|
LARGE = (70, 20) # Multiline, help
|
||||||
|
|
||||||
|
|
||||||
|
class Icons:
|
||||||
|
"""Unicode symbols used throughout UI"""
|
||||||
|
ACTIVE = "●"
|
||||||
|
INACTIVE = "○"
|
||||||
|
DIAMOND = "◆"
|
||||||
|
SQUARE = "■"
|
||||||
|
SMALL_SQUARE = "▪"
|
||||||
|
ARROW_RIGHT = "▸"
|
||||||
|
WARNING = "⚠"
|
||||||
|
HASH = "⌗"
|
||||||
|
FILTER = "◈"
|
||||||
|
VERIFIED = "✓"
|
||||||
|
FAILED = "✗"
|
||||||
|
UNSIGNED = "?"
|
||||||
|
SEPARATOR_H = "─"
|
||||||
|
SEPARATOR_V = "│"
|
||||||
|
SEPARATOR_GROUP = "│" # For grouping footer commands
|
||||||
|
BOX_TL = "┌"
|
||||||
|
BOX_BL = "└"
|
||||||
|
# Box drawing for improved empty states
|
||||||
|
BOX_DOUBLE_TL = "╔"
|
||||||
|
BOX_DOUBLE_TR = "╗"
|
||||||
|
BOX_DOUBLE_BL = "╚"
|
||||||
|
BOX_DOUBLE_BR = "╝"
|
||||||
|
BOX_DOUBLE_H = "═"
|
||||||
|
BOX_DOUBLE_V = "║"
|
||||||
|
|
||||||
|
|
||||||
|
class Timing:
|
||||||
|
"""Timing constants"""
|
||||||
|
FLASH_MESSAGE_DURATION = 3 # seconds
|
||||||
574
trace/tui_app.py
574
trace/tui_app.py
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user