mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-15 17:27:43 -05:00
Release 2.1.1
This commit is contained in:
commit
44c58a66d1
1
.gitattributes
vendored
1
.gitattributes
vendored
@ -1,6 +1,7 @@
|
|||||||
src/version.h.cmake export-subst
|
src/version.h.cmake export-subst
|
||||||
.gitattributes export-ignore
|
.gitattributes export-ignore
|
||||||
.gitignore export-ignore
|
.gitignore export-ignore
|
||||||
|
.github export-ignore
|
||||||
.travis.yml export-ignore
|
.travis.yml export-ignore
|
||||||
.tx export-ignore
|
.tx export-ignore
|
||||||
snapcraft.yaml export-ignore
|
snapcraft.yaml export-ignore
|
||||||
|
89
.github/CONTRIBUTING.md
vendored
89
.github/CONTRIBUTING.md
vendored
@ -1,31 +1,32 @@
|
|||||||
# Contributing to KeePassX Reboot
|
# Contributing to KeePassXC
|
||||||
|
|
||||||
:+1::tada: First off, thanks for taking the time to contribute! :tada::+1:
|
:+1::tada: First off, thanks for taking the time to contribute! :tada::+1:
|
||||||
|
|
||||||
The following is a set of guidelines for contributing to KeePassX Reboot on GitHub.
|
The following is a set of guidelines for contributing to KeePassXC on GitHub.
|
||||||
These are just guidelines, not rules. Use your best judgment, and feel free to propose changes to this document in a pull request.
|
These are just guidelines, not rules. Use your best judgment, and feel free to propose changes to this document in a pull request.
|
||||||
|
|
||||||
#### Table Of Contents
|
#### Table of contents
|
||||||
|
|
||||||
[What should I know before I get started?](#what-should-i-know-before-i-get-started)
|
[What should I know before I get started?](#what-should-i-know-before-i-get-started)
|
||||||
* [Open Source Contribution Policy](#open-source-contribution-policy)
|
* [Open Source Contribution Policy](#open-source-contribution-policy)
|
||||||
|
|
||||||
[How Can I Contribute?](#how-can-i-contribute)
|
[How can I contribute?](#how-can-i-contribute)
|
||||||
* [Feature Requests](#feature-requests)
|
* [Feature requests](#feature-requests)
|
||||||
* [Bug Reports](#bug-reports)
|
* [Bug reports](#bug-reports)
|
||||||
* [Your First Code Contribution](#your-first-code-contribution)
|
* [Discuss with the team](#discuss-with-the-team)
|
||||||
* [Pull Requests](#pull-requests)
|
* [Your first code contribution](#your-first-code-contribution)
|
||||||
|
* [Pull requests](#pull-requests)
|
||||||
* [Translations](#translations)
|
* [Translations](#translations)
|
||||||
|
|
||||||
[Styleguides](#styleguides)
|
[Styleguides](#styleguides)
|
||||||
* [Git Branch Strategy](#git_branch_strategy)
|
* [Git branch strategy](#git-branch-strategy)
|
||||||
* [Git Commit Messages](#git-commit-messages)
|
* [Git commit messages](#git-commit-messages)
|
||||||
* [Coding Styleguide](#coding-styleguide)
|
* [Coding styleguide](#coding-styleguide)
|
||||||
|
|
||||||
|
|
||||||
## What should I know before I get started?
|
## What should I know before I get started?
|
||||||
### Open Source Contribution Policy
|
### Open Source Contribution Policy
|
||||||
[Version 0.3, 2015–11–18](https://medium.com/@jmaynard/a-contribution-policy-for-open-source-that-works-bfc4600c9d83#.i9ntbhmad)
|
**Source**: [Version 0.3, 2015–11–18](https://medium.com/@jmaynard/a-contribution-policy-for-open-source-that-works-bfc4600c9d83#.i9ntbhmad)
|
||||||
|
|
||||||
#### Policy
|
#### Policy
|
||||||
|
|
||||||
@ -49,35 +50,35 @@ If we reject your contribution, it means only that we do not consider it suitabl
|
|||||||
* 0.3, 2011–11–19: Added “irrevocably” to “we can use” and changed “it” to “your contribution” in the “if rejected” section. Thanks to Patrick Maupin.
|
* 0.3, 2011–11–19: Added “irrevocably” to “we can use” and changed “it” to “your contribution” in the “if rejected” section. Thanks to Patrick Maupin.
|
||||||
|
|
||||||
|
|
||||||
## How Can I Contribute?
|
## How can I contribute?
|
||||||
### Feature Requests
|
### Feature requests
|
||||||
|
|
||||||
We're always looking for suggestions to improve our application. If you have a suggestion for improving an existing feature, or would like to suggest a completely new feature for KeePassX Reboot, please use the Issues section or our [Google Groups](https://groups.google.com/forum/#!forum/keepassx-reboot) forum.
|
We're always looking for suggestions to improve our application. If you have a suggestion to improve an existing feature, or would like to suggest a completely new feature for KeePassXC, please use the [issue tracker on GitHub][issues-section]. For more general discussion, try using our [Google Groups][google-groups] forum.
|
||||||
|
|
||||||
### Bug Reports
|
### Bug reports
|
||||||
|
|
||||||
Our software isn't always perfect, but we strive to always improve our work. You may file bug reports in the Issues section.
|
Our software isn't always perfect, but we strive to always improve our work. You may file bug reports in the issue tracker.
|
||||||
|
|
||||||
Before submitting a Bug Report, check if the problem has already been reported. Please refrain from opening a duplicate issue. If you want to highlight a deficiency on an existing issue, simply add a comment.
|
Before submitting a bug report, check if the problem has already been reported. Please refrain from opening a duplicate issue. If you want to add further information to an existing issue, simply add a comment on that issue.
|
||||||
|
|
||||||
### Discuss with the Team
|
### Discuss with the team
|
||||||
|
|
||||||
You can talk to the KeePassX Reboot Team about Bugs, new feature, Issue and PullRequests at our [Google Groups](https://groups.google.com/forum/#!forum/keepassx-reboot) forum
|
As with feature requests, you can talk to the KeePassXC team about bugs, new features, other issues and pull requests on the dedicated issue tracker, using the [Google Groups][google-groups] forum, or in the IRC channel on Freenode (`#keepassxc-dev` on `irc.freenode.net`, or use a [webchat link](https://webchat.freenode.net/?channels=%23keepassxc-dev)).
|
||||||
|
|
||||||
### Your First Code Contribution
|
### Your first code contribution
|
||||||
|
|
||||||
Unsure where to begin contributing to KeePassX Reboot? You can start by looking through these `beginner` and `help-wanted` issues:
|
Unsure where to begin contributing to KeePassXC? You can start by looking through these `beginner` and `help-wanted` issues:
|
||||||
|
|
||||||
* [Beginner issues][beginner] - issues which should only require a few lines of code, and a test or two.
|
* [Beginner issues][beginner] – issues which should only require a few lines of code, and a test or two.
|
||||||
* [Help wanted issues][help-wanted] - issues which should be a bit more involved than `beginner` issues.
|
* ['Help wanted' issues][help-wanted] – issues which should be a bit more involved than `beginner` issues.
|
||||||
|
|
||||||
Both issue lists are sorted by total number of comments. While not perfect, number of comments is a reasonable proxy for impact a given change will have.
|
Both issue lists are sorted by total number of comments. While not perfect, looking at the number of comments on an issue can give a general idea of how much an impact a given change will have.
|
||||||
|
|
||||||
### Pull Requests
|
### Pull requests
|
||||||
|
|
||||||
Along with our desire to hear your feedback and suggestions, we're also interested in accepting direct assistance in the form of code.
|
Along with our desire to hear your feedback and suggestions, we're also interested in accepting direct assistance in the form of code.
|
||||||
|
|
||||||
All pull requests must comply with the above requirements and with the [Styleguides](#styleguides).
|
All pull requests must comply with the above requirements and with the [styleguides](#styleguides).
|
||||||
|
|
||||||
### Translations
|
### Translations
|
||||||
|
|
||||||
@ -86,19 +87,20 @@ Please join an existing language team or request a new one if there is none.
|
|||||||
|
|
||||||
## Styleguides
|
## Styleguides
|
||||||
|
|
||||||
### Git Branch Strategy
|
### Git branch strategy
|
||||||
|
|
||||||
The Branch Strategy is based on [git-flow-lite](http://nvie.com/posts/a-successful-git-branching-model/).
|
The Branch Strategy is based on [git-flow-lite](http://nvie.com/posts/a-successful-git-branching-model/).
|
||||||
|
|
||||||
* **master** -> always points to the last release published
|
* **master** – points to the latest public release
|
||||||
* **develop** -> points to the next planned release, tested and reviewed code
|
* **develop** – points to the development of the next release, contains tested and reviewed code
|
||||||
* **feature/**[name] -> points to brand new feature in codebase, candidate for merge into develop (subject to rebase)
|
* **feature/**[name] – points to a branch with a new feature, one which is candidate for merge into develop (subject to rebase)
|
||||||
|
* **hotfix/**[id]-[description] – points to a branch with a fix for a particular issue ID
|
||||||
|
|
||||||
|
|
||||||
### Git Commit Messages
|
### Git commit messages
|
||||||
|
|
||||||
* Use the present tense ("Add feature" not "Added feature")
|
* Use the present tense ("Add feature" not "Added feature")
|
||||||
* Use the imperative mood ("Move cursor to..." not "Moves cursor to...")
|
* Use the imperative mood ("Move cursor to…" not "Moves cursor to…")
|
||||||
* Limit the first line to 72 characters or less
|
* Limit the first line to 72 characters or less
|
||||||
* Reference issues and pull requests liberally
|
* Reference issues and pull requests liberally
|
||||||
* When only changing documentation, include `[ci skip]` in the commit description
|
* When only changing documentation, include `[ci skip]` in the commit description
|
||||||
@ -114,21 +116,21 @@ The Branch Strategy is based on [git-flow-lite](http://nvie.com/posts/a-successf
|
|||||||
* :lock: `:lock:` when dealing with security
|
* :lock: `:lock:` when dealing with security
|
||||||
|
|
||||||
|
|
||||||
### Coding Styleguide
|
### Coding styleguide
|
||||||
|
|
||||||
This project follows the [Qt Coding Style](https://wiki.qt.io/Qt_Coding_Style). All submissions are expected to follow this style.
|
This project follows the [Qt Coding Style](https://wiki.qt.io/Qt_Coding_Style). All submissions are expected to follow this style.
|
||||||
|
|
||||||
In particular Code must follow the following specific rules:
|
In particular, code must stick to the following rules:
|
||||||
|
|
||||||
#### Naming Convention
|
#### Naming convention
|
||||||
`lowerCamelCase`
|
`lowerCamelCase`
|
||||||
|
|
||||||
For names made of only one word, the fist letter is lowercase.
|
For names made of only one word, the first letter should be lowercase.
|
||||||
For names made of multiple concatenated words, the first letter is lowercase and each subsequent concatenated word is capitalized.
|
For names made of multiple concatenated words, the first letter of the whole is lowercase, and the first letter of each subsequent word is capitalized.
|
||||||
|
|
||||||
#### Indention
|
#### Indention
|
||||||
For C++ files (.cpp .h): 4 spaces
|
For **C++ files** (*.cpp .h*): 4 spaces
|
||||||
For Qt-UI files (.ui): 2 spaces
|
For **Qt-UI files** (*.ui*): 2 spaces
|
||||||
|
|
||||||
#### Pointers
|
#### Pointers
|
||||||
```c
|
```c
|
||||||
@ -165,9 +167,8 @@ Use prefix: `m_*`
|
|||||||
|
|
||||||
Example: `m_variable`
|
Example: `m_variable`
|
||||||
|
|
||||||
#### GUI Widget names
|
#### GUI widget names
|
||||||
Widget names must be related to the desired program behaviour.
|
Widget names must be related to the desired program behavior, and preferably end with the widget's classname.
|
||||||
Preferably end the name with the Widget Classname
|
|
||||||
|
|
||||||
Example: `<widget class="QCheckBox" name="rememberCheckBox">`
|
Example: `<widget class="QCheckBox" name="rememberCheckBox">`
|
||||||
|
|
||||||
@ -175,3 +176,5 @@ Example: `<widget class="QCheckBox" name="rememberCheckBox">`
|
|||||||
|
|
||||||
[beginner]:https://github.com/keepassxreboot/keepassx/issues?q=is%3Aopen+is%3Aissue+label%3Abeginner+label%3A%22help+wanted%22+sort%3Acomments-desc
|
[beginner]:https://github.com/keepassxreboot/keepassx/issues?q=is%3Aopen+is%3Aissue+label%3Abeginner+label%3A%22help+wanted%22+sort%3Acomments-desc
|
||||||
[help-wanted]:https://github.com/keepassxreboot/keepassx/issues?q=is%3Aopen+is%3Aissue+label%3A%22help+wanted%22+sort%3Acomments-desc
|
[help-wanted]:https://github.com/keepassxreboot/keepassx/issues?q=is%3Aopen+is%3Aissue+label%3A%22help+wanted%22+sort%3Acomments-desc
|
||||||
|
[issues-section]:https://github.com/keepassxreboot/keepassxc/issues
|
||||||
|
[google-groups]:https://groups.google.com/forum/#!forum/keepassx-reboot
|
||||||
|
@ -39,26 +39,54 @@ mkdir -p $APP.AppDir
|
|||||||
wget -q https://github.com/probonopd/AppImages/raw/master/functions.sh -O ./functions.sh
|
wget -q https://github.com/probonopd/AppImages/raw/master/functions.sh -O ./functions.sh
|
||||||
. ./functions.sh
|
. ./functions.sh
|
||||||
|
|
||||||
|
LIB_DIR=./usr/lib
|
||||||
|
if [ -d ./usr/lib/x86_64-linux-gnu ]; then
|
||||||
|
LIB_DIR=./usr/lib/x86_64-linux-gnu
|
||||||
|
fi
|
||||||
|
|
||||||
cd $APP.AppDir
|
cd $APP.AppDir
|
||||||
cp -a ../../bin-release/* .
|
cp -a ../../bin-release/* .
|
||||||
mv ./usr/local/* ./usr
|
cp -a ./usr/local/* ./usr
|
||||||
rmdir ./usr/local
|
rm -R ./usr/local
|
||||||
patch_strings_in_file /usr/local ./
|
rmdir ./opt 2> /dev/null
|
||||||
|
patch_strings_in_file /usr/local ././
|
||||||
patch_strings_in_file /usr ./
|
patch_strings_in_file /usr ./
|
||||||
|
|
||||||
|
# bundle Qt platform plugins and themes
|
||||||
|
QXCB_PLUGIN="$(find /usr/lib -name 'libqxcb.so' 2> /dev/null)"
|
||||||
|
if [ "$QXCB_PLUGIN" == "" ]; then
|
||||||
|
QXCB_PLUGIN="$(find /opt/qt*/plugins -name 'libqxcb.so' 2> /dev/null)"
|
||||||
|
fi
|
||||||
|
QT_PLUGIN_PATH="$(dirname $(dirname $QXCB_PLUGIN))"
|
||||||
|
mkdir -p ".${QT_PLUGIN_PATH}/platforms"
|
||||||
|
cp "$QXCB_PLUGIN" ".${QT_PLUGIN_PATH}/platforms/"
|
||||||
|
|
||||||
get_apprun
|
get_apprun
|
||||||
copy_deps
|
copy_deps
|
||||||
delete_blacklisted
|
delete_blacklisted
|
||||||
|
|
||||||
|
# remove dbus and systemd libs as they are not blacklisted
|
||||||
|
find . -name libdbus-1.so.3 -exec rm {} \;
|
||||||
|
find . -name libsystemd.so.0 -exec rm {} \;
|
||||||
|
|
||||||
get_desktop
|
get_desktop
|
||||||
get_icon
|
get_icon
|
||||||
|
cat << EOF > ./usr/bin/keepassxc_env
|
||||||
|
#!/usr/bin/env bash
|
||||||
|
#export QT_QPA_PLATFORMTHEME=gtk2
|
||||||
|
export LD_LIBRARY_PATH="../opt/qt58/lib:\${LD_LIBRARY_PATH}"
|
||||||
|
export QT_PLUGIN_PATH="..${QT_PLUGIN_PATH}"
|
||||||
|
exec keepassxc "\$@"
|
||||||
|
EOF
|
||||||
|
chmod +x ./usr/bin/keepassxc_env
|
||||||
|
sed -i 's/Exec=keepassxc/Exec=keepassxc_env/' keepassxc.desktop
|
||||||
get_desktopintegration $LOWERAPP
|
get_desktopintegration $LOWERAPP
|
||||||
|
|
||||||
GLIBC_NEEDED=$(glibc_needed)
|
GLIBC_NEEDED=$(glibc_needed)
|
||||||
|
|
||||||
cd ..
|
cd ..
|
||||||
|
|
||||||
generate_appimage
|
generate_type2_appimage
|
||||||
|
|
||||||
mv ../out/*.AppImage ..
|
mv ../out/*.AppImage ..
|
||||||
rmdir ../out > /dev/null 2>&1
|
rmdir ../out > /dev/null 2>&1
|
||||||
|
10
CHANGELOG
10
CHANGELOG
@ -1,3 +1,13 @@
|
|||||||
|
2.1.1 (2017-02-06)
|
||||||
|
=========================
|
||||||
|
|
||||||
|
- Enabled HTTP plugin build; plugin is disabled by default and limited to localhost [#147]
|
||||||
|
- Escape HTML in dialog boxes [#247]
|
||||||
|
- Corrected crashes in favicon download and password generator [#233, #226]
|
||||||
|
- Increase font size of password meter [#228]
|
||||||
|
- Fixed compatibility with Qt 5.8 [#211]
|
||||||
|
- Use consistent button heights in password generator [#229]
|
||||||
|
|
||||||
2.1.0 (2017-01-22)
|
2.1.0 (2017-01-22)
|
||||||
=========================
|
=========================
|
||||||
|
|
||||||
|
@ -38,8 +38,8 @@ option(WITH_XC_AUTOTYPE "Include Autotype." OFF)
|
|||||||
option(WITH_XC_HTTP "Include KeePassHTTP." OFF)
|
option(WITH_XC_HTTP "Include KeePassHTTP." OFF)
|
||||||
option(WITH_XC_YUBIKEY "Include Yubikey support." OFF)
|
option(WITH_XC_YUBIKEY "Include Yubikey support." OFF)
|
||||||
|
|
||||||
set(KEEPASSXC_VERSION "2.1.0")
|
set(KEEPASSXC_VERSION "2.1.1")
|
||||||
set(KEEPASSXC_VERSION_NUM "2.1.0")
|
set(KEEPASSXC_VERSION_NUM "2.1.1")
|
||||||
|
|
||||||
if("${CMAKE_C_COMPILER}" MATCHES "clang$" OR "${CMAKE_C_COMPILER_ID}" STREQUAL "Clang")
|
if("${CMAKE_C_COMPILER}" MATCHES "clang$" OR "${CMAKE_C_COMPILER_ID}" STREQUAL "Clang")
|
||||||
set(CMAKE_COMPILER_IS_CLANG 1)
|
set(CMAKE_COMPILER_IS_CLANG 1)
|
||||||
@ -172,9 +172,13 @@ set(CMAKE_AUTOMOC ON)
|
|||||||
# Make sure we don't enable asserts there.
|
# Make sure we don't enable asserts there.
|
||||||
set_property(DIRECTORY APPEND PROPERTY COMPILE_DEFINITIONS_NONE QT_NO_DEBUG)
|
set_property(DIRECTORY APPEND PROPERTY COMPILE_DEFINITIONS_NONE QT_NO_DEBUG)
|
||||||
|
|
||||||
|
find_package(LibGPGError REQUIRED)
|
||||||
|
|
||||||
find_package(Gcrypt 1.6.0 REQUIRED)
|
find_package(Gcrypt 1.6.0 REQUIRED)
|
||||||
|
|
||||||
find_package(LibMicroHTTPD REQUIRED)
|
if (WITH_XC_HTTP)
|
||||||
|
find_package(LibMicroHTTPD REQUIRED)
|
||||||
|
endif(WITH_XC_HTTP)
|
||||||
|
|
||||||
find_package(ZLIB REQUIRED)
|
find_package(ZLIB REQUIRED)
|
||||||
|
|
||||||
|
32
Dockerfile
32
Dockerfile
@ -14,21 +14,41 @@
|
|||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
FROM ubuntu:16.04
|
FROM ubuntu:14.04
|
||||||
|
|
||||||
RUN set -x && apt-get update
|
|
||||||
RUN set -x \
|
RUN set -x \
|
||||||
|
&& apt-get update \
|
||||||
|
&& apt-get install --yes software-properties-common
|
||||||
|
|
||||||
|
RUN set -x \
|
||||||
|
&& add-apt-repository --yes ppa:beineri/opt-qt58-trusty
|
||||||
|
|
||||||
|
RUN set -x \
|
||||||
|
&& apt-get update \
|
||||||
&& apt-get install --yes \
|
&& apt-get install --yes \
|
||||||
|
g++ \
|
||||||
cmake \
|
cmake \
|
||||||
libgcrypt20-dev \
|
libgcrypt20-dev \
|
||||||
qtbase5-dev \
|
qt58base \
|
||||||
qttools5-dev-tools \
|
qt58tools \
|
||||||
|
qt58x11extras \
|
||||||
libmicrohttpd-dev \
|
libmicrohttpd-dev \
|
||||||
libqt5x11extras5-dev \
|
|
||||||
libxi-dev \
|
libxi-dev \
|
||||||
libxtst-dev \
|
libxtst-dev \
|
||||||
zlib1g-dev
|
zlib1g-dev \
|
||||||
|
wget \
|
||||||
|
file \
|
||||||
|
fuse \
|
||||||
|
python
|
||||||
|
|
||||||
|
RUN set -x \
|
||||||
|
&& apt-get install --yes mesa-common-dev
|
||||||
|
|
||||||
VOLUME /keepassxc/src
|
VOLUME /keepassxc/src
|
||||||
VOLUME /keepassxc/out
|
VOLUME /keepassxc/out
|
||||||
WORKDIR /keepassxc
|
WORKDIR /keepassxc
|
||||||
|
|
||||||
|
ENV CMAKE_PREFIX_PATH=/opt/qt58/lib/cmake
|
||||||
|
ENV LD_LIBRARY_PATH=/opt/qt58/lib
|
||||||
|
RUN set -x \
|
||||||
|
&& echo /opt/qt58/lib > /etc/ld.so.conf.d/qt58.conf
|
||||||
|
64
README.md
64
README.md
@ -3,57 +3,63 @@
|
|||||||
[![Travis Build Status](https://travis-ci.org/keepassxreboot/keepassxc.svg?branch=develop)](https://travis-ci.org/keepassxreboot/keepassxc) [![Coverage Status](https://coveralls.io/repos/github/keepassxreboot/keepassxc/badge.svg)](https://coveralls.io/github/keepassxreboot/keepassxc)
|
[![Travis Build Status](https://travis-ci.org/keepassxreboot/keepassxc.svg?branch=develop)](https://travis-ci.org/keepassxreboot/keepassxc) [![Coverage Status](https://coveralls.io/repos/github/keepassxreboot/keepassxc/badge.svg)](https://coveralls.io/github/keepassxreboot/keepassxc)
|
||||||
|
|
||||||
## About
|
## About
|
||||||
|
KeePassXC is a fork of [KeePassX](https://www.keepassx.org/) that [aims to incorporate stalled pull requests, features, and bug fixes that have never made it into the main KeePassX repository](https://github.com/keepassxreboot/keepassx/issues/43).
|
||||||
Fork of [KeePassX](https://www.keepassx.org/) that [aims to incorporate stalled Pull Requests, features, and bug fixes that are not being incorporated into the main KeePassX baseline](https://github.com/keepassxreboot/keepassx/issues/43).
|
|
||||||
|
|
||||||
|
|
||||||
#### Additional Reboot Features
|
## Additional features compared to KeePassX
|
||||||
- keepasshttp support for use with [PassIFox](https://addons.mozilla.org/en-us/firefox/addon/passifox/) for Mozilla Firefox and [chromeIPass](https://chrome.google.com/webstore/detail/chromeipass/ompiailgknfdndiefoaoiligalphfdae) for Google Chrome.
|
- Autotype on all three major platforms (Linux, Windows, OS X)
|
||||||
|
- Stand-alone password generator
|
||||||
|
- Password strength meter
|
||||||
|
- Use website's favicons as entry icons
|
||||||
|
- Merging of databases
|
||||||
|
- Automatic reload when the database changed on disk
|
||||||
|
- KeePassHTTP support for use with [PassIFox](https://addons.mozilla.org/en-us/firefox/addon/passifox/) in Mozilla Firefox and [chromeIPass](https://chrome.google.com/webstore/detail/chromeipass/ompiailgknfdndiefoaoiligalphfdae) in Google Chrome or Chromium.
|
||||||
|
|
||||||
KeePassHttp implementation has been forked from jdachtera's repository, which in turn was based on code from code with Francois Ferrand's [keepassx-http](https://gitorious.org/keepassx/keepassx-http/source/master) repository.
|
For a full list of features and changes, read the [CHANGELOG](CHANGELOG) document.
|
||||||
|
|
||||||
This is a rebuild from [denk-mal's keepasshttp](https://github.com/denk-mal/keepassx.git) that brings it forward to Qt5 and KeePassX v2.x.
|
|
||||||
|
|
||||||
|
### Note about KeePassHTTP
|
||||||
|
KeePassHTTP is not a highly secure protocol and has certain flaw which allow an attacker to decrypt your passwords when they manage to intercept communication between a KeePassHTTP server and PassIFox/chromeIPass over a network connection (see [here](https://github.com/pfn/keepasshttp/issues/258) and [here](https://github.com/keepassxreboot/keepassxc/issues/147)). KeePassXC therefore strictly limits communication between itself and the browser plugin to your local computer. As long as your computer is not compromised, your passwords are fairly safe that way, but still use it at your own risk!
|
||||||
|
|
||||||
### Installation
|
### Installation
|
||||||
|
Pre-compiled binaries can be found on the [downloads page](https://keepassxc.org/download). Additionally, individual Linux distributions may ship their own versions, so please check out your distribution's package list to see if KeePassXC is available.
|
||||||
|
|
||||||
Right now KeePassXC does not have a precompiled executable or an installation package.<br/>
|
### Building KeePassXC yourself
|
||||||
So you must install it from its source code.
|
|
||||||
|
|
||||||
**More detailed instructions are available in the INSTALL file or at the [Wiki page](https://github.com/keepassxreboot/keepassx/wiki/Install-Instruction-from-Source).**
|
*More detailed instructions are available in the INSTALL file or on the [Wiki page](https://github.com/keepassxreboot/keepassx/wiki/Install-Instruction-from-Source).*
|
||||||
|
|
||||||
First you must download the KeePassXC source code as ZIP file or with Git.
|
First, you must download the KeePassXC [source tarball](https://keepassxc.org/download#source) or check out the latest version from our [Git repository](https://github.com/keepassxreboot/keepassxc).
|
||||||
|
|
||||||
Generally you can build and install KeePassXC with the following commands from a Terminal in the KeePassXC folder
|
To clone the project from Git, `cd` to a suitable location and run
|
||||||
```
|
|
||||||
mkdir build
|
|
||||||
cd build
|
|
||||||
cmake -DWITH_TESTS=OFF ..
|
|
||||||
make
|
|
||||||
sudo make install
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
### Clone Repository
|
|
||||||
|
|
||||||
Clone the repository to a suitable location where you can extend and build this project.
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://github.com/keepassxreboot/keepassxc.git
|
git clone https://github.com/keepassxreboot/keepassxc.git
|
||||||
```
|
```
|
||||||
|
|
||||||
**Note:** This will clone the entire contents of the repository at the HEAD revision.
|
This will clone the entire contents of the repository and check out the current `develop` branch.
|
||||||
|
|
||||||
To update the project from within the project's folder you can run the following command:
|
To update the project from within the project's folder, you can run the following command:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git pull
|
git pull
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Once you have downloaded the source code, you can `cd` into the source code directory and build and install KeePassXC with
|
||||||
|
|
||||||
|
```
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake -DWITH_TESTS=OFF ..
|
||||||
|
make -j8
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
To enable autotype, add `-DWITH_XC_AUTOTYPE=ON` to the `cmake` command. KeePassHTTP support is compiled in by adding `-DWITH_XC_HTTP=ON`. If these options are not specified, KeePassXC will be built without these plugins.
|
||||||
|
|
||||||
|
|
||||||
### Contributing
|
### Contributing
|
||||||
|
|
||||||
We're always looking for suggestions to improve our application. If you have a suggestion for improving an existing feature,
|
We are always looking for suggestions how to improve our application. If you find any bugs or have an idea for a new feature, please let us know by opening a report in our [issue tracker](https://github.com/keepassxreboot/keepassxc/issues) on GitHub or write to our [Google Groups](https://groups.google.com/forum/#!forum/keepassx-reboot) forum.
|
||||||
or would like to suggest a completely new feature for KeePassX Reboot, please use the [Issues](https://github.com/keepassxreboot/keepassxc/issues) section or our [Google Groups](https://groups.google.com/forum/#!forum/keepassx-reboot) forum.
|
|
||||||
|
|
||||||
Please review the [CONTRIBUTING](.github/CONTRIBUTING.md) document for further information.
|
You can of course also directly contribute your own code. We are happy to accept your pull requests.
|
||||||
|
|
||||||
|
Please read the [CONTRIBUTING](.github/CONTRIBUTING.md) document for further information.
|
||||||
|
9
cmake/FindLibGPGError.cmake
Normal file
9
cmake/FindLibGPGError.cmake
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
|
||||||
|
find_path(GPGERROR_INCLUDE_DIR gpg-error.h)
|
||||||
|
|
||||||
|
find_library(GPGERROR_LIBRARIES gpg-error)
|
||||||
|
|
||||||
|
mark_as_advanced(GPGERROR_LIBRARIES GPGERROR_INCLUDE_DIR)
|
||||||
|
|
||||||
|
include(FindPackageHandleStandardArgs)
|
||||||
|
find_package_handle_standard_args(LibGPGError DEFAULT_MSG GPGERROR_LIBRARIES GPGERROR_INCLUDE_DIR)
|
350
make_release.sh
350
make_release.sh
@ -1,350 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
#
|
|
||||||
# KeePassXC Release Preparation Helper
|
|
||||||
# Copyright (C) 2017 KeePassXC team <https://keepassxc.org/>
|
|
||||||
#
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, either version 2 or (at your option)
|
|
||||||
# version 3 of the License.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
echo -e "\e[1m\e[32mKeePassXC\e[0m Release Preparation Helper"
|
|
||||||
echo -e "Copyright (C) 2017 KeePassXC Team <https://keepassxc.org/>\n"
|
|
||||||
|
|
||||||
|
|
||||||
# default values
|
|
||||||
RELEASE_NAME=""
|
|
||||||
APP_NAME="KeePassXC"
|
|
||||||
APP_NAME_LOWER="keepassxc"
|
|
||||||
SRC_DIR="."
|
|
||||||
GPG_KEY="CFB4C2166397D0D2"
|
|
||||||
GPG_GIT_KEY=""
|
|
||||||
OUTPUT_DIR="release"
|
|
||||||
BRANCH=""
|
|
||||||
RELEASE_BRANCH="master"
|
|
||||||
TAG_NAME=""
|
|
||||||
BUILD_SOURCES=false
|
|
||||||
DOCKER_IMAGE=""
|
|
||||||
DOCKER_CONTAINER_NAME="${APP_NAME_LOWER}-build-container"
|
|
||||||
CMAKE_OPTIONS=""
|
|
||||||
COMPILER="g++"
|
|
||||||
MAKE_OPTIONS="-j8"
|
|
||||||
BUILD_PLUGINS="autotype"
|
|
||||||
INSTALL_PREFIX="/usr/local"
|
|
||||||
|
|
||||||
ORIG_BRANCH="$(git rev-parse --abbrev-ref HEAD 2> /dev/null)"
|
|
||||||
ORIG_CWD="$(pwd)"
|
|
||||||
|
|
||||||
|
|
||||||
# helper functions
|
|
||||||
printUsage() {
|
|
||||||
echo -e "\e[1mUsage:\e[0m $(basename $0) [options]"
|
|
||||||
cat << EOF
|
|
||||||
|
|
||||||
Options:
|
|
||||||
-v, --version Release version number or name (required)
|
|
||||||
-a, --app-name Application name (default: '${APP_NAME}')
|
|
||||||
-s, --source-dir Source directory (default: '${SRC_DIR}')
|
|
||||||
-k, --gpg-key GPG key used to sign the release tarball
|
|
||||||
(default: '${GPG_KEY}')
|
|
||||||
-g, --gpg-git-key GPG key used to sign the merge commit and release tag,
|
|
||||||
leave empty to let Git choose your default key
|
|
||||||
(default: '${GPG_GIT_KEY}')
|
|
||||||
-o, --output-dir Output directory where to build the release
|
|
||||||
(default: '${OUTPUT_DIR}')
|
|
||||||
--develop-branch Development branch to merge from (default: 'release/VERSION')
|
|
||||||
--release-branch Target release branch to merge to (default: '${RELEASE_BRANCH}')
|
|
||||||
-t, --tag-name Override release tag name (defaults to version number)
|
|
||||||
-b, --build Build sources after exporting release
|
|
||||||
-d, --docker-image Use the specified Docker image to compile the application.
|
|
||||||
The image must have all required build dependencies installed.
|
|
||||||
This option has no effect if --build is not set.
|
|
||||||
--container-name Docker container name (default: '${DOCKER_CONTAINER_NAME}')
|
|
||||||
The container must not exist already
|
|
||||||
-c, --cmake-options Additional CMake options for compiling the sources
|
|
||||||
--compiler Compiler to use (default: '${COMPILER}')
|
|
||||||
-m, --make-options Make options for compiling sources (default: '${MAKE_OPTIONS}')
|
|
||||||
-i, --install-prefix Install prefix (default: '${INSTALL_PREFIX}')
|
|
||||||
-p, --plugins Space-separated list of plugins to build
|
|
||||||
(default: ${BUILD_PLUGINS})
|
|
||||||
-h, --help Show this help
|
|
||||||
|
|
||||||
EOF
|
|
||||||
}
|
|
||||||
|
|
||||||
logInfo() {
|
|
||||||
echo -e "\e[1m[ \e[34mINFO\e[39m ]\e[0m $1"
|
|
||||||
}
|
|
||||||
|
|
||||||
logError() {
|
|
||||||
echo -e "\e[1m[ \e[31mERROR\e[39m ]\e[0m $1" >&2
|
|
||||||
}
|
|
||||||
|
|
||||||
exitError() {
|
|
||||||
logError "$1"
|
|
||||||
if [ "" != "$ORIG_BRANCH" ]; then
|
|
||||||
git checkout "$ORIG_BRANCH" > /dev/null 2>&1
|
|
||||||
fi
|
|
||||||
cd "$ORIG_CWD"
|
|
||||||
exit 1
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
# parse command line options
|
|
||||||
while [ $# -ge 1 ]; do
|
|
||||||
arg="$1"
|
|
||||||
|
|
||||||
case "$arg" in
|
|
||||||
-a|--app-name)
|
|
||||||
APP_NAME="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-s|--source-dir)
|
|
||||||
SRC_DIR"$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-v|--version)
|
|
||||||
RELEASE_NAME="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-k|--gpg-key)
|
|
||||||
GPG_KEY="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-g|--gpg-git-key)
|
|
||||||
GPG_GIT_KEY="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-o|--output-dir)
|
|
||||||
OUTPUT_DIR="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
--develop-branch)
|
|
||||||
BRANCH="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
--release-branch)
|
|
||||||
RELEASE_BRANCH="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-t|--tag-name)
|
|
||||||
TAG_NAME="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-b|--build)
|
|
||||||
BUILD_SOURCES=true ;;
|
|
||||||
|
|
||||||
-d|--docker-image)
|
|
||||||
DOCKER_IMAGE="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
--container-name)
|
|
||||||
DOCKER_CONTAINER_NAME="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-c|--cmake-options)
|
|
||||||
CMAKE_OPTIONS="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-m|--make-options)
|
|
||||||
MAKE_OPTIONS="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
--compiler)
|
|
||||||
COMPILER="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-p|--plugins)
|
|
||||||
BUILD_PLUGINS="$2"
|
|
||||||
shift ;;
|
|
||||||
|
|
||||||
-h|--help)
|
|
||||||
printUsage
|
|
||||||
exit ;;
|
|
||||||
|
|
||||||
*)
|
|
||||||
logError "Unknown option '$arg'\n"
|
|
||||||
printUsage
|
|
||||||
exit 1 ;;
|
|
||||||
esac
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
|
|
||||||
|
|
||||||
if [ "" == "$RELEASE_NAME" ]; then
|
|
||||||
logError "Missing arguments, --version is required!\n"
|
|
||||||
printUsage
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ "" == "$TAG_NAME" ]; then
|
|
||||||
TAG_NAME="$RELEASE_NAME"
|
|
||||||
fi
|
|
||||||
if [ "" == "$BRANCH" ]; then
|
|
||||||
BRANCH="release/${RELEASE_NAME}"
|
|
||||||
fi
|
|
||||||
APP_NAME_LOWER="$(echo "$APP_NAME" | tr '[:upper:]' '[:lower:]')"
|
|
||||||
APP_NAME_UPPER="$(echo "$APP_NAME" | tr '[:lower:]' '[:upper:]')"
|
|
||||||
|
|
||||||
SRC_DIR="$(realpath "$SRC_DIR")"
|
|
||||||
OUTPUT_DIR="$(realpath "$OUTPUT_DIR")"
|
|
||||||
if [ ! -d "$SRC_DIR" ]; then
|
|
||||||
exitError "Source directory '${SRC_DIR}' does not exist!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "Changing to source directory..."
|
|
||||||
cd "${SRC_DIR}"
|
|
||||||
|
|
||||||
logInfo "Performing basic checks..."
|
|
||||||
|
|
||||||
if [ -e "$OUTPUT_DIR" ]; then
|
|
||||||
exitError "Output directory '$OUTPUT_DIR' already exists. Please choose a different location!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -d .git ] || [ ! -f CHANGELOG ]; then
|
|
||||||
exitError "Source directory is not a valid Git repository!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
git tag | grep -q "$RELEASE_NAME"
|
|
||||||
if [ $? -eq 0 ]; then
|
|
||||||
exitError "Release '$RELEASE_NAME' already exists!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
git diff-index --quiet HEAD --
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "Current working tree is not clean! Please commit or unstage any changes."
|
|
||||||
fi
|
|
||||||
|
|
||||||
git checkout "$BRANCH" > /dev/null 2>&1
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "Source branch '$BRANCH' does not exist!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
grep -q "${APP_NAME_UPPER}_VERSION \"${RELEASE_NAME}\"" CMakeLists.txt
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "${APP_NAME_UPPER}_VERSION version not updated to '${RELEASE_NAME}' in CMakeLists.txt!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
grep -q "${APP_NAME_UPPER}_VERSION_NUM \"${RELEASE_NAME}\"" CMakeLists.txt
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "${APP_NAME_UPPER}_VERSION_NUM version not updated to '${RELEASE_NAME}' in CMakeLists.txt!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f CHANGELOG ]; then
|
|
||||||
exitError "No CHANGELOG file found!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
grep -qPzo "${RELEASE_NAME} \(\d{4}-\d{2}-\d{2}\)\n=+\n" CHANGELOG
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "CHANGELOG does not contain any information about the '${RELEASE_NAME}' release!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
git checkout "$RELEASE_BRANCH" > /dev/null 2>&1
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "Release branch '$RELEASE_BRANCH' does not exist!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "All checks pass, getting our hands dirty now!"
|
|
||||||
|
|
||||||
logInfo "Merging '${BRANCH}' into '${RELEASE_BRANCH}'..."
|
|
||||||
|
|
||||||
CHANGELOG=$(grep -Pzo "(?<=${RELEASE_NAME} \(\d{4}-\d{2}-\d{2}\)\n)=+\n\n(?:.|\n)+?\n(?=\n)" \
|
|
||||||
CHANGELOG | grep -Pzo '(?<=\n\n)(.|\n)+' | tr -d \\0)
|
|
||||||
COMMIT_MSG="Release ${RELEASE_NAME}"
|
|
||||||
|
|
||||||
git merge "$BRANCH" --no-ff -m "$COMMIT_MSG" -m "${CHANGELOG}" "$BRANCH" -S"$GPG_GIT_KEY"
|
|
||||||
|
|
||||||
logInfo "Creating tag '${RELEASE_NAME}'..."
|
|
||||||
if [ "" == "$GPG_GIT_KEY" ]; then
|
|
||||||
git tag -a "$RELEASE_NAME" -m "$COMMIT_MSG" -m "${CHANGELOG}" -s
|
|
||||||
else
|
|
||||||
git tag -a "$RELEASE_NAME" -m "$COMMIT_MSG" -m "${CHANGELOG}" -s -u "$GPG_GIT_KEY"
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "Merge done, creating target directory..."
|
|
||||||
mkdir -p "$OUTPUT_DIR"
|
|
||||||
|
|
||||||
if [ $? -ne 0 ]; then
|
|
||||||
exitError "Failed to create output directory!"
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "Creating source tarball..."
|
|
||||||
TARBALL_NAME="${APP_NAME_LOWER}-${RELEASE_NAME}-src.tar.bz2"
|
|
||||||
git archive --format=tar "$RELEASE_BRANCH" --prefix="${APP_NAME_LOWER}-${RELEASE_NAME}/" \
|
|
||||||
| bzip2 -9 > "${OUTPUT_DIR}/${TARBALL_NAME}"
|
|
||||||
|
|
||||||
|
|
||||||
if $BUILD_SOURCES; then
|
|
||||||
logInfo "Creating build directory..."
|
|
||||||
mkdir -p "${OUTPUT_DIR}/build-release"
|
|
||||||
mkdir -p "${OUTPUT_DIR}/bin-release"
|
|
||||||
cd "${OUTPUT_DIR}/build-release"
|
|
||||||
|
|
||||||
logInfo "Configuring sources..."
|
|
||||||
for p in $BUILD_PLUGINS; do
|
|
||||||
CMAKE_OPTIONS="${CMAKE_OPTIONS} -DWITH_XC_$(echo $p | tr '[:lower:]' '[:upper:]')=On"
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ "$COMPILER" == "g++" ]; then
|
|
||||||
export CC=gcc
|
|
||||||
elif [ "$COMPILER" == "clang++" ]; then
|
|
||||||
export CC=clang
|
|
||||||
fi
|
|
||||||
export CXX="$COMPILER"
|
|
||||||
|
|
||||||
if [ "" == "$DOCKER_IMAGE" ]; then
|
|
||||||
cmake -DCMAKE_BUILD_TYPE=Release -DWITH_TESTS=Off $CMAKE_OPTIONS \
|
|
||||||
-DCMAKE_INSTALL_PREFIX="${INSTALL_PREFIX}" "$SRC_DIR"
|
|
||||||
|
|
||||||
logInfo "Compiling sources..."
|
|
||||||
make $MAKE_OPTIONS
|
|
||||||
|
|
||||||
logInfo "Installing to bin dir..."
|
|
||||||
make DESTDIR="${OUTPUT_DIR}/bin-release" install/strip
|
|
||||||
else
|
|
||||||
logInfo "Launching Docker container to compile sources..."
|
|
||||||
|
|
||||||
docker run --name "$DOCKER_CONTAINER_NAME" --rm \
|
|
||||||
-e "CC=${CC}" -e "CXX=${CXX}" \
|
|
||||||
-v "$(realpath "$SRC_DIR"):/keepassxc/src:ro" \
|
|
||||||
-v "$(realpath "$OUTPUT_DIR"):/keepassxc/out:rw" \
|
|
||||||
"$DOCKER_IMAGE" \
|
|
||||||
bash -c "cd /keepassxc/out/build-release && \
|
|
||||||
cmake -DCMAKE_BUILD_TYPE=Release -DWITH_TESTS=Off $CMAKE_OPTIONS \
|
|
||||||
-DCMAKE_INSTALL_PREFIX=\"${INSTALL_PREFIX}\" /keepassxc/src && \
|
|
||||||
make $MAKE_OPTIONS && make DESTDIR=/keepassxc/out/bin-release install/strip"
|
|
||||||
|
|
||||||
logInfo "Build finished, Docker container terminated."
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "Creating AppImage..."
|
|
||||||
${SRC_DIR}/AppImage-Recipe.sh "$APP_NAME" "$RELEASE_NAME"
|
|
||||||
|
|
||||||
cd ..
|
|
||||||
logInfo "Signing source tarball..."
|
|
||||||
gpg --output "${TARBALL_NAME}.sig" --armor --local-user "$GPG_KEY" --detach-sig "$TARBALL_NAME"
|
|
||||||
|
|
||||||
logInfo "Signing AppImage..."
|
|
||||||
APPIMAGE_NAME="${APP_NAME}-${RELEASE_NAME}-x86_64.AppImage"
|
|
||||||
gpg --output "${APPIMAGE_NAME}.sig" --armor --local-user "$GPG_KEY" --detach-sig "$APPIMAGE_NAME"
|
|
||||||
|
|
||||||
logInfo "Creating digests..."
|
|
||||||
sha256sum "$TARBALL_NAME" > "${TARBALL_NAME}.DIGEST"
|
|
||||||
sha256sum "$APPIMAGE_NAME" > "${APPIMAGE_NAME}.DIGEST"
|
|
||||||
fi
|
|
||||||
|
|
||||||
logInfo "Leaving source directory..."
|
|
||||||
cd "$ORIG_CWD"
|
|
||||||
git checkout "$ORIG_BRANCH" > /dev/null 2>&1
|
|
||||||
|
|
||||||
logInfo "All done!"
|
|
||||||
logInfo "Please merge the release branch back into the develop branch now and then push your changes."
|
|
||||||
logInfo "Don't forget to also push the tags using \e[1mgit push --tags\e[0m."
|
|
682
release-tool
Executable file
682
release-tool
Executable file
@ -0,0 +1,682 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
#
|
||||||
|
# KeePassXC Release Preparation Helper
|
||||||
|
# Copyright (C) 2017 KeePassXC team <https://keepassxc.org/>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 2 or (at your option)
|
||||||
|
# version 3 of the License.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
echo -e "\e[1m\e[32mKeePassXC\e[0m Release Preparation Helper"
|
||||||
|
echo -e "Copyright (C) 2017 KeePassXC Team <https://keepassxc.org/>\n"
|
||||||
|
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# global default values
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
RELEASE_NAME=""
|
||||||
|
APP_NAME="KeePassXC"
|
||||||
|
SRC_DIR="."
|
||||||
|
GPG_KEY="CFB4C2166397D0D2"
|
||||||
|
GPG_GIT_KEY=""
|
||||||
|
OUTPUT_DIR="release"
|
||||||
|
SOURCE_BRANCH=""
|
||||||
|
TARGET_BRANCH="master"
|
||||||
|
TAG_NAME=""
|
||||||
|
DOCKER_IMAGE=""
|
||||||
|
DOCKER_CONTAINER_NAME="keepassxc-build-container"
|
||||||
|
CMAKE_OPTIONS=""
|
||||||
|
COMPILER="g++"
|
||||||
|
MAKE_OPTIONS="-j8"
|
||||||
|
BUILD_PLUGINS="autotype"
|
||||||
|
INSTALL_PREFIX="/usr/local"
|
||||||
|
BUILD_SOURCE_TARBALL=true
|
||||||
|
ORIG_BRANCH=""
|
||||||
|
ORIG_CWD="$(pwd)"
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# helper functions
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
printUsage() {
|
||||||
|
local cmd
|
||||||
|
if [ "" == "$1" ] || [ "help" == "$1" ]; then
|
||||||
|
cmd="COMMAND"
|
||||||
|
elif [ "merge" == "$1" ] || [ "build" == "$1" ] || [ "sign" == "$1" ]; then
|
||||||
|
cmd="$1"
|
||||||
|
else
|
||||||
|
logError "Unknown command: '$1'\n"
|
||||||
|
cmd="COMMAND"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo -e "\e[1mUsage:\e[0m $(basename $0) $cmd [options]"
|
||||||
|
|
||||||
|
if [ "COMMAND" == "$cmd" ]; then
|
||||||
|
cat << EOF
|
||||||
|
|
||||||
|
Commands:
|
||||||
|
merge Merge release branch into main branch and create release tags
|
||||||
|
build Build and package binary release from sources
|
||||||
|
sign Sign previously compiled release packages
|
||||||
|
help Show help for the given command
|
||||||
|
EOF
|
||||||
|
elif [ "merge" == "$cmd" ]; then
|
||||||
|
cat << EOF
|
||||||
|
|
||||||
|
Merge release branch into main branch and create release tags
|
||||||
|
|
||||||
|
Options:
|
||||||
|
-v, --version Release version number or name (required)
|
||||||
|
-a, --app-name Application name (default: '${APP_NAME}')
|
||||||
|
-s, --source-dir Source directory (default: '${SRC_DIR}')
|
||||||
|
-g, --gpg-key GPG key used to sign the merge commit and release tag,
|
||||||
|
leave empty to let Git choose your default key
|
||||||
|
(default: '${GPG_GIT_KEY}')
|
||||||
|
-r, --release-branch Source release branch to merge from (default: 'release/VERSION')
|
||||||
|
--target-branch Target branch to merge to (default: '${TARGET_BRANCH}')
|
||||||
|
-t, --tag-name Override release tag name (defaults to version number)
|
||||||
|
-h, --help Show this help
|
||||||
|
EOF
|
||||||
|
elif [ "build" == "$cmd" ]; then
|
||||||
|
cat << EOF
|
||||||
|
|
||||||
|
Build and package binary release from sources
|
||||||
|
|
||||||
|
Options:
|
||||||
|
-v, --version Release version number or name (required)
|
||||||
|
-a, --app-name Application name (default: '${APP_NAME}')
|
||||||
|
-s, --source-dir Source directory (default: '${SRC_DIR}')
|
||||||
|
-o, --output-dir Output directory where to build the release
|
||||||
|
(default: '${OUTPUT_DIR}')
|
||||||
|
-t, --tag-name Release tag to check out (defaults to version number)
|
||||||
|
-b, --build Build sources after exporting release
|
||||||
|
-d, --docker-image Use the specified Docker image to compile the application.
|
||||||
|
The image must have all required build dependencies installed.
|
||||||
|
This option has no effect if --build is not set.
|
||||||
|
--container-name Docker container name (default: '${DOCKER_CONTAINER_NAME}')
|
||||||
|
The container must not exist already
|
||||||
|
-c, --cmake-options Additional CMake options for compiling the sources
|
||||||
|
--compiler Compiler to use (default: '${COMPILER}')
|
||||||
|
-m, --make-options Make options for compiling sources (default: '${MAKE_OPTIONS}')
|
||||||
|
-i, --install-prefix Install prefix (default: '${INSTALL_PREFIX}')
|
||||||
|
-p, --plugins Space-separated list of plugins to build
|
||||||
|
(default: ${BUILD_PLUGINS})
|
||||||
|
-n, --no-source-tarball Don't build source tarball
|
||||||
|
-h, --help Show this help
|
||||||
|
EOF
|
||||||
|
elif [ "sign" == "$cmd" ]; then
|
||||||
|
cat << EOF
|
||||||
|
|
||||||
|
Sign previously compiled release packages
|
||||||
|
|
||||||
|
Options:
|
||||||
|
-f, --files Files to sign (required)
|
||||||
|
-g, --gpg-key GPG key used to sign the files (default: '${GPG_KEY}')
|
||||||
|
-h, --help Show this help
|
||||||
|
EOF
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
logInfo() {
|
||||||
|
echo -e "\e[1m[ \e[34mINFO\e[39m ]\e[0m $1"
|
||||||
|
}
|
||||||
|
|
||||||
|
logError() {
|
||||||
|
echo -e "\e[1m[ \e[31mERROR\e[39m ]\e[0m $1" >&2
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
ORIG_CWD="$(pwd)"
|
||||||
|
cd "$SRC_DIR" > /dev/null 2>&1
|
||||||
|
ORIG_BRANCH="$(git rev-parse --abbrev-ref HEAD 2> /dev/null)"
|
||||||
|
cd "$ORIG_CWD"
|
||||||
|
}
|
||||||
|
|
||||||
|
cleanup() {
|
||||||
|
logInfo "Checking out original branch..."
|
||||||
|
if [ "" != "$ORIG_BRANCH" ]; then
|
||||||
|
git checkout "$ORIG_BRANCH" > /dev/null 2>&1
|
||||||
|
fi
|
||||||
|
logInfo "Leaving source directory..."
|
||||||
|
cd "$ORIG_CWD"
|
||||||
|
}
|
||||||
|
|
||||||
|
exitError() {
|
||||||
|
logError "$1"
|
||||||
|
cleanup
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
exitTrap() {
|
||||||
|
exitError "Existing upon user request..."
|
||||||
|
}
|
||||||
|
|
||||||
|
checkSourceDirExists() {
|
||||||
|
if [ ! -d "$SRC_DIR" ]; then
|
||||||
|
exitError "Source directory '${SRC_DIR}' does not exist!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkOutputDirDoesNotExist() {
|
||||||
|
if [ -e "$OUTPUT_DIR" ]; then
|
||||||
|
exitError "Output directory '$OUTPUT_DIR' already exists. Please choose a different location!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkGitRepository() {
|
||||||
|
if [ ! -d .git ] || [ ! -f CHANGELOG ]; then
|
||||||
|
exitError "Source directory is not a valid Git repository!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkTagExists() {
|
||||||
|
git tag | grep -q "$TAG_NAME"
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "Tag '${TAG_NAME}' does not exist!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkReleaseDoesNotExist() {
|
||||||
|
git tag | grep -q "$TAG_NAME"
|
||||||
|
if [ $? -eq 0 ]; then
|
||||||
|
exitError "Release '$RELEASE_NAME' (tag: '$TAG_NAME') already exists!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkWorkingTreeClean() {
|
||||||
|
git diff-index --quiet HEAD --
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "Current working tree is not clean! Please commit or unstage any changes."
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkSourceBranchExists() {
|
||||||
|
git rev-parse "$SOURCE_BRANCH" > /dev/null 2>&1
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "Source branch '$SOURCE_BRANCH' does not exist!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkTargetBranchExists() {
|
||||||
|
git rev-parse "$TARGET_BRANCH" > /dev/null 2>&1
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "Target branch '$TARGET_BRANCH' does not exist!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkVersionInCMake() {
|
||||||
|
local app_name_upper="$(echo "$APP_NAME" | tr '[:lower:]' '[:upper:]')"
|
||||||
|
|
||||||
|
grep -q "${app_name_upper}_VERSION \"${RELEASE_NAME}\"" CMakeLists.txt
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "${app_name_upper}_VERSION version not updated to '${RELEASE_NAME}' in CMakeLists.txt!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
grep -q "${app_name_upper}_VERSION_NUM \"${RELEASE_NAME}\"" CMakeLists.txt
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "${app_name_upper}_VERSION_NUM version not updated to '${RELEASE_NAME}' in CMakeLists.txt!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkChangeLog() {
|
||||||
|
if [ ! -f CHANGELOG ]; then
|
||||||
|
exitError "No CHANGELOG file found!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
grep -qPzo "${RELEASE_NAME} \(\d{4}-\d{2}-\d{2}\)\n=+\n" CHANGELOG
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "CHANGELOG does not contain any information about the '${RELEASE_NAME}' release!"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
checkTransifexCommandExists() {
|
||||||
|
command -v tx > /dev/null
|
||||||
|
if [ 0 -ne $? ]; then
|
||||||
|
exitError "Transifex tool 'tx' not installed! Please install it using 'pip install transifex-client'"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# re-implement realpath for OS X (thanks mschrag)
|
||||||
|
# https://superuser.com/questions/205127/
|
||||||
|
if $(command -v realpath > /dev/null); then
|
||||||
|
realpath() {
|
||||||
|
pushd . > /dev/null
|
||||||
|
if [ -d "$1" ]; then
|
||||||
|
cd "$1"
|
||||||
|
dirs -l +0
|
||||||
|
else
|
||||||
|
cd "$(dirname "$1")"
|
||||||
|
cur_dir=$(dirs -l +0)
|
||||||
|
|
||||||
|
if [ "$cur_dir" == "/" ]; then
|
||||||
|
echo "$cur_dir$(basename "$1")"
|
||||||
|
else
|
||||||
|
echo "$cur_dir/$(basename "$1")"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
popd > /dev/null
|
||||||
|
}
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
trap exitTrap SIGINT SIGTERM
|
||||||
|
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# merge command
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
merge() {
|
||||||
|
while [ $# -ge 1 ]; do
|
||||||
|
local arg="$1"
|
||||||
|
case "$arg" in
|
||||||
|
-v|--version)
|
||||||
|
RELEASE_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-a|--app-name)
|
||||||
|
APP_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-s|--source-dir)
|
||||||
|
SRC_DIR="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-g|--gpg-key)
|
||||||
|
GPG_GIT_KEY="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-r|--release-branch)
|
||||||
|
SOURCE_BRANCH="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
--target-branch)
|
||||||
|
TARGET_BRANCH="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-t|--tag-name)
|
||||||
|
TAG_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-h|--help)
|
||||||
|
printUsage "merge"
|
||||||
|
exit ;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
logError "Unknown option '$arg'\n"
|
||||||
|
printUsage "merge"
|
||||||
|
exit 1 ;;
|
||||||
|
esac
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "" == "$RELEASE_NAME" ]; then
|
||||||
|
logError "Missing arguments, --version is required!\n"
|
||||||
|
printUsage "merge"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "" == "$TAG_NAME" ]; then
|
||||||
|
TAG_NAME="$RELEASE_NAME"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "" == "$SOURCE_BRANCH" ]; then
|
||||||
|
SOURCE_BRANCH="release/${RELEASE_NAME}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
init
|
||||||
|
|
||||||
|
SRC_DIR="$(realpath "$SRC_DIR")"
|
||||||
|
|
||||||
|
logInfo "Performing basic checks..."
|
||||||
|
|
||||||
|
checkSourceDirExists
|
||||||
|
|
||||||
|
logInfo "Changing to source directory..."
|
||||||
|
cd "${SRC_DIR}"
|
||||||
|
|
||||||
|
checkTransifexCommandExists
|
||||||
|
checkGitRepository
|
||||||
|
checkReleaseDoesNotExist
|
||||||
|
checkWorkingTreeClean
|
||||||
|
checkSourceBranchExists
|
||||||
|
checkTargetBranchExists
|
||||||
|
checkVersionInCMake
|
||||||
|
checkChangeLog
|
||||||
|
|
||||||
|
logInfo "All checks pass, getting our hands dirty now!"
|
||||||
|
|
||||||
|
logInfo "Checking out source branch..."
|
||||||
|
git checkout "$SOURCE_BRANCH"
|
||||||
|
|
||||||
|
logInfo "Updating language files..."
|
||||||
|
./share/translations/update.sh
|
||||||
|
if [ 0 -ne $? ]; then
|
||||||
|
exitError "Updating translations failed!"
|
||||||
|
fi
|
||||||
|
git diff-index --quiet HEAD --
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
git add ./share/translations/*
|
||||||
|
logInfo "Committing changes..."
|
||||||
|
if [ "" == "$GPG_GIT_KEY" ]; then
|
||||||
|
git commit -m "Update translations"
|
||||||
|
else
|
||||||
|
git commit -m "Update translations" -S"$GPG_GIT_KEY"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
logInfo "Checking out target branch '${TARGET_BRANCH}'..."
|
||||||
|
git checkout "$TARGET_BRANCH"
|
||||||
|
|
||||||
|
logInfo "Merging '${SOURCE_BRANCH}' into '${TARGET_BRANCH}'..."
|
||||||
|
|
||||||
|
CHANGELOG=$(grep -Pzo "(?<=${RELEASE_NAME} \(\d{4}-\d{2}-\d{2}\)\n)=+\n\n?(?:.|\n)+?\n(?=\n)" \
|
||||||
|
CHANGELOG | grep -Pzo '(?<=\n\n)(.|\n)+' | tr -d \\0)
|
||||||
|
COMMIT_MSG="Release ${RELEASE_NAME}"
|
||||||
|
|
||||||
|
git merge "$SOURCE_BRANCH" --no-ff -m "$COMMIT_MSG" -m "${CHANGELOG}" "$SOURCE_BRANCH" -S"$GPG_GIT_KEY"
|
||||||
|
|
||||||
|
logInfo "Creating tag '${TAG_NAME}'..."
|
||||||
|
if [ "" == "$GPG_GIT_KEY" ]; then
|
||||||
|
git tag -a "$TAG_NAME" -m "$COMMIT_MSG" -m "${CHANGELOG}" -s
|
||||||
|
else
|
||||||
|
git tag -a "$TAG_NAME" -m "$COMMIT_MSG" -m "${CHANGELOG}" -s -u "$GPG_GIT_KEY"
|
||||||
|
fi
|
||||||
|
|
||||||
|
cleanup
|
||||||
|
|
||||||
|
logInfo "All done!"
|
||||||
|
logInfo "Please merge the release branch back into the develop branch now and then push your changes."
|
||||||
|
logInfo "Don't forget to also push the tags using \e[1mgit push --tags\e[0m."
|
||||||
|
}
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# build command
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
build() {
|
||||||
|
while [ $# -ge 1 ]; do
|
||||||
|
local arg="$1"
|
||||||
|
case "$arg" in
|
||||||
|
-v|--version)
|
||||||
|
RELEASE_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-a|--app-name)
|
||||||
|
APP_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-s|--source-dir)
|
||||||
|
SRC_DIR="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-o|--output-dir)
|
||||||
|
OUTPUT_DIR="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-t|--tag-name)
|
||||||
|
TAG_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-d|--docker-image)
|
||||||
|
DOCKER_IMAGE="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
--container-name)
|
||||||
|
DOCKER_CONTAINER_NAME="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-c|--cmake-options)
|
||||||
|
CMAKE_OPTIONS="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
--compiler)
|
||||||
|
COMPILER="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-m|--make-options)
|
||||||
|
MAKE_OPTIONS="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-i|--install-prefix)
|
||||||
|
INSTALL_PREFIX="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-p|--plugins)
|
||||||
|
BUILD_PLUGINS="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-n|--no-source-tarball)
|
||||||
|
BUILD_SOURCE_TARBALL=false ;;
|
||||||
|
|
||||||
|
-h|--help)
|
||||||
|
printUsage "build"
|
||||||
|
exit ;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
logError "Unknown option '$arg'\n"
|
||||||
|
printUsage "build"
|
||||||
|
exit 1 ;;
|
||||||
|
esac
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "" == "$RELEASE_NAME" ]; then
|
||||||
|
logError "Missing arguments, --version is required!\n"
|
||||||
|
printUsage "build"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "" == "$TAG_NAME" ]; then
|
||||||
|
TAG_NAME="$RELEASE_NAME"
|
||||||
|
fi
|
||||||
|
|
||||||
|
init
|
||||||
|
|
||||||
|
SRC_DIR="$(realpath "$SRC_DIR")"
|
||||||
|
OUTPUT_DIR="$(realpath "$OUTPUT_DIR")"
|
||||||
|
|
||||||
|
logInfo "Performing basic checks..."
|
||||||
|
|
||||||
|
checkSourceDirExists
|
||||||
|
|
||||||
|
logInfo "Changing to source directory..."
|
||||||
|
cd "${SRC_DIR}"
|
||||||
|
|
||||||
|
checkTagExists
|
||||||
|
checkGitRepository
|
||||||
|
checkWorkingTreeClean
|
||||||
|
checkOutputDirDoesNotExist
|
||||||
|
|
||||||
|
logInfo "All checks pass, getting our hands dirty now!"
|
||||||
|
|
||||||
|
logInfo "Checking out release tag '${TAG_NAME}'..."
|
||||||
|
git checkout "$TAG_NAME"
|
||||||
|
|
||||||
|
logInfo "Creating output directory..."
|
||||||
|
mkdir -p "$OUTPUT_DIR"
|
||||||
|
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
exitError "Failed to create output directory!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if $BUILD_SOURCE_TARBALL; then
|
||||||
|
logInfo "Creating source tarball..."
|
||||||
|
local app_name_lower="$(echo "$APP_NAME" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
TARBALL_NAME="${app_name_lower}-${RELEASE_NAME}-src.tar.xz"
|
||||||
|
git archive --format=tar "$TAG_NAME" --prefix="${app_name_lower}-${RELEASE_NAME}/" \
|
||||||
|
| xz -6 > "${OUTPUT_DIR}/${TARBALL_NAME}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
logInfo "Creating build directory..."
|
||||||
|
mkdir -p "${OUTPUT_DIR}/build-release"
|
||||||
|
cd "${OUTPUT_DIR}/build-release"
|
||||||
|
|
||||||
|
logInfo "Configuring sources..."
|
||||||
|
for p in $BUILD_PLUGINS; do
|
||||||
|
CMAKE_OPTIONS="${CMAKE_OPTIONS} -DWITH_XC_$(echo $p | tr '[:lower:]' '[:upper:]')=On"
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "$COMPILER" == "g++" ]; then
|
||||||
|
export CC=gcc
|
||||||
|
elif [ "$COMPILER" == "clang++" ]; then
|
||||||
|
export CC=clang
|
||||||
|
fi
|
||||||
|
export CXX="$COMPILER"
|
||||||
|
|
||||||
|
if [ "" == "$DOCKER_IMAGE" ]; then
|
||||||
|
if [ "$(uname -s)" == "Darwin" ]; then
|
||||||
|
# Building on OS X
|
||||||
|
local qt_vers="$(ls /usr/local/Cellar/qt5 2> /dev/null | sort -r | head -n1)"
|
||||||
|
if [ "" == "$qt_vers" ]; then
|
||||||
|
exitError "Couldn't find Qt5! Please make sure it is available in '/usr/local/Cellar/qt5'."
|
||||||
|
fi
|
||||||
|
export MACOSX_DEPLOYMENT_TARGET=10.7
|
||||||
|
|
||||||
|
logInfo "Configuring build..."
|
||||||
|
cmake -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX="${INSTALL_PREFIX}" \
|
||||||
|
-DCMAKE_OSX_ARCHITECTURES=x86_64 -DWITH_CXX11=OFF \
|
||||||
|
-DCMAKE_PREFIX_PATH="/usr/local/Cellar/qt5/${qt_vers}/lib/cmake" \
|
||||||
|
-DQT_BINARY_DIR="/usr/local/Cellar/qt5/${qt_vers}/bin" $CMAKE_OPTIONS "$SRC_DIR"
|
||||||
|
|
||||||
|
logInfo "Compiling and packaging sources..."
|
||||||
|
make $MAKE_OPTIONS package
|
||||||
|
|
||||||
|
mv "./${APP_NAME}-${RELEASE_NAME}.dmg" ../
|
||||||
|
elif [ "$(uname -o)" == "Msys" ]; then
|
||||||
|
# Building on Windows with Msys
|
||||||
|
logInfo "Configuring build..."
|
||||||
|
cmake -DCMAKE_BUILD_TYPE=Release -DWITH_TESTS=Off -G"MSYS Makefiles" \
|
||||||
|
-DCMAKE_INSTALL_PREFIX="${INSTALL_PREFIX}" $CMAKE_OPTIONS "$SRC_DIR"
|
||||||
|
|
||||||
|
logInfo "Compiling and packaging sources..."
|
||||||
|
make $MAKE_OPTIONS package
|
||||||
|
|
||||||
|
mv "./${APP_NAME}-${RELEASE_NAME}-"*.{exe,zip} ../
|
||||||
|
else
|
||||||
|
mkdir -p "${OUTPUT_DIR}/bin-release"
|
||||||
|
|
||||||
|
# Building on Linux without Docker container
|
||||||
|
logInfo "Configuring build..."
|
||||||
|
cmake -DCMAKE_BUILD_TYPE=Release -DWITH_TESTS=Off $CMAKE_OPTIONS \
|
||||||
|
-DCMAKE_INSTALL_PREFIX="${INSTALL_PREFIX}" "$SRC_DIR"
|
||||||
|
|
||||||
|
logInfo "Compiling sources..."
|
||||||
|
make $MAKE_OPTIONS
|
||||||
|
|
||||||
|
logInfo "Installing to bin dir..."
|
||||||
|
make DESTDIR="${OUTPUT_DIR}/bin-release" install/strip
|
||||||
|
|
||||||
|
logInfo "Creating AppImage..."
|
||||||
|
${SRC_DIR}/AppImage-Recipe.sh "$APP_NAME" "$RELEASE_NAME"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
mkdir -p "${OUTPUT_DIR}/bin-release"
|
||||||
|
|
||||||
|
logInfo "Launching Docker container to compile sources..."
|
||||||
|
|
||||||
|
docker run --name "$DOCKER_CONTAINER_NAME" --rm \
|
||||||
|
--cap-add SYS_ADMIN --device /dev/fuse \
|
||||||
|
-e "CC=${CC}" -e "CXX=${CXX}" \
|
||||||
|
-v "$(realpath "$SRC_DIR"):/keepassxc/src:ro" \
|
||||||
|
-v "$(realpath "$OUTPUT_DIR"):/keepassxc/out:rw" \
|
||||||
|
"$DOCKER_IMAGE" \
|
||||||
|
bash -c "cd /keepassxc/out/build-release && \
|
||||||
|
cmake -DCMAKE_BUILD_TYPE=Release -DWITH_TESTS=Off $CMAKE_OPTIONS \
|
||||||
|
-DCMAKE_INSTALL_PREFIX=\"${INSTALL_PREFIX}\" /keepassxc/src && \
|
||||||
|
make $MAKE_OPTIONS && make DESTDIR=/keepassxc/out/bin-release install/strip && \
|
||||||
|
/keepassxc/src/AppImage-Recipe.sh "$APP_NAME" "$RELEASE_NAME""
|
||||||
|
|
||||||
|
if [ 0 -ne $? ]; then
|
||||||
|
exitError "Docker build failed!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
logInfo "Build finished, Docker container terminated."
|
||||||
|
fi
|
||||||
|
|
||||||
|
cleanup
|
||||||
|
|
||||||
|
logInfo "All done!"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# sign command
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
sign() {
|
||||||
|
SIGN_FILES=()
|
||||||
|
|
||||||
|
while [ $# -ge 1 ]; do
|
||||||
|
local arg="$1"
|
||||||
|
case "$arg" in
|
||||||
|
-f|--files)
|
||||||
|
while [ "${2:0:1}" != "-" ] && [ $# -ge 2 ]; do
|
||||||
|
SIGN_FILES+=("$2")
|
||||||
|
shift
|
||||||
|
done ;;
|
||||||
|
|
||||||
|
-g|--gpg-key)
|
||||||
|
GPG_KEY="$2"
|
||||||
|
shift ;;
|
||||||
|
|
||||||
|
-h|--help)
|
||||||
|
printUsage "sign"
|
||||||
|
exit ;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
logError "Unknown option '$arg'\n"
|
||||||
|
printUsage "sign"
|
||||||
|
exit 1 ;;
|
||||||
|
esac
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ -z "$SIGN_FILES" ]; then
|
||||||
|
logError "Missing arguments, --files is required!\n"
|
||||||
|
printUsage "sign"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
for f in "${SIGN_FILES[@]}"; do
|
||||||
|
if [ ! -f "$f" ]; then
|
||||||
|
exitError "File '${f}' does not exist!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
logInfo "Signing file '${f}'..."
|
||||||
|
gpg --output "${f}.sig" --armor --local-user "$GPG_KEY" --detach-sig "$f"
|
||||||
|
|
||||||
|
if [ 0 -ne $? ]; then
|
||||||
|
exitError "Signing failed!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
logInfo "Creating digest for file '${f}'..."
|
||||||
|
sha256sum "$f" > "${f}.DIGEST"
|
||||||
|
done
|
||||||
|
|
||||||
|
logInfo "All done!"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# parse global command line
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
MODE="$1"
|
||||||
|
shift
|
||||||
|
if [ "" == "$MODE" ]; then
|
||||||
|
logError "Missing arguments!\n"
|
||||||
|
printUsage
|
||||||
|
exit 1
|
||||||
|
elif [ "help" == "$MODE" ]; then
|
||||||
|
printUsage "$1"
|
||||||
|
exit
|
||||||
|
elif [ "merge" == "$MODE" ] || [ "build" == "$MODE" ] || [ "sign" == "$MODE" ]; then
|
||||||
|
$MODE "$@"
|
||||||
|
else
|
||||||
|
printUsage "$MODE"
|
||||||
|
fi
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -19,6 +19,11 @@
|
|||||||
<source>KeePassXC is distributed under the term of the GNU General Public License (GPL) version 2 or (at your option) version 3.</source>
|
<source>KeePassXC is distributed under the term of the GNU General Public License (GPL) version 2 or (at your option) version 3.</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Extensions:
|
||||||
|
</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>AccessControlDialog</name>
|
<name>AccessControlDialog</name>
|
||||||
@ -475,6 +480,46 @@ Do you want to save it anyway?</source>
|
|||||||
<source>No Results</source>
|
<source>No Results</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Execute command?</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Do you really want to execute the following command?<br><br>%1<br></source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Remember my choice</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Autoreload Request</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>The database file has changed. Do you want to load the changes?</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Merge Request</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>The database file has changed and you have unsaved changes.Do you want to merge your changes?</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Autoreload Failed</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Could not parse or unlock the new database file while attempting to autoreload this database.</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Could not open the new database file while attempting to autoreload this database.</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>EditEntryWidget</name>
|
<name>EditEntryWidget</name>
|
||||||
@ -669,10 +714,6 @@ Do you want to save it anyway?</source>
|
|||||||
<source>Repeat:</source>
|
<source>Repeat:</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Gen.</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>URL:</source>
|
<source>URL:</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
@ -830,6 +871,13 @@ Do you want to save it anyway?</source>
|
|||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
|
<context>
|
||||||
|
<name>Entry</name>
|
||||||
|
<message>
|
||||||
|
<source> - Clone</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>EntryAttributesModel</name>
|
<name>EntryAttributesModel</name>
|
||||||
<message>
|
<message>
|
||||||
@ -1226,6 +1274,10 @@ This is a one-way migration. You won't be able to open the imported databas
|
|||||||
<source>Re&pair database</source>
|
<source>Re&pair database</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Password Generator</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>OptionDialog</name>
|
<name>OptionDialog</name>
|
||||||
@ -1299,14 +1351,6 @@ Only entries with the same scheme (http://, https://, ftp://, ...) are returned<
|
|||||||
<source>Automatic creates or updates are not supported for string fields!</source>
|
<source>Automatic creates or updates are not supported for string fields!</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>HTTP Host:</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<source>Default host: localhost</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>HTTP Port:</source>
|
<source>HTTP Port:</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
@ -1315,11 +1359,6 @@ Only entries with the same scheme (http://, https://, ftp://, ...) are returned<
|
|||||||
<source>Default port: 19455</source>
|
<source>Default port: 19455</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Enable KeepassXC Http protocol
|
|
||||||
This is required for accessing your databases from ChromeIPass or PassIFox</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>Re&quest to unlock the database if it is locked</source>
|
<source>Re&quest to unlock the database if it is locked</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
@ -1328,6 +1367,24 @@ This is required for accessing your databases from ChromeIPass or PassIFox</sour
|
|||||||
<source>Sort &matching entries by title</source>
|
<source>Sort &matching entries by title</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Enable KeepassXC HTTP protocol
|
||||||
|
This is required for accessing your databases from ChromeIPass or PassIFox</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>KeePassXC will listen to this port on 127.0.0.1</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Cannot bind to privileged ports</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Cannot bind to privileged ports below 1024!
|
||||||
|
Using default port 19455.</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>PasswordGeneratorWidget</name>
|
<name>PasswordGeneratorWidget</name>
|
||||||
@ -1335,10 +1392,6 @@ This is required for accessing your databases from ChromeIPass or PassIFox</sour
|
|||||||
<source>Password:</source>
|
<source>Password:</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Length:</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>Character Types</source>
|
<source>Character Types</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
@ -1364,11 +1417,63 @@ This is required for accessing your databases from ChromeIPass or PassIFox</sour
|
|||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Ensure that the password contains characters from every group</source>
|
<source>Accept</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Accept</source>
|
<source>%p%</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>strength</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>entropy</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>&Length:</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Pick characters from every group</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Generate</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Close</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Apply</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Entropy: %1 bit</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Password Quality: %1</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Poor</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Weak</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Good</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Excellent</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
@ -1415,16 +1520,20 @@ This is required for accessing your databases from ChromeIPass or PassIFox</sour
|
|||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>SearchWidget</name>
|
<name>SearchWidget</name>
|
||||||
<message>
|
|
||||||
<source>Find:</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>Case Sensitive</source>
|
<source>Case Sensitive</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Search Current Group</source>
|
<source>Search</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Find</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Clear</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
@ -1536,10 +1645,6 @@ give it a unique name to identify and accept it.</source>
|
|||||||
<source>Remember last databases</source>
|
<source>Remember last databases</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Open previous databases on startup</source>
|
|
||||||
<translation type="unfinished"></translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<source>Automatically save on exit</source>
|
<source>Automatically save on exit</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
@ -1581,11 +1686,19 @@ give it a unique name to identify and accept it.</source>
|
|||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Hide window to system tray instead of App Exit</source>
|
<source>Load previous databases on startup</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Hide window to system tray on App start</source>
|
<source>Automatically reload the database when modified externally</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Hide window to system tray instead of app exit</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Minimize window at application startup</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
@ -1615,6 +1728,10 @@ give it a unique name to identify and accept it.</source>
|
|||||||
<source>Lock databases after minimizing the window</source>
|
<source>Lock databases after minimizing the window</source>
|
||||||
<translation type="unfinished"></translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
|
<message>
|
||||||
|
<source>Don't require password repeat when it is visible</source>
|
||||||
|
<translation type="unfinished"></translation>
|
||||||
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>UnlockDatabaseWidget</name>
|
<name>UnlockDatabaseWidget</name>
|
||||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
BIN
share/windows/installer-header.bmp
Normal file
BIN
share/windows/installer-header.bmp
Normal file
Binary file not shown.
After Width: | Height: | Size: 25 KiB |
BIN
share/windows/installer-wizard.bmp
Normal file
BIN
share/windows/installer-wizard.bmp
Normal file
Binary file not shown.
After Width: | Height: | Size: 151 KiB |
@ -218,6 +218,7 @@ target_link_libraries(${PROGNAME}
|
|||||||
Qt5::Widgets
|
Qt5::Widgets
|
||||||
Qt5::Network
|
Qt5::Network
|
||||||
${GCRYPT_LIBRARIES}
|
${GCRYPT_LIBRARIES}
|
||||||
|
${GPGERROR_LIBRARIES}
|
||||||
${ZLIB_LIBRARIES})
|
${ZLIB_LIBRARIES})
|
||||||
|
|
||||||
set_target_properties(${PROGNAME} PROPERTIES ENABLE_EXPORTS ON)
|
set_target_properties(${PROGNAME} PROPERTIES ENABLE_EXPORTS ON)
|
||||||
@ -257,9 +258,25 @@ if(APPLE)
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(MINGW)
|
if(MINGW)
|
||||||
set(CPACK_GENERATOR "ZIP")
|
string(REPLACE "AMD" "Win" OUTPUT_FILE_POSTFIX "${CMAKE_HOST_SYSTEM_PROCESSOR}")
|
||||||
|
set(CPACK_GENERATOR "ZIP;NSIS")
|
||||||
set(CPACK_STRIP_FILES ON)
|
set(CPACK_STRIP_FILES ON)
|
||||||
set(CPACK_PACKAGE_FILE_NAME "${PROGNAME}-${KEEPASSXC_VERSION_NUM}")
|
set(CPACK_PACKAGE_FILE_NAME "${PROGNAME}-${KEEPASSXC_VERSION}-${OUTPUT_FILE_POSTFIX}")
|
||||||
|
set(CPACK_PACKAGE_INSTALL_DIRECTORY ${PROGNAME})
|
||||||
|
set(CPACK_PACKAGE_VERSION ${KEEPASSXC_VERSION})
|
||||||
|
set(CPACK_PACKAGE_VENDOR "${PROGNAME} Team")
|
||||||
|
string(REGEX REPLACE "/" "\\\\\\\\" CPACK_PACKAGE_ICON "${CMAKE_SOURCE_DIR}/share/windows/installer-header.bmp")
|
||||||
|
set(CPACK_RESOURCE_FILE_LICENSE "${CMAKE_SOURCE_DIR}/LICENSE.GPL-2")
|
||||||
|
set(CPACK_NSIS_MUI_ICON "${CMAKE_SOURCE_DIR}/share/windows/keepassxc.ico")
|
||||||
|
set(CPACK_NSIS_MUI_UNIICON "${CPACK_NSIS_MUI_ICON}")
|
||||||
|
set(CPACK_NSIS_INSTALLED_ICON_NAME "\\\\${PROGNAME}.exe")
|
||||||
|
string(REGEX REPLACE "/" "\\\\\\\\" CPACK_NSIS_MUI_WELCOMEFINISHPAGE_BITMAP "${CMAKE_SOURCE_DIR}/share/windows/installer-wizard.bmp")
|
||||||
|
set(CPACK_NSIS_MUI_UNWELCOMEFINISHPAGE_BITMAP "${CPACK_NSIS_MUI_WELCOMEFINISHPAGE_BITMAP}")
|
||||||
|
set(CPACK_NSIS_CREATE_ICONS_EXTRA "CreateShortCut '$SMPROGRAMS\\\\$STARTMENU_FOLDER\\\\${PROGNAME}.lnk' '$INSTDIR\\\\${PROGNAME}.exe'")
|
||||||
|
set(CPACK_NSIS_DELETE_ICONS_EXTRA "Delete '$SMPROGRAMS\\\\$START_MENU\\\\${PROGNAME}.lnk'")
|
||||||
|
set(CPACK_NSIS_URL_INFO_ABOUT "https://keepassxc.org")
|
||||||
|
set(CPACK_NSIS_PACKAGE_NAME "${PROGNAME} v${KEEPASSXC_VERSION}")
|
||||||
|
set(CPACK_NSIS_MUI_FINISHPAGE_RUN "../${PROGNAME}.exe")
|
||||||
include(CPack)
|
include(CPack)
|
||||||
|
|
||||||
install(CODE "
|
install(CODE "
|
||||||
@ -267,5 +284,9 @@ if(MINGW)
|
|||||||
" COMPONENT Runtime)
|
" COMPONENT Runtime)
|
||||||
|
|
||||||
include(DeployQt4)
|
include(DeployQt4)
|
||||||
install_qt4_executable(${PROGNAME}.exe "qjpeg;qgif;qico;qtaccessiblewidgets")
|
install_qt4_executable(${PROGNAME}.exe)
|
||||||
|
add_custom_command(TARGET ${PROGNAME} POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different ${Qt5Core_DIR}/../../../share/qt5/plugins/platforms/qwindows$<$<CONFIG:Debug>:d>.dll
|
||||||
|
$<TARGET_FILE_DIR:${PROGNAME}>)
|
||||||
|
install(FILES $<TARGET_FILE_DIR:${PROGNAME}>/qwindows$<$<CONFIG:Debug>:d>.dll DESTINATION "platforms")
|
||||||
endif()
|
endif()
|
||||||
|
@ -98,7 +98,9 @@ QString AutoTypePlatformMac::activeWindowTitle()
|
|||||||
if (windowLayer(window) == 0) {
|
if (windowLayer(window) == 0) {
|
||||||
// First toplevel window in list (front to back order)
|
// First toplevel window in list (front to back order)
|
||||||
title = windowTitle(window);
|
title = windowTitle(window);
|
||||||
break;
|
if (!title.isEmpty()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -353,6 +353,12 @@ void Entry::setTitle(const QString& title)
|
|||||||
|
|
||||||
void Entry::setUrl(const QString& url)
|
void Entry::setUrl(const QString& url)
|
||||||
{
|
{
|
||||||
|
bool remove = url != m_attributes->value(EntryAttributes::URLKey) &&
|
||||||
|
(m_attributes->value(EntryAttributes::RememberCmdExecAttr) == "1" ||
|
||||||
|
m_attributes->value(EntryAttributes::RememberCmdExecAttr) == "0");
|
||||||
|
if (remove) {
|
||||||
|
m_attributes->remove(EntryAttributes::RememberCmdExecAttr);
|
||||||
|
}
|
||||||
m_attributes->set(EntryAttributes::URLKey, url, m_attributes->isProtected(EntryAttributes::URLKey));
|
m_attributes->set(EntryAttributes::URLKey, url, m_attributes->isProtected(EntryAttributes::URLKey));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -508,7 +514,8 @@ Entry* Entry::clone(CloneFlags flags) const
|
|||||||
entry->m_data.timeInfo.setLocationChanged(now);
|
entry->m_data.timeInfo.setLocationChanged(now);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (flags & CloneRenameTitle)
|
||||||
|
entry->setTitle(entry->title() + tr(" - Clone"));
|
||||||
|
|
||||||
return entry;
|
return entry;
|
||||||
}
|
}
|
||||||
|
@ -115,7 +115,8 @@ public:
|
|||||||
CloneNoFlags = 0,
|
CloneNoFlags = 0,
|
||||||
CloneNewUuid = 1, // generate a random uuid for the clone
|
CloneNewUuid = 1, // generate a random uuid for the clone
|
||||||
CloneResetTimeInfo = 2, // set all TimeInfo attributes to the current time
|
CloneResetTimeInfo = 2, // set all TimeInfo attributes to the current time
|
||||||
CloneIncludeHistory = 4 // clone the history items
|
CloneIncludeHistory = 4, // clone the history items
|
||||||
|
CloneRenameTitle = 8 // add "-Clone" after the original title
|
||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(CloneFlags, CloneFlag)
|
Q_DECLARE_FLAGS(CloneFlags, CloneFlag)
|
||||||
|
|
||||||
|
@ -24,6 +24,7 @@ const QString EntryAttributes::URLKey = "URL";
|
|||||||
const QString EntryAttributes::NotesKey = "Notes";
|
const QString EntryAttributes::NotesKey = "Notes";
|
||||||
const QStringList EntryAttributes::DefaultAttributes(QStringList() << TitleKey << UserNameKey
|
const QStringList EntryAttributes::DefaultAttributes(QStringList() << TitleKey << UserNameKey
|
||||||
<< PasswordKey << URLKey << NotesKey);
|
<< PasswordKey << URLKey << NotesKey);
|
||||||
|
const QString EntryAttributes::RememberCmdExecAttr = "_EXEC_CMD";
|
||||||
|
|
||||||
EntryAttributes::EntryAttributes(QObject* parent)
|
EntryAttributes::EntryAttributes(QObject* parent)
|
||||||
: QObject(parent)
|
: QObject(parent)
|
||||||
|
@ -52,6 +52,7 @@ public:
|
|||||||
static const QString URLKey;
|
static const QString URLKey;
|
||||||
static const QString NotesKey;
|
static const QString NotesKey;
|
||||||
static const QStringList DefaultAttributes;
|
static const QStringList DefaultAttributes;
|
||||||
|
static const QString RememberCmdExecAttr;
|
||||||
static bool isDefaultAttribute(const QString& key);
|
static bool isDefaultAttribute(const QString& key);
|
||||||
|
|
||||||
Q_SIGNALS:
|
Q_SIGNALS:
|
||||||
|
@ -42,7 +42,11 @@ QList<Entry*> EntrySearcher::searchEntries(const QString& searchTerm, const Grou
|
|||||||
const QList<Group*> children = group->children();
|
const QList<Group*> children = group->children();
|
||||||
for (Group* childGroup : children) {
|
for (Group* childGroup : children) {
|
||||||
if (childGroup->searchingEnabled() != Group::Disable) {
|
if (childGroup->searchingEnabled() != Group::Disable) {
|
||||||
searchResult.append(searchEntries(searchTerm, childGroup, caseSensitivity));
|
if (matchGroup(searchTerm, childGroup, caseSensitivity)) {
|
||||||
|
searchResult.append(childGroup->entriesRecursive());
|
||||||
|
} else {
|
||||||
|
searchResult.append(searchEntries(searchTerm, childGroup, caseSensitivity));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,3 +73,21 @@ bool EntrySearcher::wordMatch(const QString& word, Entry* entry, Qt::CaseSensiti
|
|||||||
entry->url().contains(word, caseSensitivity) ||
|
entry->url().contains(word, caseSensitivity) ||
|
||||||
entry->notes().contains(word, caseSensitivity);
|
entry->notes().contains(word, caseSensitivity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool EntrySearcher::matchGroup(const QString& searchTerm, const Group* group, Qt::CaseSensitivity caseSensitivity)
|
||||||
|
{
|
||||||
|
const QStringList wordList = searchTerm.split(QRegExp("\\s"), QString::SkipEmptyParts);
|
||||||
|
for (const QString& word : wordList) {
|
||||||
|
if (!wordMatch(word, group, caseSensitivity)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool EntrySearcher::wordMatch(const QString& word, const Group* group, Qt::CaseSensitivity caseSensitivity)
|
||||||
|
{
|
||||||
|
return group->name().contains(word, caseSensitivity) ||
|
||||||
|
group->notes().contains(word, caseSensitivity);
|
||||||
|
}
|
||||||
|
@ -33,6 +33,8 @@ private:
|
|||||||
QList<Entry*> searchEntries(const QString& searchTerm, const Group* group, Qt::CaseSensitivity caseSensitivity);
|
QList<Entry*> searchEntries(const QString& searchTerm, const Group* group, Qt::CaseSensitivity caseSensitivity);
|
||||||
QList<Entry*> matchEntry(const QString& searchTerm, Entry* entry, Qt::CaseSensitivity caseSensitivity);
|
QList<Entry*> matchEntry(const QString& searchTerm, Entry* entry, Qt::CaseSensitivity caseSensitivity);
|
||||||
bool wordMatch(const QString& word, Entry* entry, Qt::CaseSensitivity caseSensitivity);
|
bool wordMatch(const QString& word, Entry* entry, Qt::CaseSensitivity caseSensitivity);
|
||||||
|
bool matchGroup(const QString& searchTerm, const Group* group, Qt::CaseSensitivity caseSensitivity);
|
||||||
|
bool wordMatch(const QString& word, const Group* group, Qt::CaseSensitivity caseSensitivity);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSX_ENTRYSEARCHER_H
|
#endif // KEEPASSX_ENTRYSEARCHER_H
|
||||||
|
@ -388,7 +388,7 @@ void KeePass2XmlReader::parseBinaries()
|
|||||||
QString id = attr.value("ID").toString();
|
QString id = attr.value("ID").toString();
|
||||||
|
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
if (attr.value("Compressed").compare("True", Qt::CaseInsensitive) == 0) {
|
if (attr.value("Compressed").compare(QLatin1String("True"), Qt::CaseInsensitive) == 0) {
|
||||||
data = readCompressedBinary();
|
data = readCompressedBinary();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -17,12 +17,20 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "Application.h"
|
#include "Application.h"
|
||||||
|
#include "MainWindow.h"
|
||||||
|
|
||||||
#include <QAbstractNativeEventFilter>
|
#include <QAbstractNativeEventFilter>
|
||||||
#include <QFileOpenEvent>
|
#include <QFileOpenEvent>
|
||||||
|
#include <QSocketNotifier>
|
||||||
|
|
||||||
#include "autotype/AutoType.h"
|
#include "autotype/AutoType.h"
|
||||||
|
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
#include <signal.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#if defined(Q_OS_UNIX) && !defined(Q_OS_OSX)
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_OSX)
|
||||||
class XcbEventFilter : public QAbstractNativeEventFilter
|
class XcbEventFilter : public QAbstractNativeEventFilter
|
||||||
{
|
{
|
||||||
@ -65,12 +73,18 @@ public:
|
|||||||
Application::Application(int& argc, char** argv)
|
Application::Application(int& argc, char** argv)
|
||||||
: QApplication(argc, argv)
|
: QApplication(argc, argv)
|
||||||
, m_mainWindow(nullptr)
|
, m_mainWindow(nullptr)
|
||||||
|
#ifdef Q_OS_UNIX
|
||||||
|
, m_unixSignalNotifier(nullptr)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
#if defined(Q_OS_UNIX) && !defined(Q_OS_OSX)
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_OSX)
|
||||||
installNativeEventFilter(new XcbEventFilter());
|
installNativeEventFilter(new XcbEventFilter());
|
||||||
#elif defined(Q_OS_WIN)
|
#elif defined(Q_OS_WIN)
|
||||||
installNativeEventFilter(new WinEventFilter());
|
installNativeEventFilter(new WinEventFilter());
|
||||||
#endif
|
#endif
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
registerUnixSignals();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void Application::setMainWindow(QWidget* mainWindow)
|
void Application::setMainWindow(QWidget* mainWindow)
|
||||||
@ -98,3 +112,57 @@ bool Application::event(QEvent* event)
|
|||||||
|
|
||||||
return QApplication::event(event);
|
return QApplication::event(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
int Application::unixSignalSocket[2];
|
||||||
|
|
||||||
|
void Application::registerUnixSignals()
|
||||||
|
{
|
||||||
|
int result = ::socketpair(AF_UNIX, SOCK_STREAM, 0, unixSignalSocket);
|
||||||
|
Q_ASSERT(0 == result);
|
||||||
|
if (0 != result) {
|
||||||
|
// do not register handles when socket creation failed, otherwise
|
||||||
|
// application will be unresponsive to signals such as SIGINT or SIGTERM
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVector<int> const handledSignals = { SIGQUIT, SIGINT, SIGTERM, SIGHUP };
|
||||||
|
for (auto s: handledSignals) {
|
||||||
|
struct sigaction sigAction;
|
||||||
|
|
||||||
|
sigAction.sa_handler = handleUnixSignal;
|
||||||
|
sigemptyset(&sigAction.sa_mask);
|
||||||
|
sigAction.sa_flags = 0 | SA_RESTART;
|
||||||
|
sigaction(s, &sigAction, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_unixSignalNotifier = new QSocketNotifier(unixSignalSocket[1], QSocketNotifier::Read, this);
|
||||||
|
connect(m_unixSignalNotifier, SIGNAL(activated(int)), this, SLOT(quitBySignal()));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Application::handleUnixSignal(int sig)
|
||||||
|
{
|
||||||
|
switch (sig) {
|
||||||
|
case SIGQUIT:
|
||||||
|
case SIGINT:
|
||||||
|
case SIGTERM:
|
||||||
|
{
|
||||||
|
char buf = 0;
|
||||||
|
::write(unixSignalSocket[0], &buf, sizeof(buf));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case SIGHUP:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Application::quitBySignal()
|
||||||
|
{
|
||||||
|
m_unixSignalNotifier->setEnabled(false);
|
||||||
|
char buf;
|
||||||
|
::read(unixSignalSocket[1], &buf, sizeof(buf));
|
||||||
|
|
||||||
|
if (nullptr != m_mainWindow)
|
||||||
|
static_cast<MainWindow*>(m_mainWindow)->appExit();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
@ -21,6 +21,8 @@
|
|||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
|
|
||||||
|
class QSocketNotifier;
|
||||||
|
|
||||||
class Application : public QApplication
|
class Application : public QApplication
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@ -34,8 +36,23 @@ public:
|
|||||||
Q_SIGNALS:
|
Q_SIGNALS:
|
||||||
void openFile(const QString& filename);
|
void openFile(const QString& filename);
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
void quitBySignal();
|
||||||
|
#endif
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QWidget* m_mainWindow;
|
QWidget* m_mainWindow;
|
||||||
|
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
/**
|
||||||
|
* Register Unix signals such as SIGINT and SIGTERM for clean shutdown.
|
||||||
|
*/
|
||||||
|
void registerUnixSignals();
|
||||||
|
QSocketNotifier* m_unixSignalNotifier;
|
||||||
|
static void handleUnixSignal(int sig);
|
||||||
|
static int unixSignalSocket[2];
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSX_APPLICATION_H
|
#endif // KEEPASSX_APPLICATION_H
|
||||||
|
@ -246,7 +246,7 @@ bool DatabaseTabWidget::closeDatabase(Database* db)
|
|||||||
QMessageBox::StandardButton result =
|
QMessageBox::StandardButton result =
|
||||||
MessageBox::question(
|
MessageBox::question(
|
||||||
this, tr("Close?"),
|
this, tr("Close?"),
|
||||||
tr("\"%1\" is in edit mode.\nDiscard changes and close anyway?").arg(dbName),
|
tr("\"%1\" is in edit mode.\nDiscard changes and close anyway?").arg(dbName.toHtmlEscaped()),
|
||||||
QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Cancel);
|
QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Cancel);
|
||||||
if (result == QMessageBox::Cancel) {
|
if (result == QMessageBox::Cancel) {
|
||||||
return false;
|
return false;
|
||||||
@ -262,7 +262,7 @@ bool DatabaseTabWidget::closeDatabase(Database* db)
|
|||||||
QMessageBox::StandardButton result =
|
QMessageBox::StandardButton result =
|
||||||
MessageBox::question(
|
MessageBox::question(
|
||||||
this, tr("Save changes?"),
|
this, tr("Save changes?"),
|
||||||
tr("\"%1\" was modified.\nSave changes?").arg(dbName),
|
tr("\"%1\" was modified.\nSave changes?").arg(dbName.toHtmlEscaped()),
|
||||||
QMessageBox::Yes | QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Yes);
|
QMessageBox::Yes | QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Yes);
|
||||||
if (result == QMessageBox::Yes) {
|
if (result == QMessageBox::Yes) {
|
||||||
if (!saveDatabase(db)) {
|
if (!saveDatabase(db)) {
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
|
|
||||||
#include <QAction>
|
#include <QAction>
|
||||||
#include <QDesktopServices>
|
#include <QDesktopServices>
|
||||||
|
#include <QCheckBox>
|
||||||
#include <QHBoxLayout>
|
#include <QHBoxLayout>
|
||||||
#include <QLabel>
|
#include <QLabel>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
@ -312,8 +313,10 @@ void DatabaseWidget::cloneEntry()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Entry* entry = currentEntry->clone(Entry::CloneNewUuid | Entry::CloneResetTimeInfo);
|
Entry* entry = currentEntry->clone(Entry::CloneNewUuid | Entry::CloneResetTimeInfo | Entry::CloneRenameTitle);
|
||||||
entry->setGroup(currentEntry->group());
|
entry->setGroup(currentEntry->group());
|
||||||
|
if (isInSearchMode())
|
||||||
|
search(m_lastSearchText);
|
||||||
m_entryView->setFocus();
|
m_entryView->setFocus();
|
||||||
m_entryView->setCurrentEntry(entry);
|
m_entryView->setCurrentEntry(entry);
|
||||||
}
|
}
|
||||||
@ -341,7 +344,7 @@ void DatabaseWidget::deleteEntries()
|
|||||||
result = MessageBox::question(
|
result = MessageBox::question(
|
||||||
this, tr("Delete entry?"),
|
this, tr("Delete entry?"),
|
||||||
tr("Do you really want to delete the entry \"%1\" for good?")
|
tr("Do you really want to delete the entry \"%1\" for good?")
|
||||||
.arg(selectedEntries.first()->title()),
|
.arg(selectedEntries.first()->title().toHtmlEscaped()),
|
||||||
QMessageBox::Yes | QMessageBox::No);
|
QMessageBox::Yes | QMessageBox::No);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -365,7 +368,7 @@ void DatabaseWidget::deleteEntries()
|
|||||||
result = MessageBox::question(
|
result = MessageBox::question(
|
||||||
this, tr("Move entry to recycle bin?"),
|
this, tr("Move entry to recycle bin?"),
|
||||||
tr("Do you really want to move entry \"%1\" to the recycle bin?")
|
tr("Do you really want to move entry \"%1\" to the recycle bin?")
|
||||||
.arg(selectedEntries.first()->title()),
|
.arg(selectedEntries.first()->title().toHtmlEscaped()),
|
||||||
QMessageBox::Yes | QMessageBox::No);
|
QMessageBox::Yes | QMessageBox::No);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -494,8 +497,46 @@ void DatabaseWidget::openUrlForEntry(Entry* entry)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (urlString.startsWith("cmd://")) {
|
if (urlString.startsWith("cmd://")) {
|
||||||
|
// check if decision to execute command was stored
|
||||||
|
if (entry->attributes()->hasKey(EntryAttributes::RememberCmdExecAttr)) {
|
||||||
|
if (entry->attributes()->value(EntryAttributes::RememberCmdExecAttr) == "1") {
|
||||||
|
QProcess::startDetached(urlString.mid(6));
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// otherwise ask user
|
||||||
if (urlString.length() > 6) {
|
if (urlString.length() > 6) {
|
||||||
QProcess::startDetached(urlString.mid(6));
|
QString cmdTruncated = urlString.mid(6);
|
||||||
|
if (cmdTruncated.length() > 400)
|
||||||
|
cmdTruncated = cmdTruncated.left(400) + " […]";
|
||||||
|
QMessageBox msgbox(QMessageBox::Icon::Question,
|
||||||
|
tr("Execute command?"),
|
||||||
|
tr("Do you really want to execute the following command?<br><br>%1<br>")
|
||||||
|
.arg(cmdTruncated.toHtmlEscaped()),
|
||||||
|
QMessageBox::Yes | QMessageBox::No,
|
||||||
|
this
|
||||||
|
);
|
||||||
|
msgbox.setDefaultButton(QMessageBox::No);
|
||||||
|
|
||||||
|
QCheckBox* checkbox = new QCheckBox(tr("Remember my choice"), &msgbox);
|
||||||
|
msgbox.setCheckBox(checkbox);
|
||||||
|
bool remember = false;
|
||||||
|
QObject::connect(checkbox, &QCheckBox::stateChanged, [&](int state) {
|
||||||
|
if (static_cast<Qt::CheckState>(state) == Qt::CheckState::Checked) {
|
||||||
|
remember = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
int result = msgbox.exec();
|
||||||
|
if (result == QMessageBox::Yes) {
|
||||||
|
QProcess::startDetached(urlString.mid(6));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (remember) {
|
||||||
|
entry->attributes()->set(EntryAttributes::RememberCmdExecAttr,
|
||||||
|
result == QMessageBox::Yes ? "1" : "0");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -532,7 +573,7 @@ void DatabaseWidget::deleteGroup()
|
|||||||
QMessageBox::StandardButton result = MessageBox::question(
|
QMessageBox::StandardButton result = MessageBox::question(
|
||||||
this, tr("Delete group?"),
|
this, tr("Delete group?"),
|
||||||
tr("Do you really want to delete the group \"%1\" for good?")
|
tr("Do you really want to delete the group \"%1\" for good?")
|
||||||
.arg(currentGroup->name()),
|
.arg(currentGroup->name().toHtmlEscaped()),
|
||||||
QMessageBox::Yes | QMessageBox::No);
|
QMessageBox::Yes | QMessageBox::No);
|
||||||
if (result == QMessageBox::Yes) {
|
if (result == QMessageBox::Yes) {
|
||||||
delete currentGroup;
|
delete currentGroup;
|
||||||
@ -722,15 +763,10 @@ void DatabaseWidget::unlockDatabase(bool accepted)
|
|||||||
|
|
||||||
replaceDatabase(db);
|
replaceDatabase(db);
|
||||||
|
|
||||||
const QList<Group*> groups = m_db->rootGroup()->groupsRecursive(true);
|
restoreGroupEntryFocus(m_groupBeforeLock, m_entryBeforeLock);
|
||||||
for (Group* group : groups) {
|
|
||||||
if (group->uuid() == m_groupBeforeLock) {
|
|
||||||
m_groupView->setCurrentGroup(group);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m_groupBeforeLock = Uuid();
|
m_groupBeforeLock = Uuid();
|
||||||
|
m_entryBeforeLock = Uuid();
|
||||||
|
|
||||||
setCurrentWidget(m_mainWidget);
|
setCurrentWidget(m_mainWidget);
|
||||||
m_unlockDatabaseWidget->clearForms();
|
m_unlockDatabaseWidget->clearForms();
|
||||||
Q_EMIT unlockedDatabase();
|
Q_EMIT unlockedDatabase();
|
||||||
@ -755,7 +791,7 @@ void DatabaseWidget::entryActivationSignalReceived(Entry* entry, EntryModel::Mod
|
|||||||
void DatabaseWidget::switchToEntryEdit()
|
void DatabaseWidget::switchToEntryEdit()
|
||||||
{
|
{
|
||||||
Entry* entry = m_entryView->currentEntry();
|
Entry* entry = m_entryView->currentEntry();
|
||||||
Q_ASSERT(entry);
|
|
||||||
if (!entry) {
|
if (!entry) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -766,7 +802,7 @@ void DatabaseWidget::switchToEntryEdit()
|
|||||||
void DatabaseWidget::switchToGroupEdit()
|
void DatabaseWidget::switchToGroupEdit()
|
||||||
{
|
{
|
||||||
Group* group = m_groupView->currentGroup();
|
Group* group = m_groupView->currentGroup();
|
||||||
Q_ASSERT(group);
|
|
||||||
if (!group) {
|
if (!group) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -943,6 +979,10 @@ void DatabaseWidget::lock()
|
|||||||
m_groupBeforeLock = m_db->rootGroup()->uuid();
|
m_groupBeforeLock = m_db->rootGroup()->uuid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (m_entryView->currentEntry()) {
|
||||||
|
m_entryBeforeLock = m_entryView->currentEntry()->uuid();
|
||||||
|
}
|
||||||
|
|
||||||
clearAllWidgets();
|
clearAllWidgets();
|
||||||
m_unlockDatabaseWidget->load(m_filename);
|
m_unlockDatabaseWidget->load(m_filename);
|
||||||
setCurrentWidget(m_unlockDatabaseWidget);
|
setCurrentWidget(m_unlockDatabaseWidget);
|
||||||
@ -1028,7 +1068,22 @@ void DatabaseWidget::reloadDatabaseFile()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Uuid groupBeforeReload;
|
||||||
|
if (m_groupView && m_groupView->currentGroup()) {
|
||||||
|
groupBeforeReload = m_groupView->currentGroup()->uuid();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
groupBeforeReload = m_db->rootGroup()->uuid();
|
||||||
|
}
|
||||||
|
|
||||||
|
Uuid entryBeforeReload;
|
||||||
|
if (m_entryView && m_entryView->currentEntry()) {
|
||||||
|
entryBeforeReload = m_entryView->currentEntry()->uuid();
|
||||||
|
}
|
||||||
|
|
||||||
replaceDatabase(db);
|
replaceDatabase(db);
|
||||||
|
restoreGroupEntryFocus(groupBeforeReload, entryBeforeReload);
|
||||||
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
MessageBox::critical(this, tr("Autoreload Failed"),
|
MessageBox::critical(this, tr("Autoreload Failed"),
|
||||||
@ -1061,6 +1116,35 @@ QStringList DatabaseWidget::customEntryAttributes() const
|
|||||||
return entry->attributes()->customKeys();
|
return entry->attributes()->customKeys();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Restores the focus on the group and entry that was focused
|
||||||
|
* before the database was locked or reloaded.
|
||||||
|
*/
|
||||||
|
void DatabaseWidget::restoreGroupEntryFocus(Uuid groupUuid, Uuid entryUuid)
|
||||||
|
{
|
||||||
|
Group* restoredGroup = nullptr;
|
||||||
|
const QList<Group*> groups = m_db->rootGroup()->groupsRecursive(true);
|
||||||
|
for (Group* group : groups) {
|
||||||
|
if (group->uuid() == groupUuid) {
|
||||||
|
restoredGroup = group;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (restoredGroup != nullptr) {
|
||||||
|
m_groupView->setCurrentGroup(restoredGroup);
|
||||||
|
|
||||||
|
const QList<Entry*> entries = restoredGroup->entries();
|
||||||
|
for (Entry* entry : entries) {
|
||||||
|
if (entry->uuid() == entryUuid) {
|
||||||
|
m_entryView->setCurrentEntry(entry);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
bool DatabaseWidget::isGroupSelected() const
|
bool DatabaseWidget::isGroupSelected() const
|
||||||
{
|
{
|
||||||
return m_groupView->currentGroup() != nullptr;
|
return m_groupView->currentGroup() != nullptr;
|
||||||
|
@ -163,6 +163,7 @@ private Q_SLOTS:
|
|||||||
// Database autoreload slots
|
// Database autoreload slots
|
||||||
void onWatchedFileChanged();
|
void onWatchedFileChanged();
|
||||||
void reloadDatabaseFile();
|
void reloadDatabaseFile();
|
||||||
|
void restoreGroupEntryFocus(Uuid groupUuid, Uuid EntryUuid);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void setClipboardTextAndMinimize(const QString& text);
|
void setClipboardTextAndMinimize(const QString& text);
|
||||||
@ -190,6 +191,7 @@ private:
|
|||||||
Group* m_newParent;
|
Group* m_newParent;
|
||||||
QString m_filename;
|
QString m_filename;
|
||||||
Uuid m_groupBeforeLock;
|
Uuid m_groupBeforeLock;
|
||||||
|
Uuid m_entryBeforeLock;
|
||||||
|
|
||||||
// Search state
|
// Search state
|
||||||
QString m_lastSearchText;
|
QString m_lastSearchText;
|
||||||
|
@ -190,6 +190,10 @@ void EditWidgetIcons::abortFaviconDownload(bool clearRedirect)
|
|||||||
|
|
||||||
void EditWidgetIcons::onRequestFinished(QNetworkReply *reply)
|
void EditWidgetIcons::onRequestFinished(QNetworkReply *reply)
|
||||||
{
|
{
|
||||||
|
if (m_database == nullptr) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!reply->error()) {
|
if (!reply->error()) {
|
||||||
QImage image;
|
QImage image;
|
||||||
image.loadFromData(reply->readAll());
|
image.loadFromData(reply->readAll());
|
||||||
|
@ -364,7 +364,7 @@ void MainWindow::setMenuActionState(DatabaseWidget::Mode mode)
|
|||||||
bool groupSelected = dbWidget->isGroupSelected();
|
bool groupSelected = dbWidget->isGroupSelected();
|
||||||
|
|
||||||
m_ui->actionEntryNew->setEnabled(!inSearch);
|
m_ui->actionEntryNew->setEnabled(!inSearch);
|
||||||
m_ui->actionEntryClone->setEnabled(singleEntrySelected && !inSearch);
|
m_ui->actionEntryClone->setEnabled(singleEntrySelected);
|
||||||
m_ui->actionEntryEdit->setEnabled(singleEntrySelected);
|
m_ui->actionEntryEdit->setEnabled(singleEntrySelected);
|
||||||
m_ui->actionEntryDelete->setEnabled(entriesSelected);
|
m_ui->actionEntryDelete->setEnabled(entriesSelected);
|
||||||
m_ui->actionEntryCopyTitle->setEnabled(singleEntrySelected && dbWidget->currentEntryHasTitle());
|
m_ui->actionEntryCopyTitle->setEnabled(singleEntrySelected && dbWidget->currentEntryHasTitle());
|
||||||
|
@ -42,6 +42,7 @@ public:
|
|||||||
public Q_SLOTS:
|
public Q_SLOTS:
|
||||||
void openDatabase(const QString& fileName, const QString& pw = QString(),
|
void openDatabase(const QString& fileName, const QString& pw = QString(),
|
||||||
const QString& keyFile = QString());
|
const QString& keyFile = QString());
|
||||||
|
void appExit();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void closeEvent(QCloseEvent* event) override;
|
void closeEvent(QCloseEvent* event) override;
|
||||||
@ -68,7 +69,6 @@ private Q_SLOTS:
|
|||||||
void applySettingsChanges();
|
void applySettingsChanges();
|
||||||
void trayIconTriggered(QSystemTrayIcon::ActivationReason reason);
|
void trayIconTriggered(QSystemTrayIcon::ActivationReason reason);
|
||||||
void toggleWindow();
|
void toggleWindow();
|
||||||
void appExit();
|
|
||||||
void lockDatabasesAfterInactivity();
|
void lockDatabasesAfterInactivity();
|
||||||
void repairDatabase();
|
void repairDatabase();
|
||||||
|
|
||||||
|
@ -155,15 +155,15 @@
|
|||||||
<addaction name="actionEntryCopyNotes"/>
|
<addaction name="actionEntryCopyNotes"/>
|
||||||
<addaction name="separator"/>
|
<addaction name="separator"/>
|
||||||
</widget>
|
</widget>
|
||||||
<addaction name="actionEntryNew"/>
|
|
||||||
<addaction name="actionEntryClone"/>
|
|
||||||
<addaction name="actionEntryEdit"/>
|
|
||||||
<addaction name="actionEntryDelete"/>
|
|
||||||
<addaction name="actionEntryCopyUsername"/>
|
<addaction name="actionEntryCopyUsername"/>
|
||||||
<addaction name="actionEntryCopyPassword"/>
|
<addaction name="actionEntryCopyPassword"/>
|
||||||
<addaction name="menuEntryCopyAttribute"/>
|
<addaction name="menuEntryCopyAttribute"/>
|
||||||
<addaction name="actionEntryAutoType"/>
|
<addaction name="actionEntryAutoType"/>
|
||||||
<addaction name="actionEntryOpenUrl"/>
|
<addaction name="actionEntryOpenUrl"/>
|
||||||
|
<addaction name="actionEntryEdit"/>
|
||||||
|
<addaction name="actionEntryClone"/>
|
||||||
|
<addaction name="actionEntryDelete"/>
|
||||||
|
<addaction name="actionEntryNew"/>
|
||||||
</widget>
|
</widget>
|
||||||
<widget class="QMenu" name="menuGroups">
|
<widget class="QMenu" name="menuGroups">
|
||||||
<property name="title">
|
<property name="title">
|
||||||
|
@ -45,11 +45,15 @@ PasswordGeneratorWidget::PasswordGeneratorWidget(QWidget* parent)
|
|||||||
|
|
||||||
connect(m_ui->optionButtons, SIGNAL(buttonClicked(int)), SLOT(updateGenerator()));
|
connect(m_ui->optionButtons, SIGNAL(buttonClicked(int)), SLOT(updateGenerator()));
|
||||||
|
|
||||||
// set font size of password quality and entropy labels dynamically to 80% of the default font size
|
// set font size of password quality and entropy labels dynamically to 80% of
|
||||||
|
// the default font size, but make it no smaller than 8pt
|
||||||
QFont defaultFont;
|
QFont defaultFont;
|
||||||
defaultFont.setPointSize(static_cast<int>(defaultFont.pointSize() * 0.8f));
|
int smallerSize = static_cast<int>(defaultFont.pointSize() * 0.8f);
|
||||||
m_ui->entropyLabel->setFont(defaultFont);
|
if (smallerSize >= 8) {
|
||||||
m_ui->strengthLabel->setFont(defaultFont);
|
defaultFont.setPointSize(smallerSize);
|
||||||
|
m_ui->entropyLabel->setFont(defaultFont);
|
||||||
|
m_ui->strengthLabel->setFont(defaultFont);
|
||||||
|
}
|
||||||
|
|
||||||
loadSettings();
|
loadSettings();
|
||||||
reset();
|
reset();
|
||||||
@ -132,8 +136,10 @@ void PasswordGeneratorWidget::updatePasswordStrength(const QString& password)
|
|||||||
|
|
||||||
void PasswordGeneratorWidget::generatePassword()
|
void PasswordGeneratorWidget::generatePassword()
|
||||||
{
|
{
|
||||||
QString password = m_generator->generatePassword();
|
if (m_generator->isValid()) {
|
||||||
m_ui->editNewPassword->setText(password);
|
QString password = m_generator->generatePassword();
|
||||||
|
m_ui->editNewPassword->setText(password);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void PasswordGeneratorWidget::applyPassword()
|
void PasswordGeneratorWidget::applyPassword()
|
||||||
@ -279,5 +285,11 @@ void PasswordGeneratorWidget::updateGenerator()
|
|||||||
m_generator->setCharClasses(classes);
|
m_generator->setCharClasses(classes);
|
||||||
m_generator->setFlags(flags);
|
m_generator->setFlags(flags);
|
||||||
|
|
||||||
|
if (m_generator->isValid()) {
|
||||||
|
m_ui->buttonGenerate->setEnabled(true);
|
||||||
|
} else {
|
||||||
|
m_ui->buttonGenerate->setEnabled(false);
|
||||||
|
}
|
||||||
|
|
||||||
regeneratePassword();
|
regeneratePassword();
|
||||||
}
|
}
|
||||||
|
@ -104,11 +104,6 @@ QProgressBar::chunk {
|
|||||||
<height>30</height>
|
<height>30</height>
|
||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<property name="font">
|
|
||||||
<font>
|
|
||||||
<pointsize>8</pointsize>
|
|
||||||
</font>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>strength</string>
|
<string>strength</string>
|
||||||
</property>
|
</property>
|
||||||
@ -144,11 +139,6 @@ QProgressBar::chunk {
|
|||||||
<height>0</height>
|
<height>0</height>
|
||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<property name="font">
|
|
||||||
<font>
|
|
||||||
<pointsize>8</pointsize>
|
|
||||||
</font>
|
|
||||||
</property>
|
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>entropy</string>
|
<string>entropy</string>
|
||||||
</property>
|
</property>
|
||||||
@ -295,6 +285,12 @@ QProgressBar::chunk {
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QToolButton" name="checkBoxNumbers">
|
<widget class="QToolButton" name="checkBoxNumbers">
|
||||||
|
<property name="minimumSize">
|
||||||
|
<size>
|
||||||
|
<width>0</width>
|
||||||
|
<height>26</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
<property name="focusPolicy">
|
<property name="focusPolicy">
|
||||||
<enum>Qt::StrongFocus</enum>
|
<enum>Qt::StrongFocus</enum>
|
||||||
</property>
|
</property>
|
||||||
@ -314,6 +310,12 @@ QProgressBar::chunk {
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QToolButton" name="checkBoxSpecialChars">
|
<widget class="QToolButton" name="checkBoxSpecialChars">
|
||||||
|
<property name="minimumSize">
|
||||||
|
<size>
|
||||||
|
<width>0</width>
|
||||||
|
<height>26</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
<property name="focusPolicy">
|
<property name="focusPolicy">
|
||||||
<enum>Qt::StrongFocus</enum>
|
<enum>Qt::StrongFocus</enum>
|
||||||
</property>
|
</property>
|
||||||
|
@ -41,6 +41,7 @@ SearchWidget::SearchWidget(QWidget *parent)
|
|||||||
connect(this, SIGNAL(escapePressed()), m_ui->searchEdit, SLOT(clear()));
|
connect(this, SIGNAL(escapePressed()), m_ui->searchEdit, SLOT(clear()));
|
||||||
|
|
||||||
new QShortcut(Qt::CTRL + Qt::Key_F, this, SLOT(searchFocus()), nullptr, Qt::ApplicationShortcut);
|
new QShortcut(Qt::CTRL + Qt::Key_F, this, SLOT(searchFocus()), nullptr, Qt::ApplicationShortcut);
|
||||||
|
new QShortcut(Qt::Key_Escape, m_ui->searchEdit, SLOT(clear()), nullptr, Qt::ApplicationShortcut);
|
||||||
|
|
||||||
m_ui->searchEdit->installEventFilter(this);
|
m_ui->searchEdit->installEventFilter(this);
|
||||||
|
|
||||||
|
@ -89,6 +89,7 @@ void EditEntryWidget::setupMain()
|
|||||||
add(tr("Entry"), m_mainWidget);
|
add(tr("Entry"), m_mainWidget);
|
||||||
|
|
||||||
m_mainUi->togglePasswordButton->setIcon(filePath()->onOffIcon("actions", "password-show"));
|
m_mainUi->togglePasswordButton->setIcon(filePath()->onOffIcon("actions", "password-show"));
|
||||||
|
m_mainUi->togglePasswordGeneratorButton->setIcon(filePath()->icon("actions", "password-generator", false));
|
||||||
connect(m_mainUi->togglePasswordButton, SIGNAL(toggled(bool)), m_mainUi->passwordEdit, SLOT(setShowPassword(bool)));
|
connect(m_mainUi->togglePasswordButton, SIGNAL(toggled(bool)), m_mainUi->passwordEdit, SLOT(setShowPassword(bool)));
|
||||||
connect(m_mainUi->togglePasswordGeneratorButton, SIGNAL(toggled(bool)), SLOT(togglePasswordGeneratorButton(bool)));
|
connect(m_mainUi->togglePasswordGeneratorButton, SIGNAL(toggled(bool)), SLOT(togglePasswordGeneratorButton(bool)));
|
||||||
connect(m_mainUi->expireCheck, SIGNAL(toggled(bool)), m_mainUi->expireDatePicker, SLOT(setEnabled(bool)));
|
connect(m_mainUi->expireCheck, SIGNAL(toggled(bool)), m_mainUi->expireDatePicker, SLOT(setEnabled(bool)));
|
||||||
@ -271,14 +272,15 @@ void EditEntryWidget::loadEntry(Entry* entry, bool create, bool history, const Q
|
|||||||
m_history = history;
|
m_history = history;
|
||||||
|
|
||||||
if (history) {
|
if (history) {
|
||||||
setHeadline(QString("%1 > %2").arg(parentName, tr("Entry history")));
|
setHeadline(QString("%1 > %2").arg(parentName.toHtmlEscaped(), tr("Entry history")));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (create) {
|
if (create) {
|
||||||
setHeadline(QString("%1 > %2").arg(parentName, tr("Add entry")));
|
setHeadline(QString("%1 > %2").arg(parentName.toHtmlEscaped(), tr("Add entry")));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
setHeadline(QString("%1 > %2 > %3").arg(parentName, entry->title(), tr("Edit entry")));
|
setHeadline(QString("%1 > %2 > %3").arg(parentName.toHtmlEscaped(),
|
||||||
|
entry->title().toHtmlEscaped(), tr("Edit entry")));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -433,6 +435,9 @@ void EditEntryWidget::saveEntry()
|
|||||||
|
|
||||||
void EditEntryWidget::updateEntryData(Entry* entry) const
|
void EditEntryWidget::updateEntryData(Entry* entry) const
|
||||||
{
|
{
|
||||||
|
entry->attributes()->copyCustomKeysFrom(m_entryAttributes);
|
||||||
|
entry->attachments()->copyDataFrom(m_entryAttachments);
|
||||||
|
|
||||||
entry->setTitle(m_mainUi->titleEdit->text());
|
entry->setTitle(m_mainUi->titleEdit->text());
|
||||||
entry->setUsername(m_mainUi->usernameEdit->text());
|
entry->setUsername(m_mainUi->usernameEdit->text());
|
||||||
entry->setUrl(m_mainUi->urlEdit->text());
|
entry->setUrl(m_mainUi->urlEdit->text());
|
||||||
@ -442,9 +447,6 @@ void EditEntryWidget::updateEntryData(Entry* entry) const
|
|||||||
|
|
||||||
entry->setNotes(m_mainUi->notesEdit->toPlainText());
|
entry->setNotes(m_mainUi->notesEdit->toPlainText());
|
||||||
|
|
||||||
entry->attributes()->copyCustomKeysFrom(m_entryAttributes);
|
|
||||||
entry->attachments()->copyDataFrom(m_entryAttachments);
|
|
||||||
|
|
||||||
IconStruct iconStruct = m_iconsWidget->state();
|
IconStruct iconStruct = m_iconsWidget->state();
|
||||||
|
|
||||||
if (iconStruct.number < 0) {
|
if (iconStruct.number < 0) {
|
||||||
|
@ -77,9 +77,6 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QToolButton" name="togglePasswordGeneratorButton">
|
<widget class="QToolButton" name="togglePasswordGeneratorButton">
|
||||||
<property name="text">
|
|
||||||
<string>Generate</string>
|
|
||||||
</property>
|
|
||||||
<property name="checkable">
|
<property name="checkable">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
</property>
|
</property>
|
||||||
|
@ -18,7 +18,7 @@ PasswordGenerator HttpSettings::m_generator;
|
|||||||
|
|
||||||
bool HttpSettings::isEnabled()
|
bool HttpSettings::isEnabled()
|
||||||
{
|
{
|
||||||
return config()->get("Http/Enabled", true).toBool();
|
return config()->get("Http/Enabled", false).toBool();
|
||||||
}
|
}
|
||||||
|
|
||||||
void HttpSettings::setEnabled(bool enabled)
|
void HttpSettings::setEnabled(bool enabled)
|
||||||
@ -126,18 +126,6 @@ void HttpSettings::setSupportKphFields(bool supportKphFields)
|
|||||||
config()->set("Http/SupportKphFields", supportKphFields);
|
config()->set("Http/SupportKphFields", supportKphFields);
|
||||||
}
|
}
|
||||||
|
|
||||||
QString HttpSettings::httpHost()
|
|
||||||
{
|
|
||||||
static const QString host = "localhost";
|
|
||||||
|
|
||||||
return config()->get("Http/Host", host).toString().toUtf8();
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpSettings::setHttpHost(QString host)
|
|
||||||
{
|
|
||||||
config()->set("Http/Host", host);
|
|
||||||
}
|
|
||||||
|
|
||||||
int HttpSettings::httpPort()
|
int HttpSettings::httpPort()
|
||||||
{
|
{
|
||||||
static const int PORT = 19455;
|
static const int PORT = 19455;
|
||||||
|
@ -42,8 +42,6 @@ public:
|
|||||||
static void setSearchInAllDatabases(bool searchInAllDatabases);
|
static void setSearchInAllDatabases(bool searchInAllDatabases);
|
||||||
static bool supportKphFields();
|
static bool supportKphFields();
|
||||||
static void setSupportKphFields(bool supportKphFields);
|
static void setSupportKphFields(bool supportKphFields);
|
||||||
static QString httpHost();
|
|
||||||
static void setHttpHost(QString host);
|
|
||||||
static int httpPort();
|
static int httpPort();
|
||||||
static void setHttpPort(int port);
|
static void setHttpPort(int port);
|
||||||
|
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
#include "ui_OptionDialog.h"
|
#include "ui_OptionDialog.h"
|
||||||
#include "HttpSettings.h"
|
#include "HttpSettings.h"
|
||||||
|
|
||||||
|
#include <QMessageBox>
|
||||||
|
|
||||||
OptionDialog::OptionDialog(QWidget *parent) :
|
OptionDialog::OptionDialog(QWidget *parent) :
|
||||||
QWidget(parent),
|
QWidget(parent),
|
||||||
ui(new Ui::OptionDialog())
|
ui(new Ui::OptionDialog())
|
||||||
@ -41,7 +43,6 @@ void OptionDialog::loadSettings()
|
|||||||
ui->sortByUsername->setChecked(true);
|
ui->sortByUsername->setChecked(true);
|
||||||
else
|
else
|
||||||
ui->sortByTitle->setChecked(true);
|
ui->sortByTitle->setChecked(true);
|
||||||
ui->httpHost->setText(settings.httpHost());
|
|
||||||
ui->httpPort->setText(QString::number(settings.httpPort()));
|
ui->httpPort->setText(QString::number(settings.httpPort()));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -70,8 +71,14 @@ void OptionDialog::saveSettings()
|
|||||||
settings.setUnlockDatabase(ui->unlockDatabase->isChecked());
|
settings.setUnlockDatabase(ui->unlockDatabase->isChecked());
|
||||||
settings.setMatchUrlScheme(ui->matchUrlScheme->isChecked());
|
settings.setMatchUrlScheme(ui->matchUrlScheme->isChecked());
|
||||||
settings.setSortByUsername(ui->sortByUsername->isChecked());
|
settings.setSortByUsername(ui->sortByUsername->isChecked());
|
||||||
settings.setHttpHost(ui->httpHost->text());
|
|
||||||
settings.setHttpPort(ui->httpPort->text().toInt());
|
int port = ui->httpPort->text().toInt();
|
||||||
|
if (port < 1024) {
|
||||||
|
QMessageBox::warning(this, tr("Cannot bind to privileged ports"),
|
||||||
|
tr("Cannot bind to privileged ports below 1024!\nUsing default port 19455."));
|
||||||
|
port = 19455;
|
||||||
|
}
|
||||||
|
settings.setHttpPort(port);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
settings.setPasswordUseLowercase(ui->checkBoxLower->isChecked());
|
settings.setPasswordUseLowercase(ui->checkBoxLower->isChecked());
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>605</width>
|
<width>605</width>
|
||||||
<height>389</height>
|
<height>429</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
@ -17,7 +17,7 @@
|
|||||||
<item>
|
<item>
|
||||||
<widget class="QCheckBox" name="enableHttpServer">
|
<widget class="QCheckBox" name="enableHttpServer">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Enable KeepassXC Http protocol
|
<string>Enable KeepassXC HTTP protocol
|
||||||
This is required for accessing your databases from ChromeIPass or PassIFox</string>
|
This is required for accessing your databases from ChromeIPass or PassIFox</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
@ -28,7 +28,7 @@ This is required for accessing your databases from ChromeIPass or PassIFox</stri
|
|||||||
<enum>QTabWidget::Rounded</enum>
|
<enum>QTabWidget::Rounded</enum>
|
||||||
</property>
|
</property>
|
||||||
<property name="currentIndex">
|
<property name="currentIndex">
|
||||||
<number>0</number>
|
<number>2</number>
|
||||||
</property>
|
</property>
|
||||||
<widget class="QWidget" name="tab">
|
<widget class="QWidget" name="tab">
|
||||||
<attribute name="title">
|
<attribute name="title">
|
||||||
@ -201,32 +201,41 @@ Only entries with the same scheme (http://, https://, ftp://, ...) are returned<
|
|||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout_1">
|
<spacer name="verticalSpacer_4">
|
||||||
<item>
|
<property name="orientation">
|
||||||
<widget class="QLabel" name="label_5">
|
<enum>Qt::Vertical</enum>
|
||||||
<property name="sizePolicy">
|
</property>
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
<property name="sizeType">
|
||||||
<horstretch>0</horstretch>
|
<enum>QSizePolicy::Fixed</enum>
|
||||||
<verstretch>0</verstretch>
|
</property>
|
||||||
</sizepolicy>
|
<property name="sizeHint" stdset="0">
|
||||||
</property>
|
<size>
|
||||||
<property name="text">
|
<width>20</width>
|
||||||
<string>HTTP Host:</string>
|
<height>20</height>
|
||||||
</property>
|
</size>
|
||||||
</widget>
|
</property>
|
||||||
</item>
|
</spacer>
|
||||||
<item>
|
|
||||||
<widget class="QLineEdit" name="httpHost">
|
|
||||||
<property name="placeholderText">
|
|
||||||
<string>Default host: localhost</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
</layout>
|
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout_2">
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
<item>
|
<item row="1" column="1">
|
||||||
|
<widget class="QLineEdit" name="httpPort">
|
||||||
|
<property name="inputMask">
|
||||||
|
<string notr="true">d0000</string>
|
||||||
|
</property>
|
||||||
|
<property name="placeholderText">
|
||||||
|
<string>Default port: 19455</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="2" column="1">
|
||||||
|
<widget class="QLabel" name="label_5">
|
||||||
|
<property name="text">
|
||||||
|
<string>KeePassXC will listen to this port on 127.0.0.1</string>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item row="1" column="0">
|
||||||
<widget class="QLabel" name="label_4">
|
<widget class="QLabel" name="label_4">
|
||||||
<property name="sizePolicy">
|
<property name="sizePolicy">
|
||||||
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
|
||||||
@ -237,15 +246,8 @@ Only entries with the same scheme (http://, https://, ftp://, ...) are returned<
|
|||||||
<property name="text">
|
<property name="text">
|
||||||
<string>HTTP Port:</string>
|
<string>HTTP Port:</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
<property name="alignment">
|
||||||
</item>
|
<set>Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter</set>
|
||||||
<item>
|
|
||||||
<widget class="QLineEdit" name="httpPort">
|
|
||||||
<property name="inputMask">
|
|
||||||
<string notr="true">d0000</string>
|
|
||||||
</property>
|
|
||||||
<property name="placeholderText">
|
|
||||||
<string>Default port: 19455</string>
|
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
@ -331,68 +331,34 @@ void Server::start(void)
|
|||||||
if (m_started)
|
if (m_started)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bool nohost = true;
|
// local loopback hardcoded, since KeePassHTTP handshake
|
||||||
|
// is not safe against interception
|
||||||
|
QHostAddress address("127.0.0.1");
|
||||||
int port = HttpSettings::httpPort();
|
int port = HttpSettings::httpPort();
|
||||||
|
|
||||||
QHostInfo info = QHostInfo::fromName(HttpSettings::httpHost());
|
void* addrx = NULL;
|
||||||
if (!info.addresses().isEmpty()) {
|
unsigned int flags = MHD_USE_SELECT_INTERNALLY;
|
||||||
void* addrx = NULL;
|
|
||||||
unsigned int flags = MHD_USE_SELECT_INTERNALLY;
|
|
||||||
QHostAddress address = info.addresses().first();
|
|
||||||
|
|
||||||
if (address.protocol() == QAbstractSocket::IPv4Protocol) {
|
struct sockaddr_in *addr = static_cast<struct sockaddr_in*>(calloc(1, sizeof(struct sockaddr_in)));
|
||||||
struct sockaddr_in *addr = static_cast<struct sockaddr_in*>(calloc(1, sizeof(struct sockaddr_in)));
|
addrx = static_cast<void*>(addr);
|
||||||
addrx = static_cast<void*>(addr);
|
addr->sin_family = AF_INET;
|
||||||
addr->sin_family = AF_INET;
|
addr->sin_port = htons(port);
|
||||||
addr->sin_port = htons(HttpSettings::httpPort());
|
addr->sin_addr.s_addr = htonl(address.toIPv4Address());
|
||||||
addr->sin_addr.s_addr = htonl(address.toIPv4Address());
|
|
||||||
nohost = false;
|
|
||||||
} else {
|
|
||||||
struct sockaddr_in6 *addr = static_cast<struct sockaddr_in6*>(calloc(1, sizeof(struct sockaddr_in6)));
|
|
||||||
addrx = static_cast<void*>(addr);
|
|
||||||
addr->sin6_family = AF_INET6;
|
|
||||||
addr->sin6_port = htons(HttpSettings::httpPort());
|
|
||||||
memcpy(&addr->sin6_addr, address.toIPv6Address().c, 16);
|
|
||||||
nohost = false;
|
|
||||||
flags |= MHD_USE_IPv6;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nohost) {
|
if (NULL == (daemon = MHD_start_daemon(flags, port, NULL, NULL,
|
||||||
qWarning("HTTPPlugin: Faled to get configured host!");
|
&this->request_handler_wrapper, this,
|
||||||
} else {
|
MHD_OPTION_NOTIFY_COMPLETED,
|
||||||
if (NULL == (daemon = MHD_start_daemon(flags, port, NULL, NULL,
|
this->request_completed, NULL,
|
||||||
&this->request_handler_wrapper, this,
|
MHD_OPTION_SOCK_ADDR,
|
||||||
MHD_OPTION_NOTIFY_COMPLETED,
|
addrx,
|
||||||
this->request_completed, NULL,
|
MHD_OPTION_END))) {
|
||||||
MHD_OPTION_SOCK_ADDR,
|
qWarning("HTTPPlugin: Failed to bind to localhost!");
|
||||||
addrx,
|
} else {
|
||||||
MHD_OPTION_END))) {
|
m_started = true;
|
||||||
nohost = true;
|
|
||||||
qWarning("HTTPPlugin: Failed to bind to configured host!");
|
|
||||||
} else {
|
|
||||||
nohost = false;
|
|
||||||
//qWarning("HTTPPlugin: Binded to configured host.");
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
if (addrx != NULL)
|
|
||||||
free(addrx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nohost) {
|
if (addrx != NULL)
|
||||||
if (NULL == (daemon = MHD_start_daemon(MHD_USE_SELECT_INTERNALLY, port, NULL, NULL,
|
free(addrx);
|
||||||
&this->request_handler_wrapper, this,
|
|
||||||
MHD_OPTION_NOTIFY_COMPLETED,
|
|
||||||
this->request_completed, NULL,
|
|
||||||
MHD_OPTION_END))) {
|
|
||||||
qWarning("HTTPPlugin: Fatal! Failed to bind to both configured and default hosts!");
|
|
||||||
} else {
|
|
||||||
qWarning("HTTPPlugin: Bound to fallback address 0.0.0.0/:::!");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m_started = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -480,7 +480,8 @@ void Service::updateEntry(const QString &, const QString &uuid, const QString &l
|
|||||||
//ShowNotification(QString("%0: You have an entry change prompt waiting, click to activate").arg(requestId));
|
//ShowNotification(QString("%0: You have an entry change prompt waiting, click to activate").arg(requestId));
|
||||||
if ( HttpSettings::alwaysAllowUpdate()
|
if ( HttpSettings::alwaysAllowUpdate()
|
||||||
|| QMessageBox::warning(0, tr("KeePassXC: Update Entry"),
|
|| QMessageBox::warning(0, tr("KeePassXC: Update Entry"),
|
||||||
tr("Do you want to update the information in %1 - %2?").arg(QUrl(url).host()).arg(u),
|
tr("Do you want to update the information in %1 - %2?")
|
||||||
|
.arg(QUrl(url).host().toHtmlEscaped()).arg(u.toHtmlEscaped()),
|
||||||
QMessageBox::Yes|QMessageBox::No) == QMessageBox::Yes ) {
|
QMessageBox::Yes|QMessageBox::No) == QMessageBox::Yes ) {
|
||||||
entry->beginUpdate();
|
entry->beginUpdate();
|
||||||
entry->setUsername(login);
|
entry->setUsername(login);
|
||||||
|
10
src/main.cpp
10
src/main.cpp
@ -28,6 +28,16 @@
|
|||||||
#include "gui/MainWindow.h"
|
#include "gui/MainWindow.h"
|
||||||
#include "gui/MessageBox.h"
|
#include "gui/MessageBox.h"
|
||||||
|
|
||||||
|
#ifdef QT_STATIC
|
||||||
|
#include <QtPlugin>
|
||||||
|
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
Q_IMPORT_PLUGIN(QWindowsIntegrationPlugin)
|
||||||
|
#elif Q_OS_LINUX
|
||||||
|
Q_IMPORT_PLUGIN(QXcbIntegrationPlugin)
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
int main(int argc, char** argv)
|
int main(int argc, char** argv)
|
||||||
{
|
{
|
||||||
#ifdef QT_NO_DEBUG
|
#ifdef QT_NO_DEBUG
|
||||||
|
@ -92,6 +92,7 @@ set(TEST_LIBRARIES
|
|||||||
Qt5::Widgets
|
Qt5::Widgets
|
||||||
Qt5::Test
|
Qt5::Test
|
||||||
${GCRYPT_LIBRARIES}
|
${GCRYPT_LIBRARIES}
|
||||||
|
${GPGERROR_LIBRARIES}
|
||||||
${ZLIB_LIBRARIES}
|
${ZLIB_LIBRARIES}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -481,8 +481,7 @@ void TestGui::testSearch()
|
|||||||
QCOMPARE(entry->title(), origTitle.append("_edited"));
|
QCOMPARE(entry->title(), origTitle.append("_edited"));
|
||||||
|
|
||||||
// Cancel search, should return to normal view
|
// Cancel search, should return to normal view
|
||||||
QTest::mouseClick(searchTextEdit, Qt::LeftButton);
|
QTest::keyClick(m_mainWindow, Qt::Key_Escape);
|
||||||
QTest::keyClick(searchTextEdit, Qt::Key_Escape);
|
|
||||||
QTRY_COMPARE(m_dbWidget->currentMode(), DatabaseWidget::ViewMode);
|
QTRY_COMPARE(m_dbWidget->currentMode(), DatabaseWidget::ViewMode);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -567,7 +566,7 @@ void TestGui::testCloneEntry()
|
|||||||
QCOMPARE(entryView->model()->rowCount(), 2);
|
QCOMPARE(entryView->model()->rowCount(), 2);
|
||||||
Entry* entryClone = entryView->entryFromIndex(entryView->model()->index(1, 1));
|
Entry* entryClone = entryView->entryFromIndex(entryView->model()->index(1, 1));
|
||||||
QVERIFY(entryOrg->uuid() != entryClone->uuid());
|
QVERIFY(entryOrg->uuid() != entryClone->uuid());
|
||||||
QCOMPARE(entryClone->title(), entryOrg->title());
|
QCOMPARE(entryClone->title(), entryOrg->title() + QString(" - Clone"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestGui::testDragAndDropEntry()
|
void TestGui::testDragAndDropEntry()
|
||||||
|
@ -20,6 +20,7 @@ target_link_libraries(kdbx-extract
|
|||||||
keepassx_core
|
keepassx_core
|
||||||
Qt5::Core
|
Qt5::Core
|
||||||
${GCRYPT_LIBRARIES}
|
${GCRYPT_LIBRARIES}
|
||||||
|
${GPGERROR_LIBRARIES}
|
||||||
${ZLIB_LIBRARIES})
|
${ZLIB_LIBRARIES})
|
||||||
|
|
||||||
add_executable(kdbx-merge kdbx-merge.cpp)
|
add_executable(kdbx-merge kdbx-merge.cpp)
|
||||||
@ -27,6 +28,7 @@ target_link_libraries(kdbx-merge
|
|||||||
keepassx_core
|
keepassx_core
|
||||||
Qt5::Core
|
Qt5::Core
|
||||||
${GCRYPT_LIBRARIES}
|
${GCRYPT_LIBRARIES}
|
||||||
|
${GPGERROR_LIBRARIES}
|
||||||
${ZLIB_LIBRARIES})
|
${ZLIB_LIBRARIES})
|
||||||
|
|
||||||
|
|
||||||
|
@ -101,9 +101,10 @@ int main(int argc, char **argv)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (line[0])
|
if (line[0]) {
|
||||||
calculate(line,advanced);
|
calculate(line,advanced);
|
||||||
printf("> ");
|
printf("> ");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
Loading…
Reference in New Issue
Block a user