Metadata-Version: 2.4
Name: fckprint
Version: 1.1.0
Summary: imagine a world using print for debugging, and we are happy to be not be in it now.
Home-page: https://github.com/SRSWTI/fckprint
Author: SRSWTI Research Labs
Author-email: team@srswti.com
Keywords: debugging monitoring performance security production tracing
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
Classifier: Programming Language :: Python :: 3.6
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: Implementation :: CPython
Classifier: Programming Language :: Python :: Implementation :: PyPy
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Topic :: Software Development :: Debuggers
Classifier: Topic :: Software Development :: Testing
Classifier: Topic :: System :: Monitoring
Classifier: Topic :: System :: Systems Administration
Requires-Python: >=3.6
Description-Content-Type: text/markdown
License-File: LICENSE
License-File: AUTHORS
Provides-Extra: tests
Requires-Dist: pytest; extra == "tests"
Requires-Dist: pytest-cov; extra == "tests"
Provides-Extra: monitoring
Requires-Dist: psutil>=5.0.0; extra == "monitoring"
Provides-Extra: full
Requires-Dist: psutil>=5.0.0; extra == "full"
Dynamic: author
Dynamic: author-email
Dynamic: classifier
Dynamic: description
Dynamic: description-content-type
Dynamic: home-page
Dynamic: keywords
Dynamic: license-file
Dynamic: provides-extra
Dynamic: requires-python
Dynamic: summary

# fckprint

advanced debugging and monitoring for python applications

fckprint is a powerful debugging and monitoring library that provides comprehensive tracing, performance monitoring, error tracking, caching, and production-ready features for python applications.

## why fckprint over print?

traditional debugging with print statements is slow, messy, and doesn't scale. fckprint provides structured, detailed debugging that's faster and more informative.

### traditional print debugging (slow and messy)

```python
def calculate_fibonacci(n):
    print(f"entering calculate_fibonacci with n={n}")
    if n <= 1:
        print(f"base case: returning {n}")
        return n
    
    print(f"recursive case: calling calculate_fibonacci({n-1}) + calculate_fibonacci({n-2})")
    result = calculate_fibonacci(n - 1) + calculate_fibonacci(n - 2)
    print(f"returning result: {result}")
    return result

# output is messy and hard to follow:
# entering calculate_fibonacci with n=5
# recursive case: calling calculate_fibonacci(4) + calculate_fibonacci(3)
# entering calculate_fibonacci with n=4
# recursive case: calling calculate_fibonacci(3) + calculate_fibonacci(2)
# entering calculate_fibonacci with n=3
# recursive case: calling calculate_fibonacci(2) + calculate_fibonacci(1)
# entering calculate_fibonacci with n=2
# recursive case: calling calculate_fibonacci(1) + calculate_fibonacci(0)
# entering calculate_fibonacci with n=1
# base case: returning 1
# entering calculate_fibonacci with n=0
# base case: returning 0
# returning result: 1
# returning result: 1
# returning result: 2
# entering calculate_fibonacci with n=1
# base case: returning 1
# returning result: 3
# returning result: 5
```

### fckprint debugging (fast and structured)

```python
import fckprint

@fckprint.snoop()
def calculate_fibonacci(n):
    if n <= 1:
        return n
    return calculate_fibonacci(n - 1) + calculate_fibonacci(n - 2)

# clean, structured output with timestamps and variable tracking:
# 17:21:32.924559 line        10         if n <= 1:
# 17:21:32.924657 line        11         return n
# 17:21:32.924677 line        12         return calculate_fibonacci(n - 1) + calculate_fibonacci(n - 2)
# elapsed time: 00:00:00.000181
```

### fckprint show (print replacement)

```python
# direct import (recommended)
from fckprint import show

def calculate_fibonacci(n):
    show("entering fibonacci function with n =", n)
    if n <= 1:
        show("base case: returning", n)
        return n
    
    show("recursive case: calling fibonacci", n-1, "and", n-2)
    result = calculate_fibonacci(n - 1) + calculate_fibonacci(n - 2)
    show("returning result:", result)
    return result
```

