Cleaning up
This commit is contained in:
		
							parent
							
								
									7f7a9434a7
								
							
						
					
					
						commit
						ed1827bf04
					
				
							
								
								
									
										454
									
								
								.hidden/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										454
									
								
								.hidden/.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,454 +0,0 @@ | ||||
| ## Ignore Visual Studio temporary files, build results, and | ||||
| ## files generated by popular Visual Studio add-ons. | ||||
| ## | ||||
| ## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore | ||||
| 
 | ||||
| # User-specific files | ||||
| *.rsuser | ||||
| *.suo | ||||
| *.user | ||||
| *.userosscache | ||||
| *.sln.docstates | ||||
| 
 | ||||
| # User-specific files (MonoDevelop/Xamarin Studio) | ||||
| *.userprefs | ||||
| 
 | ||||
| # Mono auto generated files | ||||
| mono_crash.* | ||||
| 
 | ||||
| # Build results | ||||
| [Dd]ebug/ | ||||
| [Dd]ebugPublic/ | ||||
| [Rr]elease/ | ||||
| [Rr]eleases/ | ||||
| x64/ | ||||
| x86/ | ||||
| [Ww][Ii][Nn]32/ | ||||
| [Aa][Rr][Mm]/ | ||||
| [Aa][Rr][Mm]64/ | ||||
| bld/ | ||||
| [Bb]in/ | ||||
| [Oo]bj/ | ||||
| [Ll]og/ | ||||
| [Ll]ogs/ | ||||
| 
 | ||||
| # Visual Studio 2015/2017 cache/options directory | ||||
| .vs/ | ||||
| # Uncomment if you have tasks that create the project's static files in wwwroot | ||||
| #wwwroot/ | ||||
| 
 | ||||
| # Visual Studio 2017 auto generated files | ||||
| Generated\ Files/ | ||||
| 
 | ||||
| # MSTest test Results | ||||
| [Tt]est[Rr]esult*/ | ||||
| [Bb]uild[Ll]og.* | ||||
| 
 | ||||
| # NUnit | ||||
| *.VisualState.xml | ||||
| TestResult.xml | ||||
| nunit-*.xml | ||||
| 
 | ||||
| # Build Results of an ATL Project | ||||
| [Dd]ebugPS/ | ||||
| [Rr]eleasePS/ | ||||
| dlldata.c | ||||
| 
 | ||||
| # Benchmark Results | ||||
| BenchmarkDotNet.Artifacts/ | ||||
| 
 | ||||
| # .NET Core | ||||
| project.lock.json | ||||
| project.fragment.lock.json | ||||
| artifacts/ | ||||
| 
 | ||||
| # Tye | ||||
| .tye/ | ||||
| 
 | ||||
| # ASP.NET Scaffolding | ||||
| ScaffoldingReadMe.txt | ||||
| 
 | ||||
| # StyleCop | ||||
| StyleCopReport.xml | ||||
| 
 | ||||
| # Files built by Visual Studio | ||||
| *_i.c | ||||
| *_p.c | ||||
| *_h.h | ||||
| *.ilk | ||||
| *.meta | ||||
| *.obj | ||||
| *.iobj | ||||
| *.pch | ||||
| *.pdb | ||||
| *.ipdb | ||||
| *.pgc | ||||
| *.pgd | ||||
| *.rsp | ||||
| *.sbr | ||||
| *.tlb | ||||
| *.tli | ||||
| *.tlh | ||||
| *.tmp | ||||
| *.tmp_proj | ||||
| *_wpftmp.csproj | ||||
| *.log | ||||
| *.vspscc | ||||
| *.vssscc | ||||
| .builds | ||||
| *.pidb | ||||
| *.svclog | ||||
| *.scc | ||||
| 
 | ||||
| # Chutzpah Test files | ||||
| _Chutzpah* | ||||
| 
 | ||||
| # Visual C++ cache files | ||||
| ipch/ | ||||
| *.aps | ||||
| *.ncb | ||||
| *.opendb | ||||
| *.opensdf | ||||
| *.sdf | ||||
| *.cachefile | ||||
| *.VC.db | ||||
| *.VC.VC.opendb | ||||
| 
 | ||||
| # Visual Studio profiler | ||||
| *.psess | ||||
| *.vsp | ||||
| *.vspx | ||||
| *.sap | ||||
| 
 | ||||
| # Visual Studio Trace Files | ||||
| *.e2e | ||||
| 
 | ||||
| # TFS 2012 Local Workspace | ||||
| $tf/ | ||||
| 
 | ||||
| # Guidance Automation Toolkit | ||||
| *.gpState | ||||
| 
 | ||||
| # ReSharper is a .NET coding add-in | ||||
| _ReSharper*/ | ||||
| *.[Rr]e[Ss]harper | ||||
| *.DotSettings.user | ||||
| 
 | ||||
| # TeamCity is a build add-in | ||||
| _TeamCity* | ||||
| 
 | ||||
| # DotCover is a Code Coverage Tool | ||||
| *.dotCover | ||||
| 
 | ||||
| # AxoCover is a Code Coverage Tool | ||||
| .axoCover/* | ||||
| !.axoCover/settings.json | ||||
| 
 | ||||
| # Coverlet is a free, cross platform Code Coverage Tool | ||||
| coverage*.json | ||||
| coverage*.xml | ||||
| coverage*.info | ||||
| 
 | ||||
| # Visual Studio code coverage results | ||||
| *.coverage | ||||
| *.coveragexml | ||||
| 
 | ||||
| # NCrunch | ||||
| _NCrunch_* | ||||
| .*crunch*.local.xml | ||||
| nCrunchTemp_* | ||||
| 
 | ||||
| # MightyMoose | ||||
| *.mm.* | ||||
| AutoTest.Net/ | ||||
| 
 | ||||
| # Web workbench (sass) | ||||
| .sass-cache/ | ||||
| 
 | ||||
| # Installshield output folder | ||||
| [Ee]xpress/ | ||||
| 
 | ||||
| # DocProject is a documentation generator add-in | ||||
| DocProject/buildhelp/ | ||||
| DocProject/Help/*.HxT | ||||
| DocProject/Help/*.HxC | ||||
| DocProject/Help/*.hhc | ||||
| DocProject/Help/*.hhk | ||||
| DocProject/Help/*.hhp | ||||
| DocProject/Help/Html2 | ||||
| DocProject/Help/html | ||||
| 
 | ||||
| # Click-Once directory | ||||
| publish/ | ||||
| 
 | ||||
| # Publish Web Output | ||||
| *.[Pp]ublish.xml | ||||
| *.azurePubxml | ||||
| # Note: Comment the next line if you want to checkin your web deploy settings, | ||||
| # but database connection strings (with potential passwords) will be unencrypted | ||||
| *.pubxml | ||||
| *.publishproj | ||||
| 
 | ||||
| # Microsoft Azure Web App publish settings. Comment the next line if you want to | ||||
| # checkin your Azure Web App publish settings, but sensitive information contained | ||||
| # in these scripts will be unencrypted | ||||
| PublishScripts/ | ||||
| 
 | ||||
| # NuGet Packages | ||||
| *.nupkg | ||||
| # NuGet Symbol Packages | ||||
| *.snupkg | ||||
| # The packages folder can be ignored because of Package Restore | ||||
| **/[Pp]ackages/* | ||||
| # except build/, which is used as an MSBuild target. | ||||
| !**/[Pp]ackages/build/ | ||||
| # Uncomment if necessary however generally it will be regenerated when needed | ||||
| #!**/[Pp]ackages/repositories.config | ||||
| # NuGet v3's project.json files produces more ignorable files | ||||
| *.nuget.props | ||||
| *.nuget.targets | ||||
| 
 | ||||
| # Microsoft Azure Build Output | ||||
| csx/ | ||||
| *.build.csdef | ||||
| 
 | ||||
| # Microsoft Azure Emulator | ||||
| ecf/ | ||||
| rcf/ | ||||
| 
 | ||||
| # Windows Store app package directories and files | ||||
| AppPackages/ | ||||
| BundleArtifacts/ | ||||
| Package.StoreAssociation.xml | ||||
| _pkginfo.txt | ||||
| *.appx | ||||
| *.appxbundle | ||||
| *.appxupload | ||||
| 
 | ||||
| # Visual Studio cache files | ||||
| # files ending in .cache can be ignored | ||||
| *.[Cc]ache | ||||
| # but keep track of directories ending in .cache | ||||
| !?*.[Cc]ache/ | ||||
| 
 | ||||
| # Others | ||||
| ClientBin/ | ||||
| ~$* | ||||
| *~ | ||||
| *.dbmdl | ||||
| *.dbproj.schemaview | ||||
| *.jfm | ||||
| *.pfx | ||||
| *.publishsettings | ||||
| orleans.codegen.cs | ||||
| 
 | ||||
| # Including strong name files can present a security risk | ||||
| # (https://github.com/github/gitignore/pull/2483#issue-259490424) | ||||
| #*.snk | ||||
| 
 | ||||
| # Since there are multiple workflows, uncomment next line to ignore bower_components | ||||
| # (https://github.com/github/gitignore/pull/1529#issuecomment-104372622) | ||||
| #bower_components/ | ||||
| 
 | ||||
| # RIA/Silverlight projects | ||||
| Generated_Code/ | ||||
| 
 | ||||
| # Backup & report files from converting an old project file | ||||
| # to a newer Visual Studio version. Backup files are not needed, | ||||
| # because we have git ;-) | ||||
| _UpgradeReport_Files/ | ||||
| Backup*/ | ||||
| UpgradeLog*.XML | ||||
| UpgradeLog*.htm | ||||
| ServiceFabricBackup/ | ||||
| *.rptproj.bak | ||||
| 
 | ||||
| # SQL Server files | ||||
| *.mdf | ||||
| *.ldf | ||||
| *.ndf | ||||
| 
 | ||||
| # Business Intelligence projects | ||||
| *.rdl.data | ||||
| *.bim.layout | ||||
| *.bim_*.settings | ||||
| *.rptproj.rsuser | ||||
| *- [Bb]ackup.rdl | ||||
| *- [Bb]ackup ([0-9]).rdl | ||||
| *- [Bb]ackup ([0-9][0-9]).rdl | ||||
| 
 | ||||
| # Microsoft Fakes | ||||
| FakesAssemblies/ | ||||
| 
 | ||||
| # GhostDoc plugin setting file | ||||
| *.GhostDoc.xml | ||||
| 
 | ||||
| # Node.js Tools for Visual Studio | ||||
| .ntvs_analysis.dat | ||||
| node_modules/ | ||||
| 
 | ||||
