@rsconcept/rstool 0.5.2 → 0.6.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -78,7 +78,7 @@ From `rsconcept/rstool` (or run `powershell -File scripts/dev/LocalDevSetup.ps1`
78
78
  - `npm test`
79
79
  - `npm run build` — produce `dist/` via tsdown
80
80
  - `npm run wrapper` — dev stdio wrapper via `tsx`
81
- - `npm run example:client`, `npm run example:build-schema`, `npm run example:build-rsmodel`
81
+ - `npm run example:client`, `npm run example:build-schema`, `npm run example:build-rsmodel` — see [`examples/README.md`](examples/README.md) for the full list (kinship scripts, session JSON)
82
82
 
83
83
  ## Stdio protocol
84
84
 
@@ -1,6 +1,8 @@
1
1
  # Portal REST API
2
2
 
3
- Справка только для чтения существующих КС, ОСС и RSModel. `rstool` сам не ходит в REST API: он работает с in-memory сессиями.
3
+ Справка для чтения существующих КС, ОСС и RSModel из Portal и переноса КС в `rstool`.
4
+ `rstool` сам не ходит в REST API: он работает с in-memory сессиями.
5
+ Агент получает JSON Portal отдельно, затем переносит конституенты через контракт `rstool`.
4
6
 
5
7
  ## Хосты
6
8
 
@@ -16,17 +18,93 @@
16
18
  - Модель → `GET /api/models/:id`.
17
19
  - OpenAPI JSON → `GET https://api.portal.acconcept.ru/schema`.
18
20
 
21
+ Для UI-ссылки `https://portal.acconcept.ru/rsforms/856` агент должен ходить на:
22
+
23
+ ```text
24
+ GET https://api.portal.acconcept.ru/api/rsforms/856
25
+ GET https://api.portal.acconcept.ru/api/rsforms/856/details
26
+ ```
27
+
28
+ Не переноси UI query/hash вроде `?tab=editor`.
29
+
30
+ ## Доступ
31
+
32
+ - `access_policy: "public"` читается без авторизации.
33
+ - Приватная схема или модель вернёт `403` и не предназначена для агентов.
34
+ - Если нужны только `title`, `alias`, `description`, `owner`, `access_policy`, не запрашивай `/details`.
35
+
36
+ ## Формат `GET /api/rsforms/:id/details`
37
+
38
+ Ответ содержит метаданные объекта библиотеки и массив `items[]` с конституентами.
39
+ Для анализа КС обычно нужны только корневые `title`, `alias`, `description` и поля каждой конституенты.
40
+
41
+ | Portal JSON | `rstool` | Примечание |
42
+ | --------------------- | ------------------------ | ----------------------------------------------------- |
43
+ | `id` | `draft.id` | Стабильный id внутри сессии |
44
+ | `alias` | `draft.alias` | `X1`, `C1`, `S1`, `D1`, `F1`, `P1`, `A1`, `T1` |
45
+ | `cst_type` | `draft.cstType` | Значения совпадают с `CstType` |
46
+ | `definition_formal` | `draft.definitionFormal` | Пустая строка допустима только для `basic`/`constant` |
47
+ | `term_raw` | `draft.term` | Для редактирования сохраняй raw, не resolved |
48
+ | `definition_raw` | `draft.definitionText` | Для редактирования сохраняй raw, не resolved |
49
+ | `convention` | `draft.convention` | Соглашение, не формальная проверка |
50
+ | `term_resolved` | не переносить | Только отображение после resolver |
51
+ | `definition_resolved` | не переносить | Только отображение после resolver |
52
+ | `typification_manual` | обычно не переносить | Переопределение типизации |
53
+ | `term_forms` | обычно не переносить | Словоформы термина |
54
+ | `crucial` | обычно не переносить | UI/Portal-флаг |
55
+ | `value_is_property` | обычно не переносить | Переопределение класса интерпретируемости |
56
+
57
+ `inheritance`, `attribution`, `oss`, `models`, `versions`, `editors` полезны для контекста Portal, но не нужны для базовой проверки RSLang в `rstool`.
58
+
19
59
  ## Как перенести КС в rstool
20
60
 
21
61
  1. Получи `GET /api/rsforms/:id/details`.
22
- 2. Создай `createSession`.
23
- 3. Перенеси конституенты через `addOrUpdateConstituenta`: `alias`, `cstType`, `definitionFormal`, `term`, `definitionText`, `convention`.
24
- 4. Работай локально: анализ, правки, вычисления.
25
- 5. Для передачи результата используй `exportSession`.
62
+ 2. Создай `createSession({ title, alias, comment: description })`.
63
+ 3. Иди по `items[]` в порядке ответа API.
64
+ 4. Для каждого элемента вызови `addOrUpdateConstituenta(sessionId, { draft })`.
65
+ 5. Проверь `listDiagnostics(sessionId)`.
66
+ 6. Работай локально: анализ, правки, вычисления, маленькая КМ.
67
+ 7. Для сохранения рабочей сессии используй `exportSession`.
68
+ 8. Для загрузки схемы обратно в Portal используй `exportPortalSchema`.
69
+
70
+ Portal JSON из `/details` **не является** форматом `exportSession` и не передаётся
71
+ в `importSession` напрямую. Его нужно перенести через `addOrUpdateConstituenta`
72
+ или через отдельный адаптер, если он появится в контракте.
73
+
74
+ Минимальный маппинг:
75
+
76
+ ```ts
77
+ const session = tool.createSession({
78
+ title: portal.title,
79
+ alias: portal.alias,
80
+ comment: portal.description
81
+ });
82
+
83
+ for (const item of portal.items) {
84
+ tool.addOrUpdateConstituenta(session.sessionId, {
85
+ draft: {
86
+ id: item.id,
87
+ alias: item.alias,
88
+ cstType: item.cst_type,
89
+ definitionFormal: item.definition_formal ?? '',
90
+ term: item.term_raw ?? '',
91
+ definitionText: item.definition_raw ?? '',
92
+ convention: item.convention ?? ''
93
+ }
94
+ });
95
+ }
96
+ ```
97
+
98
+ ## Практика для агентов
99
+
100
+ - Запрашивай API host, а не UI host: `https://api.portal.acconcept.ru/api/...`.
101
+ - Для больших схем используй увеличенный таймаут. В PowerShell вызывай `curl.exe`,
102
+ а не alias `curl` (`Invoke-WebRequest`), если нужен CLI-синтаксис curl.
103
+ - Сначала бери метаданные, если нужно только название или доступность схемы.
26
104
 
27
105
  ## Не делай
28
106
 
29
107
  - Не парси HTML SPA.
30
- - Не переноси UI query/hash вроде `?tab=editor` в REST.
31
108
  - Не запрашивай UI напрямую, используй преобразованные API запросы.
109
+ - Не передавай `/details` JSON в `importSession`.
32
110
  - Не запрашивай `/details`, если нужны только метаданные.
@@ -0,0 +1,59 @@
1
+ # Примеры rstool
2
+
3
+ Полные исполняемые сценарии поверх `RSToolWrapperClient` (stdio-обёртка) и готовые JSON-сессии для импорта. Короткие фрагменты кода для агентов — в [`../skills/rstool-helper/EXAMPLES.md`](../skills/rstool-helper/EXAMPLES.md); воркфлоу — [`../skills/rstool-helper/GUIDE.md`](../skills/rstool-helper/GUIDE.md).
4
+
5
+ ## Как запускать
6
+
7
+ **Из checkout репозитория** (`rsconcept/rstool`):
8
+
9
+ ```bash
10
+ npm install
11
+ npm run build # обёртка для скриптов
12
+ npm run example:client # и остальные example:* / kinship:cli — см. ниже
13
+ ```
14
+
15
+ **После `npm install @rsconcept/rstool`** (нужен `tsx` или аналог):
16
+
17
+ ```bash
18
+ npx tsx node_modules/@rsconcept/rstool/examples/agent-client.ts
19
+ ```
20
+
21
+ Скрипты пишут JSON рядом с собой в `examples/`; запускайте из корня пакета (каталог, где лежит `package.json` rstool).
22
+
23
+ ## Скрипты
24
+
25
+ | Файл | npm-скрипт | Назначение |
26
+ |------|------------|------------|
27
+ | [`agent-client.ts`](agent-client.ts) | `example:client` | Минимальный цикл: сессия, upsert, анализ, значения, диагностики через stdio |
28
+ | [`build-sample-rsform.ts`](build-sample-rsform.ts) | `example:build-schema` | Собрать учебную RSForm (X1, S1, D1) и сохранить сессию |
29
+ | [`build-sample-rsmodel.ts`](build-sample-rsmodel.ts) | `example:build-rsmodel` | RSForm + привязки + вычисления → учебная RSModel |
30
+ | [`build-kinship-rsform.ts`](build-kinship-rsform.ts) | `example:build-kinship-schema` | Полная концептуальная схема «родственные отношения» (много конституент) |
31
+ | [`build-kinship-rsmodel.ts`](build-kinship-rsmodel.ts) | `example:build-kinship-rsmodel` | Загрузить kinship RSForm, задать семью, вычислить модель (в т.ч. D3) |
32
+ | [`kinship/cli.ts`](kinship/cli.ts) | `kinship:cli` | Интерактивное редактирование людей (X1) и связей (S1) в kinship-модели |
33
+
34
+ ## JSON-сессии
35
+
36
+ Готовые снимки `exportSession` — для `importSession`, тестов и CLI:
37
+
38
+ | Файл | Содержимое |
39
+ |------|------------|
40
+ | [`sample-rsform-session.json`](sample-rsform-session.json) | Учебная RSForm (результат `build-sample-rsform`) |
41
+ | [`sample-rsmodel-session.json`](sample-rsmodel-session.json) | Учебная RSModel с вычисленными значениями |
42
+ | [`kinship-rsform-session.json`](kinship-rsform-session.json) | Схема родства без привязок |
43
+ | [`kinship-rsmodel-session.json`](kinship-rsmodel-session.json) | Kinship-модель с примерной семьёй (по умолчанию для `kinship:cli`) |
44
+
45
+ Пересобрать JSON: соответствующий `build-*.ts` перезапишет файл в `examples/`.
46
+
47
+ ## Папка `kinship/`
48
+
49
+ Вспомогательный код для сценария «родственные отношения», не отдельный пакет:
50
+
51
+ | Файл | Роль |
52
+ |------|------|
53
+ | [`constants.ts`](kinship/constants.ts) | Id конституент (X1, S1, D3, A1, …) и путь сессии по умолчанию |
54
+ | [`session.ts`](kinship/session.ts) | Обёртка над сессией: загрузка/сохранение, команды над X1 и S1 |
55
+ | [`x1-actions.ts`](kinship/x1-actions.ts) | Парсинг и применение изменений списка людей |
56
+ | [`x1-actions.test.ts`](kinship/x1-actions.test.ts) | Тесты логики X1 (`npm test`) |
57
+ | [`cli.ts`](kinship/cli.ts) | REPL: `list`, `add`, `remove`, `rename`, `set`, `save`, … |
58
+
59
+ См. также [`../docs/CONSTITUENTA.md`](../docs/CONSTITUENTA.md) (разбор kinship-схемы) и [`../docs/MODEL-TESTING.md`](../docs/MODEL-TESTING.md).
@@ -0,0 +1,76 @@
1
+ import { CstType, RSToolWrapperClient, type AddOrUpdateConstituentaInput, type AnalyzeExpressionInput } from '../src';
2
+
3
+ interface CreateSessionResult {
4
+ sessionId: string;
5
+ contractVersion: string;
6
+ }
7
+
8
+ async function runExample() {
9
+ const client = new RSToolWrapperClient();
10
+ try {
11
+ await client.waitUntilReady();
12
+
13
+ const session = await client.call<CreateSessionResult>('createSession');
14
+ console.log('Session created:', session);
15
+
16
+ const baseInput: AddOrUpdateConstituentaInput = {
17
+ draft: {
18
+ id: 1,
19
+ alias: 'X1',
20
+ cstType: CstType.BASE,
21
+ definitionFormal: ''
22
+ }
23
+ };
24
+ await client.call('addOrUpdateConstituenta', {
25
+ sessionId: session.sessionId,
26
+ input: baseInput
27
+ });
28
+
29
+ const upsertInput: AddOrUpdateConstituentaInput = {
30
+ draft: {
31
+ id: 2,
32
+ alias: 'D1',
33
+ cstType: CstType.TERM,
34
+ definitionFormal: '1+2'
35
+ }
36
+ };
37
+ const upsert = await client.call('addOrUpdateConstituenta', {
38
+ sessionId: session.sessionId,
39
+ input: upsertInput
40
+ });
41
+ console.log('Upsert result:', upsert);
42
+
43
+ await client.call('setConstituentaValue', {
44
+ sessionId: session.sessionId,
45
+ input: { target: 1, value: { 0: 'zero', 1: 'one' } }
46
+ });
47
+
48
+ const evalResult = await client.call('evaluateConstituenta', {
49
+ sessionId: session.sessionId,
50
+ input: { constituentId: 2 }
51
+ });
52
+ console.log('Evaluation result:', evalResult);
53
+
54
+ const analyzeInput: AnalyzeExpressionInput = {
55
+ expression: '(',
56
+ cstType: CstType.TERM
57
+ };
58
+ const analysis = await client.call('analyzeExpression', {
59
+ sessionId: session.sessionId,
60
+ input: analyzeInput
61
+ });
62
+ console.log('Analysis result:', analysis);
63
+
64
+ const diagnostics = await client.call('listDiagnostics', {
65
+ sessionId: session.sessionId
66
+ });
67
+ console.log('Diagnostics count:', Array.isArray(diagnostics) ? diagnostics.length : diagnostics);
68
+ } finally {
69
+ await client.close();
70
+ }
71
+ }
72
+
73
+ runExample().catch(error => {
74
+ console.error(error);
75
+ process.exit(1);
76
+ });