# structured output with timestamps and log levels:
# [18:08:47.183] INFO entering fibonacci function with n = 5
# [18:08:47.183] INFO recursive case: calling fibonacci 4 and 3
# [18:08:47.183] INFO entering fibonacci function with n = 4
# [18:08:47.183] INFO recursive case: calling fibonacci 3 and 2
# [18:08:47.183] INFO entering fibonacci function with n = 3
# [18:08:47.183] INFO recursive case: calling fibonacci 2 and 1
# [18:08:47.183] INFO entering fibonacci function with n = 2
# [18:08:47.183] INFO recursive case: calling fibonacci 1 and 0
# [18:08:47.183] INFO entering fibonacci function with n = 1
# [18:08:47.183] INFO base case: returning 1
# [18:08:47.183] INFO entering fibonacci function with n = 0
# [18:08:47.183] INFO base case: returning 0
# [18:08:47.183] INFO returning result: 1
# [18:08:47.183] INFO returning result: 1
# [18:08:47.183] INFO returning result: 2
# [18:08:47.183] INFO returning result: 3
# [18:08:47.183] INFO returning result: 5
```



### advanced debugging with fckprint

```python
@fckprint.snoop(watch=('x', 'y', 'result'))
def advanced_calculation(x, y):
    result = x * y + 10
    return result

# automatically tracks specific variables:
# starting var:.. x = 5
# starting var:.. y = 3
# new var:....... result = 25
# return value:.. 25
```

### why fckprint is better

1. **faster execution** - no manual print statements to slow down code
2. **structured output** - timestamps, line numbers, and variable tracking
3. **production ready** - can be disabled in production with environment variables
4. **comprehensive monitoring** - performance, errors, caching, security
5. **thread safe** - works correctly in multi-threaded applications
6. **configurable** - customize output format and verbosity
7. **non-intrusive** - minimal code changes required

## installation

```bash
uv pip install fckprint
```

or

```bash
pip install fckprint
```

## quick start

### basic function tracing:

```python
import fckprint

@fckprint.snoop()
def fibonacci(n):
    if n <= 1:
        return n
    return fibonacci(n - 1) + fibonacci(n - 2)

result = fibonacci(5)
```

### print replacement with show:

```python
# direct import (recommended)
from fckprint import show

def fibonacci(n):
    show("calculating fibonacci for", n)
    if n <= 1:
        show("base case:", n)
        return n
    
    result = fibonacci(n - 1) + fibonacci(n - 2)
    show("result:", result)
    return result

# with log levels and prefixes:
show("starting calculation", level="info")
show("cache miss", level="warning", prefix="CACHE")
show("calculation complete", level="success")
```

or using the full import:

```python
import fckprint

def fibonacci(n):
    fckprint.show("calculating fibonacci for", n)
    if n <= 1:
        fckprint.show("base case:", n)
        return n
    
    result = fibonacci(n - 1) + fibonacci(n - 2)
    fckprint.show("result:", result)
    return result
```

output:
```
17:21:32.924559 line        10         lower = min(lst)
new var:....... lower = 262
17:21:32.924657 line        11         upper = max(lst)
new var:....... upper = 900
17:21:32.924677 line        12         mid = (lower + upper) / 2
new var:....... mid = 581.0
17:21:32.924692 line        13         print(lower, mid, upper)
262 581.0 900
elapsed time: 00:00:00.000181
```

## show function

fckprint's print replacement with structured output:

```python
# direct import (recommended)
from fckprint import show

# basic usage
show("hello world")
show("x =", 5, "y =", 10)

# with log levels
show("debug info", level="debug")
show("warning message", level="warning")
show("error occurred", level="error")
show("operation successful", level="success")

# with prefixes for easy filtering
show("database query", prefix="DB", level="info")
show("cache miss", prefix="CACHE", level="warning")
show("user login", prefix="AUTH", level="success")
```

or using the full import:

```python
import fckprint

