feat: project skeleton with config, dependencies, and test setup

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
walkpan
2026-03-29 21:26:00 +08:00
parent ce10907619
commit 8b925263a2
8 changed files with 143 additions and 0 deletions

10
.env.example Normal file
View File

@@ -0,0 +1,10 @@
MQTT_HOST=192.168.0.31
MQTT_PORT=1883
MQTT_USERNAME=
MQTT_PASSWORD=
EMQX_API_URL=http://192.168.0.31:18083/api/v5
EMQX_API_KEY=your-api-key
EMQX_API_SECRET=your-secret
DATABASE_URL=sqlite+aiosqlite:///./data/mqtt_home.db
WEB_HOST=0.0.0.0
WEB_PORT=8000

11
.gitignore vendored Normal file
View File

@@ -0,0 +1,11 @@
__pycache__/
*.py[cod]
*.egg-info/
dist/
build/
.venv/
.env
data/
*.db
frontend/node_modules/
frontend/dist/

38
pyproject.toml Normal file
View File

@@ -0,0 +1,38 @@
[build-system]
requires = ["setuptools>=68.0", "wheel"]
build-backend = "setuptools.build_meta"
[project]
name = "mqtt-home"
version = "0.1.0"
description = "轻量级智能家居 MQTT 设备管理系统"
requires-python = ">=3.11"
dependencies = [
"fastapi>=0.115.0",
"uvicorn[standard]>=0.30.0",
"aiomqtt>=2.0.0",
"sqlalchemy[asyncio]>=2.0.0",
"aiosqlite>=0.20.0",
"pydantic-settings>=2.0.0",
"click>=8.1.0",
"httpx>=0.27.0",
"websockets>=12.0",
]
[project.optional-dependencies]
dev = [
"pytest>=8.0.0",
"pytest-asyncio>=0.23.0",
"pytest-httpx>=0.30.0",
"httpx>=0.27.0",
]
[project.scripts]
mqtt-home = "mqtt_home.cli:cli"
[tool.setuptools.packages.find]
where = ["src"]
[tool.pytest.ini_options]
asyncio_mode = "auto"
testpaths = ["tests"]

View File

20
src/mqtt_home/config.py Normal file
View File

@@ -0,0 +1,20 @@
from pydantic_settings import BaseSettings
class Settings(BaseSettings):
mqtt_host: str = "localhost"
mqtt_port: int = 1883
mqtt_username: str = ""
mqtt_password: str = ""
emqx_api_url: str = "http://localhost:18083/api/v5"
emqx_api_key: str = ""
emqx_api_secret: str = ""
database_url: str = "sqlite+aiosqlite:///./data/mqtt_home.db"
web_host: str = "0.0.0.0"
web_port: int = 8000
model_config = {"env_file": ".env", "env_prefix": ""}
def get_settings() -> Settings:
return Settings()

View File

@@ -0,0 +1,5 @@
from sqlalchemy.orm import DeclarativeBase
class Base(DeclarativeBase):
pass

42
tests/conftest.py Normal file
View File

@@ -0,0 +1,42 @@
import pytest
from sqlalchemy.ext.asyncio import create_async_engine, async_sessionmaker, AsyncSession
from sqlalchemy.pool import StaticPool
from mqtt_home.database import Base
from mqtt_home.config import Settings
TEST_DATABASE_URL = "sqlite+aiosqlite:///:memory:"
@pytest.fixture
def settings():
return Settings(
mqtt_host="localhost",
mqtt_port=1883,
emqx_api_url="http://localhost:18083/api/v5",
emqx_api_key="test-key",
emqx_api_secret="test-secret",
database_url=TEST_DATABASE_URL,
)
@pytest.fixture
async def engine():
engine = create_async_engine(
TEST_DATABASE_URL,
connect_args={"check_same_thread": False},
poolclass=StaticPool,
)
async with engine.begin() as conn:
await conn.run_sync(Base.metadata.create_all)
yield engine
async with engine.begin() as conn:
await conn.run_sync(Base.metadata.drop_all)
await engine.dispose()
@pytest.fixture
async def db_session(engine):
async_session = async_sessionmaker(engine, expire_on_commit=False)
async with async_session() as session:
yield session

17
tests/test_config.py Normal file
View File

@@ -0,0 +1,17 @@
import os
from mqtt_home.config import Settings
def test_default_settings():
s = Settings()
assert s.mqtt_host == "localhost"
assert s.mqtt_port == 1883
assert s.web_port == 8000
def test_settings_from_env(monkeypatch):
monkeypatch.setenv("MQTT_HOST", "192.168.1.1")
monkeypatch.setenv("MQTT_PORT", "1884")
s = Settings()
assert s.mqtt_host == "192.168.1.1"
assert s.mqtt_port == 1884