mirror of
https://github.com/jakeswenson/BitBetter.git
synced 2025-09-15 06:13:26 +00:00
Merge 0e34e638b0
into 29add24126
This commit is contained in:
commit
4e725ac498
|
@ -12,7 +12,7 @@ jobs:
|
||||||
command: ./generateKeys.sh
|
command: ./generateKeys.sh
|
||||||
- run:
|
- run:
|
||||||
name: Build script
|
name: Build script
|
||||||
command: ./build.sh y
|
command: ./build.sh update
|
||||||
- run:
|
- run:
|
||||||
name: Test generating user license
|
name: Test generating user license
|
||||||
command: ./licenseGen.sh user TestName TestEmail@example.com 4a619d4a-522d-4c70-8596-affb5b607c23
|
command: ./licenseGen.sh user TestName TestEmail@example.com 4a619d4a-522d-4c70-8596-affb5b607c23
|
||||||
|
|
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -7,5 +7,5 @@ src/bitBetter/.vs/*
|
||||||
*.pem
|
*.pem
|
||||||
.vscode/
|
.vscode/
|
||||||
*.pfx
|
*.pfx
|
||||||
*.cert
|
*.cer
|
||||||
*.vsidx
|
*.vsidx
|
||||||
|
|
|
@ -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 -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
|
# 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
|
||||||
# <OR>
|
# <OR>
|
||||||
# docker-compose -f <full-local-path>/docker-compose.yml up -d
|
# docker-compose -f <full-local-path>/docker-compose.yml up -d
|
||||||
|
|
47
README.md
47
README.md
|
@ -46,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.cert -days 36500 -outform DER -passout pass:test
|
openssl req -x509 -newkey rsa:4096 -keyout key.pem -out cert.cer -days 36500 -outform DER -passout pass:test
|
||||||
openssl x509 -inform DER -in cert.cert -out cert.pem
|
openssl x509 -inform DER -in cert.cer -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
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -70,7 +70,7 @@ 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/self-host` image called `bitwarden-patch`.
|
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-patched`.
|
||||||
|
|
||||||
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.**
|
||||||
|
|
||||||
|
@ -98,6 +98,36 @@ 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/docker-unified/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.
|
||||||
|
@ -114,6 +144,17 @@ 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.
|
||||||
|
|
||||||
# Footnotes
|
# Footnotes
|
||||||
|
|
||||||
|
|
85
build.ps1
85
build.ps1
|
@ -1,13 +1,16 @@
|
||||||
$ErrorActionPreference = 'Stop'
|
$ErrorActionPreference = 'Stop'
|
||||||
$PSNativeCommandUseErrorActionPreference = $true
|
$PSNativeCommandUseErrorActionPreference = $true
|
||||||
|
|
||||||
|
# detect buildx, ErrorActionPreference will ensure the script stops execution if not found
|
||||||
|
docker buildx version
|
||||||
|
|
||||||
# define temporary directory
|
# define temporary directory
|
||||||
$tempdirectory = "$pwd\temp"
|
$tempdirectory = "$pwd\temp"
|
||||||
# define services to patch
|
# define services to patch
|
||||||
$components = "Api","Identity"
|
$components = "Api","Identity"
|
||||||
|
|
||||||
# delete old directories / files if applicable
|
# delete old directories / files if applicable
|
||||||
if (Test-Path "$tempdirectory") {
|
if (Test-Path "$tempdirectory" -PathType Container) {
|
||||||
Remove-Item "$tempdirectory" -Recurse -Force
|
Remove-Item "$tempdirectory" -Recurse -Force
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,24 +22,27 @@ if (Test-Path -Path "$pwd\src\licenseGen\cert.pfx" -PathType Leaf) {
|
||||||
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Test-Path -Path "$pwd\src\bitBetter\cert.cert" -PathType Leaf) {
|
if (Test-Path -Path "$pwd\src\bitBetter\cert.cer" -PathType Leaf) {
|
||||||
Remove-Item "$pwd\src\bitBetter\cert.cert" -Force
|
Remove-Item "$pwd\src\bitBetter\cert.cer" -Force
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Test-Path "$pwd\.keys\cert.cert" -PathType Leaf) {
|
||||||
|
Rename-Item -Path "$pwd\.keys\cert.cert" -NewName "$pwd\.keys\cert.cer"
|
||||||
}
|
}
|
||||||
|
|
||||||
# generate keys if none are available
|
# generate keys if none are available
|
||||||
if (!(Test-Path "$pwd\.keys")) {
|
if (!(Test-Path "$pwd\.keys" -PathType Container)) {
|
||||||
.\generateKeys.ps1
|
.\generateKeys.ps1
|
||||||
}
|
}
|
||||||
|
|
||||||
# copy the key to bitBetter and licenseGen
|
# copy the key to bitBetter
|
||||||
Copy-Item "$pwd\.keys\cert.cert" -Destination "$pwd\src\bitBetter"
|
Copy-Item "$pwd\.keys\cert.cer" -Destination "$pwd\src\bitBetter"
|
||||||
Copy-Item "$pwd\.keys\cert.pfx" -Destination "$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 --no-cache -t bitbetter/bitbetter "$pwd\src\bitBetter"
|
||||||
Remove-Item "$pwd\src\bitBetter\cert.cert" -Force
|
Remove-Item "$pwd\src\bitBetter\cert.cer" -Force
|
||||||
|
|
||||||
# gather all running instances
|
# 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}}'
|
$oldinstances = docker container ps --all -f Name=bitwarden --format '{{.ID}}'
|
||||||
|
|
||||||
# stop and remove all running instances
|
# stop and remove all running instances
|
||||||
|
@ -46,25 +52,35 @@ foreach ($instance in $oldinstances) {
|
||||||
}
|
}
|
||||||
|
|
||||||
# update bitwarden itself
|
# update bitwarden itself
|
||||||
if ($args[0] -eq 'y')
|
if ($args[0] -eq 'update') {
|
||||||
{
|
|
||||||
docker pull ghcr.io/bitwarden/self-host:beta
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
}
|
} else {
|
||||||
else
|
$confirmation = Read-Host "Update (or get) bitwarden source container (y/n)"
|
||||||
{
|
|
||||||
$confirmation = Read-Host "Update (or get) bitwarden source container"
|
|
||||||
if ($confirmation -eq 'y') {
|
if ($confirmation -eq 'y') {
|
||||||
docker pull ghcr.io/bitwarden/self-host:beta
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# remove previous existing patch(ed) image
|
# stop and remove previous existing patch(ed) container
|
||||||
if (docker image ls -q bitwarden-patch) {
|
$oldinstances = docker container ps --all -f Ancestor=bitwarden-patched --format '{{.ID}}'
|
||||||
docker image rm bitwarden-patch
|
foreach ($instance in $oldinstances) {
|
||||||
|
docker stop $instance
|
||||||
|
docker rm $instance
|
||||||
|
}
|
||||||
|
$oldinstances = docker image ls bitwarden-patched --format '{{.ID}}'
|
||||||
|
foreach ($instance in $oldinstances) {
|
||||||
|
docker image rm $instance
|
||||||
|
}
|
||||||
|
|
||||||
|
# remove old extract containers
|
||||||
|
$oldinstances = docker container ps --all -f Name=bitwarden-extract --format '{{.ID}}'
|
||||||
|
foreach ($instance in $oldinstances) {
|
||||||
|
docker stop $instance
|
||||||
|
docker rm $instance
|
||||||
}
|
}
|
||||||
|
|
||||||
# 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-patch ghcr.io/bitwarden/self-host:beta
|
$patchinstance = docker run -d --name bitwarden-extract ghcr.io/bitwarden/self-host:beta
|
||||||
|
|
||||||
# create our temporary directory
|
# create our temporary directory
|
||||||
New-item -ItemType Directory -Path $tempdirectory
|
New-item -ItemType Directory -Path $tempdirectory
|
||||||
|
@ -75,33 +91,38 @@ foreach ($component in $components) {
|
||||||
docker cp $patchinstance`:/app/$component/Core.dll "$tempdirectory\$component\Core.dll"
|
docker cp $patchinstance`:/app/$component/Core.dll "$tempdirectory\$component\Core.dll"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# 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-patch --file "$pwd\src\bitBetter\Dockerfile-bitwarden-patch"
|
docker build . --tag bitwarden-patched --file "$pwd\src\bitBetter\Dockerfile-bitwarden-patch"
|
||||||
|
|
||||||
# stop and remove our temporary container
|
|
||||||
docker stop bitwarden-patch
|
|
||||||
docker rm bitwarden-patch
|
|
||||||
|
|
||||||
# copy our patched library to the licenseGen source directory
|
|
||||||
Copy-Item "$tempdirectory\Identity\Core.dll" -Destination "$pwd\src\licenseGen"
|
|
||||||
|
|
||||||
# remove our temporary directory
|
|
||||||
Remove-Item "$tempdirectory" -Recurse -Force
|
|
||||||
|
|
||||||
# start all user requested instances
|
# start all user requested instances
|
||||||
foreach($line in Get-Content "$pwd\.servers\serverlist.txt") {
|
if (Test-Path -Path "$pwd\.servers\serverlist.txt" -PathType Leaf) {
|
||||||
Invoke-Expression "& $line"
|
foreach($line in Get-Content "$pwd\.servers\serverlist.txt") {
|
||||||
|
if (!($line.StartsWith("#"))) {
|
||||||
|
Invoke-Expression "& $line"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# remove our bitBetter image
|
# remove our bitBetter image
|
||||||
docker image rm bitbetter/bitbetter
|
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
|
# build the licenseGen
|
||||||
docker build -t bitbetter/licensegen "$pwd\src\licenseGen"
|
docker build -t bitbetter/licensegen "$pwd\src\licenseGen"
|
||||||
|
|
||||||
# clean the licenseGen source directory
|
# clean the licenseGen source directory
|
||||||
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
Remove-Item "$pwd\src\licenseGen\Core.dll" -Force
|
||||||
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
Remove-Item "$pwd\src\licenseGen\cert.pfx" -Force
|
||||||
|
|
||||||
|
# remove our temporary directory
|
||||||
|
Remove-Item "$tempdirectory" -Recurse -Force
|
87
build.sh
87
build.sh
|
@ -1,6 +1,9 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
|
# detect buildx, set -e will ensure the script stops execution if not found
|
||||||
|
docker buildx version
|
||||||
|
|
||||||
# define temporary directory
|
# define temporary directory
|
||||||
TEMPDIRECTORY="$PWD/temp"
|
TEMPDIRECTORY="$PWD/temp"
|
||||||
|
|
||||||
|
@ -20,8 +23,12 @@ 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.cert" ]; then
|
if [ -f "$PWD/src/bitBetter/cert.cer" ]; then
|
||||||
rm -f "$PWD/src/bitBetter/cert.cert"
|
rm -f "$PWD/src/bitBetter/cert.cer"
|
||||||
|
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
|
||||||
|
@ -29,15 +36,14 @@ if [ ! -d "$PWD/.keys" ]; then
|
||||||
./generateKeys.sh
|
./generateKeys.sh
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# copy the key to bitBetter and licenseGen
|
# copy the key to bitBetter
|
||||||
cp -f "$PWD/.keys/cert.cert" "$PWD/src/bitBetter"
|
cp -f "$PWD/.keys/cert.cer" "$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 --no-cache -t bitbetter/bitbetter "$PWD/src/bitBetter"
|
||||||
rm -f "$PWD/src/bitBetter/cert.cert"
|
rm -f "$PWD/src/bitBetter/cert.cer"
|
||||||
|
|
||||||
# gather all running instances
|
# 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}}')
|
OLDINSTANCES=$(docker container ps --all -f Name=bitwarden --format '{{.ID}}')
|
||||||
|
|
||||||
# stop and remove all running instances
|
# stop and remove all running instances
|
||||||
|
@ -47,24 +53,35 @@ for INSTANCE in ${OLDINSTANCES[@]}; do
|
||||||
done
|
done
|
||||||
|
|
||||||
# update bitwarden itself
|
# update bitwarden itself
|
||||||
if [ "$1" = "y" ]; then
|
if [ "$1" = "update" ]; then
|
||||||
docker pull ghcr.io/bitwarden/self-host:beta
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
else
|
else
|
||||||
read -p "Update (or get) bitwarden source container: " -n 1 -r
|
read -p "Update (or get) bitwarden source container (y/n): "
|
||||||
echo
|
if [[ $REPLY =~ ^[Yy]$ ]]; then
|
||||||
if [[ $REPLY =~ ^[Yy]$ ]]
|
|
||||||
then
|
|
||||||
docker pull ghcr.io/bitwarden/self-host:beta
|
docker pull ghcr.io/bitwarden/self-host:beta
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# remove previous existing patch(ed) image
|
# stop and remove previous existing patch(ed) container
|
||||||
if [ "$(docker image ls -q bitwarden-patch)" ]; then
|
OLDINSTANCES=$(docker container ps --all -f Ancestor=bitwarden-patched --format '{{.ID}}')
|
||||||
docker image rm bitwarden-patch
|
for INSTANCE in ${OLDINSTANCES[@]}; do
|
||||||
fi
|
docker stop $INSTANCE
|
||||||
|
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-patch ghcr.io/bitwarden/self-host:beta)
|
PATCHINSTANCE=$(docker run -d --name bitwarden-extract ghcr.io/bitwarden/self-host:beta)
|
||||||
|
|
||||||
# create our temporary directory
|
# create our temporary directory
|
||||||
mkdir $TEMPDIRECTORY
|
mkdir $TEMPDIRECTORY
|
||||||
|
@ -75,34 +92,40 @@ 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-patch --file "$PWD/src/bitBetter/Dockerfile-bitwarden-patch"
|
docker build . --tag bitwarden-patched --file "$PWD/src/bitBetter/Dockerfile-bitwarden-patch"
|
||||||
|
|
||||||
# stop and remove our temporary container
|
|
||||||
docker stop bitwarden-patch
|
|
||||||
docker rm bitwarden-patch
|
|
||||||
|
|
||||||
# copy our patched library to the licenseGen source directory
|
|
||||||
cp -f "$TEMPDIRECTORY/Identity/Core.dll" "$PWD/src/licenseGen"
|
|
||||||
|
|
||||||
# remove our temporary directory
|
|
||||||
rm -rf "$TEMPDIRECTORY"
|
|
||||||
|
|
||||||
# start all user requested instances
|
# start all user requested instances
|
||||||
sed -i 's/\r$//' "$PWD/.servers/serverlist.txt"
|
if [ -f "$PWD/.servers/serverlist.txt" ]; then
|
||||||
cat "$PWD/.servers/serverlist.txt" | while read -r LINE; do
|
# convert line endings to unix
|
||||||
bash -c "$LINE"
|
sed -i 's/\r$//' "$PWD/.servers/serverlist.txt"
|
||||||
done
|
cat "$PWD/.servers/serverlist.txt" | while read -r LINE; do
|
||||||
|
if [[ $LINE != "#"* ]]; then
|
||||||
|
bash -c "$LINE"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
# remove our bitBetter image
|
# remove our bitBetter image
|
||||||
docker image rm bitbetter/bitbetter
|
docker image rm bitbetter/bitbetter
|
||||||
|
|
||||||
|
# copy our patched library to the licenseGen source directory
|
||||||
|
cp -f "$TEMPDIRECTORY/Identity/Core.dll" "$PWD/src/licenseGen"
|
||||||
|
cp -f "$PWD/.keys/cert.pfx" "$PWD/src/licenseGen"
|
||||||
|
|
||||||
# build the licenseGen
|
# build the licenseGen
|
||||||
docker build -t bitbetter/licensegen "$PWD/src/licenseGen"
|
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"
|
|
@ -20,6 +20,6 @@ if (Test-Path "$pwd\.keys")
|
||||||
New-item -ItemType Directory -Path "$pwd\.keys"
|
New-item -ItemType Directory -Path "$pwd\.keys"
|
||||||
|
|
||||||
# generate actual keys
|
# generate actual 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' 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.cert`" -out `"$pwd\.keys\cert.pem`""
|
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"
|
|
@ -15,6 +15,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.cert" -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.cer" -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.cert" -out "$DIR/cert.pem"
|
openssl x509 -inform DER -in "$DIR/cert.cer" -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
|
||||||
|
|
|
@ -10,7 +10,7 @@ if ($($args.Count) -lt 1) {
|
||||||
Exit 1
|
Exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($args[0] = "interactive") {
|
if ($args[0] -eq "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
|
||||||
|
|
|
@ -2,7 +2,7 @@ FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
||||||
WORKDIR /bitBetter
|
WORKDIR /bitBetter
|
||||||
|
|
||||||
COPY . /bitBetter
|
COPY . /bitBetter
|
||||||
COPY cert.cert /app/
|
COPY cert.cer /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 [ "/app/bitBetter" ]
|
ENTRYPOINT ["dotnet", "/app/bitBetter.dll"]
|
|
@ -1,4 +1,3 @@
|
||||||
FROM ghcr.io/bitwarden/self-host:beta
|
FROM ghcr.io/bitwarden/self-host:beta
|
||||||
|
|
||||||
COPY ./temp/Api/Core.dll /app/Api/Core.dll
|
COPY ./temp/ /app/
|
||||||
COPY ./temp/Identity/Core.dll /app/Identity/Core.dll
|
|
|
@ -14,7 +14,7 @@ internal class Program
|
||||||
{
|
{
|
||||||
private static Int32 Main()
|
private static Int32 Main()
|
||||||
{
|
{
|
||||||
const String certFile = "/app/cert.cert";
|
const String certFile = "/app/cert.cer";
|
||||||
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)
|
||||||
|
@ -23,12 +23,7 @@ internal class Program
|
||||||
ModuleDefMD moduleDefMd = ModuleDefMD.Load(file);
|
ModuleDefMD moduleDefMd = ModuleDefMD.Load(file);
|
||||||
Byte[] cert = File.ReadAllBytes(certFile);
|
Byte[] cert = File.ReadAllBytes(certFile);
|
||||||
|
|
||||||
EmbeddedResource embeddedResourceToRemove = moduleDefMd.Resources
|
EmbeddedResource embeddedResourceToRemove = moduleDefMd.Resources.OfType<EmbeddedResource>().First(r => r.Name.Equals("Bit.Core.licensing.cer"));
|
||||||
.OfType<EmbeddedResource>()
|
|
||||||
.First(r => r.Name.Equals("Bit.Core.licensing.cer"));
|
|
||||||
|
|
||||||
Console.WriteLine(embeddedResourceToRemove.Name);
|
|
||||||
|
|
||||||
EmbeddedResource embeddedResourceToAdd = new("Bit.Core.licensing.cer", cert) { Attributes = embeddedResourceToRemove.Attributes };
|
EmbeddedResource embeddedResourceToAdd = new("Bit.Core.licensing.cer", cert) { Attributes = embeddedResourceToRemove.Attributes };
|
||||||
moduleDefMd.Resources.Add(embeddedResourceToAdd);
|
moduleDefMd.Resources.Add(embeddedResourceToAdd);
|
||||||
moduleDefMd.Resources.Remove(embeddedResourceToRemove);
|
moduleDefMd.Resources.Remove(embeddedResourceToRemove);
|
||||||
|
@ -45,10 +40,7 @@ internal class Program
|
||||||
TypeDef type = services.First(t => t.Name == "LicensingService");
|
TypeDef type = services.First(t => t.Name == "LicensingService");
|
||||||
MethodDef constructor = type.FindConstructors().First();
|
MethodDef constructor = type.FindConstructors().First();
|
||||||
|
|
||||||
Instruction instructionToPatch =
|
Instruction instructionToPatch = constructor.Body.Instructions.FirstOrDefault(i => i.OpCode == OpCodes.Ldstr && String.Equals((String)i.Operand, existingCert.Thumbprint, StringComparison.InvariantCultureIgnoreCase));
|
||||||
constructor.Body.Instructions
|
|
||||||
.FirstOrDefault(i => i.OpCode == OpCodes.Ldstr
|
|
||||||
&& String.Equals((String)i.Operand, existingCert.Thumbprint, StringComparison.InvariantCultureIgnoreCase));
|
|
||||||
|
|
||||||
if (instructionToPatch != null)
|
if (instructionToPatch != null)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,12 +1,9 @@
|
||||||
<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="dnlib" Version="4.4.0" />
|
<PackageReference Include="dnlib" Version="4.5.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", "--core", "/app/Core.dll", "--cert", "/app/cert.pfx" ]
|
ENTRYPOINT ["dotnet", "/app/licenseGen.dll", "--cert=/app/cert.pfx", "--core=/app/Core.dll"]
|
||||||
|
|
|
@ -1,57 +1,31 @@
|
||||||
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 readonly CommandOption Cert = App.Option("--cert", "Certificate file", CommandOptionType.SingleValue);
|
||||||
|
private static readonly CommandOption CoreDll = App.Option("--core", "Path to Core.dll", CommandOptionType.SingleValue);
|
||||||
|
|
||||||
private static Int32 Main(String[] args)
|
private static Int32 Main(String[] args)
|
||||||
{
|
{
|
||||||
CommandLineApplication app = new();
|
App.Command("interactive", config =>
|
||||||
CommandOption cert = app.Option("--cert", "cert file", CommandOptionType.SingleValue);
|
|
||||||
CommandOption coreDll = app.Option("--core", "path to core dll", CommandOptionType.SingleValue);
|
|
||||||
|
|
||||||
Boolean CertExists()
|
|
||||||
{
|
|
||||||
return File.Exists(cert.Value());
|
|
||||||
}
|
|
||||||
|
|
||||||
Boolean CoreExists()
|
|
||||||
{
|
|
||||||
return File.Exists(coreDll.Value());
|
|
||||||
}
|
|
||||||
|
|
||||||
Boolean VerifyTopOptions()
|
|
||||||
{
|
|
||||||
return !String.IsNullOrWhiteSpace(cert.Value()) &&
|
|
||||||
!String.IsNullOrWhiteSpace(coreDll.Value()) &&
|
|
||||||
CertExists() && CoreExists();
|
|
||||||
}
|
|
||||||
|
|
||||||
app.Command("interactive", config =>
|
|
||||||
{
|
{
|
||||||
String buff, licenseType = "", name = "", email = "", businessName="";
|
String buff, licenseType = "", name = "", email = "", businessName="";
|
||||||
Int16 storage = 0;
|
Int16 storage = 0;
|
||||||
|
|
||||||
Boolean validGuid = false, validInstallid = false;
|
Boolean validGuid = false, validInstallid = false;
|
||||||
Guid guid = new(), installid = new();
|
Guid guid = Guid.Empty, installid = Guid.Empty;
|
||||||
|
|
||||||
config.OnExecute(() =>
|
config.OnExecute(() =>
|
||||||
{
|
{
|
||||||
if (!VerifyTopOptions())
|
Check();
|
||||||
{
|
|
||||||
if (!CoreExists()) config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
|
||||||
if (!CertExists()) config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
|
||||||
|
|
||||||
config.ShowHelp();
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.WriteLine("Interactive license mode...");
|
Console.WriteLine("Interactive license mode...");
|
||||||
|
|
||||||
while (licenseType == "")
|
while (licenseType == "")
|
||||||
|
@ -59,51 +33,55 @@ internal class Program
|
||||||
Console.WriteLine("What would you like to generate, a [u]ser license or an [o]rg license: ");
|
Console.WriteLine("What would you like to generate, a [u]ser license or an [o]rg license: ");
|
||||||
buff = Console.ReadLine();
|
buff = Console.ReadLine();
|
||||||
|
|
||||||
if(buff == "u")
|
switch (buff)
|
||||||
{
|
{
|
||||||
licenseType = "user";
|
case "u":
|
||||||
Console.WriteLine("Okay, we will generate a user license.");
|
|
||||||
|
|
||||||
while (validGuid == false)
|
|
||||||
{
|
{
|
||||||
Console.WriteLine("Please provide the user's guid — refer to the Readme for details on how to retrieve this. [GUID]: ");
|
licenseType = "user";
|
||||||
buff = Console.ReadLine();
|
Console.WriteLine("Okay, we will generate a user license.");
|
||||||
|
|
||||||
if (Guid.TryParse(buff, out guid))validGuid = true;
|
while (!validGuid)
|
||||||
else Console.WriteLine("The user-guid provided does not appear to be valid!");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (buff == "o")
|
|
||||||
{
|
|
||||||
licenseType = "org";
|
|
||||||
Console.WriteLine("Okay, we will generate an organization license.");
|
|
||||||
|
|
||||||
while (validInstallid == false)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Please provide your Bitwarden Install-ID — refer to the Readme for details on how to retrieve this. [Install-ID]: ");
|
|
||||||
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 (businessName == "")
|
|
||||||
{
|
|
||||||
Console.WriteLine("Please enter a business name, default is BitBetter. [Business Name]: ");
|
|
||||||
buff = Console.ReadLine();
|
|
||||||
if (buff == "")
|
|
||||||
{
|
{
|
||||||
businessName = "BitBetter";
|
Console.WriteLine("Please provide the user's guid — refer to the Readme for details on how to retrieve this. [GUID]: ");
|
||||||
}
|
buff = Console.ReadLine();
|
||||||
else if (CheckBusinessName(buff))
|
|
||||||
{
|
if (Guid.TryParse(buff, out guid))validGuid = true;
|
||||||
businessName = buff;
|
else Console.WriteLine("The user-guid provided does not appear to be valid!");
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
case "o":
|
||||||
else
|
{
|
||||||
{
|
licenseType = "org";
|
||||||
Console.WriteLine("Unrecognized option \'" + buff + "\'.");
|
Console.WriteLine("Okay, we will generate an organization license.");
|
||||||
|
|
||||||
|
while (!validInstallid)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Please provide your Bitwarden Install-ID — refer to the Readme for details on how to retrieve this. [Install-ID]: ");
|
||||||
|
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 (businessName == "")
|
||||||
|
{
|
||||||
|
Console.WriteLine("Please enter a business name, default is BitBetter. [Business Name]: ");
|
||||||
|
buff = Console.ReadLine();
|
||||||
|
if (buff == "")
|
||||||
|
{
|
||||||
|
businessName = "BitBetter";
|
||||||
|
}
|
||||||
|
else if (CheckBusinessName(buff))
|
||||||
|
{
|
||||||
|
businessName = buff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
Console.WriteLine("Unrecognized option \'" + buff + "\'.");
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -111,7 +89,7 @@ internal class Program
|
||||||
{
|
{
|
||||||
Console.WriteLine("Please provide the username this license will be registered to. [username]: ");
|
Console.WriteLine("Please provide the username this license will be registered to. [username]: ");
|
||||||
buff = Console.ReadLine();
|
buff = Console.ReadLine();
|
||||||
if ( CheckUsername(buff) ) name = buff;
|
if (CheckUsername(buff)) name = buff;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (email == "")
|
while (email == "")
|
||||||
|
@ -141,39 +119,46 @@ internal class Program
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (licenseType == "user")
|
switch (licenseType)
|
||||||
{
|
{
|
||||||
Console.WriteLine("Confirm creation of \"user\" license for username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", User-GUID: \"" + guid + "\"? Y/n");
|
case "user":
|
||||||
buff = Console.ReadLine();
|
|
||||||
if ( buff is "" or "y" or "Y" )
|
|
||||||
{
|
{
|
||||||
GenerateUserLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name, email, storage, guid, null);
|
Console.WriteLine("Confirm creation of \"user\" license for username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", User-GUID: \"" + guid + "\"? Y/n");
|
||||||
|
buff = Console.ReadLine();
|
||||||
|
if (buff is "" or "y" or "Y")
|
||||||
|
{
|
||||||
|
GenerateUserLicense(new X509Certificate2(Cert.Value(), "test"), CoreDll.Value(), name, email, storage, guid, null);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Console.WriteLine("Exiting...");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else
|
case "org":
|
||||||
{
|
{
|
||||||
Console.WriteLine("Exiting...");
|
Console.WriteLine("Confirm creation of \"organization\" license for business name: \"" + businessName + "\", username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", Install-ID: \"" + installid + "\"? Y/n");
|
||||||
return 0;
|
buff = Console.ReadLine();
|
||||||
}
|
if (buff is "" or "y" or "Y")
|
||||||
}
|
{
|
||||||
else if (licenseType == "org")
|
GenerateOrgLicense(new X509Certificate2(Cert.Value(), "test"), CoreDll.Value(), name, email, storage, installid, businessName, null);
|
||||||
{
|
}
|
||||||
Console.WriteLine("Confirm creation of \"organization\" license for business name: \"" + businessName + "\", username: \"" + name + "\", email: \"" + email + "\", Storage: \"" + storage + " GB\", Install-ID: \"" + installid + "\"? Y/n");
|
else
|
||||||
buff = Console.ReadLine();
|
{
|
||||||
if ( buff is "" or "y" or "Y" )
|
Console.WriteLine("Exiting...");
|
||||||
{
|
return 0;
|
||||||
GenerateOrgLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name, email, storage, installid, businessName, null);
|
}
|
||||||
}
|
|
||||||
else
|
break;
|
||||||
{
|
|
||||||
Console.WriteLine("Exiting...");
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
app.Command("user", config =>
|
App.Command("user", config =>
|
||||||
{
|
{
|
||||||
CommandArgument name = config.Argument("Name", "your name");
|
CommandArgument name = config.Argument("Name", "your name");
|
||||||
CommandArgument email = config.Argument("Email", "your email");
|
CommandArgument email = config.Argument("Email", "your email");
|
||||||
|
@ -183,20 +168,7 @@ internal class Program
|
||||||
|
|
||||||
config.OnExecute(() =>
|
config.OnExecute(() =>
|
||||||
{
|
{
|
||||||
if (!VerifyTopOptions())
|
Check();
|
||||||
{
|
|
||||||
if (!CoreExists())
|
|
||||||
{
|
|
||||||
config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
|
||||||
}
|
|
||||||
if (!CertExists())
|
|
||||||
{
|
|
||||||
config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
|
||||||
}
|
|
||||||
|
|
||||||
config.ShowHelp();
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value))
|
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value))
|
||||||
{
|
{
|
||||||
|
@ -225,12 +197,12 @@ internal class Program
|
||||||
storageShort = (Int16) parsedStorage;
|
storageShort = (Int16) parsedStorage;
|
||||||
}
|
}
|
||||||
|
|
||||||
GenerateUserLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name.Value, email.Value, storageShort, userId, key.Value);
|
GenerateUserLicense(new X509Certificate2(Cert.Value()!, "test"), CoreDll.Value(), name.Value, email.Value, storageShort, userId, key.Value);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
app.Command("org", config =>
|
App.Command("org", config =>
|
||||||
{
|
{
|
||||||
CommandArgument name = config.Argument("Name", "your name");
|
CommandArgument name = config.Argument("Name", "your name");
|
||||||
CommandArgument email = config.Argument("Email", "your email");
|
CommandArgument email = config.Argument("Email", "your email");
|
||||||
|
@ -241,24 +213,9 @@ internal class Program
|
||||||
|
|
||||||
config.OnExecute(() =>
|
config.OnExecute(() =>
|
||||||
{
|
{
|
||||||
if (!VerifyTopOptions())
|
Check();
|
||||||
{
|
|
||||||
if (!CoreExists())
|
|
||||||
{
|
|
||||||
config.Error.WriteLine($"Can't find core dll at: {coreDll.Value()}");
|
|
||||||
}
|
|
||||||
if (!CertExists())
|
|
||||||
{
|
|
||||||
config.Error.WriteLine($"Can't find certificate at: {cert.Value()}");
|
|
||||||
}
|
|
||||||
|
|
||||||
config.ShowHelp();
|
if (String.IsNullOrWhiteSpace(name.Value) || String.IsNullOrWhiteSpace(email.Value) || String.IsNullOrWhiteSpace(installId.Value))
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (String.IsNullOrWhiteSpace(name.Value) ||
|
|
||||||
String.IsNullOrWhiteSpace(email.Value) ||
|
|
||||||
String.IsNullOrWhiteSpace(installId.Value))
|
|
||||||
{
|
{
|
||||||
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}' InstallId='{installId.Value}'");
|
config.Error.WriteLine($"Some arguments are missing: Name='{name.Value}' Email='{email.Value}' InstallId='{installId.Value}'");
|
||||||
config.ShowHelp(true);
|
config.ShowHelp(true);
|
||||||
|
@ -283,34 +240,61 @@ internal class Program
|
||||||
config.ShowHelp(true);
|
config.ShowHelp(true);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
storageShort = (Int16) parsedStorage;
|
storageShort = (Int16)parsedStorage;
|
||||||
}
|
}
|
||||||
|
|
||||||
GenerateOrgLicense(new X509Certificate2(cert.Value(), "test"), coreDll.Value(), name.Value, email.Value, storageShort, installationId, businessName.Value, key.Value);
|
GenerateOrgLicense(new X509Certificate2(Cert.Value()!, "test"), CoreDll.Value(), name.Value, email.Value, storageShort, installationId, businessName.Value, key.Value);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
app.OnExecute(() =>
|
App.OnExecute(() =>
|
||||||
{
|
{
|
||||||
app.ShowHelp();
|
App.ShowHelp();
|
||||||
return 10;
|
return 10;
|
||||||
});
|
});
|
||||||
|
|
||||||
app.HelpOption("-? | -h | --help");
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return app.Execute(args);
|
App.HelpOption("-? | -h | --help");
|
||||||
|
return App.Execute(args);
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception exception)
|
||||||
{
|
{
|
||||||
Console.Error.WriteLine("Oops: {0}", e);
|
Console.Error.WriteLine("Oops: {0}", exception);
|
||||||
return 100;
|
return 100;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void Check()
|
||||||
|
{
|
||||||
|
if (Cert == null || String.IsNullOrWhiteSpace(Cert.Value()))
|
||||||
|
{
|
||||||
|
App.Error.WriteLine("No certificate specified");
|
||||||
|
App.ShowHelp();
|
||||||
|
Environment.Exit(1);
|
||||||
|
}
|
||||||
|
else if (CoreDll == null || String.IsNullOrWhiteSpace(CoreDll.Value()))
|
||||||
|
{
|
||||||
|
App.Error.WriteLine("No core dll specified");
|
||||||
|
App.ShowHelp();
|
||||||
|
Environment.Exit(1);
|
||||||
|
}
|
||||||
|
else if (!File.Exists(Cert.Value()))
|
||||||
|
{
|
||||||
|
App.Error.WriteLine($"Can't find certificate at: {Cert.Value()}");
|
||||||
|
App.ShowHelp();
|
||||||
|
Environment.Exit(1);
|
||||||
|
}
|
||||||
|
else if (!File.Exists(CoreDll.Value()))
|
||||||
|
{
|
||||||
|
App.Error.WriteLine($"Can't find core dll at: {CoreDll.Value()}");
|
||||||
|
App.ShowHelp();
|
||||||
|
Environment.Exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// checkUsername Checks that the username is a valid username
|
// checkUsername Checks that the username is a valid username
|
||||||
private static Boolean CheckUsername(String s)
|
private static Boolean CheckUsername(String s)
|
||||||
{
|
{
|
||||||
|
@ -356,99 +340,146 @@ internal class Program
|
||||||
return false;
|
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)
|
private static void GenerateUserLicense(X509Certificate2 cert, String corePath, String userName, String email, Int16 storage, Guid userId, String key)
|
||||||
{
|
{
|
||||||
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(corePath);
|
Assembly core = AssemblyLoadContext.Default.LoadFromAssemblyPath(Path.GetFullPath(corePath));
|
||||||
|
|
||||||
Type type = core.GetType("Bit.Core.Billing.Models.Business.UserLicense");
|
Type type = core.GetType("Bit.Core.Billing.Models.Business.UserLicense");
|
||||||
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
Type licenseTypeEnum = core.GetType("Bit.Core.Enums.LicenseType");
|
||||||
|
|
||||||
|
if (type == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find type!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (licenseTypeEnum == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find license licenseTypeEnum!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
Object license = Activator.CreateInstance(type);
|
Object license = Activator.CreateInstance(type);
|
||||||
|
|
||||||
Set("LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
MethodInfo computeHash = type.GetMethod("ComputeHash");
|
||||||
Set("Id", userId);
|
if (computeHash == null)
|
||||||
Set("Name", userName);
|
|
||||||
Set("Email", email);
|
|
||||||
Set("Premium", true);
|
|
||||||
Set("MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
|
||||||
Set("Version", 1);
|
|
||||||
Set("Issued", DateTime.UtcNow);
|
|
||||||
Set("Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
|
||||||
Set("Expires", DateTime.UtcNow.AddYears(100));
|
|
||||||
Set("Trial", false);
|
|
||||||
Set("LicenseType", Enum.Parse(licenseTypeEnum, "User"));
|
|
||||||
Set("Hash", Convert.ToBase64String((Byte[])type.GetMethod("ComputeHash").Invoke(license, [])));
|
|
||||||
Set("Signature", Convert.ToBase64String((Byte[])type.GetMethod("Sign").Invoke(license, [cert])));
|
|
||||||
|
|
||||||
Console.WriteLine(JsonConvert.SerializeObject(license, Formatting.Indented));
|
|
||||||
return;
|
|
||||||
|
|
||||||
void Set(String name, Object value)
|
|
||||||
{
|
{
|
||||||
type.GetProperty(name)?.SetValue(license, value);
|
Console.WriteLine("Could not find ComputeHash!");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
MethodInfo sign = type.GetMethod("Sign");
|
||||||
|
if (sign == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find sign!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Set(type, license, "LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
||||||
|
Set(type, license, "Id", userId);
|
||||||
|
Set(type, license, "Name", userName);
|
||||||
|
Set(type, license, "Email", email);
|
||||||
|
Set(type, license, "Premium", true);
|
||||||
|
Set(type, license, "MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
||||||
|
Set(type, license, "Version", 1);
|
||||||
|
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, "Trial", false);
|
||||||
|
Set(type, license, "LicenseType", Enum.Parse(licenseTypeEnum, "User"));
|
||||||
|
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.Billing.Organizations.Models.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");
|
||||||
|
|
||||||
|
if (type == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find type!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (licenseTypeEnum == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find licenseTypeEnum!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (planTypeEnum == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find planTypeEnum!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
Object license = Activator.CreateInstance(type);
|
Object license = Activator.CreateInstance(type);
|
||||||
|
|
||||||
Set("LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
MethodInfo computeHash = type.GetMethod("ComputeHash");
|
||||||
Set("InstallationId", instalId);
|
if (computeHash == null)
|
||||||
Set("Id", Guid.NewGuid());
|
|
||||||
Set("Name", userName);
|
|
||||||
Set("BillingEmail", email);
|
|
||||||
Set("BusinessName", String.IsNullOrWhiteSpace(businessName) ? "BitBetter" : businessName);
|
|
||||||
Set("Enabled", true);
|
|
||||||
Set("Plan", "Enterprise (Annually)");
|
|
||||||
Set("PlanType", Enum.Parse(planTypeEnum, "EnterpriseAnnually"));
|
|
||||||
Set("Seats", Int32.MaxValue);
|
|
||||||
Set("MaxCollections", Int16.MaxValue);
|
|
||||||
Set("UsePolicies", true);
|
|
||||||
Set("UseSso", true);
|
|
||||||
Set("UseKeyConnector", true);
|
|
||||||
Set("UseScim", true);
|
|
||||||
Set("UseGroups", true);
|
|
||||||
Set("UseEvents", true);
|
|
||||||
Set("UseDirectory", true);
|
|
||||||
Set("UseTotp", true);
|
|
||||||
Set("Use2fa", true);
|
|
||||||
Set("UseApi", true);
|
|
||||||
Set("UseResetPassword", true);
|
|
||||||
Set("UseCustomPermissions", true);
|
|
||||||
Set("MaxStorageGb", storage == 0 ? Int16.MaxValue : storage);
|
|
||||||
Set("SelfHost", true);
|
|
||||||
Set("UsersGetPremium", true);
|
|
||||||
Set("UsePasswordManager", true);
|
|
||||||
Set("UseSecretsManager", true);
|
|
||||||
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("Issued", DateTime.UtcNow);
|
|
||||||
Set("Refresh", DateTime.UtcNow.AddYears(100).AddMonths(-1));
|
|
||||||
Set("Expires", DateTime.UtcNow.AddYears(100));
|
|
||||||
Set("Trial", false);
|
|
||||||
Set("LicenseType", Enum.Parse(licenseTypeEnum, "Organization"));
|
|
||||||
Set("LimitCollectionCreationDeletion", true); //This will be used in the new version of BitWarden but can be applied now
|
|
||||||
Set("AllowAdminAccessToAllCollectionItems", true);
|
|
||||||
Set("UseRiskInsights", true);
|
|
||||||
Set("UseOrganizationDomains", true);
|
|
||||||
Set("UseAdminSponsoredFamilies", true);
|
|
||||||
Set("Hash", Convert.ToBase64String((Byte[])type.GetMethod("ComputeHash").Invoke(license, [])));
|
|
||||||
Set("Signature", Convert.ToBase64String((Byte[])type.GetMethod("Sign").Invoke(license, [cert])));
|
|
||||||
|
|
||||||
Console.WriteLine(JsonConvert.SerializeObject(license, Formatting.Indented));
|
|
||||||
return;
|
|
||||||
|
|
||||||
void Set(String name, Object value)
|
|
||||||
{
|
{
|
||||||
type.GetProperty(name)?.SetValue(license, value);
|
Console.WriteLine("Could not find ComputeHash!");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MethodInfo sign = type.GetMethod("Sign");
|
||||||
|
if (sign == null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Could not find sign!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Set(type, license, "LicenseKey", String.IsNullOrWhiteSpace(key) ? Guid.NewGuid().ToString("n") : key);
|
||||||
|
Set(type, license, "InstallationId", instalId);
|
||||||
|
Set(type, license, "Id", Guid.NewGuid());
|
||||||
|
Set(type, license, "Name", userName);
|
||||||
|
Set(type, license, "BillingEmail", email);
|
||||||
|
Set(type, license, "BusinessName", String.IsNullOrWhiteSpace(businessName) ? "BitBetter" : businessName);
|
||||||
|
Set(type, license, "Enabled", true);
|
||||||
|
Set(type, license, "Plan", "Enterprise (Annually)");
|
||||||
|
Set(type, license, "PlanType", Enum.Parse(planTypeEnum, "EnterpriseAnnually"));
|
||||||
|
Set(type, license, "Seats", Int32.MaxValue);
|
||||||
|
Set(type, license, "MaxCollections", Int16.MaxValue);
|
||||||
|
Set(type, license, "UsePolicies", true);
|
||||||
|
Set(type, license, "UseSso", true);
|
||||||
|
Set(type, license, "UseKeyConnector", true);
|
||||||
|
Set(type, license, "UseScim", true);
|
||||||
|
Set(type, license, "UseGroups", true);
|
||||||
|
Set(type, license, "UseEvents", true);
|
||||||
|
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,14 +1,10 @@
|
||||||
<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>
|
||||||
|
|
Loading…
Reference in New Issue
Block a user