# basic usage
fckprint.show("hello world")
fckprint.show("x =", 5, "y =", 10)
```

output:
```
[18:08:47.183] INFO hello world
[18:08:47.183] INFO x = 5 y = 10
[18:08:47.183] DEBUG debug info (/path/to/file.py:10)
[18:08:47.183] WARNING warning message
[18:08:47.183] ERROR error occurred
[18:08:47.183] SUCCESS operation successful
[DB] [18:08:47.183] INFO database query
[CACHE] [18:08:47.183] WARNING cache miss
[AUTH] [18:08:47.183] SUCCESS user login
```

### show vs print comparison

```python
# traditional print
print("starting function")
print(f"x = {x}")
print(f"y = {y}")
print("finished function")

# fckprint show (direct import)
from fckprint import show
show("starting function")
show("x =", x)
show("y =", y)
show("finished function")

# or with full import
import fckprint
fckprint.show("starting function")
fckprint.show("x =", x)
fckprint.show("y =", y)
fckprint.show("finished function")
```

advantages of show:
- **timestamps** - every message has precise timing
- **log levels** - info, debug, warning, error, success
- **prefixes** - easy filtering and categorization
- **caller info** - debug level shows file and line number
- **color coding** - different colors for different levels
- **structured output** - consistent format for parsing

## performance monitoring

monitor function execution time and memory usage:

```python
@fckprint.performance_monitor(threshold=0.5, memory_threshold=100)
def expensive_function():
    # function will be monitored for performance issues
    time.sleep(0.2)
    return "result"

result = expensive_function()
```

output:
```
starting var:.. execution_time = 0
starting var:.. memory_usage = 0
starting var:.. performance_warning = ['psutil_not_available']
new var:....... result = result
modified var:.. execution_time = 0.2091982364654541
modified var:.. performance_warning = ['slow: 0.21s > 0.1s']
return value:.. result
elapsed time: 00:00:00.209508
```

## error tracking and retry logic

automatically retry failed functions with exponential backoff:

```python
@fckprint.error_tracker(max_retries=3, log_file="api_errors.log")
def unreliable_network_call(fail_probability=0.3):
    if random.random() < fail_probability:
        raise connectionerror("network timeout")
    return {"status": "success", "data": "important_data"}

result = unreliable_network_call()
```

output:
```
starting var:.. attempt = 0
new var:....... result = {'status': 'success', 'data': 'important_data'}
new var:....... retry_success = false
return value:.. {'status': 'success', 'data': 'important_data'}
elapsed time: 00:00:00.000285
```

## caching and optimization

intelligent caching with ttl and size limits:

```python
@fckprint.cache_monitor(cache_size=50, ttl=600)
def expensive_calculation(x, y):
    # results will be cached for 10 minutes
    time.sleep(0.1)
    return x * y

# first call (cache miss)
result1 = expensive_calculation(5, 10)
# second call (cache hit)
result2 = expensive_calculation(5, 10)
```

output:
```
starting var:.. cache_hit = false
starting var:.. cache_stats = {'hits': 0, 'misses': 1, 'evictions': 0}
new var:....... result = 50
elapsed time: 00:00:00.640305

starting var:.. cache_hit = true
starting var:.. cache_stats = {'hits': 1, 'misses': 1, 'evictions': 0}
return value:.. 50
elapsed time: 00:00:00.000545
```

## thread safety monitoring

detect potential race conditions and high concurrency:

```python
@fckprint.thread_monitor(max_concurrent=3)
def database_operation(operation_id):
    time.sleep(0.1)
    return f"db result for operation {operation_id}"

# simulate concurrent access
import threading
threads = []
for i in range(5):
    thread = threading.thread(target=database_operation, args=(i,))
    threads.append(thread)
    thread.start()

for thread in threads:
    thread.join()
```

output:
```
starting var:.. concurrent_warning = ['high_concurrency: 4 > 3', 'race_condition_risk: 4 instances']
new var:....... result = 'db result for operation 0'
elapsed time: 00:00:00.107264
```

## data validation

validate input and output data against schemas:

```python
@fckprint.validate_data(
    input_schema={'required_args': 2, 'required_kwargs': ['email']},
    output_schema={'type': dict, 'not_none': true}
)
def create_user_profile(name, age, email=none):
    return {'name': name, 'age': age, 'email': email, 'created_at': datetime.now().isoformat()}

