init: add contract definitions and database schema with migrations skeleton
This commit is contained in:
13
.gitignore
vendored
13
.gitignore
vendored
@@ -30,3 +30,16 @@ build/
|
||||
sdkconfig
|
||||
sdkconfig.old
|
||||
Kconfig
|
||||
|
||||
.git/
|
||||
|
||||
# ide
|
||||
.idea/
|
||||
.vscode/
|
||||
|
||||
# env
|
||||
.env
|
||||
*.env
|
||||
|
||||
application.yaml
|
||||
application.conf
|
||||
16
backend/src/main/resources/application.conf.example
Normal file
16
backend/src/main/resources/application.conf.example
Normal file
@@ -0,0 +1,16 @@
|
||||
ktor {
|
||||
host = "0.0.0.0"
|
||||
port = 8080
|
||||
}
|
||||
|
||||
app {
|
||||
name = "iot-backend"
|
||||
}
|
||||
|
||||
api {
|
||||
prefix = "/api/v1"
|
||||
}
|
||||
|
||||
ws {
|
||||
path = "/ws"
|
||||
}
|
||||
12
backend/src/main/resources/application.yaml.example
Normal file
12
backend/src/main/resources/application.yaml.example
Normal file
@@ -0,0 +1,12 @@
|
||||
ktor:
|
||||
host: 0.0.0.0
|
||||
port: 8080
|
||||
|
||||
app:
|
||||
name: iot-backend
|
||||
|
||||
api:
|
||||
prefix: /api/v1
|
||||
|
||||
ws:
|
||||
path: /ws
|
||||
17
contract/api/common.schema.json
Normal file
17
contract/api/common.schema.json
Normal file
@@ -0,0 +1,17 @@
|
||||
{
|
||||
"$id": "contract/api/common.schema.json",
|
||||
"type": "object", "required": ["ok"],
|
||||
"properties": {
|
||||
"ok": {
|
||||
"type": "boolean"
|
||||
},
|
||||
"data": {},
|
||||
"error": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"code": { "type": "string" },
|
||||
"message": { "type": "string" }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
0
contract/api/device.schema.json
Normal file
0
contract/api/device.schema.json
Normal file
0
contract/api/stats.schema.json
Normal file
0
contract/api/stats.schema.json
Normal file
18
contract/api/timeseries.schema.json
Normal file
18
contract/api/timeseries.schema.json
Normal file
@@ -0,0 +1,18 @@
|
||||
{
|
||||
"type": "object",
|
||||
"required": ["points"],
|
||||
"properties": {
|
||||
"points": {
|
||||
"type": "array",
|
||||
"items": {
|
||||
"type": "object",
|
||||
"required": ["ts", "value"],
|
||||
"properties": {
|
||||
"ts": { "type": "integer" },
|
||||
"value": {}
|
||||
},
|
||||
"additionalProperties": false
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
37
contract/ingest/common.schema.json
Normal file
37
contract/ingest/common.schema.json
Normal file
@@ -0,0 +1,37 @@
|
||||
{
|
||||
"$schema": "http://json-schema.org/draft-07/schema#",
|
||||
"$id": "contract/ingest/common.schema.json",
|
||||
"type": "object",
|
||||
"required": ["v", "id", "type", "ts", "deviceId", "payload"],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"v": {
|
||||
"type": "integer",
|
||||
"const": 1
|
||||
},
|
||||
"id": {
|
||||
"type": "string",
|
||||
"minLength": 1
|
||||
},
|
||||
"type": {
|
||||
"type": "string",
|
||||
"enum": ["telemetry", "event"]
|
||||
},
|
||||
"ts": {
|
||||
"type": "integer",
|
||||
"minimum": 1600000000000,
|
||||
"description": "Unix time in milliseconds (UTC)"
|
||||
},
|
||||
"deviceId": {
|
||||
"type": "string",
|
||||
"minLength": 1
|
||||
},
|
||||
"payload": {
|
||||
"type": "object"
|
||||
},
|
||||
"meta": {
|
||||
"type": "object",
|
||||
"additionalProperties": true
|
||||
}
|
||||
}
|
||||
}
|
||||
38
contract/ingest/event.schema.json
Normal file
38
contract/ingest/event.schema.json
Normal file
@@ -0,0 +1,38 @@
|
||||
{
|
||||
"$schema": "http://json-schema.org/draft-07/schema#",
|
||||
"$id": "contract/ingest/event.schema.json",
|
||||
"allOf": [
|
||||
{ "$ref": "contract/ingest/common.schema.json" },
|
||||
{
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"type": {
|
||||
"const": "event"
|
||||
},
|
||||
"payload": {
|
||||
"type": "object",
|
||||
"required": ["name"],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"name": {
|
||||
"type": "string",
|
||||
"minLength": 1,
|
||||
"description": "Event identifier, e.g. device.overheat"
|
||||
},
|
||||
"severity": {
|
||||
"type": "string",
|
||||
"enum": ["debug", "info", "warn", "error", "fatal"]
|
||||
},
|
||||
"message": {
|
||||
"type": "string"
|
||||
},
|
||||
"data": {
|
||||
"type": "object",
|
||||
"additionalProperties": true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
46
contract/ingest/telemetry.schema.json
Normal file
46
contract/ingest/telemetry.schema.json
Normal file
@@ -0,0 +1,46 @@
|
||||
{
|
||||
"$schema": "http://json-schema.org/draft-07/schema#",
|
||||
"$id": "contract/ingest/telemetry.schema.json",
|
||||
"allOf": [
|
||||
{ "$ref": "contract/ingest/common.schema.json" },
|
||||
{
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"payload": {
|
||||
"type": "object",
|
||||
"required": ["measurements"],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"measurements": {
|
||||
"type": "array",
|
||||
"minItems": 1,
|
||||
"items": {
|
||||
"type": "object",
|
||||
"required": ["metric", "value"],
|
||||
"additionalProperties": false,
|
||||
"properties": {
|
||||
"metric": {
|
||||
"type": "string",
|
||||
"minLength": 1
|
||||
},
|
||||
"value": {
|
||||
"type": ["number", "string", "boolean"]
|
||||
},
|
||||
"unit": {
|
||||
"type": "string"
|
||||
},
|
||||
"source": {
|
||||
"type": "string"
|
||||
},
|
||||
"ts": {
|
||||
"type": "integer"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
18
contract/readme.md
Normal file
18
contract/readme.md
Normal file
@@ -0,0 +1,18 @@
|
||||
## Контракты
|
||||
|
||||
В системе используются два независимых контракта:
|
||||
|
||||
### 1. Ingest (devices → backend)
|
||||
|
||||
- поток сырых данных
|
||||
- оптимизирован под запись
|
||||
- минимальный размер сообщений
|
||||
- формат: telemetry / event
|
||||
|
||||
### 2. API (backend → mobile)
|
||||
|
||||
- агрегированные данные
|
||||
- оптимизирован под чтение
|
||||
- формат зависит от UI (графики, таблицы, статусы)
|
||||
|
||||
⚠️ Эти контракты НЕ обязаны совпадать
|
||||
4
db/README.md
Normal file
4
db/README.md
Normal file
@@ -0,0 +1,4 @@
|
||||
- все изменения только через migrations
|
||||
- schema.sql = текущее состояние
|
||||
- старые migrations не редактируем
|
||||
- версия базы = schema_version
|
||||
5
db/schema.sql
Normal file
5
db/schema.sql
Normal file
@@ -0,0 +1,5 @@
|
||||
CREATE TABLE schema_version (
|
||||
version INT PRIMARY KEY,
|
||||
applied_at TIMESTAMP NOT NULL,
|
||||
checksum VARCHAR(64) NOT NULL
|
||||
);
|
||||
@@ -1,12 +1,31 @@
|
||||
{
|
||||
"type": "object",
|
||||
"required": ["v", "id", "type", "ts", "deviceId", "payload"],
|
||||
"required": ["v", "type", "ts", "deviceId", "payload"],
|
||||
"properties": {
|
||||
"v": { "type": "integer" },
|
||||
"id": { "type": "string" },
|
||||
"type": { "type": "string" },
|
||||
"ts": { "type": "integer" },
|
||||
"deviceId": { "type": "string" },
|
||||
"payload": { "type": "object" }
|
||||
"v": {
|
||||
"type": "integer",
|
||||
"description": "Protocol version"
|
||||
},
|
||||
|
||||
"type": {
|
||||
"type": "string",
|
||||
"description": "Message type"
|
||||
},
|
||||
|
||||
"ts": {
|
||||
"type": "integer",
|
||||
"description": "Unix time in milliseconds"
|
||||
},
|
||||
|
||||
"deviceId": {
|
||||
"type": "string",
|
||||
"minLength": 1,
|
||||
"description": "Unique device identifier"
|
||||
},
|
||||
|
||||
"payload": {
|
||||
"type": "object",
|
||||
"description": "Message-specific data"
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user