| # Visual Studio 6 build log | ||||
| *.plg | ||||
| 
 | ||||
| # Visual Studio 6 workspace options file | ||||
| *.opt | ||||
| 
 | ||||
| # Visual Studio 6 auto-generated workspace file (contains which files were open etc.) | ||||
| *.vbw | ||||
| 
 | ||||
| # Visual Studio LightSwitch build output | ||||
| **/*.HTMLClient/GeneratedArtifacts | ||||
| **/*.DesktopClient/GeneratedArtifacts | ||||
| **/*.DesktopClient/ModelManifest.xml | ||||
| **/*.Server/GeneratedArtifacts | ||||
| **/*.Server/ModelManifest.xml | ||||
| _Pvt_Extensions | ||||
| 
 | ||||
| # Paket dependency manager | ||||
| .paket/paket.exe | ||||
| paket-files/ | ||||
| 
 | ||||
| # FAKE - F# Make | ||||
| .fake/ | ||||
| 
 | ||||
| # CodeRush personal settings | ||||
| .cr/personal | ||||
| 
 | ||||
| # Python Tools for Visual Studio (PTVS) | ||||
| __pycache__/ | ||||
| *.pyc | ||||
| 
 | ||||
| # Cake - Uncomment if you are using it | ||||
| # tools/** | ||||
| # !tools/packages.config | ||||
| 
 | ||||
| # Tabs Studio | ||||
| *.tss | ||||
| 
 | ||||
| # Telerik's JustMock configuration file | ||||
| *.jmconfig | ||||
| 
 | ||||
| # BizTalk build output | ||||
| *.btp.cs | ||||
| *.btm.cs | ||||
| *.odx.cs | ||||
| *.xsd.cs | ||||
| 
 | ||||
| # OpenCover UI analysis results | ||||
| OpenCover/ | ||||
| 
 | ||||
| # Azure Stream Analytics local run output | ||||
| ASALocalRun/ | ||||
| 
 | ||||
| # MSBuild Binary and Structured Log | ||||
| *.binlog | ||||
| 
 | ||||
| # NVidia Nsight GPU debugger configuration file | ||||
| *.nvuser | ||||
| 
 | ||||
| # MFractors (Xamarin productivity tool) working folder | ||||
| .mfractor/ | ||||
| 
 | ||||
| # Local History for Visual Studio | ||||
| .localhistory/ | ||||
| 
 | ||||
| # BeatPulse healthcheck temp database | ||||
| healthchecksdb | ||||
| 
 | ||||
| # Backup folder for Package Reference Convert tool in Visual Studio 2017 | ||||
| MigrationBackup/ | ||||
| 
 | ||||
| # Ionide (cross platform F# VS Code tools) working folder | ||||
| .ionide/ | ||||
| 
 | ||||
| # Fody - auto-generated XML schema | ||||
| FodyWeavers.xsd | ||||
| 
 | ||||
| ## | ||||
| ## Visual studio for Mac | ||||
| ## | ||||
| 
 | ||||
| 
 | ||||
| # globs | ||||
| Makefile.in | ||||
| *.userprefs | ||||
| *.usertasks | ||||
| config.make | ||||
| config.status | ||||
| aclocal.m4 | ||||
| install-sh | ||||
| autom4te.cache/ | ||||
| *.tar.gz | ||||
| tarballs/ | ||||
| test-results/ | ||||
| 
 | ||||
| # Mac bundle stuff | ||||
| *.dmg | ||||
| *.app | ||||
| 
 | ||||
| # content below from: https://github.com/github/gitignore/blob/master/Global/macOS.gitignore | ||||
| # General | ||||
| .DS_Store | ||||
| .AppleDouble | ||||
| .LSOverride | ||||
| 
 | ||||
| # Icon must end with two \r | ||||
| Icon | ||||
| 
 | ||||
| 
 | ||||
| # Thumbnails | ||||
| ._* | ||||
| 
 | ||||
| # Files that might appear in the root of a volume | ||||
| .DocumentRevisions-V100 | ||||
| .fseventsd | ||||
| .Spotlight-V100 | ||||
| .TemporaryItems | ||||
| .Trashes | ||||
| .VolumeIcon.icns | ||||
| .com.apple.timemachine.donotpresent | ||||
| 
 | ||||
| # Directories potentially created on remote AFP share | ||||
| .AppleDB | ||||
| .AppleDesktop | ||||
| Network Trash Folder | ||||
| Temporary Items | ||||
| .apdisk | ||||
| 
 | ||||
| # content below from: https://github.com/github/gitignore/blob/master/Global/Windows.gitignore | ||||
| # Windows thumbnail cache files | ||||
| Thumbs.db | ||||
| ehthumbs.db | ||||
| ehthumbs_vista.db | ||||
| 
 | ||||
| # Dump file | ||||
| *.stackdump | ||||
| 
 | ||||
| # Folder config file | ||||
| [Dd]esktop.ini | ||||
| 
 | ||||
| # Recycle Bin used on file shares | ||||
| $RECYCLE.BIN/ | ||||
| 
 | ||||
| # Windows Installer files | ||||
| *.cab | ||||
| *.msi | ||||
| *.msix | ||||
| *.msm | ||||
| *.msp | ||||
| 
 | ||||
| # Windows shortcuts | ||||
| *.lnk | ||||
| 
 | ||||
| # JetBrains Rider | ||||
| .idea/ | ||||
| *.sln.iml | ||||
| 
 | ||||
| ## | ||||
| ## Visual Studio Code | ||||
| ## | ||||
| .vscode/* | ||||
| !.vscode/settings.json | ||||
| !.vscode/tasks.json | ||||
| !.vscode/launch.json | ||||
| !.vscode/extensions.json | ||||
| @ -1,34 +0,0 @@ | ||||
|  | ||||
| Microsoft Visual Studio Solution File, Format Version 12.00 | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SessionZero.Shared", "src\SessionZero.Shared\SessionZero.Shared.csproj", "{86CD0D66-D47D-41DA-B857-F8371AC6E6A9}" | ||||
| EndProject | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SessionZero.Client", "src\SessionZero.Client\SessionZero.Client.csproj", "{A6455219-BBF2-4A67-B13C-449249083703}" | ||||
| EndProject | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SessionZero.Server", "src\SessionZero.Server\SessionZero.Server.csproj", "{54824F5A-0499-4BC9-AC8E-88E943C66256}" | ||||
| EndProject | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "szpack", "tools\szpack\szpack.csproj", "{70FDB950-CAE6-4C38-A476-3FC10BFCF6E6}" | ||||
| EndProject | ||||
| Global | ||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||
| 		Debug|Any CPU = Debug|Any CPU | ||||
| 		Release|Any CPU = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| 	GlobalSection(ProjectConfigurationPlatforms) = postSolution | ||||
| 		{86CD0D66-D47D-41DA-B857-F8371AC6E6A9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{86CD0D66-D47D-41DA-B857-F8371AC6E6A9}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{86CD0D66-D47D-41DA-B857-F8371AC6E6A9}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{86CD0D66-D47D-41DA-B857-F8371AC6E6A9}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 		{A6455219-BBF2-4A67-B13C-449249083703}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{A6455219-BBF2-4A67-B13C-449249083703}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{A6455219-BBF2-4A67-B13C-449249083703}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{A6455219-BBF2-4A67-B13C-449249083703}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 		{54824F5A-0499-4BC9-AC8E-88E943C66256}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{54824F5A-0499-4BC9-AC8E-88E943C66256}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{54824F5A-0499-4BC9-AC8E-88E943C66256}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{54824F5A-0499-4BC9-AC8E-88E943C66256}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 		{70FDB950-CAE6-4C38-A476-3FC10BFCF6E6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{70FDB950-CAE6-4C38-A476-3FC10BFCF6E6}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{70FDB950-CAE6-4C38-A476-3FC10BFCF6E6}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{70FDB950-CAE6-4C38-A476-3FC10BFCF6E6}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| EndGlobal | ||||
| @ -1,354 +0,0 @@ | ||||
| # **SessionZero Design Document** | ||||
| 
 | ||||
| ## **1. Introduction** | ||||
| 
 | ||||
| ### **1.1 Purpose** | ||||
| 
 | ||||
| SessionZero is a free and open-source TTRPG companion app designed to streamline tabletop role-playing game management. It emphasizes flexibility, modularity, and federation through a **system-agnostic datapack format**, supporting both offline and self-hostable online play. The goal is to empower players and GMs with full control over their data, rulesets, and session management. | ||||
| 
 | ||||
| ### **1.2 Scope** | ||||
| 
 | ||||
| The project will begin as a cross-platform desktop application (Windows and Linux), offering tools for creating and managing TTRPG data such as characters, datasets, templates, and sessions. The backend supports user accounts, a database for shared content (SessionZeroDB), and prepares for future federation capabilities. Phase 2 introduces federation, while Phase 3 focuses on mobile support and plugin extensibility. | ||||
| 
 | ||||
| ### **1.3 Objectives** | ||||
| 
 | ||||
| * Provide a modular, open-source companion app for any TTRPG system.   | ||||
| * Implement a JSON-based datapack format for extensible content storage.   | ||||
| * Support both local use and self-hostable online instances.   | ||||
| * Create a foundation for future federation and mobile support. | ||||
| 
 | ||||
| ## **2. System Overview** | ||||
| 
 | ||||
| SessionZero consists of three main layers: | ||||
| 
 | ||||
| 1. **Frontend (Client):** Cross-platform desktop application (.NET Avalonia).   | ||||
| 2. **Backend (Server):** ASP.NET backend managing user accounts, SessionZeroDB, and P2P connections.   | ||||
| 3. **Datapack System:** A structured archive format for storing and sharing game data. | ||||
| 
 | ||||
| ## **3. Technology Stack** | ||||
| 
 | ||||
| * **Core:** C# / .NET 9.0 (minimal API)   | ||||
| * **Client:** Avalonia UI (cross-platform desktop)   | ||||
| * **Database:** PostgreSQL (primary data store for shared/online content)   | ||||
| * **Offline/Local:** Local file storage, perhaps SQLite for local instances.   | ||||
| * **Networking:** WebSockets for real-time session updates; SignalR/gRPC considered for P2P/federation. | ||||
| 
 | ||||
| ## **4. Core Data Concepts** | ||||
| 
 | ||||
| ### **4.1 Datapack** | ||||
| 
 | ||||
| The top-level container (.szp file), bundling all system-specific data (templates, datasets, assets). | ||||
| 
 | ||||
| ### **4.2 Template** | ||||
| 
 | ||||
| A reusable definition of a data structure (e.g., a **Character Template** defining character sheet fields, or a **Session Template** defining campaign structure). | ||||
| 
 | ||||
| ### **4.3 Dataset** | ||||
| 
 | ||||
| A structured collection of external data records (e.g., a bestiary, a list of items, or spells). | ||||
| 
 | ||||
| ### **4.4 Instance** | ||||
| 
 | ||||
| The runtime data created from a Template (e.g., "Aelric's Character Sheet" based on the "Fantasy Character" Template). | ||||
| 
 | ||||
| ### **4.5 Schema** | ||||
| 
 | ||||
| Formal JSON Schemas used to validate the structure of Datapacks and their contents. | ||||
| 
 | ||||
| ## **5. Datapack Specification** | ||||
| 
 | ||||
| A datapack is a self-contained archive (.szp) that holds all data and assets for a specific TTRPG system, module, or expansion. It serves as the foundation for SessionZero's extensible content model. Each pack can define **datasets**, **templates**, and optionally contain instance data (though instances are usually stored separately as save files). The structure emphasizes modularity, portability, and reliable dependency management between packs. | ||||
| 
 | ||||
| ### **5.1 Directory Structure** | ||||
| 
 | ||||
| ```text | ||||
| pack-name/ | ||||
| ├── szpack.json                 # Pack metadata and manifest | ||||
| ├── media/ | ||||
| │   └── images/                 # Image assets (referenced by objects) | ||||
| └── objects/ | ||||
|     ├── datasets/               # User-defined structured data (e.g. items, NPCs) | ||||
|     ├── character-templates/    # Templates defining character structures | ||||
|     └── session-templates/      # Templates defining session/campaign structures | ||||
| ``` | ||||
| 
 | ||||
| ### **5.2 Pack Metadata (szpack.json)** | ||||
| 
 | ||||
| Each datapack contains a root metadata file that defines its identity, authorship, dependencies, and compatibility information. This structure aligns with the C# Datapack and DatapackDependency models.   | ||||
| ```json | ||||
| { | ||||
|   "id": "c9a3f9c8-8b8a-4f1f-9f3b-2d8c7f2c6c62", | ||||
|   "name": "Example Pack", | ||||
|   "version": "1.0.0", | ||||
|   "author": "UserName", | ||||
|   "license": "CC-BY-SA-4.0", | ||||
|   "description": "A fantasy setting datapack containing items and NPCs.", | ||||
|   "createdAt": "2025-10-15T00:00:00Z", | ||||
|   "sessionZeroVersion": "1.0.0", | ||||
|   "dependencies": [ | ||||
|     { | ||||
|       "id": "c2e6d0a4-2a8f-4a51-b36e-9f8f2c7b1d11", | ||||
|       "name": "Core-Ruleset", | ||||
|       "version": "1.2.0" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### **5.3 Common Object Metadata** | ||||
| 
 | ||||
| Objects within the datapack (datasets, templates) inherit from **SzObject**. The **szType** field is required to identify the object type. **Image paths are relative to the media/images folder.**   | ||||
| ```json | ||||
| { | ||||
|   "id": "core-items", | ||||
|   "name": "Core Items", | ||||
|   "szType": "dataset", | ||||
|   "description": "A collection of basic weapons and armor.", | ||||
|   "icon": "core-items.png", | ||||
|   "version": "1.0.0", | ||||
|   "schemaVersion": "1.0.0" | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| *Note: The system resolves the icon field path (e.g., "core-items.png") to datapack/media/images/core-items.png.* | ||||
| 
 | ||||
| ### **5.4 Dataset Objects** | ||||
| 
 | ||||
| Datasets are structured collections of data entries. The entry structure includes TopLevelFields and optional Groups of fields, aligning with the DatasetEntry C# model.   | ||||
| ```json | ||||
| { | ||||
|   "id": "core-items", | ||||
|   "szType": "dataset", | ||||
|   "datasetType": "items", | ||||
|   "name": "Core Items", | ||||
|   "description": "Weapons, armor, and consumables for the base ruleset.", | ||||
|   "version": "1.0.0", | ||||
|   "schemaVersion": "1.0.0", | ||||
|   "entries": { | ||||
|     "sword": { | ||||
|       "id": "sword", | ||||
|       "name": "Sword", | ||||
|       "description": "A basic weapon.", | ||||
|       "icon": "weapons/sword.png", | ||||
|       "topLevelFields": { | ||||
|         "damage": { "type": "Number", "value": 10 }, | ||||
|         "weight": { "type": "Number", "value": 3 } | ||||
|       }, | ||||
|       "groups": [ | ||||
|         { | ||||
|           "id": "stats", | ||||
|           "name": "Stats", | ||||
|           "fields": { | ||||
|             "rarity": { "type": "Text", "value": "Common" } | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }, | ||||
|     "potion": { | ||||
|       "id": "potion", | ||||
|       "name": "Healing Potion", | ||||
|       "description": "Restores a small amount of HP.", | ||||
|       "icon": "consumables/potion.png", | ||||
|       "topLevelFields": { | ||||
|         "healAmount": { "type": "Number", "value": 20 }, | ||||
|         "consumable": { "type": "Boolean", "value": true } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| *Note: The field retrieval path is expected to be EntryName.FieldName or EntryName.GroupName.FieldName, matching the logic in Dataset.cs.* | ||||
| 
 | ||||
| ### **5.5 Character Templates** | ||||
| 
 | ||||
| Character templates define the fields and structure for characters. Note the use of the specific **DatasetLink** object for list fields, requiring a Datapack ID, Dataset ID, and Version.   | ||||
| ```json | ||||
| { | ||||
|   "id": "default-character", | ||||
|   "szType": "character-template", | ||||
|   "name": "Generic Character", | ||||
|   "description": "A base character layout usable across multiple systems.", | ||||
|   "icon": "character.png", | ||||
|   "version": "1.0.0", | ||||
|   "schemaVersion": "1.0.0", | ||||
|   "sections": [ | ||||
|     { | ||||
|       "id": "core", | ||||
|       "name": "Core Stats", | ||||
|       "groups": [ | ||||
|         { | ||||
|           "id": "attributes", | ||||
|           "name": "Attributes", | ||||
|           "fields": [ | ||||
|             { "id": "strength", "name": "Strength", "type": "Number", "defaultValue": 10 }, | ||||
|             { "id": "dexterity", "name": "Dexterity", "type": "Number", "defaultValue": 10 }, | ||||
|             { "id": "intelligence", "name": "Intelligence", "type": "Number", "defaultValue": 10 } | ||||
|           ] | ||||
|         }, | ||||
|         { | ||||
|           "id": "inventory", | ||||
|           "name": "Inventory", | ||||
|           "fields": [ | ||||
|             { | ||||
|               "id": "equipment", | ||||
|               "name": "Equipment", | ||||
|               "type": "List", | ||||
|               "datasetLink": { | ||||
|                 "datapackId": "c9a3f9c8-8b8a-4f1f-9f3b-2d8c7f2c6c62", | ||||
|                 "datasetId": "core-items", | ||||
|                 "version": "1.0.0" | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### **5.6 Session Templates** | ||||
| 
 | ||||
| Session templates are blueprints for setting up a campaign. This example uses **CharacterTemplateLink** and a list of **DatasetLink** objects for **RequiredDatasets** as defined in SessionTemplate.cs.   | ||||
| ```json | ||||
| { | ||||
|   "id": "basic-session", | ||||
|   "szType": "session-template", | ||||
|   "name": "Basic Fantasy Campaign", | ||||
|   "description": "A classic campaign setup with standard fantasy rules.", | ||||
|   "icon": "session.png", | ||||
|   "version": "1.0.0", | ||||
|   "schemaVersion": "1.0.0", | ||||
|   "characterTemplateLink": { | ||||
|     "datapackId": "c9a3f9c8-8b8a-4f1f-9f3b-2d8c7f2c6c62", | ||||
|     "templateId": "default-character", | ||||
|     "version": "1.0.0" | ||||
|   }, | ||||
|   "requiredDatasets": [ | ||||
|     { | ||||
|       "datapackId": "c9a3f9c8-8b8a-4f1f-9f3b-2d8c7f2c6c62", | ||||
|       "datasetId": "core-items", | ||||
|       "version": "1.0.0" | ||||
|     } | ||||
|   ], | ||||
|   "sections": [ | ||||
|     { | ||||
|       "id": "session-info", | ||||
|       "name": "Session Info", | ||||
|       "groups": [ | ||||
|         { | ||||
|           "id": "details", | ||||
|           "name": "Details", | ||||
|           "fields": [ | ||||
|             { "id": "setting", "name": "Setting", "type": "Text", "defaultValue": "Thalindra" }, | ||||
|             { "id": "gmNotes", "name": "GM Notes", "type": "MultiText" } | ||||
|           ] | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### **5.7 Instances (Stored Separately)** | ||||
| 
 | ||||
| Instances, such as a specific character or a running campaign's state, are typically stored outside of the core, reusable datapacks but follow a similar structure, referencing a template.   | ||||
| ```json | ||||
| { | ||||
|   "szType": "character-instance", | ||||
|   "templateId": "default-character", | ||||
|   "name": "Aelric Stormhand", | ||||
|   "fields": { | ||||
|     "strength": 14, | ||||
|     "dexterity": 11, | ||||
|     "inventory": ["sword", "potion"] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### **5.8 Validation and Schema Versioning** | ||||
| 
 | ||||
| Formal, versioned JSON Schemas define the structure of szpack.json and core object types. | ||||
| 
 | ||||
| * Each object declares a **schemaVersion** (global versioning per SessionZero release).   | ||||
| * Validation tools confirm compliance before import/export.   | ||||
| * Internal references (DatasetLink, CharacterTemplateLink, etc.) are resolved within the pack automatically.   | ||||
| * External references are verified via dependencies in szpack.json. | ||||
| 
 | ||||
| **Session Datapack vs Templates:** | ||||
| 
 | ||||
| * **Session Template:** A reusable blueprint for campaign/session setup. Lives under objects/session-templates/.   | ||||
| * **Session Datapack:** The persisted, evolving state of an ongoing campaign. It is stored as a special pack and does not appear alongside normal content packs in the UI. Snapshots are versioned with timestamps. Export/import uses the same .szp format with a distinct type flag in szpack.json. | ||||
| 
 | ||||
| ## **6. Networking & Synchronization** | ||||
| 
 | ||||
| ### **6.1 Session Model** | ||||
| 
 | ||||
| A "Session" is a single game instance, hosted by a GM. It uses a centralized, shared document model (SessionZeroDB/PostgreSQL) with real-time updates pushed via WebSockets. | ||||
| 
 | ||||
| ### **6.2 Data Flow** | ||||
| 
 | ||||
| 1. **GM Action:** GM updates an NPC in their local client.   | ||||
| 2. **Client:** Pushes the update to the server via API/WebSockets.   | ||||
| 3. **Server:** Validates, commits to DB, and broadcasts the change to all connected players (including the GM).   | ||||
| 4. **Players:** Receive the update and apply it to their local model. | ||||
| 
 | ||||
| ### **6.3 Offline Mode** | ||||
| 
 | ||||
| The application supports full offline functionality, using local storage. When connectivity is restored, the system uses conflict-resolution logic (timestamp-based or operational transformation) to reconcile changes with the SessionZeroDB. | ||||
| 
 | ||||
| ## **7. Federation** | ||||
| 
 | ||||
| ### **7.1 Concept** | ||||
| 
 | ||||
| Federation allows different SessionZero instances (e.g., self-hosted servers) to share and discover content (Datapacks, public Datasets, Templates). This requires standardized APIs for content retrieval and identity verification. | ||||
| 
 | ||||
| ### **7.2 Identity** | ||||
| 
 | ||||
| * Users have a unique global identifier (UID).   | ||||
| * Datapacks are identified by their UUID (id). | ||||
| 
 | ||||
| ### **7.3 Data Sharing** | ||||
| 
 | ||||
| Publicly shared Datapacks are discoverable through a federation layer, allowing users to import content from other trusted servers. | ||||
| 
 | ||||
| ## **8. Licensing** | ||||
| 
 | ||||
| ### **8.1 Source Code License** | ||||
| 
 | ||||
| * **AGPL-3.0 (Affero General Public License):** Ensures open-source continuity and attribution, prevents uncredited commercial rebranding. | ||||
| 
 | ||||
| ### **8.2 User Content License** | ||||
| 
 | ||||
| * **Creative Commons BY-SA 4.0:** Allows sharing and adaptation with attribution. | ||||
| 
 | ||||
| ## **9. Development Phases** | ||||
| 
 | ||||
| ### **Phase 1: Core Platform (MVP)** | ||||
| 
 | ||||
| * Client (UI, datapack editing, session management)   | ||||
| * Backend (accounts, SessionZeroDB, P2P prep)   | ||||
| * JSON datapack format implementation | ||||
| 
 | ||||
| ### **Phase 2: Federation** | ||||
| 
 | ||||
| * Cross-server communication   | ||||
| * Access controls and search indexing | ||||
| 
 | ||||
| ### **Phase 3: Mobile and Plugins** | ||||
| 
 | ||||
| * Cross-platform mobile apps   | ||||
| * Plugin and API extensibility | ||||
| 
 | ||||
| ## **10. Testing & Quality** | ||||
| 
 | ||||
| * **Unit Tests:** Core services (DataPackService, StorageService, NetworkingService) with fixtures for characters/datasets/templates.   | ||||
| * **Schema Validation:** CI job validates sample packs against JSON Schemas (see section 5.8) and rejects PRs with invalid packs.   | ||||
| * **Integration Tests:** Import/export round-trips, dependency resolution, and session snapshot/recovery.   | ||||
| * **Compatibility Matrix:** Validate on Windows and Linux for net9.0; browser target smoke tests where applicable. | ||||
| 
 | ||||
| ## **11. Accessibility & Internationalization** | ||||
| 
 | ||||
| * **Accessibility:** Keyboard navigation across all primary views, sufficient color contrast, focus indicators, and screen-reader-friendly labels.   | ||||
| * **Internationalization:** Resource-based localization for UI strings; UTF-8 throughout; prepare for RTL support; date/number formatting via invariant + locale-aware overlays. | ||||
| @ -1,2 +0,0 @@ | ||||
| Samples: | ||||
| - This folder will include sample datapacks | ||||
| @ -1,15 +0,0 @@ | ||||
| <Application xmlns="https://github.com/avaloniaui" | ||||
|              xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||
|              x:Class="SessionZero.Client.App" | ||||
|              xmlns:local="using:SessionZero.Client" | ||||
|              RequestedThemeVariant="Default"> | ||||
|              <!-- "Default" ThemeVariant follows system theme variant. "Dark" or "Light" are other available options. --> | ||||
| 
 | ||||
|     <Application.DataTemplates> | ||||
|         <local:ViewLocator/> | ||||
|     </Application.DataTemplates> | ||||
|    | ||||
|     <Application.Styles> | ||||
|         <FluentTheme /> | ||||
|     </Application.Styles> | ||||
| </Application> | ||||
| @ -1,47 +0,0 @@ | ||||
| using Avalonia; | ||||
| using Avalonia.Controls.ApplicationLifetimes; | ||||
| using Avalonia.Data.Core; | ||||
| using Avalonia.Data.Core.Plugins; | ||||
| using System.Linq; | ||||
| using Avalonia.Markup.Xaml; | ||||
| using SessionZero.Client.ViewModels; | ||||
| using SessionZero.Client.Views; | ||||
| 
 | ||||
| namespace SessionZero.Client; | ||||
| 
 | ||||
| public partial class App : Application | ||||
| { | ||||
|     public override void Initialize() | ||||
|     { | ||||
|         AvaloniaXamlLoader.Load(this); | ||||
|     } | ||||
| 
 | ||||
|     public override void OnFrameworkInitializationCompleted() | ||||
|     { | ||||
|         if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop) | ||||
|         { | ||||
|             // Avoid duplicate validations from both Avalonia and the CommunityToolkit.  | ||||
|             // More info: https://docs.avaloniaui.net/docs/guides/development-guides/data-validation#manage-validationplugins | ||||
|             DisableAvaloniaDataAnnotationValidation(); | ||||
|             desktop.MainWindow = new MainWindow | ||||
|             { | ||||
|                 DataContext = new MainWindowViewModel(), | ||||
|             }; | ||||
|         } | ||||
| 
 | ||||
|         base.OnFrameworkInitializationCompleted(); | ||||
|     } | ||||
| 
 | ||||
|     private void DisableAvaloniaDataAnnotationValidation() | ||||
|     { | ||||
|         // Get an array of plugins to remove | ||||
|         var dataValidationPluginsToRemove = | ||||
|             BindingPlugins.DataValidators.OfType<DataAnnotationsValidationPlugin>().ToArray(); | ||||
| 
 | ||||
|         // remove each entry found | ||||
|         foreach (var plugin in dataValidationPluginsToRemove) | ||||
|         { | ||||
|             BindingPlugins.DataValidators.Remove(plugin); | ||||
|         } | ||||
|     } | ||||
| } | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 172 KiB | 
| @ -1,21 +0,0 @@ | ||||
| using Avalonia; | ||||
| using System; | ||||
| 
 | ||||
| namespace SessionZero.Client; | ||||
| 
 | ||||
| sealed class Program | ||||
| { | ||||
|     // Initialization code. Don't use any Avalonia, third-party APIs or any | ||||
|     // SynchronizationContext-reliant code before AppMain is called: things aren't initialized | ||||
|     // yet and stuff might break. | ||||
|     [STAThread] | ||||
|     public static void Main(string[] args) => BuildAvaloniaApp() | ||||
|         .StartWithClassicDesktopLifetime(args); | ||||
| 
 | ||||
|     // Avalonia configuration, don't remove; also used by visual designer. | ||||
|     public static AppBuilder BuildAvaloniaApp() | ||||
|         => AppBuilder.Configure<App>() | ||||
|             .UsePlatformDetect() | ||||
|             .WithInterFont() | ||||
|             .LogToTrace(); | ||||
| } | ||||
| @ -1,32 +0,0 @@ | ||||
| <Project Sdk="Microsoft.NET.Sdk"> | ||||
|     <PropertyGroup> | ||||
|         <OutputType>WinExe</OutputType> | ||||
|         <TargetFramework>net9.0</TargetFramework> | ||||
|         <Nullable>enable</Nullable> | ||||
|         <BuiltInComInteropSupport>true</BuiltInComInteropSupport> | ||||
|         <ApplicationManifest>app.manifest</ApplicationManifest> | ||||
|         <AvaloniaUseCompiledBindingsByDefault>true</AvaloniaUseCompiledBindingsByDefault> | ||||
|     </PropertyGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|         <Folder Include="Models\"/> | ||||
|         <AvaloniaResource Include="Assets\**"/> | ||||
|     </ItemGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|         <PackageReference Include="Avalonia" Version="11.3.6"/> | ||||
|         <PackageReference Include="Avalonia.Desktop" Version="11.3.6"/> | ||||
|         <PackageReference Include="Avalonia.Themes.Fluent" Version="11.3.6"/> | ||||
|         <PackageReference Include="Avalonia.Fonts.Inter" Version="11.3.6"/> | ||||
|         <!--Condition below is needed to remove Avalonia.Diagnostics package from build output in Release configuration.--> | ||||
|         <PackageReference Include="Avalonia.Diagnostics" Version="11.3.6"> | ||||
|             <IncludeAssets Condition="'$(Configuration)' != 'Debug'">None</IncludeAssets> | ||||
|             <PrivateAssets Condition="'$(Configuration)' != 'Debug'">All</PrivateAssets> | ||||
|         </PackageReference> | ||||
|         <PackageReference Include="CommunityToolkit.Mvvm" Version="8.2.1"/> | ||||
|     </ItemGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|       <ProjectReference Include="..\SessionZero.Shared\SessionZero.Shared.csproj" /> | ||||
|     </ItemGroup> | ||||
| </Project> | ||||
| @ -1,30 +0,0 @@ | ||||
| using System; | ||||
| using Avalonia.Controls; | ||||
| using Avalonia.Controls.Templates; | ||||
| using SessionZero.Client.ViewModels; | ||||
| 
 | ||||
| namespace SessionZero.Client; | ||||
| 
 | ||||
| public class ViewLocator : IDataTemplate | ||||
| { | ||||
|     public Control? Build(object? param) | ||||
|     { | ||||
|         if (param is null) | ||||
|             return null; | ||||
| 
 | ||||
|         var name = param.GetType().FullName!.Replace("ViewModel", "View", StringComparison.Ordinal); | ||||
|         var type = Type.GetType(name); | ||||
| 
 | ||||
|         if (type != null) | ||||
|         { | ||||
|             return (Control)Activator.CreateInstance(type)!; | ||||
|         } | ||||
| 
 | ||||
|         return new TextBlock { Text = "Not Found: " + name }; | ||||
|     } | ||||
| 
 | ||||
|     public bool Match(object? data) | ||||
|     { | ||||
|         return data is ViewModelBase; | ||||
|     } | ||||
| } | ||||
| @ -1,6 +0,0 @@ | ||||
| namespace SessionZero.Client.ViewModels; | ||||
| 
 | ||||
| public partial class MainWindowViewModel : ViewModelBase | ||||
| { | ||||
|     public string Greeting { get; } = "Welcome to Avalonia!"; | ||||
| } | ||||
| @ -1,7 +0,0 @@ | ||||
| using CommunityToolkit.Mvvm.ComponentModel; | ||||
| 
 | ||||
| namespace SessionZero.Client.ViewModels; | ||||
| 
 | ||||
| public class ViewModelBase : ObservableObject | ||||
| { | ||||
| } | ||||
| @ -1,20 +0,0 @@ | ||||
| <Window xmlns="https://github.com/avaloniaui" | ||||
|         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||
|         xmlns:vm="using:SessionZero.Client.ViewModels" | ||||
|         xmlns:d="http://schemas.microsoft.com/expression/blend/2008" | ||||
|         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" | ||||
|         mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" | ||||
|         x:Class="SessionZero.Client.Views.MainWindow" | ||||
|         x:DataType="vm:MainWindowViewModel" | ||||
|         Icon="/Assets/avalonia-logo.ico" | ||||
|         Title="SessionZero.Client"> | ||||
| 
 | ||||
|     <Design.DataContext> | ||||
|         <!-- This only sets the DataContext for the previewer in an IDE, | ||||
|              to set the actual DataContext for runtime, set the DataContext property in code (look at App.axaml.cs) --> | ||||
|         <vm:MainWindowViewModel/> | ||||
|     </Design.DataContext> | ||||
| 
 | ||||
|     <TextBlock Text="{Binding Greeting}" HorizontalAlignment="Center" VerticalAlignment="Center"/> | ||||
| 
 | ||||
| </Window> | ||||
| @ -1,11 +0,0 @@ | ||||
| using Avalonia.Controls; | ||||
| 
 | ||||
| namespace SessionZero.Client.Views; | ||||
| 
 | ||||
| public partial class MainWindow : Window | ||||
| { | ||||
|     public MainWindow() | ||||
|     { | ||||
|         InitializeComponent(); | ||||
|     } | ||||
| } | ||||
| @ -1,18 +0,0 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1"> | ||||
|   <!-- This manifest is used on Windows only. | ||||
|        Don't remove it as it might cause problems with window transparency and embedded controls. | ||||
|        For more details visit https://learn.microsoft.com/en-us/windows/win32/sbscs/application-manifests --> | ||||
|   <assemblyIdentity version="1.0.0.0" name="SessionZero.Client.Desktop"/> | ||||
| 
 | ||||
|   <compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1"> | ||||
|     <application> | ||||
|       <!-- A list of the Windows versions that this application has been tested on | ||||
|            and is designed to work with. Uncomment the appropriate elements | ||||
|            and Windows will automatically select the most compatible environment. --> | ||||
| 
 | ||||
|       <!-- Windows 10 --> | ||||
|       <supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}" /> | ||||
|     </application> | ||||
|   </compatibility> | ||||
| </assembly> | ||||
| @ -1,41 +0,0 @@ | ||||
| var builder = WebApplication.CreateBuilder(args); | ||||
| 
 | ||||
| // Add services to the container. | ||||
| // Learn more about configuring OpenAPI at https://aka.ms/aspnet/openapi | ||||
| builder.Services.AddOpenApi(); | ||||
| 
 | ||||
| var app = builder.Build(); | ||||
| 
 | ||||
| // Configure the HTTP request pipeline. | ||||
| if (app.Environment.IsDevelopment()) | ||||
| { | ||||
|     app.MapOpenApi(); | ||||
| } | ||||
| 
 | ||||
| app.UseHttpsRedirection(); | ||||
| 
 | ||||
| var summaries = new[] | ||||
| { | ||||
|     "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching" | ||||
| }; | ||||
| 
 | ||||
| app.MapGet("/weatherforecast", () => | ||||
|     { | ||||
|         var forecast = Enumerable.Range(1, 5).Select(index => | ||||
|                 new WeatherForecast | ||||
|                 ( | ||||
|                     DateOnly.FromDateTime(DateTime.Now.AddDays(index)), | ||||
|                     Random.Shared.Next(-20, 55), | ||||
|                     summaries[Random.Shared.Next(summaries.Length)] | ||||
|                 )) | ||||
|             .ToArray(); | ||||
|         return forecast; | ||||
|     }) | ||||
|     .WithName("GetWeatherForecast"); | ||||
| 
 | ||||
| app.Run(); | ||||
| 
 | ||||
| record WeatherForecast(DateOnly Date, int TemperatureC, string? Summary) | ||||
| { | ||||
|     public int TemperatureF => 32 + (int)(TemperatureC / 0.5556); | ||||
| } | ||||
| @ -1,23 +0,0 @@ | ||||
| { | ||||
|   "$schema": "https://json.schemastore.org/launchsettings.json", | ||||
|   "profiles": { | ||||
|     "http": { | ||||
|       "commandName": "Project", | ||||
|       "dotnetRunMessages": true, | ||||
|       "launchBrowser": false, | ||||
|       "applicationUrl": "http://localhost:5227", | ||||
|       "environmentVariables": { | ||||
|         "ASPNETCORE_ENVIRONMENT": "Development" | ||||
|       } | ||||
|     }, | ||||
|     "https": { | ||||
|       "commandName": "Project", | ||||
|       "dotnetRunMessages": true, | ||||
|       "launchBrowser": false, | ||||
|       "applicationUrl": "https://localhost:7090;http://localhost:5227", | ||||
|       "environmentVariables": { | ||||
|         "ASPNETCORE_ENVIRONMENT": "Development" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -1,13 +0,0 @@ | ||||
| <Project Sdk="Microsoft.NET.Sdk.Web"> | ||||
| 
 | ||||
|     <PropertyGroup> | ||||
|         <TargetFramework>net9.0</TargetFramework> | ||||
|         <Nullable>enable</Nullable> | ||||
|         <ImplicitUsings>enable</ImplicitUsings> | ||||
|     </PropertyGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|         <PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="9.0.7"/> | ||||
|     </ItemGroup> | ||||
| 
 | ||||
| </Project> | ||||
| @ -1,6 +0,0 @@ | ||||
| @SessionZero.Server_HostAddress = http://localhost:5227 | ||||
| 
 | ||||
| GET {{SessionZero.Server_HostAddress}}/weatherforecast/ | ||||
| Accept: application/json | ||||
| 
 | ||||
| ### | ||||
| @ -1,8 +0,0 @@ | ||||
| { | ||||
|   "Logging": { | ||||
|     "LogLevel": { | ||||
|       "Default": "Information", | ||||
|       "Microsoft.AspNetCore": "Warning" | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -1,9 +0,0 @@ | ||||
| { | ||||
|   "Logging": { | ||||
|     "LogLevel": { | ||||
|       "Default": "Information", | ||||
|       "Microsoft.AspNetCore": "Warning" | ||||
|     } | ||||
|   }, | ||||
|   "AllowedHosts": "*" | ||||
| } | ||||
| @ -1,6 +0,0 @@ | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| public class CharacterTemplate : TemplateBase | ||||
| { | ||||
| 
 | ||||
| } | ||||
| @ -1,28 +0,0 @@ | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| /// <summary> | ||||
| /// The model for a datapack (used for the szpack.json file). | ||||
| /// </summary> | ||||
| public class Datapack | ||||
| { | ||||
|     public required Guid Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required string Version { get; set; } | ||||
|     public required string Author { get; set; } | ||||
|     public required string License { get; set; } | ||||
|     public string Description { get; set; } = string.Empty; | ||||
|     public required DateTime CreatedAt { get; set; } | ||||
|     public required string SessionZeroVersion { get; set; } | ||||
| 
 | ||||
|     public List<DatapackDependency> Dependencies { get; set; } = new(); | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a dependency of a datapack. | ||||
| /// </summary> | ||||
| public class DatapackDependency | ||||
| { | ||||
|     public required Guid Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required string Version { get; set; } | ||||
| } | ||||
| @ -1,80 +0,0 @@ | ||||
| using SessionZero.Shared.Schema; | ||||
| 
 | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| public class Dataset : SzObject | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Arbitrary string used to identify the dataset type (e.g. "items", "quests", "npcs", etc.). | ||||
|     /// </summary> | ||||
|     public required string DatasetType { get; set; } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// The entries in the dataset. | ||||
|     /// </summary> | ||||
|     public required Dictionary<string, DatasetEntry> Entries { get; set; } = new(); | ||||
| 
 | ||||
|     /// <summary> | ||||
|     /// Retrieves a FieldValue from the dataset using a dot-separated path. | ||||
|     /// Format: [EntryName].[FieldName] or [EntryName].[GroupName].[FieldName] | ||||
|     /// </summary> | ||||
|     /// <param name="fieldName">The dot-separated path to the field.</param> | ||||
|     /// <returns>The FieldValue if found, otherwise null.</returns> | ||||
|     public FieldValue? GetField(string fieldName) | ||||
|     { | ||||
|         var split = fieldName.Split('.'); | ||||
|          | ||||
|         // Path must contain at least EntryName.FieldName (length 2) | ||||
|         if (split.Length < 2 || split.Length > 3) | ||||
|         { | ||||
|             return null; | ||||
|         } | ||||
| 
 | ||||
|         var entryName = split[0]; | ||||
|         if (!Entries.TryGetValue(entryName, out var entry)) | ||||
|         { | ||||
|             return null; | ||||
|         } | ||||
|          | ||||
|         if (split.Length == 2) | ||||
|         { | ||||
|             // Format: EntryName.FieldName (Top-level field) | ||||
|             var fName = split[1]; | ||||
|              | ||||
|             if (entry.TopLevelFields is null) return null; | ||||
|              | ||||
|             return entry.TopLevelFields.TryGetValue(fName, out var value) ? value : null; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             // Format: EntryName.GroupName.FieldName | ||||
|             var groupName = split[1]; | ||||
|             var fName = split[2]; | ||||
|              | ||||
|             if (entry.Groups is null) return null; | ||||
|              | ||||
|             var dataGroup = entry.Groups.FirstOrDefault(g => g.Name == groupName); | ||||
|              | ||||
|             if (dataGroup is null || dataGroup.Fields is null) | ||||
|             { | ||||
|                 return null; | ||||
|             } | ||||
|              | ||||
|             return dataGroup.Fields.TryGetValue(fName, out var value) ? value : null; | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// Sets a FieldValue in the dataset using a dot-separated path. | ||||
|     /// Format: [EntryName].[FieldName] or [EntryName].[GroupName].[FieldName] | ||||
|     /// </summary> | ||||
|     /// <param name="fieldName">The dot-separated path to the field.</param> | ||||
|     /// <param name="value">The value to set.</param> | ||||
|     /// <returns>True if the value was set, false otherwise</returns> | ||||
|     public static bool SetField(string fieldName, FieldValue value) | ||||
|     { | ||||
|         // TODO: Implementation for SetField | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -1,17 +0,0 @@ | ||||
| using SessionZero.Shared.Schema; | ||||
| 
 | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| public class SessionTemplate : TemplateBase | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Reference to the required Character Template for character creation in this session. | ||||
|     /// </summary> | ||||
|     public required CharacterTemplateLink CharacterTemplateLink { get; set; } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// List of required Datasets that this session template depends on | ||||
|     /// (datapack ids must be present in the DatapackDependency list in the datapack object) | ||||
|     /// </summary> | ||||
|     public required List<DatasetLink> RequiredDatasets { get; set; } = new(); | ||||
| } | ||||
| @ -1,12 +0,0 @@ | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| public class SzObject | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required string SzType { get; set; } | ||||
|     public string Description { get; set; } = string.Empty; | ||||
|     public string Icon { get; set; } = string.Empty; | ||||
|     public required string Version { get; set; } | ||||
|     public required string SchemaVersion { get; set; } | ||||
| } | ||||
| @ -1,11 +0,0 @@ | ||||
| using SessionZero.Shared.Schema; | ||||
| 
 | ||||
| namespace SessionZero.Shared.Models; | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Base class for templates. | ||||
| /// </summary> | ||||
| public class TemplateBase : SzObject | ||||
| { | ||||
|     public required List<TemplateSection> Sections { get; set; } = new(); | ||||
| } | ||||
| @ -1,120 +0,0 @@ | ||||
| namespace SessionZero.Shared.Schema; | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Valid types for fields. | ||||
| /// </summary> | ||||
| public enum FieldType | ||||
| { | ||||
|     Text, | ||||
|     MultiText, | ||||
|     Number, | ||||
|     Boolean, | ||||
|     Formula, | ||||
|     List | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a list of allowed values for a field with type List. | ||||
| /// </summary> | ||||
| public class SzListType | ||||
| { | ||||
|     // NOTE: Only one of these lists should be allowed | ||||
|     public List<string>? AllowedDatasetTypes { get; set; } | ||||
|     public List<DatasetLink>? AllowedDatasetIds { get; set; } | ||||
|     public string? CustomValues { get; set; } | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a formula field. | ||||
| /// </summary> | ||||
| public class SzFormulaType | ||||
| { | ||||
|     // TODO: Implement a way to parse formulas | ||||
|     public string? Formula { get; set; } | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Used to link a dataset in other data or template objects. | ||||
| /// </summary> | ||||
| public class DatasetLink | ||||
| { | ||||
|     public required Guid DatapackId { get; set; } | ||||
|     public required string DatasetId { get; set; } | ||||
|     public required string Version { get; set; } | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Used to link a character template in other template objects. | ||||
| /// </summary> | ||||
| public class CharacterTemplateLink | ||||
| { | ||||
|     public required Guid DatapackId { get; set; } | ||||
|     public required string TemplateId { get; set; } | ||||
|     public required string Version { get; set; } | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a field value. | ||||
| /// </summary> | ||||
| public class FieldValue | ||||
| { | ||||
|     public required FieldType Type { get; set; } | ||||
|     public required object Value { get; set; } | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a group of fields in a dataset. | ||||
| /// </summary> | ||||
| public class DataGroup | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required Dictionary<string, FieldValue> Fields { get; set; } = new(); | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a dataset entry used in datasets. | ||||
| /// </summary> | ||||
| public class DatasetEntry | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public string Description { get; set; } = string.Empty; | ||||
|     public string Icon { get; set; } = string.Empty; | ||||
| 
 | ||||
|     public Dictionary<string, FieldValue>? TopLevelFields { get; set; } = new(); | ||||
|     public List<DataGroup>? Groups { get; set; } = new(); | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a section in a template that houses multiple groups. | ||||
| /// </summary> | ||||
| public class TemplateSection | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required List<TemplateGroup> Groups { get; set; } = new(); | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a group of fields in a template. | ||||
| /// </summary> | ||||
| public class TemplateGroup | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required List<TemplateFieldDefinition>? Fields { get; set; } = new(); | ||||
| } | ||||
| 
 | ||||
| /// <summary> | ||||
| /// Represents a field definition in a template | ||||
| /// (Does not hold actual field values as templates are meant to be filled in upon instance creation). | ||||
| /// </summary> | ||||
| public class TemplateFieldDefinition | ||||
| { | ||||
|     public required string Id { get; set; } | ||||
|     public required string Name { get; set; } | ||||
|     public required FieldType Type { get; set; } | ||||
|     public object? DefaultValue { get; set; } | ||||
|     public DatasetLink? DatasetLink { get; set; } | ||||
| } | ||||
| @ -1,172 +0,0 @@ | ||||
| using System.Text.Json; | ||||
| using SessionZero.Shared.Models; | ||||
| using System.IO; | ||||
| using System.Threading.Tasks; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.IO.Compression; | ||||
| 
 | ||||
| namespace SessionZero.Shared.Services; | ||||
| 
 | ||||
| public static class DatapackService | ||||
| { | ||||
|     private static readonly JsonSerializerOptions JsonOptions = new() | ||||
|     { | ||||
|         WriteIndented = true, | ||||
|         PropertyNamingPolicy = JsonNamingPolicy.CamelCase | ||||
|     }; | ||||
| 
 | ||||
|     /// <summary> | ||||
|     /// Creates a new Datapack model with required metadata. | ||||
|     /// </summary> | ||||
|     /// <param name="name">The display name of the datapack.</param> | ||||
|     /// <param name="version">The initial version string (e.g., "1.0.0").</param> | ||||
|     /// <param name="author">The pack creator's name.</param> | ||||
|     /// <param name="license">The license (e.g., "CC-BY-SA-4.0").</param> | ||||
|     /// <returns>A fully initialized Datapack model.</returns> | ||||
|     public static Datapack CreateEmptyDatapack(string name, string version, string author, string license) | ||||
|     { | ||||
|         return new Datapack | ||||
|         { | ||||
|             Id = Guid.NewGuid(), | ||||
|             Name = name, | ||||
|             Version = version, | ||||
|             Author = author, | ||||
|             License = license, | ||||
|             Description = String.Empty, | ||||
|             CreatedAt = DateTime.UtcNow, | ||||
|             SessionZeroVersion = "0.0.1",  | ||||
|             Dependencies = new() | ||||
|         }; | ||||
|     } | ||||
| 
 | ||||
|     /// <summary> | ||||
|     /// Serializes a Datapack model and saves it to a file named 'szpack.json' | ||||
|     /// inside a new subdirectory named after the datapack's slug. | ||||
|     /// </summary> | ||||
|     /// <param name="datapack">The Datapack object to save.</param> | ||||
|     /// <param name="parentDirectoryPath">The directory where the new datapack folder will be created.</param> | ||||
|     /// <returns>The full path to the created szpack.json file.</returns> | ||||
|     public static async Task<string> SaveDatapackMetadataAsync(Datapack datapack, string parentDirectoryPath) | ||||
|     { | ||||
|         var packDirectoryName = datapack.Name.ToLower().Replace(' ', '-').Trim(); | ||||
|         var packRootDirectory = Path.Combine(parentDirectoryPath, packDirectoryName); | ||||
| 
 | ||||
|         Directory.CreateDirectory(packRootDirectory); | ||||
|          | ||||
|         var filePath = Path.Combine(packRootDirectory, "szpack.json"); | ||||
|          | ||||
|         var jsonString = JsonSerializer.Serialize(datapack, JsonOptions); | ||||
|         await File.WriteAllTextAsync(filePath, jsonString); | ||||
|          | ||||
|         return filePath; | ||||
|     } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// Serializes an SzObject (Dataset, Template, etc.) and saves it to a file. | ||||
|     /// </summary> | ||||
|     /// <typeparam name="T">The type of the object, must inherit from SzObject.</typeparam> | ||||
|     /// <param name="szObject">The object to save.</param> | ||||
|     /// <param name="directoryPath">The specific subdirectory (e.g., 'datasets') within the datapack root.</param> | ||||
|     /// <returns>The full path to the created JSON file.</returns> | ||||
|     public static async Task<string> SaveSzObjectAsync<T>(T szObject, string directoryPath) where T : SzObject | ||||
|     { | ||||
|         Directory.CreateDirectory(directoryPath); | ||||
|          | ||||
|         var fileName = $"{szObject.Id}.json";  | ||||
|         var filePath = Path.Combine(directoryPath, fileName); | ||||
|         var jsonString = JsonSerializer.Serialize(szObject, JsonOptions); | ||||
|         await File.WriteAllTextAsync(filePath, jsonString); | ||||
|          | ||||
|         return filePath; | ||||
|     } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// Defines and creates the standard directory structure for a SessionZero datapack. | ||||
|     /// </summary> | ||||
|     /// <param name="rootPath">The root directory path of the new datapack.</param> | ||||
|     /// <returns>A dictionary containing the standard folder names and their absolute paths.</returns> | ||||
|     public static Dictionary<string, string> CreateDatapackDirectoryStructure(string rootPath) | ||||
|     { | ||||
|         var structure = new Dictionary<string, string> | ||||
|         { | ||||
|             { "datasets", Path.Combine(rootPath, "datasets") }, | ||||
|             { "character_templates", Path.Combine(rootPath, "characters") }, | ||||
|             { "session_templates", Path.Combine(rootPath, "sessions") }, | ||||
|             { "media", Path.Combine(rootPath, "media") }, | ||||
|             { "images", Path.Combine(rootPath, "media", "images") } | ||||
|         }; | ||||
|          | ||||
|         foreach (var path in structure.Values) | ||||
|         { | ||||
|             Directory.CreateDirectory(path); | ||||
|         } | ||||
| 
 | ||||
|         return structure; | ||||
|     } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// Compresses a datapack directory into a .szpack zip archive. | ||||
|     /// </summary> | ||||
|     /// <param name="sourceDirectoryPath">The path to the datapack's root directory.</param> | ||||
|     /// <param name="outputFilePath">The full path and filename for the output .szpack file.</param> | ||||
|     public static void PackDatapack(string sourceDirectoryPath, string outputFilePath) | ||||
|     { | ||||
|         if (!Directory.Exists(sourceDirectoryPath)) | ||||
|         { | ||||
|             throw new DirectoryNotFoundException($"Source directory not found: {sourceDirectoryPath}"); | ||||
|         } | ||||
| 
 | ||||
|         if (File.Exists(outputFilePath)) | ||||
|         { | ||||
|             File.Delete(outputFilePath); | ||||
|         } | ||||
| 
 | ||||
|         ZipFile.CreateFromDirectory(sourceDirectoryPath, outputFilePath); | ||||
|     } | ||||
| 
 | ||||
|     /// <summary> | ||||
|     /// Extracts a .szpack zip archive into a target directory. | ||||
|     /// </summary> | ||||
|     /// <param name="sourceFilePath">The path to the .szpack file.</param> | ||||
|     /// <param name="destinationDirectoryPath">The directory where the contents will be extracted.</param> | ||||
|     public static void UnpackDatapack(string sourceFilePath, string destinationDirectoryPath) | ||||
|     { | ||||
|         if (!File.Exists(sourceFilePath)) | ||||
|         { | ||||
|             throw new FileNotFoundException($"Datapack file not found: {sourceFilePath}"); | ||||
|         } | ||||
| 
 | ||||
|         Directory.CreateDirectory(destinationDirectoryPath); | ||||
| 
 | ||||
|         ZipFile.ExtractToDirectory(sourceFilePath, destinationDirectoryPath, overwriteFiles: true); | ||||
|     } | ||||
|      | ||||
|     /// <summary> | ||||
|     /// Deserializes and loads the core Datapack metadata (szpack.json) from a directory. | ||||
|     /// </summary> | ||||
|     /// <param name="datapackRootPath">The path to the datapack's root directory.</param> | ||||
|     /// <returns>A Datapack model containing the metadata.</returns> | ||||
|     /// <exception cref="FileNotFoundException">Thrown if szpack.json is not found.</exception> | ||||
|     /// <exception cref="JsonException">Thrown if deserialization fails.</exception> | ||||
|     public static async Task<Datapack> LoadDatapackMetadataAsync(string datapackRootPath) | ||||
|     { | ||||
|         var filePath = Path.Combine(datapackRootPath, "szpack.json"); | ||||
| 
 | ||||
|         if (!File.Exists(filePath)) | ||||
|         { | ||||
|             throw new FileNotFoundException($"The required metadata file 'szpack.json' was not found in: {datapackRootPath}", filePath); | ||||
|         } | ||||
|          | ||||
|         var jsonString = await File.ReadAllTextAsync(filePath); | ||||
|          | ||||
|         var datapack = JsonSerializer.Deserialize<Datapack>(jsonString, JsonOptions); | ||||
| 
 | ||||
|         if (datapack is null) | ||||
|         { | ||||
|             throw new JsonException($"Failed to deserialize szpack.json from {filePath}. The file may be corrupt or invalid."); | ||||
|         } | ||||
|          | ||||
|         return datapack; | ||||
|     } | ||||
| } | ||||
| @ -1,9 +0,0 @@ | ||||
| <Project Sdk="Microsoft.NET.Sdk"> | ||||
| 
 | ||||
|     <PropertyGroup> | ||||
|         <TargetFramework>net9.0</TargetFramework> | ||||
|         <ImplicitUsings>enable</ImplicitUsings> | ||||
|         <Nullable>enable</Nullable> | ||||
|     </PropertyGroup> | ||||
| 
 | ||||
| </Project> | ||||
| @ -1,45 +0,0 @@ | ||||
| using SessionZero.Shared.Models; | ||||
| 
 | ||||
| namespace SessionZero.Shared.Validation; | ||||
| 
 | ||||
| public static class DatapackValidator | ||||
| { | ||||
|     public static List<string> ValidateDatapack(Datapack datapack) | ||||
|     { | ||||
|         var errors = new List<string>(); | ||||
| 
 | ||||
|         if (datapack.Id == Guid.Empty) errors.Add("Datapack 'Id' is required and must not be an empty GUID."); | ||||
|          | ||||
|         ValidateRequiredString(errors, datapack.Name, nameof(datapack.Name)); | ||||
|         ValidateRequiredString(errors, datapack.Version, nameof(datapack.Version)); | ||||
|         ValidateRequiredString(errors, datapack.Author, nameof(datapack.Author)); | ||||
|         ValidateRequiredString(errors, datapack.License, nameof(datapack.License)); | ||||
|         ValidateRequiredString(errors, datapack.SessionZeroVersion, nameof(datapack.SessionZeroVersion)); | ||||
|          | ||||
|         if (datapack.CreatedAt == DateTime.MinValue) errors.Add("Datapack 'CreatedAt' is required and must not be an empty DateTime."); | ||||
|          | ||||
|         return errors; | ||||
|     } | ||||
|      | ||||
|     public static List<string> ValidateSzObject(SzObject szObject) | ||||
|     { | ||||
|         var errors = new List<string>(); | ||||
|         var objectName = szObject.Id; | ||||
|          | ||||
|         ValidateRequiredString(errors, szObject.Id, $"{objectName}'s Id"); | ||||
|         ValidateRequiredString(errors, szObject.Name, $"{objectName}'s Name"); | ||||
|         ValidateRequiredString(errors, szObject.SzType, $"{objectName}'s SzType"); | ||||
|         ValidateRequiredString(errors, szObject.Version, $"{objectName}'s Version"); | ||||
|         ValidateRequiredString(errors, szObject.SchemaVersion, $"{objectName}'s SchemaVersion"); | ||||
|          | ||||
|         // 2. Validate Icon path format (optional/future: e.g., only allows alphanumeric + slashes + extension) | ||||
|         // For now, we'll only check if it's not null, which is done by the property definition in SzObject. | ||||
|          | ||||
|         return errors; | ||||
|     } | ||||
| 
 | ||||
|     private static void ValidateRequiredString(List<string> errors, string? value, string fieldName) | ||||
|     { | ||||
|         if (string.IsNullOrWhiteSpace(value)) errors.Add($"'{fieldName}' is required and must not be empty."); | ||||
|     } | ||||
| } | ||||
| @ -1,107 +0,0 @@ | ||||
| /* | ||||
|  * WARNING: | ||||
|  * This tool was created by an LLM based on the SessionZero shared lib project, therefore it is subject to errors and does not reflect the architecture of the SessionZero project. | ||||
|  * It was created to be used as a quick and dirty validation tool for the szpack format. | ||||
| */ | ||||
| 
 | ||||