# valid call
user1 = create_user_profile('alice', 30, email='alice@example.com')

# invalid call (missing email)
user2 = create_user_profile('bob', 25)
```

output:
```
starting var:.. validation_errors = ['ok']
new var:....... result = {'name': 'alice', 'age': 30, 'email': 'alice@example.com', 'created_at': '2025-08-26t17:21:48.507150'}
elapsed time: 00:00:00.001011

starting var:.. validation_errors = ["missing kwargs: {'email'}"]
new var:....... result = {'name': 'bob', 'age': 25, 'email': none, 'created_at': '2025-08-26t17:21:48.508061'}
elapsed time: 00:00:00.000828
```

## security monitoring

detect potential security threats in function inputs:

```python
@fckprint.security_monitor(check_inputs=true, mask_sensitive=true)
def process_user_data(user_input, password=none):
    return f"processing: {user_input}"

# normal data
result1 = process_user_data("normal user input", password="secret123")

# suspicious data
result2 = process_user_data("select * from users; drop table users;")
```

output:
```
starting var:.. security_warnings = ['ok']
starting var:.. sensitive_data_detected = true
return value:.. 'processing: normal user input'
elapsed time: 00:00:00.002813

starting var:.. security_warnings = ['potential_sql_injection: drop table', 'potential_sql_injection: ;']
starting var:.. input_sanitized = false
return value:.. 'executing: select * from users; drop table users;'
elapsed time: 00:00:00.001514
```

## circuit breaker pattern

prevent cascading failures in distributed systems:

```python
@fckprint.circuit_breaker(failure_threshold=2, recovery_timeout=10)
def external_service_call(should_fail=false):
    if should_fail:
        raise runtimeerror("external service unavailable")
    return "service response"

# successful calls
result1 = external_service_call(should_fail=false)

# failing calls that trigger circuit breaker
try:
    result2 = external_service_call(should_fail=true)
except exception as e:
    print(f"attempt 1 failed: {e}")

try:
    result3 = external_service_call(should_fail=true)
except exception as e:
    print(f"attempt 2 failed: {e}")

# circuit breaker opens
try:
    result4 = external_service_call(should_fail=true)
except exception as e:
    print(f"attempt 3 failed: {e}")
```

output:
```
starting var:.. circuit_open = false
starting var:.. failure_count = 0
return value:.. 'service response'
elapsed time: 00:00:00.000319

starting var:.. failure_count = 1
starting var:.. last_failure_time = 1756243307.7477162
call ended by exception
elapsed time: 00:00:00.000466

starting var:.. failure_count = 2
starting var:.. circuit_open = true
circuit breaker opened for 'external_service_call' after 2 failures
call ended by exception
elapsed time: 00:00:00.000451

starting var:.. circuit_open = true
call ended by exception
elapsed time: 00:00:00.000238
```

## feature flags

enable/disable functions based on environment variables:

```python
@fckprint.feature_flag('new_algorithm', default_enabled=true, environment_var='enable_new_algo')
def new_sorting_algorithm(data):
    print("using new sorting algorithm!")
    return sorted(data, reverse=true)

@fckprint.feature_flag('experimental_feature', default_enabled=false)
def experimental_feature():
    return "experimental result"

# enabled feature
result1 = new_sorting_algorithm([3, 1, 4, 1, 5, 9, 2, 6])

# disabled feature
result2 = experimental_feature()
```

output:
```
feature 'new_algorithm' is enabled, executing 'new_sorting_algorithm'
using new sorting algorithm!
return value:.. [9, 6, 5, 4, 3, 2, 1, 1]
elapsed time: 00:00:00.000253

