mirror of
https://github.com/jakeswenson/BitBetter.git
synced 2025-12-20 21:46:41 +00:00
Compare commits
1 Commits
lite
...
229894f944
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
229894f944 |
@@ -1,21 +1,15 @@
|
|||||||
version: 2.1
|
version: 2.1
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
machine: true
|
machine: true
|
||||||
steps:
|
steps:
|
||||||
- checkout
|
- checkout
|
||||||
- run:
|
- run:
|
||||||
name: Print the Current Time
|
name: Print the Current Time
|
||||||
command: date
|
command: date
|
||||||
- run:
|
- run:
|
||||||
name: Generate Keys
|
name: Generate Keys
|
||||||
command: ./generateKeys.sh
|
command: ./generateKeys.sh
|
||||||
- run:
|
- run:
|
||||||
name: Build script
|
name: Build script
|
||||||
command: ./build.sh update
|
command: ./build.sh y
|
||||||
- run:
|
|
||||||
name: Test generating user license
|
|
||||||
command: ./licenseGen.sh user TestName TestEmail@example.com 4a619d4a-522d-4c70-8596-affb5b607c23
|
|
||||||
- run:
|
|
||||||
name: Test generating organization license
|
|
||||||
command: ./licenseGen.sh org TestName TestEmail@example.com 4a619d4a-522d-4c70-8596-affb5b607c23
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
root=true
|
|
||||||
|
|
||||||
###############################
|
|
||||||
# Core EditorConfig Options #
|
|
||||||
###############################
|
|
||||||
# All files
|
|
||||||
[*]
|
|
||||||
indent_style=tab
|
|
||||||
indent_size=4
|
|
||||||
trim_trailing_whitespace=true
|
|
||||||
end_of_line=lf
|
|
||||||
charset=utf-8
|
|
||||||
|
|
||||||
[*.{cs}]
|
|
||||||
insert_final_newline=false
|
|
||||||
|
|
||||||
[*.{md,mkdn}]
|
|
||||||
trim_trailing_whitespace = true
|
|
||||||
indent_style = space
|
|
||||||
1
.gitattributes
vendored
1
.gitattributes
vendored
@@ -1 +0,0 @@
|
|||||||
* text eol=lf
|
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -7,6 +7,5 @@ src/bitBetter/.vs/*
|
|||||||
*.pem
|
*.pem
|
||||||
.vscode/
|
.vscode/
|
||||||
*.pfx
|
*.pfx
|
||||||
*.cer
|
*.cert
|
||||||
*.vsidx
|
*.vsidx
|
||||||
.DS_Store
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Uncomment a line below and fill in the missing values or add your own. Every line in this file will be called by build.[sh|ps1] once the patched image is built.
|
# Uncomment a line below and fill in the missing values or add your own. Every line in this file will be called by build.[sh|ps1] once the patched image is built.
|
||||||
# docker run -d --name bitwarden --restart=always -v <full-local-path>\logs:/var/log/bitwarden -v <full-local-path>\bwdata:/etc/bitwarden -p 80:8080 --env-file <full-local-path>\settings.env bitwarden-patched
|
# docker run -d --name bitwarden -v <full-local-path>\logs:/var/log/bitwarden -v <full-local-path>\bwdata:/etc/bitwarden -p 80:8080 --env-file <full-local-path>\settings.env bitwarden-patch
|
||||||
# <OR>
|
# <OR>
|
||||||
# docker-compose -f <full-local-path>/docker-compose.yml up -d
|
# docker-compose -f <full-local-path>/docker-compose.yml up -d
|
||||||
|
|||||||
68
README.md
68
README.md
@@ -1,14 +1,12 @@
|
|||||||
# BitBetter lite
|
# BitBetter
|
||||||
|
|
||||||
BitBetter is is a tool to modify Bitwarden's core dll to allow you to generate your own individual and organisation licenses.
|
BitBetter is is a tool to modify Bitwarden's core dll to allow you to generate your own individual and organisation licenses.
|
||||||
|
|
||||||
Please see the FAQ below for details on why this software was created.
|
Please see the FAQ below for details on why this software was created.
|
||||||
|
|
||||||
Be aware that this branch is **only** for the lite (formerly unified) version of bitwarden. It has been rewritten and works in different ways than the master branch.
|
_Beware! BitBetter does some semi janky stuff to rewrite the bitwarden core dll and allow the installation of a self signed certificate. Use at your own risk!_
|
||||||
|
|
||||||
_Beware! BitBetter is a solution that generates a personal certificate and uses that to generate custom licences. This requires (automated) modifying of libraries. Use at your own risk!_
|
Credit to https://github.com/h44z/BitBetter and https://github.com/jakeswenson/BitBetter
|
||||||
|
|
||||||
Credit to https://github.com/h44z/BitBetter and https://github.com/jakeswenson/BitBetter and https://github.com/GieltjE/BitBetter
|
|
||||||
|
|
||||||
# Table of Contents
|
# Table of Contents
|
||||||
- [BitBetter](#bitbetter)
|
- [BitBetter](#bitbetter)
|
||||||
@@ -32,7 +30,7 @@ The following instructions are for unix-based systems (Linux, BSD, macOS) and Wi
|
|||||||
## Dependencies
|
## Dependencies
|
||||||
Aside from docker, which you also need for Bitwarden, BitBetter requires the following:
|
Aside from docker, which you also need for Bitwarden, BitBetter requires the following:
|
||||||
|
|
||||||
* Bitwarden (tested with 2025.11.1 might work on lower versions), for safety always stay up to date
|
* Bitwarden (tested with 1.47.1, might work on lower versions)
|
||||||
* openssl (probably already installed on most Linux or WSL systems, any version should work, on Windows it will be auto installed using winget)
|
* openssl (probably already installed on most Linux or WSL systems, any version should work, on Windows it will be auto installed using winget)
|
||||||
|
|
||||||
## Setting up BitBetter
|
## Setting up BitBetter
|
||||||
@@ -48,8 +46,8 @@ If you wish to generate your self-signed cert & key manually, you can run the fo
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd .keys
|
cd .keys
|
||||||
openssl req -x509 -newkey rsa:4096 -keyout key.pem -out cert.cer -days 36500 -outform DER -passout pass:test
|
openssl req -x509 -newkey rsa:4096 -keyout key.pem -out cert.cert -days 36500 -outform DER -passout pass:test
|
||||||
openssl x509 -inform DER -in cert.cer -out cert.pem
|
openssl x509 -inform DER -in cert.cert -out cert.pem
|
||||||
openssl pkcs12 -export -out cert.pfx -inkey key.pem -in cert.pem -passin pass:test -passout pass:test
|
openssl pkcs12 -export -out cert.pfx -inkey key.pem -in cert.pem -passin pass:test -passout pass:test
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -65,14 +63,14 @@ The scripts supports running and patching multi instances.
|
|||||||
Edit the .servers/serverlist.txt file and fill in the missing values, they can be replaced with existing installation values.
|
Edit the .servers/serverlist.txt file and fill in the missing values, they can be replaced with existing installation values.
|
||||||
This file may be empty, but there will be no containers will be spun up automatically.
|
This file may be empty, but there will be no containers will be spun up automatically.
|
||||||
|
|
||||||
Now it is time to **run the main build script** to generate a modified version of the `ghcr.io/bitwarden/lite` docker image and the license generator.
|
Now it is time to **run the main build script** to generate a modified version of the `ghcr.io/bitwarden/self-host` docker image and the license generator.
|
||||||
|
|
||||||
From the BitBetter directory, simply run:
|
From the BitBetter directory, simply run:
|
||||||
```
|
```
|
||||||
./build.[sh|ps1]
|
./build.[sh|ps1]
|
||||||
```
|
```
|
||||||
|
|
||||||
This will create a new self-signed certificate in the `.keys` directory if one does not already exist and then create a modified version of the official `ghcr.io/bitwarden/lite` image called `bitwarden-patched`.
|
This will create a new self-signed certificate in the `.keys` directory if one does not already exist and then create a modified version of the official `ghcr.io/bitwarden/self-host` image called `bitwarden-patch`.
|
||||||
|
|
||||||
Afterwards it will automatically generate the license generator and start all previously specified containers which are **now ready to accept self-issued licenses.**
|
Afterwards it will automatically generate the license generator and start all previously specified containers which are **now ready to accept self-issued licenses.**
|
||||||
|
|
||||||
@@ -100,36 +98,6 @@ If you ran the build script, you can **simply run the license gen in interactive
|
|||||||
**The license generator will spit out a JSON-formatted license which can then be used within the Bitwarden web front-end to license your user or org!**
|
**The license generator will spit out a JSON-formatted license which can then be used within the Bitwarden web front-end to license your user or org!**
|
||||||
|
|
||||||
|
|
||||||
## Migrating from mssql to a real database
|
|
||||||
|
|
||||||
Prepare a new database and bwdata directory, download and prepare the new settings.env (https://raw.githubusercontent.com/bitwarden/self-host/refs/heads/main/bitwarden-lite/settings.env)
|
|
||||||
|
|
||||||
Make sure you can get the data from either the backup file or by connecting directly to the mssql database (navicat has a trial).
|
|
||||||
|
|
||||||
If required (e.g. you cannot connect to your docker mssql server directly) download Microsoft SQL Server 2022 and SQL Server Management Studio (the latter can be used to import the .bak file)
|
|
||||||
|
|
||||||
After cloning this repo and modifying .servers/serverlist.txt to suit your new environment do the following:
|
|
||||||
|
|
||||||
```
|
|
||||||
docker exec -i bitwarden-mssql /backup-db.sh
|
|
||||||
./bitwarden.sh stop
|
|
||||||
```
|
|
||||||
|
|
||||||
Run build.sh and ensure your new instance serves a webpage AND has populated the new database with the tables (should be empty now)
|
|
||||||
|
|
||||||
Proceed to stop the new container for now.
|
|
||||||
|
|
||||||
Copy from the old to the new bwdata directory (do not copy/overwrite identity.pfx!):
|
|
||||||
- bwdata/core/licenses to bwdata-new/licenses
|
|
||||||
- bwdata/core/aspnet-dataprotection to bwdata-new/data-protection
|
|
||||||
- bwdata/core/attachments to bwdata-new/attachments
|
|
||||||
|
|
||||||
Export data only from the old sql server database, if needed import the .bak file to a local mssql instance.
|
|
||||||
|
|
||||||
Only export tables that have rows, makes it much quicker, .json is the easiest with navicat.
|
|
||||||
|
|
||||||
Import the rows to the real database, start the new docker container.
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
# FAQ: Questions you might have.
|
# FAQ: Questions you might have.
|
||||||
@@ -146,26 +114,6 @@ In the past we have done so but they were not focused on the type of customer th
|
|||||||
|
|
||||||
UPDATE: Bitwarden now offers a cheap license called [Families Organization](https://bitwarden.com/pricing/) that provides premium features and the ability to self-host Bitwarden for six persons.
|
UPDATE: Bitwarden now offers a cheap license called [Families Organization](https://bitwarden.com/pricing/) that provides premium features and the ability to self-host Bitwarden for six persons.
|
||||||
|
|
||||||
## 2fa doesn't work
|
|
||||||
|
|
||||||
Unfortunately the new BitWarden container doesn't set the timezone and ignores TZ= from the environment, can be fixed by:
|
|
||||||
|
|
||||||
```
|
|
||||||
docker exec bitwarden ln -s /usr/share/zoneinfo/Europe/Amsterdam /etc/localtime
|
|
||||||
```
|
|
||||||
|
|
||||||
## Changes in settings.env
|
|
||||||
|
|
||||||
Require a recreation of the docker container, build.sh will suffice too.
|
|
||||||
|
|
||||||
## Migrating from the old unified branch
|
|
||||||
|
|
||||||
```
|
|
||||||
git branch -m unified lite
|
|
||||||
git fetch origin
|
|
||||||
git branch -u origin/lite lite
|
|
||||||
git remote set-head origin -a
|
|
||||||
```
|
|
||||||
|
|
||||||
# Footnotes
|
# Footnotes
|
||||||
|
|
||||||
|
|||||||
236
build.ps1
236
build.ps1
@@ -1,132 +1,104 @@
|
|||||||
$ErrorActionPreference = 'Stop'
|
# define temporary directory
|
||||||
$PSNativeCommandUseErrorActionPreference = $true
|
$tempdirectory = "$pwd\temp"
|
||||||
|
# define services to patch
|
||||||
# detect buildx, ErrorActionPreference will ensure the script stops execution if not found
|
$components = "Api","Identity"
|
||||||
docker buildx version
|
|
||||||
|
# delete old directories / files if applicable
|
||||||
# Enable BuildKit for better build experience and to ensure platform args are populated
|
if (Test-Path "$tempdirectory") {
|
||||||
$env:DOCKER_BUILDKIT=1
|
Remove-Item "$tempdirectory" -Recurse -Force
|
||||||
$env:COMPOSE_DOCKER_CLI_BUILD=1
|
}
|
||||||
|
|
||||||
# define temporary directory
|
if (Test-Path -Path "$pwd\src\licenseGen\Core.dll" -PathType Leaf) {
|
||||||
$tempdirectory = "$pwd\temp"
|
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
||||||
# define services to patch
|
}
|
||||||
$components = "Api","Identity"
|
|
||||||
|
if (Test-Path -Path "$pwd\src\licenseGen\cert.pfx" -PathType Leaf) {
|
||||||
# delete old directories / files if applicable
|
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
||||||
if (Test-Path "$tempdirectory" -PathType Container) {
|
}
|
||||||
Remove-Item "$tempdirectory" -Recurse -Force
|
|
||||||
}
|
if (Test-Path -Path "$pwd\src\bitBetter\cert.cert" -PathType Leaf) {
|
||||||
|
Remove-Item "$pwd\src\bitBetter\cert.cert" -Force
|
||||||
if (Test-Path -Path "$pwd\src\licenseGen\Core.dll" -PathType Leaf) {
|
}
|
||||||
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
|
||||||
}
|
# generate keys if none are available
|
||||||
|
if (!(Test-Path "$pwd\.keys")) {
|
||||||
if (Test-Path -Path "$pwd\src\licenseGen\cert.pfx" -PathType Leaf) {
|
.\generateKeys.ps1
|
||||||
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
}
|
||||||
}
|
|
||||||
|
# copy the key to bitBetter and licenseGen
|
||||||
if (Test-Path -Path "$pwd\src\bitBetter\cert.cer" -PathType Leaf) {
|
Copy-Item "$pwd\.keys\cert.cert" -Destination "$pwd\src\bitBetter"
|
||||||
Remove-Item "$pwd\src\bitBetter\cert.cer" -Force
|
Copy-Item "$pwd\.keys\cert.pfx" -Destination "$pwd\src\licenseGen"
|
||||||
}
|
|
||||||
|
# build bitBetter and clean the source directory after
|
||||||
if (Test-Path "$pwd\.keys\cert.cert" -PathType Leaf) {
|
docker build -t bitbetter/bitbetter "$pwd\src\bitBetter"
|
||||||
Rename-Item -Path "$pwd\.keys\cert.cert" -NewName "$pwd\.keys\cert.cer"
|
Remove-Item "$pwd\src\bitBetter\cert.cert" -Force
|
||||||
}
|
|
||||||
|
# gather all running instances
|
||||||
# generate keys if none are available
|
$oldinstances = docker container ps --all -f Name=bitwarden --format '{{.ID}}'
|
||||||
if (!(Test-Path "$pwd\.keys" -PathType Container)) {
|
|
||||||
.\generateKeys.ps1
|
# stop all running instances
|
||||||
}
|
foreach ($instance in $oldinstances) {
|
||||||
|
docker stop $instance
|
||||||
# copy the key to bitBetter
|
docker rm $instance
|
||||||
Copy-Item "$pwd\.keys\cert.cer" -Destination "$pwd\src\bitBetter"
|
}
|
||||||
|
|
||||||
# build bitBetter and clean the source directory after
|
# update bitwarden itself
|
||||||
docker build --no-cache -t bitbetter/bitbetter "$pwd\src\bitBetter"
|
if ($args[0] -eq 'y')
|
||||||
Remove-Item "$pwd\src\bitBetter\cert.cer" -Force
|
{
|
||||||
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
# gather all running instances, cannot run a wildcard filter on Ancestor= :(, does find all where name = *bitwarden*
|
}
|
||||||
$oldinstances = docker container ps --all -f Name=bitwarden --format '{{.ID}}'
|
else
|
||||||
|
{
|
||||||
# stop and remove all running instances
|
$confirmation = Read-Host "Update (or get) bitwarden source container"
|
||||||
foreach ($instance in $oldinstances) {
|
if ($confirmation -eq 'y') {
|
||||||
docker stop $instance
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
docker rm $instance
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# update bitwarden itself
|
# stop and remove previous existing patch(ed) container
|
||||||
if ($args[0] -eq 'update') {
|
docker stop bitwarden-patch
|
||||||
docker pull ghcr.io/bitwarden/lite:latest
|
docker rm bitwarden-patch
|
||||||
} else {
|
docker image rm bitwarden-patch
|
||||||
$confirmation = Read-Host "Update (or get) bitwarden source container (y/n)"
|
|
||||||
if ($confirmation -eq 'y') {
|
# start a new bitwarden instance so we can patch it
|
||||||
docker pull ghcr.io/bitwarden/lite:latest
|
$patchinstance = docker run -d --name bitwarden-patch ghcr.io/bitwarden/self-host:beta
|
||||||
}
|
|
||||||
}
|
# create our temporary directory
|
||||||
|
New-item -ItemType Directory -Path $tempdirectory
|
||||||
# stop and remove previous existing patch(ed) container
|
|
||||||
$oldinstances = docker container ps --all -f Ancestor=bitwarden-patched --format '{{.ID}}'
|
# extract the files that need to be patched from the services that need to be patched into our temporary directory
|
||||||
foreach ($instance in $oldinstances) {
|
foreach ($component in $components) {
|
||||||
docker stop $instance
|
New-item -itemtype Directory -path "$tempdirectory\$component"
|
||||||
docker rm $instance
|
docker cp $patchinstance`:/app/$component/Core.dll "$tempdirectory\$component\Core.dll"
|
||||||
}
|
}
|
||||||
$oldinstances = docker image ls bitwarden-patched --format '{{.ID}}'
|
|
||||||
foreach ($instance in $oldinstances) {
|
# run bitBetter, this applies our patches to the required files
|
||||||
docker image rm $instance
|
docker run -v "$tempdirectory`:/app/mount" --rm bitbetter/bitbetter
|
||||||
}
|
|
||||||
|
# create a new image with the patched files
|
||||||
# remove old extract containers
|
docker build . --tag bitwarden-patch --file "$pwd\src\bitBetter\Dockerfile-bitwarden-patch"
|
||||||
$oldinstances = docker container ps --all -f Name=bitwarden-extract --format '{{.ID}}'
|
|
||||||
foreach ($instance in $oldinstances) {
|
# stop and remove our temporary container
|
||||||
docker stop $instance
|
docker stop bitwarden-patch
|
||||||
docker rm $instance
|
docker rm bitwarden-patch
|
||||||
}
|
|
||||||
|
# copy our patched library to the licenseGen source directory
|
||||||
# start a new bitwarden instance so we can patch it
|
Copy-Item "$tempdirectory\Identity\Core.dll" -Destination "$pwd\src\licenseGen"
|
||||||
$patchinstance = docker run -d --name bitwarden-extract ghcr.io/bitwarden/lite:latest
|
|
||||||
|
# remove our temporary directory
|
||||||
# create our temporary directory
|
Remove-Item "$tempdirectory" -Recurse -Force
|
||||||
New-item -ItemType Directory -Path $tempdirectory
|
|
||||||
|
# start all user requested instances
|
||||||
# extract the files that need to be patched from the services that need to be patched into our temporary directory
|
foreach($line in Get-Content "$pwd\.servers\serverlist.txt") {
|
||||||
foreach ($component in $components) {
|
Invoke-Expression "& $line"
|
||||||
New-item -itemtype Directory -path "$tempdirectory\$component"
|
}
|
||||||
docker cp $patchinstance`:/app/$component/Core.dll "$tempdirectory\$component\Core.dll"
|
|
||||||
}
|
# remove our bitBetter image
|
||||||
|
docker image rm bitbetter/bitbetter
|
||||||
# stop and remove our temporary container
|
|
||||||
docker stop bitwarden-extract
|
# build the licenseGen
|
||||||
docker rm bitwarden-extract
|
docker build -t bitbetter/licensegen "$pwd\src\licenseGen"
|
||||||
|
|
||||||
# run bitBetter, this applies our patches to the required files
|
# clean the licenseGen source directory
|
||||||
docker run -v "$tempdirectory`:/app/mount" --rm bitbetter/bitbetter
|
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
||||||
|
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
||||||
# create a new image with the patched files
|
|
||||||
docker build . --tag bitwarden-patched --file "$pwd\src\bitBetter\Dockerfile-bitwarden-patch"
|
|
||||||
|
|
||||||
# start all user requested instances
|
|
||||||
if (Test-Path -Path "$pwd\.servers\serverlist.txt" -PathType Leaf) {
|
|
||||||
foreach($line in Get-Content "$pwd\.servers\serverlist.txt") {
|
|
||||||
if (!($line.StartsWith("#"))) {
|
|
||||||
Invoke-Expression "& $line"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
# remove our bitBetter image
|
|
||||||
docker image rm bitbetter/bitbetter
|
|
||||||
|
|
||||||
# copy our patched library to the licenseGen source directory
|
|
||||||
Copy-Item "$tempdirectory\Identity\Core.dll" -Destination "$pwd\src\licenseGen"
|
|
||||||
Copy-Item "$pwd\.keys\cert.pfx" -Destination "$pwd\src\licenseGen"
|
|
||||||
|
|
||||||
# build the licenseGen
|
|
||||||
docker build -t bitbetter/licensegen "$pwd\src\licenseGen"
|
|
||||||
|
|
||||||
# clean the licenseGen source directory
|
|
||||||
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
|
||||||
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
|
||||||
|
|
||||||
# remove our temporary directory
|
|
||||||
Remove-Item "$tempdirectory" -Recurse -Force
|
|
||||||
|
|||||||
103
build.sh
103
build.sh
@@ -1,12 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -e
|
|
||||||
|
|
||||||
# detect buildx, set -e will ensure the script stops execution if not found
|
|
||||||
docker buildx version
|
|
||||||
|
|
||||||
# Enable BuildKit for better build experience and to ensure platform args are populated
|
|
||||||
export DOCKER_BUILDKIT=1
|
|
||||||
export COMPOSE_DOCKER_CLI_BUILD=1
|
|
||||||
|
|
||||||
# define temporary directory
|
# define temporary directory
|
||||||
TEMPDIRECTORY="$PWD/temp"
|
TEMPDIRECTORY="$PWD/temp"
|
||||||
@@ -20,19 +12,15 @@ if [ -d "$TEMPDIRECTORY" ]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -f "$PWD/src/licenseGen/Core.dll" ]; then
|
if [ -f "$PWD/src/licenseGen/Core.dll" ]; then
|
||||||
rm -f "$PWD/src/licenseGen/Core.dll"
|
rm -f "$PWD/src/licenseGen/Core.dll"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -f "$PWD/src/licenseGen/cert.pfx" ]; then
|
if [ -f "$PWD/src/licenseGen/cert.pfx" ]; then
|
||||||
rm -f "$PWD/src/licenseGen/cert.pfx"
|
rm -f "$PWD/src/licenseGen/cert.pfx"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -f "$PWD/src/bitBetter/cert.cer" ]; then
|
if [ -f "$PWD/src/bitBetter/cert.cert" ]; then
|
||||||
rm -f "$PWD/src/bitBetter/cert.cer"
|
rm -f "$PWD/src/bitBetter/cert.cert"
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f "$PWD/.keys/cert.cert" ]; then
|
|
||||||
mv "$PWD/.keys/cert.cert" "$PWD/.keys/cert.cer"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# generate keys if none are available
|
# generate keys if none are available
|
||||||
@@ -40,52 +28,42 @@ if [ ! -d "$PWD/.keys" ]; then
|
|||||||
./generateKeys.sh
|
./generateKeys.sh
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# copy the key to bitBetter
|
# copy the key to bitBetter and licenseGen
|
||||||
cp -f "$PWD/.keys/cert.cer" "$PWD/src/bitBetter"
|
cp -f "$PWD/.keys/cert.cert" "$PWD/src/bitBetter"
|
||||||
|
cp -f "$PWD/.keys/cert.pfx" "$PWD/src/licenseGen"
|
||||||
|
|
||||||
# build bitBetter and clean the source directory after
|
# build bitBetter and clean the source directory after
|
||||||
docker build --no-cache -t bitbetter/bitbetter "$PWD/src/bitBetter"
|
docker build -t bitbetter/bitbetter "$PWD/src/bitBetter"
|
||||||
rm -f "$PWD/src/bitBetter/cert.cer"
|
rm -f "$PWD/src/bitBetter/cert.cert"
|
||||||
|
|
||||||
# gather all running instances, cannot run a wildcard filter on Ancestor= :(, does find all where name = *bitwarden*
|
# gather all running instances
|
||||||
OLDINSTANCES=$(docker container ps --all -f Name=bitwarden --format '{{.ID}}')
|
OLDINSTANCES=$(docker container ps --all -f Name=bitwarden --format '{{.ID}}')
|
||||||
|
|
||||||
# stop and remove all running instances
|
# stop all running instances
|
||||||
for INSTANCE in ${OLDINSTANCES[@]}; do
|
for INSTANCE in ${OLDINSTANCES[@]}; do
|
||||||
docker stop $INSTANCE
|
docker stop $INSTANCE
|
||||||
docker rm $INSTANCE
|
docker rm $INSTANCE
|
||||||
done
|
done
|
||||||
|
|
||||||
# update bitwarden itself
|
# update bitwarden itself
|
||||||
if [ "$1" = "update" ]; then
|
if [ "$1" = "y" ]; then
|
||||||
docker pull ghcr.io/bitwarden/lite:latest
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
else
|
else
|
||||||
read -p "Update (or get) bitwarden source container (y/n): "
|
read -p "Update (or get) bitwarden source container: " -n 1 -r
|
||||||
if [[ $REPLY =~ ^[Yy]$ ]]; then
|
echo
|
||||||
docker pull ghcr.io/bitwarden/lite:latest
|
if [[ $REPLY =~ ^[Yy]$ ]]
|
||||||
|
then
|
||||||
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# stop and remove previous existing patch(ed) container
|
# stop and remove previous existing patch(ed) container
|
||||||
OLDINSTANCES=$(docker container ps --all -f Ancestor=bitwarden-patched --format '{{.ID}}')
|
docker stop bitwarden-patch
|
||||||
for INSTANCE in ${OLDINSTANCES[@]}; do
|
docker rm bitwarden-patch
|
||||||
docker stop $INSTANCE
|
docker image rm bitwarden-patch
|
||||||
docker rm $INSTANCE
|
|
||||||
done
|
|
||||||
OLDINSTANCES=$(docker image ls bitwarden-patched --format '{{.ID}}')
|
|
||||||
for INSTANCE in ${OLDINSTANCES[@]}; do
|
|
||||||
docker image rm $INSTANCE
|
|
||||||
done
|
|
||||||
|
|
||||||
# remove old extract containers
|
|
||||||
OLDINSTANCES=$(docker container ps --all -f Name=bitwarden-extract --format '{{.ID}}')
|
|
||||||
for INSTANCE in ${OLDINSTANCES[@]}; do
|
|
||||||
docker stop $INSTANCE
|
|
||||||
docker rm $INSTANCE
|
|
||||||
done
|
|
||||||
|
|
||||||
# start a new bitwarden instance so we can patch it
|
# start a new bitwarden instance so we can patch it
|
||||||
PATCHINSTANCE=$(docker run -d --name bitwarden-extract ghcr.io/bitwarden/lite:latest)
|
PATCHINSTANCE=$(docker run -d --name bitwarden-patch ghcr.io/bitwarden/self-host:beta)
|
||||||
|
|
||||||
# create our temporary directory
|
# create our temporary directory
|
||||||
mkdir $TEMPDIRECTORY
|
mkdir $TEMPDIRECTORY
|
||||||
@@ -96,33 +74,29 @@ for COMPONENT in ${COMPONENTS[@]}; do
|
|||||||
docker cp $PATCHINSTANCE:/app/$COMPONENT/Core.dll "$TEMPDIRECTORY/$COMPONENT/Core.dll"
|
docker cp $PATCHINSTANCE:/app/$COMPONENT/Core.dll "$TEMPDIRECTORY/$COMPONENT/Core.dll"
|
||||||
done
|
done
|
||||||
|
|
||||||
# stop and remove our temporary container
|
|
||||||
docker stop bitwarden-extract
|
|
||||||
docker rm bitwarden-extract
|
|
||||||
|
|
||||||
# run bitBetter, this applies our patches to the required files
|
# run bitBetter, this applies our patches to the required files
|
||||||
docker run -v "$TEMPDIRECTORY:/app/mount" --rm bitbetter/bitbetter
|
docker run -v "$TEMPDIRECTORY:/app/mount" --rm bitbetter/bitbetter
|
||||||
|
|
||||||
# create a new image with the patched files
|
# create a new image with the patched files
|
||||||
docker build . --tag bitwarden-patched --file "$PWD/src/bitBetter/Dockerfile-bitwarden-patch"
|
docker build . --tag bitwarden-patch --file "$PWD/src/bitBetter/Dockerfile-bitwarden-patch"
|
||||||
|
|
||||||
# start all user requested instances
|
# stop and remove our temporary container
|
||||||
if [ -f "$PWD/.servers/serverlist.txt" ]; then
|
docker stop bitwarden-patch
|
||||||
# convert line endings to unix
|
docker rm bitwarden-patch
|
||||||
sed -i 's/\r$//' "$PWD/.servers/serverlist.txt"
|
|
||||||
cat "$PWD/.servers/serverlist.txt" | while read -r LINE; do
|
|
||||||
if [[ $LINE != "#"* ]]; then
|
|
||||||
bash -c "$LINE"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
fi
|
|
||||||
|
|
||||||
# remove our bitBetter image
|
|
||||||
docker image rm bitbetter/bitbetter
|
|
||||||
|
|
||||||
# copy our patched library to the licenseGen source directory
|
# copy our patched library to the licenseGen source directory
|
||||||
cp -f "$TEMPDIRECTORY/Identity/Core.dll" "$PWD/src/licenseGen"
|
cp -f "$TEMPDIRECTORY/Identity/Core.dll" "$PWD/src/licenseGen"
|
||||||
cp -f "$PWD/.keys/cert.pfx" "$PWD/src/licenseGen"
|
|
||||||
|
# remove our temporary directory
|
||||||
|
rm -rf "$TEMPDIRECTORY"
|
||||||
|
|
||||||
|
# start all user requested instances
|
||||||
|
cat "$PWD/.servers/serverlist.txt" | while read LINE; do
|
||||||
|
bash -c "$LINE"
|
||||||
|
done
|
||||||
|
|
||||||
|
# remove our bitBetter image
|
||||||
|
docker image rm bitbetter/bitbetter
|
||||||
|
|
||||||
# build the licenseGen
|
# build the licenseGen
|
||||||
docker build -t bitbetter/licensegen "$PWD/src/licenseGen"
|
docker build -t bitbetter/licensegen "$PWD/src/licenseGen"
|
||||||
@@ -130,6 +104,3 @@ docker build -t bitbetter/licensegen "$PWD/src/licenseGen"
|
|||||||
# clean the licenseGen source directory
|
# clean the licenseGen source directory
|
||||||
rm -f "$PWD/src/licenseGen/Core.dll"
|
rm -f "$PWD/src/licenseGen/Core.dll"
|
||||||
rm -f "$PWD/src/licenseGen/cert.pfx"
|
rm -f "$PWD/src/licenseGen/cert.pfx"
|
||||||
|
|
||||||
# remove our temporary directory
|
|
||||||
rm -rf "$TEMPDIRECTORY"
|
|
||||||
|
|||||||
@@ -1,25 +1,22 @@
|
|||||||
$ErrorActionPreference = 'Stop'
|
# get the basic openssl binary path
|
||||||
$PSNativeCommandUseErrorActionPreference = $true
|
$opensslbinary = "$Env:Programfiles\OpenSSL-Win64\bin\openssl.exe"
|
||||||
|
|
||||||
# get the basic openssl binary path
|
# if openssl is not installed attempt to install it
|
||||||
$opensslbinary = "$Env:Programfiles\OpenSSL-Win64\bin\openssl.exe"
|
if (!(Get-Command $opensslbinary -errorAction SilentlyContinue))
|
||||||
|
{
|
||||||
# if openssl is not installed attempt to install it
|
winget install openssl
|
||||||
if (!(Get-Command $opensslbinary -errorAction SilentlyContinue))
|
}
|
||||||
{
|
|
||||||
winget install openssl
|
# if previous keys exist, remove them
|
||||||
}
|
if (Test-Path "$pwd\.keys")
|
||||||
|
{
|
||||||
# if previous keys exist, remove them
|
Remove-Item "$pwd\.keys" -Recurse -Force
|
||||||
if (Test-Path "$pwd\.keys")
|
}
|
||||||
{
|
|
||||||
Remove-Item "$pwd\.keys" -Recurse -Force
|
# create new directory
|
||||||
}
|
New-item -ItemType Directory -Path "$pwd\.keys"
|
||||||
|
|
||||||
# create new directory
|
# generate actual keys
|
||||||
New-item -ItemType Directory -Path "$pwd\.keys"
|
Invoke-Expression "& '$opensslbinary' req -x509 -newkey rsa:4096 -keyout `"$pwd\.keys\key.pem`" -out `"$pwd\.keys\cert.cert`" -days 36500 -subj '/CN=www.mydom.com/O=My Company Name LTD./C=US' -outform DER -passout pass:test"
|
||||||
|
Invoke-Expression "& '$opensslbinary' x509 -inform DER -in `"$pwd\.keys\cert.cert`" -out `"$pwd\.keys\cert.pem`""
|
||||||
# generate actual keys
|
|
||||||
Invoke-Expression "& '$opensslbinary' req -x509 -newkey rsa:4096 -keyout `"$pwd\.keys\key.pem`" -out `"$pwd\.keys\cert.cer`" -days 36500 -subj '/CN=www.mydom.com/O=My Company Name LTD./C=US' -outform DER -passout pass:test"
|
|
||||||
Invoke-Expression "& '$opensslbinary' x509 -inform DER -in `"$pwd\.keys\cert.cer`" -out `"$pwd\.keys\cert.pem`""
|
|
||||||
Invoke-Expression "& '$opensslbinary' pkcs12 -export -out `"$pwd\.keys\cert.pfx`" -inkey `"$pwd\.keys\key.pem`" -in `"$pwd\.keys\cert.pem`" -passin pass:test -passout pass:test"
|
Invoke-Expression "& '$opensslbinary' pkcs12 -export -out `"$pwd\.keys\cert.pfx`" -inkey `"$pwd\.keys\key.pem`" -in `"$pwd\.keys\cert.pem`" -passin pass:test -passout pass:test"
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -e
|
|
||||||
|
|
||||||
# Check for openssl
|
# Check for openssl
|
||||||
command -v openssl >/dev/null 2>&1 || { echo >&2 "openssl required but not found. Aborting."; exit 1; }
|
command -v openssl >/dev/null 2>&1 || { echo >&2 "openssl required but not found. Aborting."; exit 1; }
|
||||||
@@ -15,6 +14,6 @@ fi
|
|||||||
mkdir "$DIR"
|
mkdir "$DIR"
|
||||||
|
|
||||||
# Generate new keys
|
# Generate new keys
|
||||||
openssl req -x509 -newkey rsa:4096 -keyout "$DIR/key.pem" -out "$DIR/cert.cer" -days 36500 -subj '/CN=www.mydom.com/O=My Company Name LTD./C=US' -outform DER -passout pass:test
|
openssl req -x509 -newkey rsa:4096 -keyout "$DIR/key.pem" -out "$DIR/cert.cert" -days 36500 -subj '/CN=www.mydom.com/O=My Company Name LTD./C=US' -outform DER -passout pass:test
|
||||||
openssl x509 -inform DER -in "$DIR/cert.cer" -out "$DIR/cert.pem"
|
openssl x509 -inform DER -in "$DIR/cert.cert" -out "$DIR/cert.pem"
|
||||||
openssl pkcs12 -export -out "$DIR/cert.pfx" -inkey "$DIR/key.pem" -in "$DIR/cert.pem" -passin pass:test -passout pass:test
|
openssl pkcs12 -export -out "$DIR/cert.pfx" -inkey "$DIR/key.pem" -in "$DIR/cert.pem" -passin pass:test -passout pass:test
|
||||||
|
|||||||
@@ -1,17 +1,14 @@
|
|||||||
$ErrorActionPreference = 'Stop'
|
|
||||||
$PSNativeCommandUseErrorActionPreference = $true
|
|
||||||
|
|
||||||
if ($($args.Count) -lt 1) {
|
if ($($args.Count) -lt 1) {
|
||||||
echo "USAGE: <License Gen action> [License Gen args...]"
|
echo "USAGE: <License Gen action> [License Gen args...]"
|
||||||
echo "ACTIONS:"
|
echo "ACTIONS:"
|
||||||
echo " interactive"
|
echo " interactive"
|
||||||
echo " user"
|
echo " user"
|
||||||
echo " org"
|
echo " org"
|
||||||
Exit 1
|
Exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($args[0] -eq "interactive") {
|
if ($args[0] = "interactive") {
|
||||||
docker run -it --rm bitbetter/licensegen interactive
|
docker run -it --rm bitbetter/licensegen interactive
|
||||||
} else {
|
} else {
|
||||||
docker run bitbetter/licensegen $args
|
docker run bitbetter/licensegen $args
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,12 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -e
|
|
||||||
|
|
||||||
if [ $# -lt 1 ]; then
|
if [ $# -lt 1 ]; then
|
||||||
echo "USAGE: <License Gen action> [License Gen args...]"
|
echo "USAGE: <License Gen action> [License Gen args...]"
|
||||||
echo "ACTIONS:"
|
echo "ACTIONS:"
|
||||||
echo " interactive"
|
echo " interactive"
|
||||||
echo " user"
|
echo " user"
|
||||||
echo " org"
|
echo " org"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ "$1" = "interactive" ]; then
|
if [ "$1" = "interactive" ]; then
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
|||||||
WORKDIR /bitBetter
|
WORKDIR /bitBetter
|
||||||
|
|
||||||
COPY . /bitBetter
|
COPY . /bitBetter
|
||||||
COPY cert.cer /app/
|
COPY cert.cert /app/
|
||||||
|
|
||||||
RUN dotnet restore
|
RUN dotnet restore
|
||||||
RUN dotnet publish -c Release -o /app --no-restore
|
RUN dotnet publish -c Release -o /app --no-restore
|
||||||
@@ -11,4 +11,4 @@ FROM mcr.microsoft.com/dotnet/sdk:8.0
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=build /app .
|
COPY --from=build /app .
|
||||||
|
|
||||||
ENTRYPOINT ["dotnet", "/app/bitBetter.dll"]
|
ENTRYPOINT [ "/app/bitBetter" ]
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
FROM ghcr.io/bitwarden/lite:latest
|
FROM ghcr.io/bitwarden/self-host:beta
|
||||||
|
|
||||||
COPY ./temp/ /app/
|
COPY ./temp/Api/Core.dll /app/Api/Core.dll
|
||||||
|
COPY ./temp/Identity/Core.dll /app/Identity/Core.dll
|
||||||
|
|||||||
@@ -1,67 +1,75 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Security.Cryptography.X509Certificates;
|
using System.Security.Cryptography.X509Certificates;
|
||||||
using dnlib.DotNet;
|
using dnlib.DotNet;
|
||||||
using dnlib.DotNet.Emit;
|
using dnlib.DotNet.Emit;
|
||||||
using dnlib.DotNet.Writer;
|
using dnlib.DotNet.Writer;
|
||||||
using dnlib.IO;
|
using dnlib.IO;
|
||||||
|
|
||||||
namespace bitBetter;
|
namespace bitBetter;
|
||||||
|
|
||||||
internal class Program
|
internal class Program
|
||||||
{
|
{
|
||||||
private static Int32 Main()
|
private static Int32 Main()
|
||||||
{
|
{
|
||||||
const String certFile = "/app/cert.cer";
|
const String certFile = "/app/cert.cert";
|
||||||
String[] files = Directory.GetFiles("/app/mount", "Core.dll", SearchOption.AllDirectories);
|
String[] files = Directory.GetFiles("/app/mount", "Core.dll", SearchOption.AllDirectories);
|
||||||
|
|
||||||
foreach (String file in files)
|
foreach (String file in files)
|
||||||
{
|
{
|
||||||
Console.WriteLine(file);
|
Console.WriteLine(file);
|
||||||
ModuleDefMD moduleDefMd = ModuleDefMD.Load(file);
|
ModuleDefMD moduleDefMd = ModuleDefMD.Load(file);
|
||||||
Byte[] cert = File.ReadAllBytes(certFile);
|
Byte[] cert = File.ReadAllBytes(certFile);
|
||||||
|
|
||||||
EmbeddedResource embeddedResourceToRemove = moduleDefMd.Resources.OfType<EmbeddedResource>().First(r => r.Name.Equals("Bit.Core.licensing.cer"));
|
EmbeddedResource embeddedResourceToRemove = moduleDefMd.Resources
|
||||||
EmbeddedResource embeddedResourceToAdd = new("Bit.Core.licensing.cer", cert) { Attributes = embeddedResourceToRemove.Attributes };
|
.OfType<EmbeddedResource>()
|
||||||
moduleDefMd.Resources.Add(embeddedResourceToAdd);
|
.First(r => r.Name.Equals("Bit.Core.licensing.cer"));
|
||||||
moduleDefMd.Resources.Remove(embeddedResourceToRemove);
|
|
||||||
|
Console.WriteLine(embeddedResourceToRemove.Name);
|
||||||
DataReader reader = embeddedResourceToRemove.CreateReader();
|
|
||||||
X509Certificate2 existingCert = new(reader.ReadRemainingBytes());
|
EmbeddedResource embeddedResourceToAdd = new("Bit.Core.licensing.cer", cert) {Attributes = embeddedResourceToRemove.Attributes };
|
||||||
|
moduleDefMd.Resources.Add(embeddedResourceToAdd);
|
||||||
Console.WriteLine($"Existing Cert Thumbprint: {existingCert.Thumbprint}");
|
moduleDefMd.Resources.Remove(embeddedResourceToRemove);
|
||||||
X509Certificate2 certificate = new(cert);
|
|
||||||
|
DataReader reader = embeddedResourceToRemove.CreateReader();
|
||||||
Console.WriteLine($"New Cert Thumbprint: {certificate.Thumbprint}");
|
X509Certificate2 existingCert = new(reader.ReadRemainingBytes());
|
||||||
|
|
||||||
IEnumerable<TypeDef> services = moduleDefMd.Types.Where(t => t.Namespace == "Bit.Core.Billing.Services");
|
Console.WriteLine($"Existing Cert Thumbprint: {existingCert.Thumbprint}");
|
||||||
TypeDef type = services.First(t => t.Name == "LicensingService");
|
X509Certificate2 certificate = new(cert);
|
||||||
MethodDef constructor = type.FindConstructors().First();
|
|
||||||
|
Console.WriteLine($"New Cert Thumbprint: {certificate.Thumbprint}");
|
||||||
Instruction instructionToPatch = constructor.Body.Instructions.FirstOrDefault(i => i.OpCode == OpCodes.Ldstr && String.Equals((String)i.Operand, existingCert.Thumbprint, StringComparison.InvariantCultureIgnoreCase));
|
|
||||||
|
IEnumerable<TypeDef> services = moduleDefMd.Types.Where(t => t.Namespace == "Bit.Core.Services");
|
||||||
if (instructionToPatch != null)
|
TypeDef type = services.First(t => t.Name == "LicensingService");
|
||||||
{
|
MethodDef constructor = type.FindConstructors().First();
|
||||||
instructionToPatch.Operand = certificate.Thumbprint;
|
|
||||||
}
|
Instruction instructionToPatch =
|
||||||
else
|
constructor.Body.Instructions
|
||||||
{
|
.FirstOrDefault(i => i.OpCode == OpCodes.Ldstr
|
||||||
Console.WriteLine("Can't find constructor to patch");
|
&& String.Equals((String)i.Operand, existingCert.Thumbprint, StringComparison.InvariantCultureIgnoreCase));
|
||||||
}
|
|
||||||
|
if (instructionToPatch != null)
|
||||||
ModuleWriterOptions moduleWriterOptions = new(moduleDefMd);
|
{
|
||||||
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.KeepOldMaxStack;
|
instructionToPatch.Operand = certificate.Thumbprint;
|
||||||
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.PreserveAll;
|
}
|
||||||
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.PreserveRids;
|
else
|
||||||
|
{
|
||||||
moduleDefMd.Write(file + ".new");
|
Console.WriteLine("Can't find constructor to patch");
|
||||||
moduleDefMd.Dispose();
|
}
|
||||||
File.Delete(file);
|
|
||||||
File.Move(file + ".new", file);
|
ModuleWriterOptions moduleWriterOptions = new(moduleDefMd);
|
||||||
}
|
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.KeepOldMaxStack;
|
||||||
|
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.PreserveAll;
|
||||||
return 0;
|
moduleWriterOptions.MetadataOptions.Flags |= MetadataFlags.PreserveRids;
|
||||||
}
|
|
||||||
|
moduleDefMd.Write(file + ".new");
|
||||||
|
moduleDefMd.Dispose();
|
||||||
|
File.Delete(file);
|
||||||
|
File.Move(file + ".new", file);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -1,9 +1,12 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="dnlib" Version="4.5.0" />
|
<ItemGroup>
|
||||||
|
<PackageReference Include="dnlib" Version="4.4.0" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
|
||||||
|
</Project>
|
||||||
|
|||||||
@@ -12,4 +12,4 @@ FROM mcr.microsoft.com/dotnet/sdk:8.0
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=build /app .
|
COPY --from=build /app .
|
||||||
|
|
||||||
ENTRYPOINT ["dotnet", "/app/licenseGen.dll", "--cert=/app/cert.pfx", "--core=/app/Core.dll"]
|
ENTRYPOINT [ "dotnet", "/app/licenseGen.dll", "--core", "/app/Core.dll", "--cert", "/app/cert.pfx" ]
|
||||||
|
|||||||
@@ -1,485 +1,453 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Text.Json;
|
using System.Reflection;
|
||||||
using System.Reflection;
|
using System.Runtime.Loader;
|
||||||
using System.Runtime.Loader;
|
using System.Security.Cryptography.X509Certificates;
|
||||||
using System.Security.Cryptography.X509Certificates;
|
using McMaster.Extensions.CommandLineUtils;
|
||||||
using McMaster.Extensions.CommandLineUtils;
|
using Newtonsoft.Json;
|
||||||
|
|
||||||
namespace licenseGen;
|
namespace licenseGen;
|
||||||
|
|
||||||
internal class Program
|
internal class Program
|
||||||
{
|
{
|
||||||
private static readonly CommandLineApplication App = new();
|
private static Int32 Main(String[] args)
|
||||||
private static readonly CommandOption Cert = App.Option("--cert", "Certificate file", CommandOptionType.SingleValue);
|
{
|
||||||
private static readonly CommandOption CoreDll = App.Option("--core", "Path to Core.dll", CommandOptionType.SingleValue);
|
CommandLineApplication app = new();
|
||||||
|
CommandOption cert = app.Option("--cert", "cert file", CommandOptionType.SingleValue);
|
||||||
private static Int32 Main(String[] args)
|
CommandOption coreDll = app.Option("--core", "path to core dll", CommandOptionType.SingleValue);
|
||||||
{
|
|
||||||
App.Command("interactive", config =>
|
Boolean CertExists()
|
||||||
{
|
{
|
||||||
String buff, licenseType = "", name = "", email = "", businessName="";
|
return File.Exists(cert.Value());
|
||||||
Int16 storage = 0;
|
}
|
||||||
Boolean validGuid = false, validInstallid = false;
|
|
||||||
Guid guid = Guid.Empty, installid = Guid.Empty;
|
Boolean CoreExists()
|
||||||
|
{
|
||||||
config.OnExecute(() =>
|
return File.Exists(coreDll.Value());
|
||||||
{
|
}
|
||||||
Check();
|
|
||||||
Console.WriteLine("Interactive license mode...");
|
Boolean VerifyTopOptions()
|
||||||
|
{
|
||||||
while (licenseType == "")
|
return !String.IsNullOrWhiteSpace(cert.Value()) &&
|
||||||
{
|
!String.IsNullOrWhiteSpace(coreDll.Value()) &&
|
||||||
Console.WriteLine("What would you like to generate, a [u]ser license or an [o]rg license: ");
|
CertExists() && CoreExists();
|
||||||
buff = Console.ReadLine();
|
}
|
||||||
|
|
||||||
switch (buff)
|
app.Command("interactive", config =>
|
||||||
{
|
{
|
||||||
case "u":
|
String buff, licenseType = "", name = "", email = "", businessName="";
|
||||||
{
|
Int16 storage = 0;
|
||||||
licenseType = "user";
|
|
||||||
Console.WriteLine("Okay, we will generate a user license.");
|
Boolean validGuid = false, validInstallid = false;
|
||||||
|
Guid guid = new(), installid = new();
|
||||||
while (!validGuid)
|
|
||||||
{
|
config.OnExecute(() =>
|
||||||
Console.WriteLine("Please provide the user's guid — refer to the Readme for details on how to retrieve this. [GUID]: ");
|
{
|
||||||
buff = Console.ReadLine();
|
if (!VerifyTopOptions())
|
||||||
|
{
|
||||||
if (Guid.TryParse(buff, out guid))validGuid = true;
|
if (!CoreExists()) config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
||||||
else Console.WriteLine("The user-guid provided does not appear to be valid!");
|
if (!CertExists()) config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
||||||
}
|
|
||||||
break;
|
config.ShowHelp();
|
||||||
}
|
return 1;
|
||||||
case "o":
|
}
|
||||||
{
|
|
||||||
licenseType = "org";
|
Console.WriteLine("Interactive license mode...");
|
||||||
Console.WriteLine("Okay, we will generate an organization license.");
|
|
||||||
|
while (licenseType == "")
|
||||||
while (!validInstallid)
|
{
|
||||||
{
|
Console.WriteLine("What would you like to generate, a [u]ser license or an [o]rg license: ");
|
||||||
Console.WriteLine("Please provide your Bitwarden Install-ID — refer to the Readme for details on how to retrieve this. [Install-ID]: ");
|
buff = Console.ReadLine();
|
||||||
buff = Console.ReadLine();
|
|
||||||
|
if(buff == "u")
|
||||||
if (Guid.TryParse(buff, out installid)) validInstallid = true;
|
{
|
||||||
else Console.WriteLine("The install-id provided does not appear to be valid.");
|
licenseType = "user";
|
||||||
}
|
Console.WriteLine("Okay, we will generate a user license.");
|
||||||
|
|
||||||
while (businessName == "")
|
while (validGuid == false)
|
||||||
{
|
{
|
||||||
Console.WriteLine("Please enter a business name, default is BitBetter. [Business Name]: ");
|
Console.WriteLine("Please provide the user's guid — refer to the Readme for details on how to retrieve this. [GUID]: ");
|
||||||
buff = Console.ReadLine();
|
buff = Console.ReadLine();
|
||||||
if (buff == "")
|
|
||||||
{
|
if (Guid.TryParse(buff, out guid))validGuid = true;
|
||||||
businessName = "BitBetter";
|
else Console.WriteLine("The user-guid provided does not appear to be valid!");
|
||||||
}
|
}
|
||||||
else if (CheckBusinessName(buff))
|
}
|
||||||
{
|
else if (buff == "o")
|
||||||
businessName = buff;
|
{
|
||||||
}
|
licenseType = "org";
|
||||||
}
|
Console.WriteLine("Okay, we will generate an organization license.");
|
||||||
break;
|
|
||||||
}
|
while (validInstallid == false)
|
||||||
default:
|
{
|
||||||
Console.WriteLine("Unrecognized option \'" + buff + "\'.");
|
Console.WriteLine("Please provide your Bitwarden Install-ID — refer to the Readme for details on how to retrieve this. [Install-ID]: ");
|
||||||
break;
|
buff = Console.ReadLine();
|
||||||
}
|
|
||||||
}
|
if (Guid.TryParse(buff, out installid)) validInstallid = true;
|
||||||
|
else Console.WriteLine("The install-id provided does not appear to be valid.");
|
||||||
while (name == "")
|
}
|
||||||
{
|
|
||||||
Console.WriteLine("Please provide the username this license will be registered to. [username]: ");
|
while (businessName == "")
|
||||||
buff = Console.ReadLine();
|
{
|
||||||
if (CheckUsername(buff)) name = buff;
|
Console.WriteLine("Please enter a business name, default is BitBetter. [Business Name]: ");
|
||||||
}
|
buff = Console.ReadLine();
|
||||||
|
if (buff == "")
|
||||||
while (email == "")
|
{
|
||||||
{
|
businessName = "BitBetter";
|
||||||
Console.WriteLine("Please provide the email address for the user " + name + ". [email]: ");
|
}
|
||||||
buff = Console.ReadLine();
|
else if (CheckBusinessName(buff))
|
||||||
if (CheckEmail(buff))
|
{
|
||||||
{
|
businessName = buff;
|
||||||
email = buff;
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
while (storage == 0)
|
{
|
||||||
{
|
Console.WriteLine("Unrecognized option \'" + buff + "\'.");
|
||||||
Console.WriteLine("Extra storage space for the user " + name + ". (max.: " + Int16.MaxValue + "). Defaults to maximum value. [storage]");
|
}
|
||||||
buff = Console.ReadLine();
|
}
|
||||||
if (String.IsNullOrWhiteSpace(buff))
|
|
||||||
{
|
while (name == "")
|
||||||
storage = Int16.MaxValue;
|
{
|
||||||
}
|
Console.WriteLine("Please provide the username this license will be registered to. [username]: ");
|
||||||
else
|
buff = Console.ReadLine();
|
||||||
{
|
if ( CheckUsername(buff) ) name = buff;
|
||||||
if (CheckStorage(buff))
|
}
|
||||||
{
|
|
||||||
storage = Int16.Parse(buff);
|
while (email == "")
|
||||||
}
|
{
|
||||||
}
|
Console.WriteLine("Please provide the email address for the user " + name + ". [email]: ");
|
||||||
}
|
buff = Console.ReadLine();
|
||||||
|
if (CheckEmail(buff))
|
||||||
switch (licenseType)
|
{
|
||||||
{
|
email = buff;
|
||||||
case "user":
|
}
|
||||||
{
|
}
|
||||||
Console.WriteLine("Confirm creation of \"user\" license for username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", User-GUID: \"" + guid + "\"? Y/n");
|
|
||||||
buff = Console.ReadLine();
|
while (storage == 0)
|
||||||
if (buff is "" or "y" or "Y")
|
{
|
||||||
{
|
Console.WriteLine("Extra storage space for the user " + name + ". (max.: " + Int16.MaxValue + "). Defaults to maximum value. [storage]");
|
||||||
GenerateUserLicense(new X509Certificate2(Cert.Value(), "test"), CoreDll.Value(), name, email, storage, guid, null);
|
buff = Console.ReadLine();
|
||||||
}
|
if (String.IsNullOrWhiteSpace(buff))
|
||||||
else
|
{
|
||||||
{
|
storage = Int16.MaxValue;
|
||||||
Console.WriteLine("Exiting...");
|
}
|
||||||
return 0;
|
else
|
||||||
}
|
{
|
||||||
|
if (CheckStorage(buff))
|
||||||
break;
|
{
|
||||||
}
|
storage = Int16.Parse(buff);
|
||||||
case "org":
|
}
|
||||||
{
|
}
|
||||||
Console.WriteLine("Confirm creation of \"organization\" license for business name: \"" + businessName + "\", username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", Install-ID: \"" + installid + "\"? Y/n");
|
}
|
||||||
buff = Console.ReadLine();
|
|
||||||
if (buff is "" or "y" or "Y")
|
if (licenseType == "user")
|
||||||
{
|
{
|
||||||
GenerateOrgLicense(new X509Certificate2(Cert.Value(), "test"), CoreDll.Value(), name, email, storage, installid, businessName, null);
|
Console.WriteLine("Confirm creation of \"user\" license for username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", User-GUID: \"" + guid + "\"? Y/n");
|
||||||
}
|
buff = Console.ReadLine();
|
||||||
else
|
if ( buff is "" or "y" or "Y" )
|
||||||
{
|
{
|
||||||
Console.WriteLine("Exiting...");
|
GenerateUserLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name, email, storage, guid, null);
|
||||||
return 0;
|
}
|
||||||
}
|
else
|
||||||
|
{
|
||||||
break;
|
Console.WriteLine("Exiting...");
|
||||||
}
|
return 0;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return 0;
|
else if (licenseType == "org")
|
||||||
});
|
{
|
||||||
});
|
Console.WriteLine("Confirm creation of \"organization\" license for business name: \"" + businessName + "\", username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", Install-ID: \"" + installid + "\"? Y/n");
|
||||||
App.Command("user", config =>
|
buff = Console.ReadLine();
|
||||||
{
|
if ( buff is "" or "y" or "Y" )
|
||||||
CommandArgument name = config.Argument("Name", "your name");
|
{
|
||||||
CommandArgument email = config.Argument("Email", "your email");
|
GenerateOrgLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name, email, storage, installid, businessName, null);
|
||||||
CommandArgument userIdArg = config.Argument("User ID", "your user id");
|
}
|
||||||
CommandArgument storage = config.Argument("Storage", "extra storage space in GB. Maximum is " + Int16.MaxValue + " (optional, default = max)");
|
else
|
||||||
CommandArgument key = config.Argument("Key", "your key id (optional)");
|
{
|
||||||
|
Console.WriteLine("Exiting...");
|
||||||
config.OnExecute(() =>
|
return 0;
|
||||||
{
|
}
|
||||||
Check();
|
}
|
||||||
|
|
||||||
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value))
|
return 0;
|
||||||
{
|
});
|
||||||
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}'");
|
});
|
||||||
config.ShowHelp(true);
|
app.Command("user", config =>
|
||||||
return 1;
|
{
|
||||||
}
|
CommandArgument name = config.Argument("Name", "your name");
|
||||||
|
CommandArgument email = config.Argument("Email", "your email");
|
||||||
if (String.IsNullOrWhiteSpace(userIdArg.Value) || !Guid.TryParse(userIdArg.Value, out Guid userId))
|
CommandArgument userIdArg = config.Argument("User ID", "your user id");
|
||||||
{
|
CommandArgument storage = config.Argument("Storage", "extra storage space in GB. Maximum is " + Int16.MaxValue + " (optional, default = max)");
|
||||||
config.Error.WriteLine("User ID not provided");
|
CommandArgument key = config.Argument("Key", "your key id (optional)");
|
||||||
config.ShowHelp(true);
|
|
||||||
return 1;
|
config.OnExecute(() =>
|
||||||
}
|
{
|
||||||
|
if (!VerifyTopOptions())
|
||||||
Int16 storageShort = 0;
|
{
|
||||||
if (!String.IsNullOrWhiteSpace(storage.Value))
|
if (!CoreExists())
|
||||||
{
|
{
|
||||||
Double parsedStorage = Double.Parse(storage.Value);
|
config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
||||||
if (parsedStorage is > Int16.MaxValue or < 0)
|
}
|
||||||
{
|
if (!CertExists())
|
||||||
config.Error.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]");
|
{
|
||||||
config.ShowHelp(true);
|
config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
||||||
return 1;
|
}
|
||||||
}
|
|
||||||
storageShort = (Int16) parsedStorage;
|
config.ShowHelp();
|
||||||
}
|
return 1;
|
||||||
|
}
|
||||||
GenerateUserLicense(new X509Certificate2(Cert.Value()!, "test"), CoreDll.Value(), name.Value, email.Value, storageShort, userId, key.Value);
|
|
||||||
|
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value))
|
||||||
return 0;
|
{
|
||||||
});
|
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}'");
|
||||||
});
|
config.ShowHelp(true);
|
||||||
App.Command("org", config =>
|
return 1;
|
||||||
{
|
}
|
||||||
CommandArgument name = config.Argument("Name", "your name");
|
|
||||||
CommandArgument email = config.Argument("Email", "your email");
|
if (String.IsNullOrWhiteSpace(userIdArg.Value) || !Guid.TryParse(userIdArg.Value, out Guid userId))
|
||||||
CommandArgument installId = config.Argument("InstallId", "your installation id (GUID)");
|
{
|
||||||
CommandArgument storage = config.Argument("Storage", "extra storage space in GB. Maximum is " + Int16.MaxValue + " (optional, default = max)");
|
config.Error.WriteLine("User ID not provided");
|
||||||
CommandArgument businessName = config.Argument("BusinessName", "name for the organization (optional)");
|
config.ShowHelp(true);
|
||||||
CommandArgument key = config.Argument("Key", "your key id (optional)");
|
return 1;
|
||||||
|
}
|
||||||
config.OnExecute(() =>
|
|
||||||
{
|
Int16 storageShort = 0;
|
||||||
Check();
|
if (!String.IsNullOrWhiteSpace(storage.Value))
|
||||||
|
{
|
||||||
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value) || String.IsNullOrWhiteSpace(installId.Value))
|
Double parsedStorage = Double.Parse(storage.Value);
|
||||||
{
|
if (parsedStorage is > Int16.MaxValue or < 0)
|
||||||
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}' InstallId='{installId.Value}'");
|
{
|
||||||
config.ShowHelp(true);
|
config.Error.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]");
|
||||||
return 1;
|
config.ShowHelp(true);
|
||||||
}
|
return 1;
|
||||||
|
}
|
||||||
if (!Guid.TryParse(installId.Value, out Guid installationId))
|
storageShort = (Int16) parsedStorage;
|
||||||
{
|
}
|
||||||
config.Error.WriteLine("Unable to parse your installation id as a GUID");
|
|
||||||
config.Error.WriteLine($"Here's a new guid: {Guid.NewGuid()}");
|
GenerateUserLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name.Value, email.Value, storageShort, userId, key.Value);
|
||||||
config.ShowHelp(true);
|
|
||||||
return 1;
|
return 0;
|
||||||
}
|
});
|
||||||
|
});
|
||||||
Int16 storageShort = 0;
|
app.Command("org", config =>
|
||||||
if (!String.IsNullOrWhiteSpace(storage.Value))
|
{
|
||||||
{
|
CommandArgument name = config.Argument("Name", "your name");
|
||||||
Double parsedStorage = Double.Parse(storage.Value);
|
CommandArgument email = config.Argument("Email", "your email");
|
||||||
if (parsedStorage is > Int16.MaxValue or < 0)
|
CommandArgument installId = config.Argument("InstallId", "your installation id (GUID)");
|
||||||
{
|
CommandArgument storage = config.Argument("Storage", "extra storage space in GB. Maximum is " + Int16.MaxValue + " (optional, default = max)");
|
||||||
config.Error.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]");
|
CommandArgument businessName = config.Argument("BusinessName", "name for the organization (optional)");
|
||||||
config.ShowHelp(true);
|
CommandArgument key = config.Argument("Key", "your key id (optional)");
|
||||||
return 1;
|
|
||||||
}
|
config.OnExecute(() =>
|
||||||
storageShort = (Int16)parsedStorage;
|
{
|
||||||
}
|
if (!VerifyTopOptions())
|
||||||
|
{
|
||||||
GenerateOrgLicense(new X509Certificate2(Cert.Value()!, "test"), CoreDll.Value(), name.Value, email.Value, storageShort, installationId, businessName.Value, key.Value);
|
if (!CoreExists())
|
||||||
|
{
|
||||||
return 0;
|
config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
||||||
});
|
}
|
||||||
});
|
if (!CertExists())
|
||||||
|
{
|
||||||
App.OnExecute(() =>
|
config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
||||||
{
|
}
|
||||||
App.ShowHelp();
|
|
||||||
return 10;
|
config.ShowHelp();
|
||||||
});
|
return 1;
|
||||||
|
}
|
||||||
try
|
|
||||||
{
|
if (String.IsNullOrWhiteSpace(name.Value) ||
|
||||||
App.HelpOption("-? | -h | --help");
|
String.IsNullOrWhiteSpace(email.Value) ||
|
||||||
return App.Execute(args);
|
String.IsNullOrWhiteSpace(installId.Value))
|
||||||
}
|
{
|
||||||
catch (Exception exception)
|
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}' InstallId='{installId.Value}'");
|
||||||
{
|
config.ShowHelp(true);
|
||||||
Console.Error.WriteLine("Oops: {0}", exception);
|
return 1;
|
||||||
return 100;
|
}
|
||||||
}
|
|
||||||
}
|
if (!Guid.TryParse(installId.Value, out Guid installationId))
|
||||||
|
{
|
||||||
private static void Check()
|
config.Error.WriteLine("Unable to parse your installation id as a GUID");
|
||||||
{
|
config.Error.WriteLine($"Here's a new guid: {Guid.NewGuid()}");
|
||||||
if (Cert == null || String.IsNullOrWhiteSpace(Cert.Value()))
|
config.ShowHelp(true);
|
||||||
{
|
return 1;
|
||||||
App.Error.WriteLine("No certificate specified");
|
}
|
||||||
App.ShowHelp();
|
|
||||||
Environment.Exit(1);
|
Int16 storageShort = 0;
|
||||||
}
|
if (!String.IsNullOrWhiteSpace(storage.Value))
|
||||||
else if (CoreDll == null || String.IsNullOrWhiteSpace(CoreDll.Value()))
|
{
|
||||||
{
|
Double parsedStorage = Double.Parse(storage.Value);
|
||||||
App.Error.WriteLine("No core dll specified");
|
if (parsedStorage is > Int16.MaxValue or < 0)
|
||||||
App.ShowHelp();
|
{
|
||||||
Environment.Exit(1);
|
config.Error.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]");
|
||||||
}
|
config.ShowHelp(true);
|
||||||
else if (!File.Exists(Cert.Value()))
|
return 1;
|
||||||
{
|
}
|
||||||
App.Error.WriteLine($"Can't find certificate at: {Cert.Value()}");
|
storageShort = (Int16) parsedStorage;
|
||||||
App.ShowHelp();
|
}
|
||||||
Environment.Exit(1);
|
|
||||||
}
|
GenerateOrgLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name.Value, email.Value, storageShort, installationId, businessName.Value, key.Value);
|
||||||
else if (!File.Exists(CoreDll.Value()))
|
|
||||||
{
|
return 0;
|
||||||
App.Error.WriteLine($"Can't find core dll at: {CoreDll.Value()}");
|
});
|
||||||
App.ShowHelp();
|
});
|
||||||
Environment.Exit(1);
|
|
||||||
}
|
app.OnExecute(() =>
|
||||||
}
|
{
|
||||||
|
app.ShowHelp();
|
||||||
// checkUsername Checks that the username is a valid username
|
return 10;
|
||||||
private static Boolean CheckUsername(String s)
|
});
|
||||||
{
|
|
||||||
// TODO: Actually validate
|
app.HelpOption("-? | -h | --help");
|
||||||
if (!String.IsNullOrWhiteSpace(s)) return true;
|
|
||||||
|
try
|
||||||
Console.WriteLine("The username provided doesn't appear to be valid!");
|
{
|
||||||
return false;
|
return app.Execute(args);
|
||||||
}
|
}
|
||||||
|
catch (Exception e)
|
||||||
// checkBusinessName Checks that the Business Name is a valid username
|
{
|
||||||
private static Boolean CheckBusinessName(String s)
|
Console.Error.WriteLine("Oops: {0}", e);
|
||||||
{
|
return 100;
|
||||||
// TODO: Actually validate
|
}
|
||||||
if (!String.IsNullOrWhiteSpace(s)) return true;
|
}
|
||||||
|
|
||||||
Console.WriteLine("The Business Name provided doesn't appear to be valid!");
|
// checkUsername Checks that the username is a valid username
|
||||||
return false;
|
private static Boolean CheckUsername(String s)
|
||||||
}
|
{
|
||||||
|
// TODO: Actually validate
|
||||||
// checkEmail Checks that the email address is a valid email address
|
if (!String.IsNullOrWhiteSpace(s)) return true;
|
||||||
private static Boolean CheckEmail(String s)
|
|
||||||
{
|
Console.WriteLine("The username provided doesn't appear to be valid!");
|
||||||
// TODO: Actually validate
|
return false;
|
||||||
if (!String.IsNullOrWhiteSpace(s)) return true;
|
}
|
||||||
|
|
||||||
Console.WriteLine("The email provided doesn't appear to be valid!");
|
// checkBusinessName Checks that the Business Name is a valid username
|
||||||
return false;
|
private static Boolean CheckBusinessName(String s)
|
||||||
}
|
{
|
||||||
|
// TODO: Actually validate
|
||||||
// checkStorage Checks that the storage is in a valid range
|
if (!String.IsNullOrWhiteSpace(s)) return true;
|
||||||
private static Boolean CheckStorage(String s)
|
|
||||||
{
|
Console.WriteLine("The Business Name provided doesn't appear to be valid!");
|
||||||
if (String.IsNullOrWhiteSpace(s))
|
return false;
|
||||||
{
|
}
|
||||||
Console.WriteLine("The storage provided doesn't appear to be valid!");
|
|
||||||
return false;
|
// checkEmail Checks that the email address is a valid email address
|
||||||
}
|
private static Boolean CheckEmail(String s)
|
||||||
|
{
|
||||||
if (!(Double.Parse(s) > Int16.MaxValue) && !(Double.Parse(s) < 0)) return true;
|
// TODO: Actually validate
|
||||||
|
if (!String.IsNullOrWhiteSpace(s)) return true;
|
||||||
Console.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]!");
|
|
||||||
return false;
|
Console.WriteLine("The email provided doesn't appear to be valid!");
|
||||||
}
|
return false;
|
||||||
|
}
|
||||||
private static readonly JsonSerializerOptions JsonOptions = new() { WriteIndented = true };
|
|
||||||
private static void GenerateUserLicense(X509Certificate2 cert, String corePath, String userName, String email, Int16 storage, Guid userId, String key)
|
// checkStorage Checks that the storage is in a valid range
|
||||||
{
|
private static Boolean CheckStorage(String s)
|
||||||
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(Path.GetFullPath(corePath));
|
{
|
||||||
|
if (String.IsNullOrWhiteSpace(s))
|
||||||
Type type = core.GetType("Bit.Core.Billing.Models.Business.UserLicense");
|
{
|
||||||
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
Console.WriteLine("The storage provided doesn't appear to be valid!");
|
||||||
|
return false;
|
||||||
if (type == null)
|
}
|
||||||
{
|
|
||||||
Console.WriteLine("Could not find type!");
|
if (!(Double.Parse(s) > Int16.MaxValue) && !(Double.Parse(s) < 0)) return true;
|
||||||
return;
|
|
||||||
}
|
Console.WriteLine("The storage value provided is outside the accepted range of [0-" + Int16.MaxValue + "]!");
|
||||||
if (licenseTypeEnum == null)
|
return false;
|
||||||
{
|
}
|
||||||
Console.WriteLine("Could not find license licenseTypeEnum!");
|
|
||||||
return;
|
private static void GenerateUserLicense(X509Certificate2 cert, String corePath, String userName, String email, Int16 storage, Guid userId, String key)
|
||||||
}
|
{
|
||||||
|
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(corePath);
|
||||||
Object license = Activator.CreateInstance(type);
|
|
||||||
|
Type type = core.GetType("Bit.Core.Models.Business.UserLicense");
|
||||||
MethodInfo computeHash = type.GetMethod("ComputeHash");
|
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
||||||
if (computeHash == null)
|
|
||||||
{
|
Object license = Activator.CreateInstance(type);
|
||||||
Console.WriteLine("Could not find ComputeHash!");
|
|
||||||
return;
|
Set("LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
||||||
}
|
Set("Id", userId);
|
||||||
|
Set("Name", userName);
|
||||||
MethodInfo sign = type.GetMethod("Sign");
|
Set("Email", email);
|
||||||
if (sign == null)
|
Set("Premium", true);
|
||||||
{
|
Set("MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
||||||
Console.WriteLine("Could not find sign!");
|
Set("Version", 1);
|
||||||
return;
|
Set("Issued", DateTime.UtcNow);
|
||||||
}
|
Set("Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
||||||
|
Set("Expires", DateTime.UtcNow.AddYears(100));
|
||||||
Set(type, license, "LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
Set("Trial", false);
|
||||||
Set(type, license, "Id", userId);
|
Set("LicenseType", Enum.Parse(licenseTypeEnum, "User"));
|
||||||
Set(type, license, "Name", userName);
|
|
||||||
Set(type, license, "Email", email);
|
Set("Hash", Convert.ToBase64String((Byte[])type.GetMethod("ComputeHash").Invoke(license, [])));
|
||||||
Set(type, license, "Premium", true);
|
Set("Signature", Convert.ToBase64String((Byte[])type.GetMethod("Sign").Invoke(license, [cert])));
|
||||||
Set(type, license, "MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
|
||||||
Set(type, license, "Version", 1);
|
Console.WriteLine(JsonConvert.SerializeObject(license, Formatting.Indented));
|
||||||
Set(type, license, "Issued", DateTime.UtcNow);
|
return;
|
||||||
Set(type, license, "Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
|
||||||
Set(type, license, "Expires", DateTime.UtcNow.AddYears(100));
|
void Set(String name, Object value)
|
||||||
Set(type, license, "Trial", false);
|
{
|
||||||
Set(type, license, "LicenseType", Enum.Parse(licenseTypeEnum, "User"));
|
type.GetProperty(name)?.SetValue(license, value);
|
||||||
Set(type, license, "Hash", Convert.ToBase64String(((Byte[])computeHash.Invoke(license, []))!));
|
}
|
||||||
Set(type, license, "Signature", Convert.ToBase64String((Byte[])sign.Invoke(license, [cert])!));
|
}
|
||||||
|
|
||||||
Console.WriteLine(JsonSerializer.Serialize(license, JsonOptions));
|
private static void GenerateOrgLicense(X509Certificate2 cert, String corePath, String userName, String email, Int16 storage, Guid instalId, String businessName, String key)
|
||||||
}
|
{
|
||||||
private static void GenerateOrgLicense(X509Certificate2 cert, String corePath, String userName, String email, Int16 storage, Guid instalId, String businessName, String key)
|
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(corePath);
|
||||||
{
|
|
||||||
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(Path.GetFullPath(corePath));
|
Type type = core.GetType("Bit.Core.Models.Business.OrganizationLicense");
|
||||||
Type type = core.GetType("Bit.Core.Billing.Organizations.Models.OrganizationLicense");
|
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
||||||
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
Type planTypeEnum = core.GetType("Bit.Core.Billing.Enums.PlanType");
|
||||||
Type planTypeEnum = core.GetType("Bit.Core.Billing.Enums.PlanType");
|
|
||||||
|
Object license = Activator.CreateInstance(type);
|
||||||
if (type == null)
|
|
||||||
{
|
Set("LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
||||||
Console.WriteLine("Could not find type!");
|
Set("InstallationId", instalId);
|
||||||
return;
|
Set("Id", Guid.NewGuid());
|
||||||
}
|
Set("Name", userName);
|
||||||
if (licenseTypeEnum == null)
|
Set("BillingEmail", email);
|
||||||
{
|
Set("BusinessName", String.IsNullOrWhiteSpace(businessName) ? "BitBetter" : businessName);
|
||||||
Console.WriteLine("Could not find licenseTypeEnum!");
|
Set("Enabled", true);
|
||||||
return;
|
Set("Plan", "Enterprise (Annually)");
|
||||||
}
|
Set("PlanType", Enum.Parse(planTypeEnum, "EnterpriseAnnually"));
|
||||||
if (planTypeEnum == null)
|
Set("Seats", Int32.MaxValue);
|
||||||
{
|
Set("MaxCollections", Int16.MaxValue);
|
||||||
Console.WriteLine("Could not find planTypeEnum!");
|
Set("UsePolicies", true);
|
||||||
return;
|
Set("UseSso", true);
|
||||||
}
|
Set("UseKeyConnector", true);
|
||||||
|
Set("UseScim", true);
|
||||||
Object license = Activator.CreateInstance(type);
|
Set("UseGroups", true);
|
||||||
|
Set("UseEvents", true);
|
||||||
MethodInfo computeHash = type.GetMethod("ComputeHash");
|
Set("UseDirectory", true);
|
||||||
if (computeHash == null)
|
Set("UseTotp", true);
|
||||||
{
|
Set("Use2fa", true);
|
||||||
Console.WriteLine("Could not find ComputeHash!");
|
Set("UseApi", true);
|
||||||
return;
|
Set("UseResetPassword", true);
|
||||||
}
|
Set("UseCustomPermissions", true);
|
||||||
|
Set("MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
||||||
MethodInfo sign = type.GetMethod("Sign");
|
Set("SelfHost", true);
|
||||||
if (sign == null)
|
Set("UsersGetPremium", true);
|
||||||
{
|
Set("UsePasswordManager", true);
|
||||||
Console.WriteLine("Could not find sign!");
|
Set("UseSecretsManager", true);
|
||||||
return;
|
Set("SmSeats", Int32.MaxValue);
|
||||||
}
|
Set("SmServiceAccounts", Int32.MaxValue);
|
||||||
|
Set("Version", 15); //This is set to 15 to use AllowAdminAccessToAllCollectionItems can be changed to 13 to just use Secrets Manager
|
||||||
Set(type, license, "LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
Set("Issued", DateTime.UtcNow);
|
||||||
Set(type, license, "InstallationId", instalId);
|
Set("Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
||||||
Set(type, license, "Id", Guid.NewGuid());
|
Set("Expires", DateTime.UtcNow.AddYears(100));
|
||||||
Set(type, license, "Name", userName);
|
Set("Trial", false);
|
||||||
Set(type, license, "BillingEmail", email);
|
Set("LicenseType", Enum.Parse(licenseTypeEnum, "Organization"));
|
||||||
Set(type, license, "BusinessName", String.IsNullOrWhiteSpace(businessName) ? "BitBetter" : businessName);
|
Set("LimitCollectionCreationDeletion", true); //This will be used in the new version of BitWarden but can be applied now
|
||||||
Set(type, license, "Enabled", true);
|
Set("AllowAdminAccessToAllCollectionItems", true);
|
||||||
Set(type, license, "Plan", "Enterprise (Annually)");
|
|
||||||
Set(type, license, "PlanType", Enum.Parse(planTypeEnum, "EnterpriseAnnually"));
|
Set("Hash", Convert.ToBase64String((Byte[])type.GetMethod("ComputeHash").Invoke(license, [])));
|
||||||
Set(type, license, "Seats", Int32.MaxValue);
|
Set("Signature", Convert.ToBase64String((Byte[])type.GetMethod("Sign").Invoke(license, [cert])));
|
||||||
Set(type, license, "MaxCollections", Int16.MaxValue);
|
|
||||||
Set(type, license, "UsePolicies", true);
|
Console.WriteLine(JsonConvert.SerializeObject(license, Formatting.Indented));
|
||||||
Set(type, license, "UseSso", true);
|
return;
|
||||||
Set(type, license, "UseKeyConnector", true);
|
|
||||||
Set(type, license, "UseScim", true);
|
void Set(String name, Object value)
|
||||||
Set(type, license, "UseGroups", true);
|
{
|
||||||
Set(type, license, "UseEvents", true);
|
type.GetProperty(name)?.SetValue(license, value);
|
||||||
Set(type, license, "UseDirectory", true);
|
}
|
||||||
Set(type, license, "UseTotp", true);
|
}
|
||||||
Set(type, license, "Use2fa", true);
|
|
||||||
Set(type, license, "UseApi", true);
|
|
||||||
Set(type, license, "UseResetPassword", true);
|
|
||||||
Set(type, license, "MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
|
||||||
Set(type, license, "SelfHost", true);
|
|
||||||
Set(type, license, "UsersGetPremium", true);
|
|
||||||
Set(type, license, "UseCustomPermissions", true);
|
|
||||||
Set(type, license, "Version", 16);
|
|
||||||
Set(type, license, "Issued", DateTime.UtcNow);
|
|
||||||
Set(type, license, "Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
|
||||||
Set(type, license, "Expires", DateTime.UtcNow.AddYears(100));
|
|
||||||
Set(type, license, "ExpirationWithoutGracePeriod", DateTime.UtcNow.AddYears(100));
|
|
||||||
Set(type, license, "UsePasswordManager", true);
|
|
||||||
Set(type, license, "UseSecretsManager", true);
|
|
||||||
Set(type, license, "SmSeats", Int32.MaxValue);
|
|
||||||
Set(type, license, "SmServiceAccounts", Int32.MaxValue);
|
|
||||||
Set(type, license, "UseRiskInsights", true);
|
|
||||||
Set(type, license, "LimitCollectionCreationDeletion", true);
|
|
||||||
Set(type, license, "AllowAdminAccessToAllCollectionItems", true);
|
|
||||||
Set(type, license, "Trial", false);
|
|
||||||
Set(type, license, "LicenseType", Enum.Parse(licenseTypeEnum, "Organization"));
|
|
||||||
Set(type, license, "UseOrganizationDomains", true);
|
|
||||||
Set(type, license, "UseAdminSponsoredFamilies", true);
|
|
||||||
Set(type, license, "Hash", Convert.ToBase64String((Byte[])computeHash.Invoke(license, [])!));
|
|
||||||
Set(type, license, "Signature", Convert.ToBase64String((Byte[])sign.Invoke(license, [cert])!));
|
|
||||||
|
|
||||||
Console.WriteLine(JsonSerializer.Serialize(license, JsonOptions));
|
|
||||||
}
|
|
||||||
private static void Set(Type type, Object license, String name, Object value)
|
|
||||||
{
|
|
||||||
type.GetProperty(name)?.SetValue(license, value);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -1,10 +1,14 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<OutputType>Exe</OutputType>
|
<OutputType>Exe</OutputType>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="McMaster.Extensions.CommandLineUtils" Version="4.1.1" />
|
<PackageReference Include="McMaster.Extensions.CommandLineUtils" Version="4.1.1" />
|
||||||
|
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
|
||||||
<PackageReference Include="System.Runtime.Loader" Version="4.3.0" />
|
<PackageReference Include="System.Runtime.Loader" Version="4.3.0" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
|
||||||
|
</Project>
|
||||||
|
|||||||
Reference in New Issue
Block a user