Collaborative session implementation, 1
This commit is contained in:
10
.gitignore
vendored
Normal file
10
.gitignore
vendored
Normal file
@@ -0,0 +1,10 @@
|
||||
# Python-generated files
|
||||
__pycache__/
|
||||
*.py[oc]
|
||||
build/
|
||||
dist/
|
||||
wheels/
|
||||
*.egg-info
|
||||
|
||||
# Virtual environments
|
||||
.venv
|
||||
1
.python-version
Normal file
1
.python-version
Normal file
@@ -0,0 +1 @@
|
||||
3.13
|
||||
3
.vscode/settings.json
vendored
Normal file
3
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
{
|
||||
"python.analysis.typeCheckingMode": "standard"
|
||||
}
|
||||
3
NOTES.md
Normal file
3
NOTES.md
Normal file
@@ -0,0 +1,3 @@
|
||||
To create a Typescript app with Preact
|
||||
|
||||
npm create vite@latest frontend -- --template preact-ts
|
||||
24
frontend/.gitignore
vendored
Normal file
24
frontend/.gitignore
vendored
Normal file
@@ -0,0 +1,24 @@
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
pnpm-debug.log*
|
||||
lerna-debug.log*
|
||||
|
||||
node_modules
|
||||
dist
|
||||
dist-ssr
|
||||
*.local
|
||||
|
||||
# Editor directories and files
|
||||
.vscode/*
|
||||
!.vscode/extensions.json
|
||||
.idea
|
||||
.DS_Store
|
||||
*.suo
|
||||
*.ntvs*
|
||||
*.njsproj
|
||||
*.sln
|
||||
*.sw?
|
||||
12
frontend/index.html
Normal file
12
frontend/index.html
Normal file
@@ -0,0 +1,12 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>frontend</title>
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
<script type="module" src="/src/main.tsx"></script>
|
||||
</body>
|
||||
</html>
|
||||
1838
frontend/package-lock.json
generated
Normal file
1838
frontend/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
24
frontend/package.json
Normal file
24
frontend/package.json
Normal file
@@ -0,0 +1,24 @@
|
||||
{
|
||||
"name": "frontend",
|
||||
"private": true,
|
||||
"version": "0.0.0",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dev": "vite",
|
||||
"build": "tsc -b && vite build",
|
||||
"preview": "vite preview"
|
||||
},
|
||||
"dependencies": {
|
||||
"preact": "^10.27.2",
|
||||
"uuidv7": "^1.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@preact/preset-vite": "^2.10.2",
|
||||
"@types/node": "^24.10.1",
|
||||
"typescript": "~5.9.3",
|
||||
"vite": "^8.0.0-beta.13"
|
||||
},
|
||||
"overrides": {
|
||||
"vite": "^8.0.0-beta.13"
|
||||
}
|
||||
}
|
||||
1
frontend/public/vite.svg
Normal file
1
frontend/public/vite.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
|
||||
|
After Width: | Height: | Size: 1.5 KiB |
25
frontend/src/app.css
Normal file
25
frontend/src/app.css
Normal file
@@ -0,0 +1,25 @@
|
||||
#app {
|
||||
max-width: 1280px;
|
||||
margin: 0 auto;
|
||||
padding: 2rem;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.logo {
|
||||
height: 6em;
|
||||
padding: 1.5em;
|
||||
}
|
||||
.logo:hover {
|
||||
filter: drop-shadow(0 0 2em #646cffaa);
|
||||
}
|
||||
.logo.preact:hover {
|
||||
filter: drop-shadow(0 0 2em #673ab8aa);
|
||||
}
|
||||
|
||||
.card {
|
||||
padding: 2em;
|
||||
}
|
||||
|
||||
.read-the-docs {
|
||||
color: #888;
|
||||
}
|
||||
263
frontend/src/app.tsx
Normal file
263
frontend/src/app.tsx
Normal file
@@ -0,0 +1,263 @@
|
||||
import { useCallback, useEffect, useRef, useState } from 'preact/hooks'
|
||||
import './app.css'
|
||||
import { uuidv7 } from 'uuidv7'
|
||||
import { useSyncExternalStore, version } from 'preact/compat'
|
||||
|
||||
const UPDATE_ENDPOINT = "http://localhost:8000/update"
|
||||
const EVENTS_ENDPOINT = "http://localhost:8000/events"
|
||||
|
||||
type Snapshot = { cursor: number, values: Record<string, RRecord> }
|
||||
type LocalTweak = { cursor: number, tweak: Update };
|
||||
|
||||
type ObjectId = string;
|
||||
type RNull = { type: "null", id: ObjectId };
|
||||
type RUser = { type: "user", id: ObjectId; name: string; }
|
||||
type RMessage = { type: "message", id: ObjectId; sender: string, content: string }
|
||||
type RRecord = RNull | RUser | RMessage;
|
||||
|
||||
type CTrue = { type: "true" }
|
||||
type CEquivalentTo = { type: "equivalent_to", expected: RRecord }
|
||||
type Condition = CTrue | CEquivalentTo;
|
||||
|
||||
type Update = {
|
||||
condition: Condition,
|
||||
new: RRecord,
|
||||
}
|
||||
type UpdateResult = {
|
||||
index: number,
|
||||
}
|
||||
|
||||
class Tracker {
|
||||
#remoteCursor: number
|
||||
#remoteValues: Record<string, RRecord>
|
||||
#localTweaks: LocalTweak[];
|
||||
#mergedSnapshot: Record<string, RRecord>
|
||||
#stopListening: (() => void) | null;
|
||||
#onChangedCallbacks: (() => void)[];
|
||||
|
||||
constructor() {
|
||||
this.#remoteCursor = 0;
|
||||
this.#remoteValues = {};
|
||||
this.#localTweaks = [];
|
||||
this.#mergedSnapshot = {};
|
||||
this.#stopListening = null;
|
||||
this.#onChangedCallbacks = [];
|
||||
}
|
||||
|
||||
start() {
|
||||
if (this.#stopListening != null) {
|
||||
throw "already started!";
|
||||
}
|
||||
let body = async() => {
|
||||
while (true) {
|
||||
if (cancellation.canceled) { return; }
|
||||
let result = await fetch(EVENTS_ENDPOINT + "?" + new URLSearchParams({ start: this.#remoteCursor.toString() }).toString(),
|
||||
{
|
||||
method: "GET",
|
||||
}
|
||||
);
|
||||
let versions: RRecord[] = await result.json();
|
||||
for (let version of versions) {
|
||||
console.log(`Got version: ${JSON.stringify(version)}`)
|
||||
this.#remoteCursor += 1;
|
||||
this.#remoteValues[version.id] = version;
|
||||
}
|
||||
if (versions.length > 0) {
|
||||
console.log("Local tweaks before:");
|
||||
console.log(this.#localTweaks);
|
||||
this.#localTweaks = this.#localTweaks.filter((i) => i.cursor >= this.#remoteCursor);
|
||||
console.log("Local tweaks after:");
|
||||
console.log(this.#localTweaks);
|
||||
this.#rebuildSnapshot();
|
||||
}
|
||||
}
|
||||
}
|
||||
var cancellation = { canceled: false };
|
||||
this.#stopListening = () => { cancellation.canceled = true; };
|
||||
body();
|
||||
}
|
||||
|
||||
stop() {
|
||||
if (this.#stopListening == null) {
|
||||
throw "never started";
|
||||
}
|
||||
this.#stopListening();
|
||||
}
|
||||
|
||||
async sendUpdate(update: Update) {
|
||||
let tweak = {
|
||||
cursor: Number.MAX_SAFE_INTEGER,
|
||||
tweak: update,
|
||||
};
|
||||
this.#localTweaks.push(tweak);
|
||||
this.#rebuildSnapshot();
|
||||
let result = await fetch(UPDATE_ENDPOINT, {
|
||||
method: "POST",
|
||||
body: JSON.stringify(update)
|
||||
});
|
||||
if (result.status == 409) {
|
||||
let ix = this.#localTweaks.indexOf(tweak);
|
||||
if (ix > -1) {
|
||||
this.#localTweaks.splice(ix, 1);
|
||||
} else {
|
||||
throw new Error("tweak unexpectedly disappeared");
|
||||
}
|
||||
this.#rebuildSnapshot();
|
||||
} else {
|
||||
let updateResult: UpdateResult = await result.json();
|
||||
console.log("updateResult: ")
|
||||
console.log(updateResult)
|
||||
tweak.cursor = updateResult.index;
|
||||
this.#localTweaks.sort((a, b) => a.cursor - b.cursor);
|
||||
this.#rebuildSnapshot();
|
||||
}
|
||||
}
|
||||
|
||||
#rebuildSnapshot() {
|
||||
let newSnapshot = {...this.#remoteValues};
|
||||
for (var i of this.#localTweaks) {
|
||||
if (i.tweak.new.type == "null") {
|
||||
delete newSnapshot[i.tweak.new.id]
|
||||
} else {
|
||||
newSnapshot[i.tweak.new.id] = i.tweak.new;
|
||||
}
|
||||
}
|
||||
this.#mergedSnapshot = newSnapshot;
|
||||
for (var cb of this.#onChangedCallbacks) {
|
||||
cb();
|
||||
}
|
||||
}
|
||||
|
||||
getSnapshot() {
|
||||
return this.#mergedSnapshot;
|
||||
}
|
||||
|
||||
onChanged(cb: () => void) {
|
||||
this.#onChangedCallbacks.push(cb);
|
||||
}
|
||||
}
|
||||
|
||||
export function useTrack() {
|
||||
const store = useRef(new Tracker());
|
||||
const subscribe = useCallback(
|
||||
(callback: () => void) => { store.current.onChanged(callback); store.current.start(); return () => store.current.stop(); },
|
||||
[store.current]
|
||||
)
|
||||
const snapshot = useSyncExternalStore(
|
||||
subscribe,
|
||||
() => { return store.current.getSnapshot(); }
|
||||
)
|
||||
const sendUpdate = (update: Update) => store.current.sendUpdate(update);
|
||||
|
||||
return {
|
||||
snapshot,
|
||||
sendUpdate
|
||||
};
|
||||
}
|
||||
|
||||
export function App() {
|
||||
const [identity, setIdentity] = useState<string | null>(null);
|
||||
if (identity == null) {
|
||||
return <>
|
||||
<p>Please set your name!</p>
|
||||
<Editor content="" onDone={setIdentity}></Editor>
|
||||
</>
|
||||
}
|
||||
return <>
|
||||
<Chat name={identity}></Chat>
|
||||
</>
|
||||
}
|
||||
export function Chat({name}: {name: string}) {
|
||||
const [me, setMe] = useState(uuidv7());
|
||||
const {snapshot, sendUpdate} = useTrack()
|
||||
|
||||
useEffect(() => {
|
||||
sendUpdate({
|
||||
condition: { type: "true" },
|
||||
new: {
|
||||
type: "user",
|
||||
id: me,
|
||||
name: name,
|
||||
}
|
||||
})
|
||||
}, [])
|
||||
|
||||
const sendMessage = (message: string) => {
|
||||
sendUpdate({
|
||||
condition: { type: "true" },
|
||||
new: {
|
||||
type: "message",
|
||||
id: uuidv7(),
|
||||
sender: me,
|
||||
content: message,
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
const users = Object.values(snapshot).filter((i) => i.type == "user").sort((a, b) => a.id.localeCompare(b.id));
|
||||
const messages = Object.values(snapshot).filter((i) => i.type == "message").sort((a, b) => a.id.localeCompare(b.id));
|
||||
|
||||
let usersByName: Record<string, string> = {};
|
||||
for (var u of users) {
|
||||
usersByName[u.id] = u.name;
|
||||
}
|
||||
console.log(usersByName);
|
||||
|
||||
return <>
|
||||
<h1>Users</h1>
|
||||
<ul>
|
||||
{users.map((i) => <li key={i.id}>{i.name}</li>)}
|
||||
</ul>
|
||||
<h1>Messages</h1>
|
||||
{messages.map((m) => <Message key={m.id} me={me} message={m} usersByName={usersByName} sendUpdate={sendUpdate}></Message>
|
||||
)}
|
||||
<Editor content={""} onDone={sendMessage}></Editor>
|
||||
</>
|
||||
}
|
||||
|
||||
export function Message({
|
||||
me,
|
||||
message,
|
||||
usersByName,
|
||||
sendUpdate,
|
||||
}: {
|
||||
me: string,
|
||||
message: RMessage,
|
||||
usersByName: Record<string, string>,
|
||||
sendUpdate: (update: Update) => void
|
||||
}) {
|
||||
const [isEditing, setIsEditing] = useState(false)
|
||||
const submitNewContent = (newContent: string) => {
|
||||
sendUpdate(
|
||||
{
|
||||
condition: {type: "true"},
|
||||
new: {
|
||||
type: "message",
|
||||
id: message.id,
|
||||
sender: me,
|
||||
content: newContent,
|
||||
}
|
||||
}
|
||||
);
|
||||
setIsEditing(false);
|
||||
}
|
||||
|
||||
return <div>
|
||||
<span>({usersByName[message.sender]})
|
||||
{isEditing ? <>
|
||||
<Editor content={message.content} onDone={submitNewContent} ></Editor>
|
||||
</> : <>{ message.content }</>}
|
||||
{(message.sender == me) && <a onClick={() => setIsEditing(!isEditing)}>EDIT</a>}
|
||||
</span>
|
||||
</div>;
|
||||
|
||||
}
|
||||
|
||||
export function Editor({ content, onDone }: {content: string, onDone: (content: string) => void }) {
|
||||
const [currentContent, setCurrentContent] = useState(content);
|
||||
return <input onChange={(e) => setCurrentContent(e.currentTarget.value)} onKeyDown={(e) => {
|
||||
if (e.key == "Enter") { onDone(currentContent) ; }
|
||||
setCurrentContent("");
|
||||
}}>
|
||||
</input>
|
||||
}
|
||||
68
frontend/src/index.css
Normal file
68
frontend/src/index.css
Normal file
@@ -0,0 +1,68 @@
|
||||
:root {
|
||||
font-family: system-ui, Avenir, Helvetica, Arial, sans-serif;
|
||||
line-height: 1.5;
|
||||
font-weight: 400;
|
||||
|
||||
color-scheme: light dark;
|
||||
color: rgba(255, 255, 255, 0.87);
|
||||
background-color: #242424;
|
||||
|
||||
font-synthesis: none;
|
||||
text-rendering: optimizeLegibility;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
}
|
||||
|
||||
a {
|
||||
font-weight: 500;
|
||||
color: #646cff;
|
||||
text-decoration: inherit;
|
||||
}
|
||||
a:hover {
|
||||
color: #535bf2;
|
||||
}
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
display: flex;
|
||||
place-items: center;
|
||||
min-width: 320px;
|
||||
min-height: 100vh;
|
||||
}
|
||||
|
||||
h1 {
|
||||
font-size: 3.2em;
|
||||
line-height: 1.1;
|
||||
}
|
||||
|
||||
button {
|
||||
border-radius: 8px;
|
||||
border: 1px solid transparent;
|
||||
padding: 0.6em 1.2em;
|
||||
font-size: 1em;
|
||||
font-weight: 500;
|
||||
font-family: inherit;
|
||||
background-color: #1a1a1a;
|
||||
cursor: pointer;
|
||||
transition: border-color 0.25s;
|
||||
}
|
||||
button:hover {
|
||||
border-color: #646cff;
|
||||
}
|
||||
button:focus,
|
||||
button:focus-visible {
|
||||
outline: 4px auto -webkit-focus-ring-color;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: light) {
|
||||
:root {
|
||||
color: #213547;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
a:hover {
|
||||
color: #747bff;
|
||||
}
|
||||
button {
|
||||
background-color: #f9f9f9;
|
||||
}
|
||||
}
|
||||
5
frontend/src/main.tsx
Normal file
5
frontend/src/main.tsx
Normal file
@@ -0,0 +1,5 @@
|
||||
import { render } from 'preact'
|
||||
import './index.css'
|
||||
import { App } from './app.tsx'
|
||||
|
||||
render(<App />, document.getElementById('app')!)
|
||||
33
frontend/tsconfig.app.json
Normal file
33
frontend/tsconfig.app.json
Normal file
@@ -0,0 +1,33 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
||||
"target": "ES2022",
|
||||
"useDefineForClassFields": true,
|
||||
"module": "ESNext",
|
||||
"lib": ["ES2022", "DOM", "DOM.Iterable"],
|
||||
"types": ["vite/client"],
|
||||
"skipLibCheck": true,
|
||||
"paths": {
|
||||
"react": ["./node_modules/preact/compat/"],
|
||||
"react-dom": ["./node_modules/preact/compat/"]
|
||||
},
|
||||
|
||||
/* Bundler mode */
|
||||
"moduleResolution": "bundler",
|
||||
"allowImportingTsExtensions": true,
|
||||
"verbatimModuleSyntax": true,
|
||||
"moduleDetection": "force",
|
||||
"noEmit": true,
|
||||
"jsx": "react-jsx",
|
||||
"jsxImportSource": "preact",
|
||||
|
||||
/* Linting */
|
||||
"strict": true,
|
||||
"noUnusedLocals": true,
|
||||
"noUnusedParameters": true,
|
||||
"erasableSyntaxOnly": true,
|
||||
"noFallthroughCasesInSwitch": true,
|
||||
"noUncheckedSideEffectImports": true
|
||||
},
|
||||
"include": ["src"]
|
||||
}
|
||||
7
frontend/tsconfig.json
Normal file
7
frontend/tsconfig.json
Normal file
@@ -0,0 +1,7 @@
|
||||
{
|
||||
"files": [],
|
||||
"references": [
|
||||
{ "path": "./tsconfig.app.json" },
|
||||
{ "path": "./tsconfig.node.json" }
|
||||
]
|
||||
}
|
||||
26
frontend/tsconfig.node.json
Normal file
26
frontend/tsconfig.node.json
Normal file
@@ -0,0 +1,26 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo",
|
||||
"target": "ES2023",
|
||||
"lib": ["ES2023"],
|
||||
"module": "ESNext",
|
||||
"types": ["node"],
|
||||
"skipLibCheck": true,
|
||||
|
||||
/* Bundler mode */
|
||||
"moduleResolution": "bundler",
|
||||
"allowImportingTsExtensions": true,
|
||||
"verbatimModuleSyntax": true,
|
||||
"moduleDetection": "force",
|
||||
"noEmit": true,
|
||||
|
||||
/* Linting */
|
||||
"strict": true,
|
||||
"noUnusedLocals": true,
|
||||
"noUnusedParameters": true,
|
||||
"erasableSyntaxOnly": true,
|
||||
"noFallthroughCasesInSwitch": true,
|
||||
"noUncheckedSideEffectImports": true
|
||||
},
|
||||
"include": ["vite.config.ts"]
|
||||
}
|
||||
7
frontend/vite.config.ts
Normal file
7
frontend/vite.config.ts
Normal file
@@ -0,0 +1,7 @@
|
||||
import { defineConfig } from 'vite'
|
||||
import preact from '@preact/preset-vite'
|
||||
|
||||
// https://vite.dev/config/
|
||||
export default defineConfig({
|
||||
plugins: [preact()],
|
||||
})
|
||||
11
pyproject.toml
Normal file
11
pyproject.toml
Normal file
@@ -0,0 +1,11 @@
|
||||
[project]
|
||||
name = "syncing-1"
|
||||
version = "0.1.0"
|
||||
description = "Add your description here"
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.13"
|
||||
dependencies = [
|
||||
"pydantic>=2.12.5",
|
||||
"starlette>=0.52.1",
|
||||
"uvicorn>=0.40.0",
|
||||
]
|
||||
94
synctoy/data_model.py
Normal file
94
synctoy/data_model.py
Normal file
@@ -0,0 +1,94 @@
|
||||
from enum import Enum, StrEnum
|
||||
from typing import Literal, NewType
|
||||
|
||||
from pydantic import BaseModel, JsonValue
|
||||
|
||||
from synctoy.state_machine import StateMachine
|
||||
|
||||
|
||||
class Type(StrEnum):
|
||||
Null = "null"
|
||||
User = "user"
|
||||
Message = "message"
|
||||
|
||||
ObjectId = NewType("ObjectId", str)
|
||||
|
||||
class RNull(BaseModel):
|
||||
type: Literal[Type.Null] = Type.Null
|
||||
id: ObjectId
|
||||
|
||||
def equivalent_to(self, other: StateMachine):
|
||||
return self == other
|
||||
|
||||
def can_transition_from(self, old: StateMachine):
|
||||
return True
|
||||
|
||||
def can_transition_to(self, new: StateMachine):
|
||||
return True
|
||||
|
||||
class RUser(BaseModel):
|
||||
type: Literal[Type.User] = Type.User
|
||||
id: ObjectId
|
||||
name: str
|
||||
|
||||
def equivalent_to(self, other: StateMachine):
|
||||
return self == other
|
||||
|
||||
def can_transition_from(self, old: StateMachine):
|
||||
if isinstance(old, RNull):
|
||||
return True
|
||||
return False
|
||||
|
||||
def can_transition_to(self, new: StateMachine):
|
||||
return False
|
||||
|
||||
class RMessage(BaseModel):
|
||||
type: Literal[Type.Message] = Type.Message
|
||||
id: ObjectId
|
||||
sender: ObjectId
|
||||
content: str
|
||||
|
||||
def equivalent_to(self, other: StateMachine):
|
||||
return self == other
|
||||
|
||||
def can_transition_from(self, old: StateMachine):
|
||||
if isinstance(old, RNull):
|
||||
return True
|
||||
if isinstance(old, RMessage):
|
||||
return True
|
||||
return False
|
||||
|
||||
def can_transition_to(self, new: StateMachine):
|
||||
if isinstance(new, RMessage):
|
||||
return new.sender == self.sender
|
||||
return True
|
||||
|
||||
NonNullRecord = RUser | RMessage
|
||||
Record = RNull | NonNullRecord
|
||||
|
||||
|
||||
|
||||
def as_state_machine(r: Record) -> StateMachine:
|
||||
return r
|
||||
|
||||
|
||||
class ConditionType(StrEnum):
|
||||
True_ = "true"
|
||||
EquivalentTo = "equivalent_to"
|
||||
|
||||
|
||||
class CTrue(BaseModel):
|
||||
type: Literal[ConditionType.True_] = ConditionType.True_
|
||||
|
||||
def is_met(self, old: Record, new: Record):
|
||||
return True
|
||||
|
||||
class CEquivalentTo(BaseModel):
|
||||
type: Literal[ConditionType.EquivalentTo] = ConditionType.EquivalentTo
|
||||
|
||||
expected: Record
|
||||
|
||||
def is_met(self, old: Record, new: Record):
|
||||
return old.equivalent_to(self.expected)
|
||||
|
||||
Condition = CTrue | CEquivalentTo
|
||||
55
synctoy/main.py
Normal file
55
synctoy/main.py
Normal file
@@ -0,0 +1,55 @@
|
||||
import asyncio
|
||||
import json
|
||||
from pydantic import BaseModel, TypeAdapter
|
||||
from starlette.applications import Starlette
|
||||
from starlette.middleware import Middleware
|
||||
from starlette.middleware.cors import CORSMiddleware
|
||||
from starlette.middleware.trustedhost import TrustedHostMiddleware
|
||||
from starlette.requests import Request
|
||||
from starlette.responses import JSONResponse, Response
|
||||
from starlette.routing import Route
|
||||
|
||||
from synctoy.data_model import ObjectId, NonNullRecord, Record
|
||||
from synctoy.snapshot import SAll
|
||||
from synctoy.store import Store, Update
|
||||
|
||||
|
||||
store = Store()
|
||||
|
||||
_TIMEOUT = 5.0
|
||||
|
||||
async def snapshot(request: Request):
|
||||
snapshot = store.snapshot(SAll())
|
||||
populated = snapshot.populated_records()
|
||||
return JSONResponse(TypeAdapter(dict[ObjectId, NonNullRecord]).dump_python(populated, mode="json"))
|
||||
|
||||
class UpdateResult(BaseModel):
|
||||
index: int
|
||||
|
||||
async def update(request: Request):
|
||||
value = Update.model_validate_json(await request.body())
|
||||
version = store.update([value])
|
||||
return JSONResponse(UpdateResult(index=version).model_dump(mode="json"))
|
||||
|
||||
async def events(request: Request):
|
||||
start = request.query_params.get("start")
|
||||
if start is None:
|
||||
start = 0
|
||||
else:
|
||||
start = int(start)
|
||||
|
||||
result = store.observe(start_from=start)
|
||||
if len(result) == 0:
|
||||
await store.wait(_TIMEOUT)
|
||||
result = store.observe(start_from=start)
|
||||
|
||||
await asyncio.sleep(1.0) # simulated delay! wow
|
||||
return JSONResponse(TypeAdapter(list[Record]).dump_python(result))
|
||||
|
||||
app = Starlette(debug=True, routes=[
|
||||
Route("/snapshot", snapshot, methods=["GET"]),
|
||||
Route("/update", update, methods=["POST"]),
|
||||
Route("/events", events, methods=["GET"])
|
||||
],
|
||||
)
|
||||
app = CORSMiddleware(app=app, allow_origins=["*"])
|
||||
43
synctoy/snapshot.py
Normal file
43
synctoy/snapshot.py
Normal file
@@ -0,0 +1,43 @@
|
||||
from typing import Protocol
|
||||
|
||||
from synctoy.data_model import NonNullRecord, ObjectId, RNull, Record
|
||||
|
||||
|
||||
class Selector(Protocol):
|
||||
def includes_object_id(self, object_id: ObjectId):
|
||||
raise NotImplementedError
|
||||
|
||||
|
||||
class SAll(object):
|
||||
def includes_object_id(self, object_id: ObjectId):
|
||||
return True
|
||||
|
||||
class SList(object):
|
||||
def __init__(self, object_ids: list[ObjectId]):
|
||||
self._object_ids = object_ids
|
||||
|
||||
def includes_object_id(self, object_id: ObjectId):
|
||||
return object_id in self._object_ids
|
||||
|
||||
class Snapshot(object):
|
||||
def __init__(self, selector: Selector, data: dict[ObjectId, NonNullRecord]):
|
||||
self._selector = selector
|
||||
self._data = data
|
||||
|
||||
def __getitem__(self, object_id: ObjectId):
|
||||
if not self._selector.includes_object_id(object_id):
|
||||
raise KeyError(f"{object_id} was not selected and cannot be viewed")
|
||||
|
||||
return self._data.get(object_id) or RNull(id=object_id)
|
||||
|
||||
def __setitem__(self, object_id: ObjectId, value: Record):
|
||||
if not self._selector.includes_object_id(object_id):
|
||||
raise KeyError(f"{object_id} was not selected and cannot be staged")
|
||||
|
||||
if isinstance(value, RNull):
|
||||
self._data.pop(object_id, None)
|
||||
else:
|
||||
self._data[object_id] = value
|
||||
|
||||
def populated_records(self) -> dict[ObjectId, NonNullRecord]:
|
||||
return dict(self._data)
|
||||
12
synctoy/state_machine.py
Normal file
12
synctoy/state_machine.py
Normal file
@@ -0,0 +1,12 @@
|
||||
from typing import Protocol
|
||||
|
||||
|
||||
class StateMachine(Protocol):
|
||||
def equivalent_to(self, other: "StateMachine") -> bool:
|
||||
raise NotImplementedError
|
||||
|
||||
def can_transition_from(self, old: "StateMachine") -> bool:
|
||||
raise NotImplementedError
|
||||
|
||||
def can_transition_to(self, new: "StateMachine") -> bool:
|
||||
raise NotImplementedError
|
||||
65
synctoy/store.py
Normal file
65
synctoy/store.py
Normal file
@@ -0,0 +1,65 @@
|
||||
import asyncio
|
||||
from typing import NewType, Protocol
|
||||
from pydantic import BaseModel, JsonValue
|
||||
|
||||
from synctoy.data_model import Condition, NonNullRecord, ObjectId, RNull, Record
|
||||
from synctoy.snapshot import SList, Selector, Snapshot
|
||||
|
||||
class Update(BaseModel):
|
||||
condition: Condition
|
||||
new: Record
|
||||
|
||||
class PreconditionException(Exception):
|
||||
pass
|
||||
|
||||
class Store(object):
|
||||
def __init__(self):
|
||||
self._event = asyncio.Event()
|
||||
self._versions: list[Record] = []
|
||||
|
||||
async def wait(self, max_timeout: float):
|
||||
try:
|
||||
await asyncio.wait_for(self._event.wait(), max_timeout)
|
||||
except TimeoutError:
|
||||
return
|
||||
|
||||
def observe(self, start_from: int):
|
||||
return self._versions[start_from:]
|
||||
|
||||
def update(self, updates: list[Update]) -> int:
|
||||
object_ids: set[ObjectId] = set(u.new.id for u in updates)
|
||||
snapshot: Snapshot = self.snapshot(SList(object_ids=list(object_ids)))
|
||||
|
||||
for update in updates:
|
||||
id = update.new.id
|
||||
old_object = snapshot[id]
|
||||
new_object = update.new
|
||||
|
||||
if not old_object.can_transition_to(new_object):
|
||||
raise PreconditionException(f"can't transition from {old_object} to {new_object}")
|
||||
if not new_object.can_transition_from(old_object):
|
||||
raise PreconditionException(f"can't transition from {old_object} to {new_object}")
|
||||
|
||||
if not update.condition.is_met(old_object, new_object):
|
||||
raise PreconditionException(f"failed condition: {update.condition}")
|
||||
|
||||
snapshot[id] = new_object
|
||||
|
||||
for new_update in updates:
|
||||
self._versions.append(new_update.new)
|
||||
self._event.set()
|
||||
self._event = asyncio.Event()
|
||||
return len(self._versions)
|
||||
|
||||
def snapshot(self, selector: Selector) -> Snapshot:
|
||||
pre_snapshot: dict[ObjectId, Record] = {}
|
||||
for row in self._versions:
|
||||
if not selector.includes_object_id(row.id):
|
||||
continue
|
||||
|
||||
pre_snapshot[row.id] = row
|
||||
filtered: dict[ObjectId, NonNullRecord] = {
|
||||
id: row for id, row in pre_snapshot.items() if not isinstance(row, RNull)
|
||||
}
|
||||
|
||||
return Snapshot(selector, filtered)
|
||||
194
uv.lock
generated
Normal file
194
uv.lock
generated
Normal file
@@ -0,0 +1,194 @@
|
||||
version = 1
|
||||
revision = 2
|
||||
requires-python = ">=3.13"
|
||||
|
||||
[[package]]
|
||||
name = "annotated-types"
|
||||
version = "0.7.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/ee/67/531ea369ba64dcff5ec9c3402f9f51bf748cec26dde048a2f973a4eea7f5/annotated_types-0.7.0.tar.gz", hash = "sha256:aff07c09a53a08bc8cfccb9c85b05f1aa9a2a6f23728d790723543408344ce89", size = 16081, upload-time = "2024-05-20T21:33:25.928Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/78/b6/6307fbef88d9b5ee7421e68d78a9f162e0da4900bc5f5793f6d3d0e34fb8/annotated_types-0.7.0-py3-none-any.whl", hash = "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53", size = 13643, upload-time = "2024-05-20T21:33:24.1Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anyio"
|
||||
version = "4.12.1"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "idna" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/96/f0/5eb65b2bb0d09ac6776f2eb54adee6abe8228ea05b20a5ad0e4945de8aac/anyio-4.12.1.tar.gz", hash = "sha256:41cfcc3a4c85d3f05c932da7c26d0201ac36f72abd4435ba90d0464a3ffed703", size = 228685, upload-time = "2026-01-06T11:45:21.246Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/38/0e/27be9fdef66e72d64c0cdc3cc2823101b80585f8119b5c112c2e8f5f7dab/anyio-4.12.1-py3-none-any.whl", hash = "sha256:d405828884fc140aa80a3c667b8beed277f1dfedec42ba031bd6ac3db606ab6c", size = 113592, upload-time = "2026-01-06T11:45:19.497Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "click"
|
||||
version = "8.3.1"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "colorama", marker = "sys_platform == 'win32'" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/3d/fa/656b739db8587d7b5dfa22e22ed02566950fbfbcdc20311993483657a5c0/click-8.3.1.tar.gz", hash = "sha256:12ff4785d337a1bb490bb7e9c2b1ee5da3112e94a8622f26a6c77f5d2fc6842a", size = 295065, upload-time = "2025-11-15T20:45:42.706Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/98/78/01c019cdb5d6498122777c1a43056ebb3ebfeef2076d9d026bfe15583b2b/click-8.3.1-py3-none-any.whl", hash = "sha256:981153a64e25f12d547d3426c367a4857371575ee7ad18df2a6183ab0545b2a6", size = 108274, upload-time = "2025-11-15T20:45:41.139Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "colorama"
|
||||
version = "0.4.6"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "h11"
|
||||
version = "0.16.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/01/ee/02a2c011bdab74c6fb3c75474d40b3052059d95df7e73351460c8588d963/h11-0.16.0.tar.gz", hash = "sha256:4e35b956cf45792e4caa5885e69fba00bdbc6ffafbfa020300e549b208ee5ff1", size = 101250, upload-time = "2025-04-24T03:35:25.427Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/04/4b/29cac41a4d98d144bf5f6d33995617b185d14b22401f75ca86f384e87ff1/h11-0.16.0-py3-none-any.whl", hash = "sha256:63cf8bbe7522de3bf65932fda1d9c2772064ffb3dae62d55932da54b31cb6c86", size = 37515, upload-time = "2025-04-24T03:35:24.344Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "idna"
|
||||
version = "3.11"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/6f/6d/0703ccc57f3a7233505399edb88de3cbd678da106337b9fcde432b65ed60/idna-3.11.tar.gz", hash = "sha256:795dafcc9c04ed0c1fb032c2aa73654d8e8c5023a7df64a53f39190ada629902", size = 194582, upload-time = "2025-10-12T14:55:20.501Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/0e/61/66938bbb5fc52dbdf84594873d5b51fb1f7c7794e9c0f5bd885f30bc507b/idna-3.11-py3-none-any.whl", hash = "sha256:771a87f49d9defaf64091e6e6fe9c18d4833f140bd19464795bc32d966ca37ea", size = 71008, upload-time = "2025-10-12T14:55:18.883Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pydantic"
|
||||
version = "2.12.5"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "annotated-types" },
|
||||
{ name = "pydantic-core" },
|
||||
{ name = "typing-extensions" },
|
||||
{ name = "typing-inspection" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/69/44/36f1a6e523abc58ae5f928898e4aca2e0ea509b5aa6f6f392a5d882be928/pydantic-2.12.5.tar.gz", hash = "sha256:4d351024c75c0f085a9febbb665ce8c0c6ec5d30e903bdb6394b7ede26aebb49", size = 821591, upload-time = "2025-11-26T15:11:46.471Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/5a/87/b70ad306ebb6f9b585f114d0ac2137d792b48be34d732d60e597c2f8465a/pydantic-2.12.5-py3-none-any.whl", hash = "sha256:e561593fccf61e8a20fc46dfc2dfe075b8be7d0188df33f221ad1f0139180f9d", size = 463580, upload-time = "2025-11-26T15:11:44.605Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pydantic-core"
|
||||
version = "2.41.5"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "typing-extensions" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/71/70/23b021c950c2addd24ec408e9ab05d59b035b39d97cdc1130e1bce647bb6/pydantic_core-2.41.5.tar.gz", hash = "sha256:08daa51ea16ad373ffd5e7606252cc32f07bc72b28284b6bc9c6df804816476e", size = 460952, upload-time = "2025-11-04T13:43:49.098Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/87/06/8806241ff1f70d9939f9af039c6c35f2360cf16e93c2ca76f184e76b1564/pydantic_core-2.41.5-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:941103c9be18ac8daf7b7adca8228f8ed6bb7a1849020f643b3a14d15b1924d9", size = 2120403, upload-time = "2025-11-04T13:40:25.248Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/94/02/abfa0e0bda67faa65fef1c84971c7e45928e108fe24333c81f3bfe35d5f5/pydantic_core-2.41.5-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:112e305c3314f40c93998e567879e887a3160bb8689ef3d2c04b6cc62c33ac34", size = 1896206, upload-time = "2025-11-04T13:40:27.099Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/15/df/a4c740c0943e93e6500f9eb23f4ca7ec9bf71b19e608ae5b579678c8d02f/pydantic_core-2.41.5-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0cbaad15cb0c90aa221d43c00e77bb33c93e8d36e0bf74760cd00e732d10a6a0", size = 1919307, upload-time = "2025-11-04T13:40:29.806Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9a/e3/6324802931ae1d123528988e0e86587c2072ac2e5394b4bc2bc34b61ff6e/pydantic_core-2.41.5-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:03ca43e12fab6023fc79d28ca6b39b05f794ad08ec2feccc59a339b02f2b3d33", size = 2063258, upload-time = "2025-11-04T13:40:33.544Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/c9/d4/2230d7151d4957dd79c3044ea26346c148c98fbf0ee6ebd41056f2d62ab5/pydantic_core-2.41.5-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dc799088c08fa04e43144b164feb0c13f9a0bc40503f8df3e9fde58a3c0c101e", size = 2214917, upload-time = "2025-11-04T13:40:35.479Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e6/9f/eaac5df17a3672fef0081b6c1bb0b82b33ee89aa5cec0d7b05f52fd4a1fa/pydantic_core-2.41.5-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:97aeba56665b4c3235a0e52b2c2f5ae9cd071b8a8310ad27bddb3f7fb30e9aa2", size = 2332186, upload-time = "2025-11-04T13:40:37.436Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/cf/4e/35a80cae583a37cf15604b44240e45c05e04e86f9cfd766623149297e971/pydantic_core-2.41.5-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:406bf18d345822d6c21366031003612b9c77b3e29ffdb0f612367352aab7d586", size = 2073164, upload-time = "2025-11-04T13:40:40.289Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bf/e3/f6e262673c6140dd3305d144d032f7bd5f7497d3871c1428521f19f9efa2/pydantic_core-2.41.5-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:b93590ae81f7010dbe380cdeab6f515902ebcbefe0b9327cc4804d74e93ae69d", size = 2179146, upload-time = "2025-11-04T13:40:42.809Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/75/c7/20bd7fc05f0c6ea2056a4565c6f36f8968c0924f19b7d97bbfea55780e73/pydantic_core-2.41.5-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:01a3d0ab748ee531f4ea6c3e48ad9dac84ddba4b0d82291f87248f2f9de8d740", size = 2137788, upload-time = "2025-11-04T13:40:44.752Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3a/8d/34318ef985c45196e004bc46c6eab2eda437e744c124ef0dbe1ff2c9d06b/pydantic_core-2.41.5-cp313-cp313-musllinux_1_1_armv7l.whl", hash = "sha256:6561e94ba9dacc9c61bce40e2d6bdc3bfaa0259d3ff36ace3b1e6901936d2e3e", size = 2340133, upload-time = "2025-11-04T13:40:46.66Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9c/59/013626bf8c78a5a5d9350d12e7697d3d4de951a75565496abd40ccd46bee/pydantic_core-2.41.5-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:915c3d10f81bec3a74fbd4faebe8391013ba61e5a1a8d48c4455b923bdda7858", size = 2324852, upload-time = "2025-11-04T13:40:48.575Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/1a/d9/c248c103856f807ef70c18a4f986693a46a8ffe1602e5d361485da502d20/pydantic_core-2.41.5-cp313-cp313-win32.whl", hash = "sha256:650ae77860b45cfa6e2cdafc42618ceafab3a2d9a3811fcfbd3bbf8ac3c40d36", size = 1994679, upload-time = "2025-11-04T13:40:50.619Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9e/8b/341991b158ddab181cff136acd2552c9f35bd30380422a639c0671e99a91/pydantic_core-2.41.5-cp313-cp313-win_amd64.whl", hash = "sha256:79ec52ec461e99e13791ec6508c722742ad745571f234ea6255bed38c6480f11", size = 2019766, upload-time = "2025-11-04T13:40:52.631Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/73/7d/f2f9db34af103bea3e09735bb40b021788a5e834c81eedb541991badf8f5/pydantic_core-2.41.5-cp313-cp313-win_arm64.whl", hash = "sha256:3f84d5c1b4ab906093bdc1ff10484838aca54ef08de4afa9de0f5f14d69639cd", size = 1981005, upload-time = "2025-11-04T13:40:54.734Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ea/28/46b7c5c9635ae96ea0fbb779e271a38129df2550f763937659ee6c5dbc65/pydantic_core-2.41.5-cp314-cp314-macosx_10_12_x86_64.whl", hash = "sha256:3f37a19d7ebcdd20b96485056ba9e8b304e27d9904d233d7b1015db320e51f0a", size = 2119622, upload-time = "2025-11-04T13:40:56.68Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/74/1a/145646e5687e8d9a1e8d09acb278c8535ebe9e972e1f162ed338a622f193/pydantic_core-2.41.5-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:1d1d9764366c73f996edd17abb6d9d7649a7eb690006ab6adbda117717099b14", size = 1891725, upload-time = "2025-11-04T13:40:58.807Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/23/04/e89c29e267b8060b40dca97bfc64a19b2a3cf99018167ea1677d96368273/pydantic_core-2.41.5-cp314-cp314-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:25e1c2af0fce638d5f1988b686f3b3ea8cd7de5f244ca147c777769e798a9cd1", size = 1915040, upload-time = "2025-11-04T13:41:00.853Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/84/a3/15a82ac7bd97992a82257f777b3583d3e84bdb06ba6858f745daa2ec8a85/pydantic_core-2.41.5-cp314-cp314-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:506d766a8727beef16b7adaeb8ee6217c64fc813646b424d0804d67c16eddb66", size = 2063691, upload-time = "2025-11-04T13:41:03.504Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/74/9b/0046701313c6ef08c0c1cf0e028c67c770a4e1275ca73131563c5f2a310a/pydantic_core-2.41.5-cp314-cp314-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4819fa52133c9aa3c387b3328f25c1facc356491e6135b459f1de698ff64d869", size = 2213897, upload-time = "2025-11-04T13:41:05.804Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/8a/cd/6bac76ecd1b27e75a95ca3a9a559c643b3afcd2dd62086d4b7a32a18b169/pydantic_core-2.41.5-cp314-cp314-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2b761d210c9ea91feda40d25b4efe82a1707da2ef62901466a42492c028553a2", size = 2333302, upload-time = "2025-11-04T13:41:07.809Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4c/d2/ef2074dc020dd6e109611a8be4449b98cd25e1b9b8a303c2f0fca2f2bcf7/pydantic_core-2.41.5-cp314-cp314-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:22f0fb8c1c583a3b6f24df2470833b40207e907b90c928cc8d3594b76f874375", size = 2064877, upload-time = "2025-11-04T13:41:09.827Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/18/66/e9db17a9a763d72f03de903883c057b2592c09509ccfe468187f2a2eef29/pydantic_core-2.41.5-cp314-cp314-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2782c870e99878c634505236d81e5443092fba820f0373997ff75f90f68cd553", size = 2180680, upload-time = "2025-11-04T13:41:12.379Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d3/9e/3ce66cebb929f3ced22be85d4c2399b8e85b622db77dad36b73c5387f8f8/pydantic_core-2.41.5-cp314-cp314-musllinux_1_1_aarch64.whl", hash = "sha256:0177272f88ab8312479336e1d777f6b124537d47f2123f89cb37e0accea97f90", size = 2138960, upload-time = "2025-11-04T13:41:14.627Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/a6/62/205a998f4327d2079326b01abee48e502ea739d174f0a89295c481a2272e/pydantic_core-2.41.5-cp314-cp314-musllinux_1_1_armv7l.whl", hash = "sha256:63510af5e38f8955b8ee5687740d6ebf7c2a0886d15a6d65c32814613681bc07", size = 2339102, upload-time = "2025-11-04T13:41:16.868Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3c/0d/f05e79471e889d74d3d88f5bd20d0ed189ad94c2423d81ff8d0000aab4ff/pydantic_core-2.41.5-cp314-cp314-musllinux_1_1_x86_64.whl", hash = "sha256:e56ba91f47764cc14f1daacd723e3e82d1a89d783f0f5afe9c364b8bb491ccdb", size = 2326039, upload-time = "2025-11-04T13:41:18.934Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/ec/e1/e08a6208bb100da7e0c4b288eed624a703f4d129bde2da475721a80cab32/pydantic_core-2.41.5-cp314-cp314-win32.whl", hash = "sha256:aec5cf2fd867b4ff45b9959f8b20ea3993fc93e63c7363fe6851424c8a7e7c23", size = 1995126, upload-time = "2025-11-04T13:41:21.418Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/48/5d/56ba7b24e9557f99c9237e29f5c09913c81eeb2f3217e40e922353668092/pydantic_core-2.41.5-cp314-cp314-win_amd64.whl", hash = "sha256:8e7c86f27c585ef37c35e56a96363ab8de4e549a95512445b85c96d3e2f7c1bf", size = 2015489, upload-time = "2025-11-04T13:41:24.076Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/4e/bb/f7a190991ec9e3e0ba22e4993d8755bbc4a32925c0b5b42775c03e8148f9/pydantic_core-2.41.5-cp314-cp314-win_arm64.whl", hash = "sha256:e672ba74fbc2dc8eea59fb6d4aed6845e6905fc2a8afe93175d94a83ba2a01a0", size = 1977288, upload-time = "2025-11-04T13:41:26.33Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/92/ed/77542d0c51538e32e15afe7899d79efce4b81eee631d99850edc2f5e9349/pydantic_core-2.41.5-cp314-cp314t-macosx_10_12_x86_64.whl", hash = "sha256:8566def80554c3faa0e65ac30ab0932b9e3a5cd7f8323764303d468e5c37595a", size = 2120255, upload-time = "2025-11-04T13:41:28.569Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/bb/3d/6913dde84d5be21e284439676168b28d8bbba5600d838b9dca99de0fad71/pydantic_core-2.41.5-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:b80aa5095cd3109962a298ce14110ae16b8c1aece8b72f9dafe81cf597ad80b3", size = 1863760, upload-time = "2025-11-04T13:41:31.055Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/5a/f0/e5e6b99d4191da102f2b0eb9687aaa7f5bea5d9964071a84effc3e40f997/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3006c3dd9ba34b0c094c544c6006cc79e87d8612999f1a5d43b769b89181f23c", size = 1878092, upload-time = "2025-11-04T13:41:33.21Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/71/48/36fb760642d568925953bcc8116455513d6e34c4beaa37544118c36aba6d/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:72f6c8b11857a856bcfa48c86f5368439f74453563f951e473514579d44aa612", size = 2053385, upload-time = "2025-11-04T13:41:35.508Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/20/25/92dc684dd8eb75a234bc1c764b4210cf2646479d54b47bf46061657292a8/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5cb1b2f9742240e4bb26b652a5aeb840aa4b417c7748b6f8387927bc6e45e40d", size = 2218832, upload-time = "2025-11-04T13:41:37.732Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/e2/09/f53e0b05023d3e30357d82eb35835d0f6340ca344720a4599cd663dca599/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bd3d54f38609ff308209bd43acea66061494157703364ae40c951f83ba99a1a9", size = 2327585, upload-time = "2025-11-04T13:41:40Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/aa/4e/2ae1aa85d6af35a39b236b1b1641de73f5a6ac4d5a7509f77b814885760c/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2ff4321e56e879ee8d2a879501c8e469414d948f4aba74a2d4593184eb326660", size = 2041078, upload-time = "2025-11-04T13:41:42.323Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/cd/13/2e215f17f0ef326fc72afe94776edb77525142c693767fc347ed6288728d/pydantic_core-2.41.5-cp314-cp314t-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:d0d2568a8c11bf8225044aa94409e21da0cb09dcdafe9ecd10250b2baad531a9", size = 2173914, upload-time = "2025-11-04T13:41:45.221Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/02/7a/f999a6dcbcd0e5660bc348a3991c8915ce6599f4f2c6ac22f01d7a10816c/pydantic_core-2.41.5-cp314-cp314t-musllinux_1_1_aarch64.whl", hash = "sha256:a39455728aabd58ceabb03c90e12f71fd30fa69615760a075b9fec596456ccc3", size = 2129560, upload-time = "2025-11-04T13:41:47.474Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/3a/b1/6c990ac65e3b4c079a4fb9f5b05f5b013afa0f4ed6780a3dd236d2cbdc64/pydantic_core-2.41.5-cp314-cp314t-musllinux_1_1_armv7l.whl", hash = "sha256:239edca560d05757817c13dc17c50766136d21f7cd0fac50295499ae24f90fdf", size = 2329244, upload-time = "2025-11-04T13:41:49.992Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/d9/02/3c562f3a51afd4d88fff8dffb1771b30cfdfd79befd9883ee094f5b6c0d8/pydantic_core-2.41.5-cp314-cp314t-musllinux_1_1_x86_64.whl", hash = "sha256:2a5e06546e19f24c6a96a129142a75cee553cc018ffee48a460059b1185f4470", size = 2331955, upload-time = "2025-11-04T13:41:54.079Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/5c/96/5fb7d8c3c17bc8c62fdb031c47d77a1af698f1d7a406b0f79aaa1338f9ad/pydantic_core-2.41.5-cp314-cp314t-win32.whl", hash = "sha256:b4ececa40ac28afa90871c2cc2b9ffd2ff0bf749380fbdf57d165fd23da353aa", size = 1988906, upload-time = "2025-11-04T13:41:56.606Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/22/ed/182129d83032702912c2e2d8bbe33c036f342cc735737064668585dac28f/pydantic_core-2.41.5-cp314-cp314t-win_amd64.whl", hash = "sha256:80aa89cad80b32a912a65332f64a4450ed00966111b6615ca6816153d3585a8c", size = 1981607, upload-time = "2025-11-04T13:41:58.889Z" },
|
||||
{ url = "https://files.pythonhosted.org/packages/9f/ed/068e41660b832bb0b1aa5b58011dea2a3fe0ba7861ff38c4d4904c1c1a99/pydantic_core-2.41.5-cp314-cp314t-win_arm64.whl", hash = "sha256:35b44f37a3199f771c3eaa53051bc8a70cd7b54f333531c59e29fd4db5d15008", size = 1974769, upload-time = "2025-11-04T13:42:01.186Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "starlette"
|
||||
version = "0.52.1"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "anyio" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/c4/68/79977123bb7be889ad680d79a40f339082c1978b5cfcf62c2d8d196873ac/starlette-0.52.1.tar.gz", hash = "sha256:834edd1b0a23167694292e94f597773bc3f89f362be6effee198165a35d62933", size = 2653702, upload-time = "2026-01-18T13:34:11.062Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/81/0d/13d1d239a25cbfb19e740db83143e95c772a1fe10202dda4b76792b114dd/starlette-0.52.1-py3-none-any.whl", hash = "sha256:0029d43eb3d273bc4f83a08720b4912ea4b071087a3b48db01b7c839f7954d74", size = 74272, upload-time = "2026-01-18T13:34:09.188Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "syncing-1"
|
||||
version = "0.1.0"
|
||||
source = { virtual = "." }
|
||||
dependencies = [
|
||||
{ name = "pydantic" },
|
||||
{ name = "starlette" },
|
||||
{ name = "uvicorn" },
|
||||
]
|
||||
|
||||
[package.metadata]
|
||||
requires-dist = [
|
||||
{ name = "pydantic", specifier = ">=2.12.5" },
|
||||
{ name = "starlette", specifier = ">=0.52.1" },
|
||||
{ name = "uvicorn", specifier = ">=0.40.0" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "typing-extensions"
|
||||
version = "4.15.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391, upload-time = "2025-08-25T13:49:26.313Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614, upload-time = "2025-08-25T13:49:24.86Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "typing-inspection"
|
||||
version = "0.4.2"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "typing-extensions" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/55/e3/70399cb7dd41c10ac53367ae42139cf4b1ca5f36bb3dc6c9d33acdb43655/typing_inspection-0.4.2.tar.gz", hash = "sha256:ba561c48a67c5958007083d386c3295464928b01faa735ab8547c5692e87f464", size = 75949, upload-time = "2025-10-01T02:14:41.687Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/dc/9b/47798a6c91d8bdb567fe2698fe81e0c6b7cb7ef4d13da4114b41d239f65d/typing_inspection-0.4.2-py3-none-any.whl", hash = "sha256:4ed1cacbdc298c220f1bd249ed5287caa16f34d44ef4e9c3d0cbad5b521545e7", size = 14611, upload-time = "2025-10-01T02:14:40.154Z" },
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "uvicorn"
|
||||
version = "0.40.0"
|
||||
source = { registry = "https://pypi.org/simple" }
|
||||
dependencies = [
|
||||
{ name = "click" },
|
||||
{ name = "h11" },
|
||||
]
|
||||
sdist = { url = "https://files.pythonhosted.org/packages/c3/d1/8f3c683c9561a4e6689dd3b1d345c815f10f86acd044ee1fb9a4dcd0b8c5/uvicorn-0.40.0.tar.gz", hash = "sha256:839676675e87e73694518b5574fd0f24c9d97b46bea16df7b8c05ea1a51071ea", size = 81761, upload-time = "2025-12-21T14:16:22.45Z" }
|
||||
wheels = [
|
||||
{ url = "https://files.pythonhosted.org/packages/3d/d8/2083a1daa7439a66f3a48589a57d576aa117726762618f6bb09fe3798796/uvicorn-0.40.0-py3-none-any.whl", hash = "sha256:c6c8f55bc8bf13eb6fa9ff87ad62308bbbc33d0b67f84293151efe87e0d5f2ee", size = 68502, upload-time = "2025-12-21T14:16:21.041Z" },
|
||||
]
|
||||
Reference in New Issue
Block a user