[project]
name = "lmtune"
version = "0.1.0"
description = "Given the instructions in the prompt, the agent writes a Python script and executes it."
requires-python = ">=3.11"
dependencies = [
    "openai>=1.0.0",
    "python-dotenv>=1.0.0",
    "prompt_toolkit>=3.0.0",
    "langchain-community>=0.3.18",
    "langchain>=0.3.18",
    "langchain-openai>=0.3.5",
    "langchain-anthropic>=0.3.7",
    "langchain-google-genai>=2.0.9",
    "google-generativeai>=0.7.0",
    "langchain-experimental>=0.3.4",
    "langgraph>=0.2.74",
    "rich>=13.9.4",
    "packaging>=24.2",
    "protobuf>=5.29.4",
    "pydantic-core>=2.33.2",
    "networkx>=3.4.2",
    "numpy>=1.24.0",
    "scipy>=1.10.0",
    "pymzn>=0.18.3",
    "smac>=2.3.1",
]

[project.optional-dependencies]
dev = [
    "pytest>=7.0.0",
    "mypy>=0.971",
    "hatchling>=1.21.0",
    "ruff>=0.1.0"
]

[project.scripts]
lmtune-agent = "lmtune.agent:main"
lmtune-run = "lmtune.run:main"
dzn-to-json = "lmtune.dzn_to_json:main"

[build-system]
requires = ["hatchling>=1.21.0"]
build-backend = "hatchling.build"

[tool.mypy]
python_version = "3.11"
warn_return_any = true
warn_unused_configs = true

[tool.pytest.ini_options]
minversion = "7.0"
addopts = "-q"

[tool.ruff]
# Usage:
# Format code: uv run ruff format .
# Check code: uv run ruff check .
# Fix issues: uv run ruff check --fix .

# Python 3.11+
target-version = "py311"

# Black-compatible
line-length = 88

# Auto-fix issues
fix = true

[tool.ruff.format]
# Black-compatible formatting
quote-style = "double"
indent-style = "space"
docstring-code-format = true

[tool.ruff.lint]
# Essential rules that catch real issues
select = [
    "E",    # pycodestyle errors
    "F",    # Pyflakes
    "I",    # isort (import sorting)
    "UP",   # pyupgrade (Python 3.11+ syntax)
    "B",    # flake8-bugbear (likely bugs)
    "SIM",  # flake8-simplify (code simplification)
    "RUF",  # Ruff-specific rules
]

# Ignore annoying/controversial rules
ignore = [
    "E501",   # Line length (formatter handles this)
    "B008",   # Function calls in defaults
    "SIM108", # Ternary operator (if/else can be clearer)
    "E402",   # Module level import not at top of file (common in test files)
    "E722",   # Do not use bare 'except'
    "F401",   # Imported but unused - many template modules have purposeful imports
    "F841",   # Local variable is assigned but never used
    "SIM105", # Use contextlib.suppress - too many changes needed
    "SIM102", # Use a single if statement - readability preference
    "SIM117", # Use a single with statement - readability preference
    "RUF001", # String contains ambiguous character - intentional emoji usage
    "RUF012", # Mutable class attributes - existing pattern
    "RUF013", # PEP 484 prohibits implicit Optional - backward compatibility
    "UP007",  # Use X | Y for type annotations - backward compatibility
    "B904",   # Raise from err - existing error handling pattern
    "SIM118", # Use key in dict instead of key in dict.keys() - readability preference
    "F403",   # Star imports - used intentionally for re-exports
    "F821",   # Undefined name - used for type annotations with quotes
    "UP038",  # Use X | Y in isinstance - existing pattern
    "SIM101", # Multiple isinstance calls - readability preference
    "B007",   # Loop control variable not used - simple iteration patterns
    "RUF022"  # __all__ is not sorted - custom ordering preferences
]

[tool.ruff.lint.isort]
# Clean import grouping
combine-as-imports = true
lines-after-imports = 2

[tool.ruff.lint.per-file-ignores]
# Tests can use assert and have unused imports
"tests/*" = ["S101", "F401", "F841"]
"__init__.py" = ["F401"]