| using SessionZero.Shared.Models; | ||||
| using SessionZero.Shared.Services; | ||||
| using Spectre.Console; | ||||
| using Spectre.Console.Cli; | ||||
| using System.ComponentModel; | ||||
| using System.Diagnostics.CodeAnalysis; | ||||
| 
 | ||||
| namespace SessionZero.Tools.Packer; | ||||
| 
 | ||||
| // Settings class defines the command-line arguments | ||||
| public class CreateSettings : CommandSettings | ||||
| { | ||||
|     [CommandArgument(0, "<name>")] | ||||
|     [Description("The display name of the datapack (e.g., 'My Fantasy Spells').")] | ||||
|     public required string Name { get; init; } | ||||
| 
 | ||||
|     [CommandArgument(1, "[author]")]
 | ||||
|     [Description("The author's name.")] | ||||
|     [DefaultValue("Test Author")] | ||||
|     public string Author { get; init; } = "Test Author"; | ||||
| 
 | ||||
|     [CommandOption("-o|--output")] | ||||
|     [Description("The parent directory where the new pack folder will be created.")] | ||||
|     [DefaultValue(".")] | ||||
|     public string Output { get; init; } = Environment.CurrentDirectory; | ||||
| } | ||||
| 
 | ||||
| public class CreateCommand : AsyncCommand<CreateSettings> | ||||
| { | ||||
|     public override async Task<int> ExecuteAsync([NotNull] CommandContext context, [NotNull] CreateSettings settings, CancellationToken cancellationToken) | ||||
|     { | ||||
|         AnsiConsole.MarkupLine($"\n[bold white on blue] --- Creating Datapack: '{settings.Name}' --- [/]"); | ||||
| 
 | ||||
|         try | ||||
|         { | ||||
|             // 1. Create the top-level model and save szpack.json | ||||
|             var newPack = DatapackService.CreateEmptyDatapack(settings.Name, "1.0.0", settings.Author, "MIT"); | ||||
|             var szpackPath = await DatapackService.SaveDatapackMetadataAsync(newPack, settings.Output); | ||||
|             var packRootDirectory = Path.GetDirectoryName(szpackPath)!; | ||||
|              | ||||
|             AnsiConsole.MarkupLine($"[green]✅ Created metadata file at: {szpackPath}[/]"); | ||||
| 
 | ||||
|             // 2. Create the standard directory structure | ||||
|             var structure = DatapackService.CreateDatapackDirectoryStructure(packRootDirectory); | ||||
|             AnsiConsole.MarkupLine($"[green]✅ Created standard directories at: {packRootDirectory}[/]"); | ||||
|              | ||||
|             // 3. Create and save test objects | ||||
|             await CreateTestObjects(packRootDirectory, structure); | ||||
|              | ||||
|             AnsiConsole.MarkupLine("\n[bold]Creation Complete![/] Use 'szpack pack' to compress it."); | ||||
|             return 0; | ||||
|         } | ||||
|         catch (Exception ex) | ||||
|         { | ||||
|             AnsiConsole.MarkupLine($"\n[bold white on red]❌ ERROR:[/] Failed to create datapack. Details: {ex.Message}"); | ||||
|             return 1; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // Helper method for creating test objects (copied from previous Program.cs) | ||||
|     private static async Task CreateTestObjects(string rootPath, Dictionary<string, string> structure) | ||||
|     { | ||||
|         // ... (Test object creation logic remains the same, ensure you have the necessary usings in this file) | ||||
|          | ||||
|         // --- Test object creation logic (omitted for brevity, assume the previous logic is here) --- | ||||
|         var testDataset = new Dataset { /* ... test data ... */ Id = "basic-attributes", Name = "Basic Character Attributes", SzType = "Dataset", Version = "1.0.0", SchemaVersion = "1.0.0", DatasetType = "Attribute", Entries = new() }; | ||||
|         var dsPath = await DatapackService.SaveSzObjectAsync(testDataset, structure["datasets"]); | ||||
|         AnsiConsole.MarkupLine($"  -> Saved Test Dataset: [yellow]{Path.GetFileName(dsPath)}[/]"); | ||||
| 
 | ||||
|         var testCharTemplate = new CharacterTemplate { /* ... test data ... */ Id = "default-char-sheet", Name = "Default Character Template", SzType = "Template", Version = "1.0.0", SchemaVersion = "1.0.0", Sections = new() }; | ||||
|         var charPath = await DatapackService.SaveSzObjectAsync(testCharTemplate, structure["character_templates"]); | ||||
|         AnsiConsole.MarkupLine($"  -> Saved Test Character Template: [yellow]{Path.GetFileName(charPath)}[/]"); | ||||
| 
 | ||||
|         var testSessionTemplate = new SessionTemplate | ||||
|         { | ||||
|             /* ... test data ... */ Id = "basic-encounter", | ||||
|             Name = "Basic Encounter Template",  | ||||
|             SzType = "Template",  | ||||
|             Version = "1.0.0",  | ||||
|             SchemaVersion = "1.0.0",  | ||||
|             CharacterTemplateLink = new() | ||||
|             { | ||||
|                 DatapackId = Guid.Empty, | ||||
|                 TemplateId = testCharTemplate.Id, | ||||
|                 Version = testCharTemplate.Version | ||||
|             },  | ||||
|             RequiredDatasets = new() | ||||
|             { | ||||
|                 new() | ||||
|                 { | ||||
|                     DatapackId = Guid.Empty, | ||||
|                     DatasetId = testDataset.Id, | ||||
|                     Version = testDataset.Version | ||||
|                 } | ||||
|             },  | ||||
|             Sections = new() | ||||
|         }; | ||||
|         var sessionPath = await DatapackService.SaveSzObjectAsync(testSessionTemplate, structure["session_templates"]); | ||||
|         AnsiConsole.MarkupLine($"  -> Saved Test Session Template: [yellow]{Path.GetFileName(sessionPath)}[/]"); | ||||
|     } | ||||
| } | ||||
| @ -1,43 +0,0 @@ | ||||
| /* | ||||
|  * WARNING: | ||||
|  * This tool was created by an LLM based on the SessionZero shared lib project, therefore it is subject to errors and does not reflect the architecture of the SessionZero project. | ||||
|  * It was created to be used as a quick and dirty validation tool for the szpack format. | ||||
| */ | ||||
| 
 | ||||
| using SessionZero.Shared.Services; | ||||
| using Spectre.Console; | ||||
| using Spectre.Console.Cli; | ||||
| using System.ComponentModel; | ||||
| using System.Diagnostics.CodeAnalysis; | ||||
| 
 | ||||
| namespace SessionZero.Tools.Packer; | ||||
| 
 | ||||
| public class PackSettings : CommandSettings | ||||
| { | ||||
|     [CommandArgument(0, "<input>")] | ||||
|     [Description("The root directory of the datapack to pack.")] | ||||
|     public required string Input { get; init; } | ||||
| } | ||||
| 
 | ||||
| public class PackCommand : Command<PackSettings> | ||||
| { | ||||
|     public override int Execute([NotNull] CommandContext context, [NotNull] PackSettings settings, CancellationToken cancellationToken) | ||||
|     { | ||||
|         AnsiConsole.MarkupLine($"\n[bold white on blue] --- Packing Datapack: {settings.Input} --- [/]"); | ||||
|         try | ||||
|         { | ||||
|             var packDirectoryName = new DirectoryInfo(settings.Input).Name; | ||||
|             var outputFilePath = Path.Combine(Path.GetDirectoryName(settings.Input) ?? Environment.CurrentDirectory, $"{packDirectoryName}.szpack"); | ||||
|              | ||||
|             DatapackService.PackDatapack(settings.Input, outputFilePath); | ||||
|              | ||||
|             AnsiConsole.MarkupLine($"[green]✅ Successfully created archive: {outputFilePath}[/]"); | ||||
|             return 0; | ||||
|         } | ||||
|         catch (Exception ex) | ||||
|         { | ||||
|             AnsiConsole.MarkupLine($"\n[bold white on red]❌ ERROR:[/] Failed to pack datapack. Details: {ex.Message}"); | ||||
|             return 1; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -1,36 +0,0 @@ | ||||
| /* | ||||
|  * WARNING: | ||||
|  * This tool was created by an LLM based on the SessionZero shared lib project, therefore it is subject to errors and does not reflect the architecture of the SessionZero project. | ||||
|  * It was created to be used as a quick and dirty validation tool for the szpack format. | ||||
| */ | ||||
| 
 | ||||
| using Spectre.Console.Cli; | ||||
| 
 | ||||
| namespace SessionZero.Tools.Packer; | ||||
| 
 | ||||
| internal class Program | ||||
| { | ||||
|     static int Main(string[] args) | ||||
|     { | ||||
|         var app = new CommandApp(); | ||||
| 
 | ||||
|         app.Configure(config => | ||||
|         { | ||||
|             config.SetApplicationName("szpack"); | ||||
|              | ||||
|             config.AddCommand<CreateCommand>("create") | ||||
|                 .WithDescription("Creates a new datapack directory with test objects."); | ||||
|              | ||||
|             config.AddCommand<PackCommand>("pack") | ||||
|                 .WithDescription("Compresses a datapack directory into a .szpack file."); | ||||
|              | ||||
|             config.AddCommand<UnpackCommand>("unpack") | ||||
|                 .WithDescription("Extracts a .szpack file and displays its metadata."); | ||||
|              | ||||
|             // Optional: Set a default command if no command is specified | ||||
|             // config.SetDefaultCommand<HelpCommand>(); | ||||
|         }); | ||||
| 
 | ||||
|         return app.Run(args); | ||||
|     } | ||||
| } | ||||
| @ -1,55 +0,0 @@ | ||||
| /* | ||||
|  * WARNING: | ||||
|  * This tool was created by an LLM based on the SessionZero shared lib project, therefore it is subject to errors and does not reflect the architecture of the SessionZero project. | ||||
|  * It was created to be used as a quick and dirty validation tool for the szpack format. | ||||
| */ | ||||
| 
 | ||||
| using SessionZero.Shared.Services; | ||||
| using Spectre.Console; | ||||
| using Spectre.Console.Cli; | ||||
| using System.ComponentModel; | ||||
| using System.Diagnostics.CodeAnalysis; | ||||
| 
 | ||||
| namespace SessionZero.Tools.Packer; | ||||
| 
 | ||||
| public class UnpackSettings : CommandSettings | ||||
| { | ||||
|     [CommandArgument(0, "<input>")] | ||||
|     [Description("The path to the .szpack file.")] | ||||
|     public required string Input { get; init; } | ||||
| 
 | ||||
|     [CommandOption("-o|--output")] | ||||
|     [Description("The directory where the pack will be extracted.")] | ||||
|     [DefaultValue("unpacked")] | ||||
|     public string Output { get; init; } = "unpacked"; | ||||
| } | ||||
| 
 | ||||
| public class UnpackCommand : AsyncCommand<UnpackSettings> | ||||
| { | ||||
|     public override async Task<int> ExecuteAsync([NotNull] CommandContext context, [NotNull] UnpackSettings settings, CancellationToken cancellationToken) | ||||
|     { | ||||
|         AnsiConsole.MarkupLine($"\n[bold white on blue] --- Unpacking Datapack: {settings.Input} --- [/]"); | ||||
|         try | ||||
|         { | ||||
|             // Unpack the archive | ||||
|             DatapackService.UnpackDatapack(settings.Input, settings.Output); | ||||
|             AnsiConsole.MarkupLine($"[green]✅ Successfully unpacked archive to: {settings.Output}[/]"); | ||||
|              | ||||
|             // Load and display the core data | ||||
|             var datapack = await DatapackService.LoadDatapackMetadataAsync(settings.Output); | ||||
|              | ||||
|             AnsiConsole.MarkupLine("\n[bold]--- Datapack Info (szpack.json) ---[/]"); | ||||
|             AnsiConsole.MarkupLine($"[yellow]ID:[/]\t\t{datapack.Id}"); | ||||
|             AnsiConsole.MarkupLine($"[yellow]Name:[/]\t{datapack.Name}"); | ||||
|             AnsiConsole.MarkupLine($"[yellow]Version:[/]\t{datapack.Version}"); | ||||
|             AnsiConsole.MarkupLine($"[yellow]Author:[/]\t{datapack.Author}"); | ||||
|              | ||||
|             return 0; | ||||
|         } | ||||
|         catch (Exception ex) | ||||
|         { | ||||
|             AnsiConsole.MarkupLine($"\n[bold white on red]❌ ERROR:[/] Failed to unpack or load datapack. Details: {ex.Message}"); | ||||
|             return 1; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -1,10 +0,0 @@ | ||||
| # SZPACK CLI Tool | ||||
| 
 | ||||
| ## ***WARNING:*** | ||||
| This tool was created by an LLM based on the SessionZero shared lib project, therefore it is subject to errors and does not reflect the architecture of the SessionZero project. | ||||
| It was created to be used as a quick and dirty validation tool for the szpack format. | ||||
| 
 | ||||
| ## Usage | ||||
| * `szpack create <name>`: Creates a sample datapack directory with a given name | ||||
| * `szpack pack <directory>`: Packs a datapack directory into a szpack file (must be a valid datapack) | ||||
| * `szpack unpack <name.szpack>`: Unpacks a .szpack file into a datapack directory and displays some metadata | ||||
| @ -1,19 +0,0 @@ | ||||
| <Project Sdk="Microsoft.NET.Sdk"> | ||||
| 
 | ||||
|     <PropertyGroup> | ||||
|         <OutputType>Exe</OutputType> | ||||
|         <TargetFramework>net9.0</TargetFramework> | ||||
|         <ImplicitUsings>enable</ImplicitUsings> | ||||
|         <Nullable>enable</Nullable> | ||||
|     </PropertyGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|       <ProjectReference Include="..\..\src\SessionZero.Shared\SessionZero.Shared.csproj" /> | ||||
|     </ItemGroup> | ||||
| 
 | ||||
|     <ItemGroup> | ||||
|       <PackageReference Include="Spectre.Console" Version="0.52.1-preview.0.5" /> | ||||
|       <PackageReference Include="Spectre.Console.Cli" Version="0.52.1-preview.0.5" /> | ||||
|     </ItemGroup> | ||||
| 
 | ||||
| </Project> | ||||
| @ -1,2 +0,0 @@ | ||||
| Planned tools: | ||||
| - szpack: a cli tool for validating and packing szpack archives | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user