feature 'experimental_feature' is disabled, skipping 'experimental_feature'
return value:.. none
elapsed time: 00:00:00.000210
```

## audit trail

create compliance audit logs for sensitive operations:

```python
@fckprint.audit_trail(log_file="user_actions.log", include_args=true)
def delete_user(user_id):
    print(f"deleting user {user_id}")
    return f"user {user_id} deleted"

@fckprint.audit_trail(log_file="user_actions.log", include_args=false)
def sensitive_operation():
    print("performing sensitive operation")
    return "operation_completed"

result1 = delete_user(456)
result2 = sensitive_operation()
```

output:
```
starting var:.. audit_logged = true
deleting user 456
return value:.. 'user 456 deleted'
elapsed time: 00:00:00.000902

starting var:.. audit_logged = true
performing sensitive operation
return value:.. 'operation_completed'
elapsed time: 00:00:00.000546
```

## production monitoring

comprehensive monitoring combining multiple decorators:

```python
@fckprint.production_monitor(
    performance_threshold=1.0,
    max_retries=3,
    cache_ttl=600,
    rate_limit=500
)
def critical_api_endpoint(operation_type, data):
    if operation_type == "slow":
        time.sleep(0.6)
    return {
        'operation': operation_type,
        'result': 'processed_9_items',
        'timestamp': datetime.now().isoformat()
    }

result1 = critical_api_endpoint('normal', 'test_data')
result2 = critical_api_endpoint('slow', 'slow_data')
```

output:
```
starting var:.. security_warnings = ['ok']
starting var:.. cache_hit = false
starting var:.. performance_warning = ['ok']
return value:.. {'operation': 'normal', 'result': 'processed_9_items', 'timestamp': '2025-08-26t17:21:48.780563'}
elapsed time: 00:00:00.004234

starting var:.. performance_warning = ['slow: 0.60s > 0.5s']
return value:.. {'operation': 'slow', 'result': 'processed_9_items', 'timestamp': '2025-08-26t17:21:49.388684'}
elapsed time: 00:00:00.608647
```

## advanced features

### custom variable watching

```python
@fckprint.snoop(watch=('x', 'y', 'result'))
def calculate(x, y):
    result = x * y + 10
    return result

calculate(5, 3)
```

### watch explosion for complex objects

```python
@fckprint.snoop(watch_explode=('user', 'config'))
def process_user(user, config):
    # automatically expand all attributes of user and config objects
    return user.name + config.environment
```

### thread information

```python
@fckprint.snoop(thread_info=true)
def threaded_function():
    return "executed in thread"
```

### custom prefixes for easy grepping

```python
@fckprint.snoop(prefix='debug: ')
def debug_function():
    return "debug output"
```

## configuration

### environment variables

```bash
# disable fckprint completely
export fckprint_disabled=1

# set custom log file
export fckprint_log_file=my_app.log

# enable debug mode
export fckprint_debug=1
```

### global settings

```python
import fckprint

# set global configuration
fckprint.set_config(
    max_variable_length=200,
    color=false,
    normalize=true,
    relative_time=true
)
```

## log files

fckprint creates several log files for different purposes:

- `fckprint_errors.log` - error tracking and retry attempts
- `fckprint_audit.log` - audit trail for compliance
- `demo_errors.log` - custom error logs
- `demo_audit.log` - custom audit logs

## tips for production use

1. **combine decorators** for comprehensive monitoring
2. **use environment variables** to control feature flags
3. **adjust thresholds** based on your application needs
4. **monitor log files** for production insights
5. **use caching** for expensive operations
6. **implement circuit breakers** for external services
7. **validate data** at function boundaries
8. **audit sensitive operations** for compliance

## examples

see the `tests/` directory for comprehensive examples:

- `fckprint_advanced_demo.py` - complete demonstration of all features
- `fckprint_custom_decorators.py` - custom decorator examples
- `ml_advanced_examples.py` - machine learning monitoring examples
- `api_debugging_examples.py` - api debugging patterns

## license

mit license - see license file for details

## contributing

contributions are welcome! please read the contributing guidelines and submit pull requests.

## support

for support and questions:
- open an issue on github
- check the documentation
- review the examples in the tests directory
