Summary
Publish the TimePro CLI package to NuGet.org as a .NET tool so users can install and update tp directly, without cloning the repository and packing a local .nupkg first.
Why
The current install path works, but it is developer-heavy:
git clone https://github.com/SSWConsulting/timepro.tools.git
cd timepro.tools
dotnet pack src/SSW.TimePro.Cli/ -o src/SSW.TimePro.Cli/nupkg
dotnet tool install -g --add-source src/SSW.TimePro.Cli/nupkg SSW.TimePro.Cli
Publishing to NuGet would make the install experience much cleaner:
dotnet tool install -g SSW.TimePro.Cli
And updates become:
dotnet tool update -g SSW.TimePro.Cli
This helps because:
- New users do not need to clone, build, or understand local NuGet sources.
- Existing users can update with the standard .NET tool workflow.
- Agent setup becomes easier because instructions can depend on one stable install command.
- Releases become traceable and repeatable through CI rather than local hand-built packages.
- NuGet.org gives the package a discoverable public install surface.
Current state
The project is already mostly set up as a .NET tool in src/SSW.TimePro.Cli/SSW.TimePro.Cli.csproj:
<PackAsTool>true</PackAsTool>
<ToolCommandName>tp</ToolCommandName>
<PackageId>SSW.TimePro.Cli</PackageId>
<Version>0.1.0</Version>
<Authors>SSW</Authors>
<Description>CLI-first tool with MCP support for managing SSW TimePro timesheets</Description>
<PackageLicenseExpression>MIT</PackageLicenseExpression>
Local verification on 2026-06-17:
dotnet pack src/SSW.TimePro.Cli/SSW.TimePro.Cli.csproj -c Release -o /tmp/timepro-tool-pack-test
dotnet tool install SSW.TimePro.Cli --tool-path /tmp/timepro-tool-install-test --add-source /tmp/timepro-tool-pack-test --version 0.1.0
/tmp/timepro-tool-install-test/tp --help
Result: package builds, installs, and tp --help works.
Tests also passed locally:
dotnet test tests/SSW.TimePro.Cli.Tests/SSW.TimePro.Cli.Tests.csproj --no-restore
dotnet test tests/SSW.TimePro.Cli.Integration/SSW.TimePro.Cli.Integration.csproj --no-restore
Result: 8 unit tests passed, 41 integration tests passed.
NuGet.org status checked on 2026-06-17:
dotnet tool search SSW.TimePro.Cli returned no results.
https://api.nuget.org/v3-flatcontainer/ssw.timepro.cli/index.json returned 404.
That suggests the package ID is currently available, but NuGet.org will still be the final authority at publish time.
Scope
This issue is only about making the existing CLI installable and updateable as a NuGet-hosted .NET tool.
Do not include unrelated package upgrades or broad runtime modernization here unless they are strictly required to make packaging, installation, or publishing work.
What needs to change
1. Add NuGet package metadata
The current package works, but the generated .nupkg is bare and dotnet pack warns that it has no package readme.
Add metadata such as:
<PackageReadmeFile>README.md</PackageReadmeFile>
<PackageProjectUrl>https://github.com/SSWConsulting/timepro.tools</PackageProjectUrl>
<RepositoryUrl>https://github.com/SSWConsulting/timepro.tools.git</RepositoryUrl>
<RepositoryType>git</RepositoryType>
<PackageTags>timepro;timesheets;cli;mcp;dotnet-tool</PackageTags>
<PackageReleaseNotes>Initial NuGet tool release.</PackageReleaseNotes>
Include the root README in the package from the project file:
<ItemGroup>
<None Include="../../README.md" Pack="true" PackagePath="\" />
</ItemGroup>
Optional but nice:
- Add
PackageIcon if we have a small approved icon asset.
- Add Source Link/package provenance later if needed.
2. Use one source of truth for the version
tp --version currently comes from a hardcoded value in Program.cs:
config.SetApplicationVersion("0.1.0");
That should read from assembly/package metadata instead, so release bumps do not drift between CLI output and NuGet package version.
3. Document the runtime requirement
The package currently targets net10.0, so users need a compatible .NET 10 runtime/SDK.
Document that clearly in the README and release notes. Investigating a lower target framework can be a separate issue if needed.
4. Add CI release workflow
Add a GitHub Actions workflow that:
- Runs on tags such as
v* or on GitHub Releases.
- Restores NuGet packages.
- Runs unit and integration tests.
- Packs the tool in Release mode.
- Publishes the
.nupkg to NuGet.org.
Preferred publishing auth:
- Use NuGet.org Trusted Publishing with GitHub Actions OIDC if available.
- Fallback: use a scoped NuGet API key stored as a GitHub Actions secret.
- Do not commit API keys or write them to repo files.
Trusted Publishing shape:
permissions:
id-token: write
steps:
- uses: actions/checkout@v4
- uses: actions/setup-dotnet@v4
with:
dotnet-version: '10.0.x'
- run: dotnet restore SSW.TimePro.Timesheets.Cli.slnx
- run: dotnet test SSW.TimePro.Timesheets.Cli.slnx -c Release --no-restore
- run: dotnet pack src/SSW.TimePro.Cli/SSW.TimePro.Cli.csproj -c Release -o artifacts/packages --no-restore
- uses: NuGet/login@v1
id: login
with:
user: ${{ secrets.NUGET_USER }}
- run: dotnet nuget push "artifacts/packages/*.nupkg" --api-key ${{ steps.login.outputs.NUGET_API_KEY }} --source https://api.nuget.org/v3/index.json --skip-duplicate
If Trusted Publishing is not available, use NUGET_API_KEY as a GitHub secret and keep it scoped to this package.
Test plan
Before publishing:
dotnet restore SSW.TimePro.Timesheets.Cli.slnx
dotnet test tests/SSW.TimePro.Cli.Tests/SSW.TimePro.Cli.Tests.csproj --no-restore
dotnet test tests/SSW.TimePro.Cli.Integration/SSW.TimePro.Cli.Integration.csproj --no-restore
dotnet pack src/SSW.TimePro.Cli/SSW.TimePro.Cli.csproj -c Release -o /tmp/timepro-tool-pack-test --no-restore
Inspect the package contents and manifest:
unzip -l /tmp/timepro-tool-pack-test/SSW.TimePro.Cli.*.nupkg
unzip -p /tmp/timepro-tool-pack-test/SSW.TimePro.Cli.*.nupkg SSW.TimePro.Cli.nuspec
unzip -p /tmp/timepro-tool-pack-test/SSW.TimePro.Cli.*.nupkg tools/net10.0/any/DotnetToolSettings.xml
Verify a clean local install from the generated package:
rm -rf /tmp/timepro-tool-install-test
dotnet tool install SSW.TimePro.Cli \
--tool-path /tmp/timepro-tool-install-test \
--add-source /tmp/timepro-tool-pack-test \
--version <VERSION> \
--ignore-failed-sources
/tmp/timepro-tool-install-test/tp --help
/tmp/timepro-tool-install-test/tp --version
Verify the global install/update flow in a controlled way before publishing by using a prerelease or incremented local version:
dotnet tool uninstall -g SSW.TimePro.Cli || true
dotnet tool install -g SSW.TimePro.Cli --add-source /tmp/timepro-tool-pack-test --version <VERSION> --ignore-failed-sources
tp --help
tp --version
dotnet tool update -g SSW.TimePro.Cli --add-source /tmp/timepro-tool-pack-test --version <NEXT_VERSION> --ignore-failed-sources
tp --version
dotnet tool uninstall -g SSW.TimePro.Cli
After publishing to NuGet.org, verify the real user path from NuGet.org only:
dotnet tool install -g SSW.TimePro.Cli --version <VERSION>
tp --help
tp --version
dotnet tool update -g SSW.TimePro.Cli
dotnet tool uninstall -g SSW.TimePro.Cli
Acceptance criteria
dotnet tool install -g SSW.TimePro.Cli
dotnet tool update -g SSW.TimePro.Cli
References
Summary
Publish the TimePro CLI package to NuGet.org as a .NET tool so users can install and update
tpdirectly, without cloning the repository and packing a local.nupkgfirst.Why
The current install path works, but it is developer-heavy:
git clone https://github.com/SSWConsulting/timepro.tools.git cd timepro.tools dotnet pack src/SSW.TimePro.Cli/ -o src/SSW.TimePro.Cli/nupkg dotnet tool install -g --add-source src/SSW.TimePro.Cli/nupkg SSW.TimePro.CliPublishing to NuGet would make the install experience much cleaner:
And updates become:
This helps because:
Current state
The project is already mostly set up as a .NET tool in
src/SSW.TimePro.Cli/SSW.TimePro.Cli.csproj:Local verification on 2026-06-17:
Result: package builds, installs, and
tp --helpworks.Tests also passed locally:
Result: 8 unit tests passed, 41 integration tests passed.
NuGet.org status checked on 2026-06-17:
dotnet tool search SSW.TimePro.Clireturned no results.https://api.nuget.org/v3-flatcontainer/ssw.timepro.cli/index.jsonreturned 404.That suggests the package ID is currently available, but NuGet.org will still be the final authority at publish time.
Scope
This issue is only about making the existing CLI installable and updateable as a NuGet-hosted .NET tool.
Do not include unrelated package upgrades or broad runtime modernization here unless they are strictly required to make packaging, installation, or publishing work.
What needs to change
1. Add NuGet package metadata
The current package works, but the generated
.nupkgis bare anddotnet packwarns that it has no package readme.Add metadata such as:
Include the root README in the package from the project file:
Optional but nice:
PackageIconif we have a small approved icon asset.2. Use one source of truth for the version
tp --versioncurrently comes from a hardcoded value inProgram.cs:That should read from assembly/package metadata instead, so release bumps do not drift between CLI output and NuGet package version.
3. Document the runtime requirement
The package currently targets
net10.0, so users need a compatible .NET 10 runtime/SDK.Document that clearly in the README and release notes. Investigating a lower target framework can be a separate issue if needed.
4. Add CI release workflow
Add a GitHub Actions workflow that:
v*or on GitHub Releases..nupkgto NuGet.org.Preferred publishing auth:
Trusted Publishing shape:
If Trusted Publishing is not available, use
NUGET_API_KEYas a GitHub secret and keep it scoped to this package.Test plan
Before publishing:
Inspect the package contents and manifest:
Verify a clean local install from the generated package:
Verify the global install/update flow in a controlled way before publishing by using a prerelease or incremented local version:
After publishing to NuGet.org, verify the real user path from NuGet.org only:
Acceptance criteria
tp --versioncomes from assembly/package metadata, not a second hardcoded version string..nupkgin a clean tool path.References