Compare commits
180 Commits
2.5.0.0bet
...
2.5.0.0bet
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4f9625a3e1 | ||
|
|
7688ebd29b | ||
|
|
3f2a7f1eb3 | ||
|
|
b3d067d0c8 | ||
|
|
a3b4ad5ac1 | ||
|
|
e3b329d27f | ||
|
|
7d10c43822 | ||
|
|
fcb0d45d39 | ||
|
|
5492db0223 | ||
|
|
2207b05f5f | ||
|
|
f15a0c2591 | ||
|
|
92fb22129c | ||
|
|
ccca9c4400 | ||
|
|
0597cb4416 | ||
|
|
0602174e50 | ||
|
|
6fddc92ce7 | ||
|
|
aa30df6454 | ||
|
|
f6c61ab407 | ||
|
|
2ab81ed77c | ||
|
|
0ade035f43 | ||
|
|
73b62035d8 | ||
|
|
28837db308 | ||
|
|
85befef260 | ||
|
|
e7bbb47422 | ||
|
|
846bc7edb1 | ||
|
|
99a9842a1f | ||
|
|
33009138c3 | ||
|
|
c74b82ebd8 | ||
|
|
49ea92a4a5 | ||
|
|
6cc4eeaa30 | ||
|
|
1b89f79888 | ||
|
|
07d99de3ea | ||
|
|
ae757affa1 | ||
|
|
482b6296fc | ||
|
|
d37ce729f0 | ||
|
|
e92ce232bc | ||
|
|
78ae44b160 | ||
|
|
49ce87a8e0 | ||
|
|
d2fe5ce884 | ||
|
|
5e695756de | ||
|
|
0506a75417 | ||
|
|
34e1316144 | ||
|
|
18dffa6c75 | ||
|
|
03621c378e | ||
|
|
aff9312419 | ||
|
|
4651b1be96 | ||
|
|
32497a22d2 | ||
|
|
3b5ef56c16 | ||
|
|
d4cc3a58d8 | ||
|
|
afedbb38b2 | ||
|
|
fe70bf3877 | ||
|
|
f14ee6cf1c | ||
|
|
1791b15f85 | ||
|
|
6c5b3b3a0d | ||
|
|
6ea2287679 | ||
|
|
cad4ec22b4 | ||
|
|
f277983caf | ||
|
|
5e561e5321 | ||
|
|
ef83dbcae2 | ||
|
|
ab6c69adcb | ||
|
|
1c90747476 | ||
|
|
c3b3d8482c | ||
|
|
86b4d92599 | ||
|
|
1f6786b1f8 | ||
|
|
4cd2b153f4 | ||
|
|
27b0810688 | ||
|
|
94c72a4cf6 | ||
|
|
e634116e71 | ||
|
|
9021cb5bc8 | ||
|
|
3674900a54 | ||
|
|
9855ae79c3 | ||
|
|
fe1a4985f1 | ||
|
|
015a368a1d | ||
|
|
07f59e071d | ||
|
|
670e2bfe33 | ||
|
|
992b6382a3 | ||
|
|
c3ac550c93 | ||
|
|
acba7fc5de | ||
|
|
ccb500fdf4 | ||
|
|
44111507e7 | ||
|
|
5f14596ed2 | ||
|
|
4ad65c8f4a | ||
|
|
b4b215fe30 | ||
|
|
d8a8005d70 | ||
|
|
37ca15b77c | ||
|
|
6c18fe5591 | ||
|
|
fdcb05467c | ||
|
|
ef8db46ae7 | ||
|
|
14f56c77e8 | ||
|
|
44e21084e4 | ||
|
|
b7f1275789 | ||
|
|
b1be05db4d | ||
|
|
09c776fd7e | ||
|
|
e8a24790a5 | ||
|
|
8b3be79266 | ||
|
|
e5f0572c1c | ||
|
|
68df3bc6c3 | ||
|
|
a3da960c26 | ||
|
|
d9490f9840 | ||
|
|
3b24f9d821 | ||
|
|
bf35897e92 | ||
|
|
977705b42d | ||
|
|
459bc40515 | ||
|
|
eb8ca9355c | ||
|
|
31f7b0d5be | ||
|
|
7c54946c4b | ||
|
|
5aaf2c222a | ||
|
|
9424feefce | ||
|
|
a3917ccab6 | ||
|
|
5286a60142 | ||
|
|
d4459de49b | ||
|
|
05dba6668c | ||
|
|
c15a1c6eaa | ||
|
|
fe8c962f73 | ||
|
|
4192cf2403 | ||
|
|
1817f1aa9e | ||
|
|
afacf352ed | ||
|
|
d3ebbba2a1 | ||
|
|
460edf1745 | ||
|
|
dacb19d412 | ||
|
|
fe8158db85 | ||
|
|
a36e4fbcd0 | ||
|
|
9442c7ef07 | ||
|
|
d4a0b59eb1 | ||
|
|
07600949ab | ||
|
|
7efaad1818 | ||
|
|
e32b0d1c25 | ||
|
|
0fdcc29aa2 | ||
|
|
37bedbffc9 | ||
|
|
46505150c4 | ||
|
|
e18c5c90cc | ||
|
|
1f5649d9d2 | ||
|
|
7bbd55a9fd | ||
|
|
92eeccf84e | ||
|
|
7bcc289518 | ||
|
|
507f758c0d | ||
|
|
6afffb7245 | ||
|
|
c62f4ae0b3 | ||
|
|
ee80c614e0 | ||
|
|
fea7af6910 | ||
|
|
c72aa0e97d | ||
|
|
3dd60b5392 | ||
|
|
a357267552 | ||
|
|
1badeb2eef | ||
|
|
0bb8d2a417 | ||
|
|
4a215db83c | ||
|
|
512e55a170 | ||
|
|
1ebe8dc022 | ||
|
|
cafeabdbc3 | ||
|
|
b342b26409 | ||
|
|
938c3a07af | ||
|
|
16288f98f7 | ||
|
|
a932156e85 | ||
|
|
64e9b9fb6d | ||
|
|
7af28550da | ||
|
|
60e2d786dd | ||
|
|
ec08f3430d | ||
|
|
a50aa0fb95 | ||
|
|
cf026e8eaa | ||
|
|
77614d0c4a | ||
|
|
e8d71039d7 | ||
|
|
4164b5bd37 | ||
|
|
3c66ec82b5 | ||
|
|
b796c16877 | ||
|
|
42d34d8f0c | ||
|
|
5fbb37df82 | ||
|
|
52e12d7cbd | ||
|
|
5cf5719e8e | ||
|
|
af3a80143e | ||
|
|
90db27c3fe | ||
|
|
3d584b76f7 | ||
|
|
fbf3dec421 | ||
|
|
1d36683128 | ||
|
|
cefc546c0a | ||
|
|
7aeb51813c | ||
|
|
d6fc29ec79 | ||
|
|
9af182e0c3 | ||
|
|
c7120b997f | ||
|
|
e61571fcec | ||
|
|
9a900b32b2 |
1
.gitignore
vendored
@@ -71,6 +71,7 @@ app/app.iml
|
|||||||
# Art
|
# Art
|
||||||
art/screen*.png
|
art/screen*.png
|
||||||
art/logo_512.png
|
art/logo_512.png
|
||||||
|
art/store_screens/
|
||||||
|
|
||||||
# Dir linux
|
# Dir linux
|
||||||
.directory
|
.directory
|
||||||
|
|||||||
37
CHANGELOG
@@ -1,3 +1,40 @@
|
|||||||
|
KeepassDX (2.5.0.0beta15)
|
||||||
|
* Read only mode
|
||||||
|
* Best group recovery for the navigation fragment
|
||||||
|
* Fix copies in notifications
|
||||||
|
* Fix orientation
|
||||||
|
* Added translations
|
||||||
|
|
||||||
|
KeepassDX (2.5.0.0beta14)
|
||||||
|
* Optimize all the memory with parcelables / fix search
|
||||||
|
|
||||||
|
KeepassDX (2.5.0.0beta13)
|
||||||
|
* Fix memory issue with parcelable (crash in beta12 version)
|
||||||
|
|
||||||
|
KeepassDX (2.5.0.0beta12)
|
||||||
|
* Added the Magikeyboard to fill the forms (settings still in development)
|
||||||
|
* Added move and copy for groups and entries
|
||||||
|
* New navigation in a single screen / new animations between activities
|
||||||
|
* New icons for the material pack / vectorization
|
||||||
|
* New adaptive launcher icon
|
||||||
|
* Icons depends now of setting of list items
|
||||||
|
* Added a setting to disable the open button when no password is identified
|
||||||
|
* Added a setting to disable the education screens
|
||||||
|
* Added a setting to disable the copy of protected custom fields
|
||||||
|
* Fix the fingerprint recognition (WARNING : The keystore is reinit, you must delete the old keys)
|
||||||
|
* Fix small bugs
|
||||||
|
|
||||||
|
KeepassDX (2.5.0.0beta11)
|
||||||
|
* Fix crash in beta10 version
|
||||||
|
|
||||||
|
KeepassDX (2.5.0.0beta10)
|
||||||
|
* Dynamically change Algorithm and Key Derivation Function in settings
|
||||||
|
* Upgrade translations
|
||||||
|
* New red volcano theme, fix classic dark theme
|
||||||
|
* Add Material Icon Pack to the Free version
|
||||||
|
* Update fingerprint state with checkbox
|
||||||
|
* Fix bugs
|
||||||
|
|
||||||
KeepassDX (2.5.0.0beta9)
|
KeepassDX (2.5.0.0beta9)
|
||||||
* Education Screens to learn how to use the app
|
* Education Screens to learn how to use the app
|
||||||
* New designs
|
* New designs
|
||||||
|
|||||||
55
FAQ.md
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
# F.A.Q.
|
||||||
|
|
||||||
|
## Why KeePass DX?
|
||||||
|
|
||||||
|
KeePass DX is an **Android password manager** implemented from Keepass password manager.
|
||||||
|
|
||||||
|
KeePass DX was created to meet the security and usability needs of a KeePass application on Android :
|
||||||
|
|
||||||
|
- To be easy to use with **secure password management and form filling tools**.
|
||||||
|
- To use only tools under **open source license** to guarantee the security of the application (With [open source store](https://f-droid.org/en/) and no closed API).
|
||||||
|
- To be in a **native langage** (java) for weight, security and a better integration of the application.
|
||||||
|
- To respect **Android design, architecture and ergonomic**.
|
||||||
|
|
||||||
|
## What makes KeePass DX stand out from other password managers?
|
||||||
|
|
||||||
|
- We **do not recover your sensitive data** on a private server or a closed cloud, you have control of your passwords.
|
||||||
|
- We respect **KeePass file standards** to maintain compatibility and data porting on different devices (computers and portable devices with different operating system).
|
||||||
|
- The code is **open source**, which implies increased **security**, you can check how the encryption algorithms are implemented.
|
||||||
|
- We remain attentive to **your needs** and we can even integrate the features that you have defined.
|
||||||
|
- We **do not put advertising** even in the free version.
|
||||||
|
|
||||||
|
## How am I sure my passwords are safely stored on the application?
|
||||||
|
|
||||||
|
- We allow users to save and use passwords, keys and digital identities in a secure way by **integrating the last encryption algorithms** and **Android architecture standards**.
|
||||||
|
- You can increase the security of your database by increasing the rounds of encryption keys. *(In Settings -> Database Settings when your database is open)* **Warning**: *Increase the number of rounds sparingly to have a reasonable opening time.*
|
||||||
|
|
||||||
|
## Can I store my data on a cloud storage?
|
||||||
|
|
||||||
|
**Yes** this is possible. Otherwise, we **recommend using cloud with personal server and open source license**, like [NextCloud](https://f-droid.org/en/packages/com.nextcloud.client/) to be sure how your databases are stored.
|
||||||
|
|
||||||
|
## Can I recover my passwords on another device if I loose my main device?
|
||||||
|
|
||||||
|
**Yes** you can, but you **must first save the .kdb or .kdbx file from your database to an external storage** *(like a hardrive or a cloud)*.
|
||||||
|
We recommend you save your data after each modification so incase you loose your android device you could retrieve the data and import it into the new KeePass DX installed on the new android device.
|
||||||
|
|
||||||
|
## Why are updates not available at the same time on all stores?
|
||||||
|
|
||||||
|
- **PlayStore** only needs an APK generated and manually signed to be available on the store, it usually takes **20 minutes** to be available because it is deployed with fastlane. But the management of the APK and its data by the google servers is obscure.
|
||||||
|
- **F-Droid**, to **ensure that the code is open source**, checks the sources directly on git repository (by checking the presence of new tags) and builds itself the APK that the server signs during the compilation of code and dependencies. Updating the project will take **1-10 days** for F-Droid to analyze all available repositories, build sources and deploy the generated APK. So F-Droid is slower for deployment but it is run by **volunteers** and guaranteed a **clean APK**. :)
|
||||||
|
|
||||||
|
## Why not an online version?
|
||||||
|
|
||||||
|
The offline and online client concepts only exists with Keepass2Android because the file access network tools are directly integrated into the code of the main application. Which is a very dubious choice knowing that **it is not normally the purpose of a password management application to take care of external file synchronization on clouds** (which can be under closed licensed and recover your data base), it is rather the purpose of the [file management application](https://developer.android.com/guide/topics/providers/document-provider).
|
||||||
|
|
||||||
|
## Can I open my database easily other than with a password?
|
||||||
|
|
||||||
|
**Yes**, we have integrated a secure openning option of fingerprint for android devices that support this feature, so no one can access the application without scanning his/her fingerprint or fill a master key.
|
||||||
|
|
||||||
|
## Can I open my database without my master key (master password and/or key file)?
|
||||||
|
|
||||||
|
**No**, you can not open a database file without the master password (and / or) the associated key file. Be sure to remember your master password and save the key file in a safe place.
|
||||||
|
|
||||||
|
## Can I suggest features and report bugs for the application?
|
||||||
|
**Yes**, we welcome this you could go ahead and do that on our github:
|
||||||
|
https://github.com/Kunzisoft/KeePassDX
|
||||||
47
ReadMe.md
@@ -1,34 +1,32 @@
|
|||||||
# Android Keepass DX
|
# Android Keepass DX
|
||||||
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/art/icon.png"> Keepass DX is a multi-format KeePass manager for Android devices. The application allows to create keys and passwords in a secure way by integrating with the Android design standards.
|
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/art/icon.png"> Keepass DX is a **multi-format KeePass manager for Android devices**. The application allows to create keys and passwords in a secure way by integrating with the Android design standards.
|
||||||
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/art/screen.jpg" width="220">
|
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/art/screen.jpg" width="220">
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
|
|
||||||
* Create database files / entries and groups
|
* Create database files / entries and groups
|
||||||
* Support for .kdb and .kdbx files (version 1 to 4) with AES - Twofish - ChaCha20 - Argon2 algorithm
|
* Support for **.kdb** and **.kdbx** files (version 1 to 4) with AES - Twofish - ChaCha20 - Argon2 algorithm
|
||||||
* Compatible with the majority of alternative programs (KeePass, KeePassX, KeePass XC...)
|
* **Compatible** with the majority of alternative programs (KeePass, KeePassX, KeePass XC...)
|
||||||
* Allows fast copy of fields and opening of URI / URL
|
* Allows **fast copy** of fields and opening of URI / URL
|
||||||
* Fingerprint for fast unlocking
|
* **Fingerprint** for fast unlocking
|
||||||
* Material design with themes
|
* Material design with **themes**
|
||||||
* AutoFill and Integration
|
* **AutoFill** and Integration
|
||||||
* Precise management of settings
|
* Field filling **keyboard**
|
||||||
|
* Precise management of **settings**
|
||||||
|
|
||||||
Keepass DX is opensource and ad-free.
|
Keepass DX is **open source** and **ad-free**.
|
||||||
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/fastlane/metadata/android/en-US/images/phoneScreenshots/screen1.jpg" width="220">
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/fastlane/metadata/android/en-US/images/phoneScreenshots/screen2.jpg" width="220">
|
|
||||||
|
|
||||||
## What is KeePass DX?
|
## What is KeePass DX?
|
||||||
|
|
||||||
Today you need to remember many passwords. You need a password for your e-mail account, your website's FTP password, online passwords (like website member account), etc. etc. etc. The list is endless. Also, you should use different passwords for each account. Because if you use only one password everywhere and someone gets this password you have a problem... A serious problem. The thief would have access to your e-mail account, website, etc. Unimaginable.
|
Today you need to remember many passwords. You need a password for your e-mail account, your website's FTP password, online passwords (like website member account), etc. etc. etc. The list is endless. Also, you **should use different passwords for each account**. Because if you use only one password everywhere and someone gets this password you have a problem... A serious problem. The thief would have access to your e-mail account, website, etc. Unimaginable.
|
||||||
|
|
||||||
KeePass DX is a free open source password manager for Android, which helps you to manage your passwords in a secure way. You can put all your passwords in one database, which is locked with one master key or a key file. So you only have to remember one single master password or select the key file to unlock the whole database. The databases are encrypted using the best and most secure encryption algorithms currently known.
|
KeePass DX is a **free open source password manager for Android**, which helps you to **manage your passwords in a secure way**. You can put all your passwords in one database, which is locked with one **master key** or a **key file**. So you **only have to remember one single master password or select the key file** to unlock the whole database. The databases are encrypted using the best and **most secure encryption algorithms** currently known.
|
||||||
|
|
||||||
## Is it really free?
|
## Is it really free?
|
||||||
|
|
||||||
Yes, KeePass DX is under free license (OSI certified) and without advertising. You can have a look at its full source and check whether the encryption algorithms are implemented correctly.
|
Yes, KeePass DX is under **free license (OSI certified)** and **without advertising**. You can have a look at its full source and check whether the encryption algorithms are implemented correctly.
|
||||||
|
|
||||||
*Note : If you access the application from a store, visual features may not be available to incentivize the contribution to the work of open source projects. These optional visuals are accessible after a donation (and a small congratulation message :) or the purchase of an extended version, but do not worry, the main features remain completely free.*
|
*Note : If you access the application from a store, visual features may not be available to incentivize the contribution to the work of open source projects. These optional visuals are accessible after a donation (and a small congratulation message :) or the purchase of an extended version, but do not worry, the main features remain completely free.*
|
||||||
|
|
||||||
@@ -36,16 +34,15 @@ Yes, KeePass DX is under free license (OSI certified) and without advertising. Y
|
|||||||
|
|
||||||
You can contribute in different ways to help us on our work.
|
You can contribute in different ways to help us on our work.
|
||||||
|
|
||||||
* Add features by a **pull request**.
|
* Add features by a **[pull request](https://help.github.com/articles/about-pull-requests/)**.
|
||||||
* Help to **translate** into your language
|
* Help to **[translate](https://hosted.weblate.org/projects/keepass-dx/strings/)** into your language (By using [Weblate](https://hosted.weblate.org/projects/keepass-dx/) or with a manual [pull request](https://help.github.com/articles/about-pull-requests/))
|
||||||
* **[Donate](https://www.kunzisoft.com/donation)** 人◕ ‿‿ ◕人Y for a better service and a quick development of your features.
|
* **[Donate](https://www.kunzisoft.com/donation)** 人◕ ‿‿ ◕人Y for a better service and a quick development of your features.
|
||||||
* Buy the **[Pro version](https://play.google.com/store/apps/details?id=com.kunzisoft.keepass.pro)** of KeePass DX
|
* Buy the **[Pro version](https://play.google.com/store/apps/details?id=com.kunzisoft.keepass.pro)** of KeePass DX
|
||||||
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/fastlane/metadata/android/en-US/images/phoneScreenshots/screen4.jpg" width="220">
|
|
||||||
<img src="https://raw.githubusercontent.com/Kunzisoft/KeePassDX/master/fastlane/metadata/android/en-US/images/phoneScreenshots/screen5.jpg" width="220">
|
|
||||||
|
|
||||||
## Download
|
## Download
|
||||||
|
|
||||||
|
*We recommend the installation from [F-Droid](https://f-droid.org/en/packages/com.kunzisoft.keepass.libre/) which verifies that all libraries and application code are open source.*
|
||||||
|
|
||||||
[<img src="https://f-droid.org/badge/get-it-on.png"
|
[<img src="https://f-droid.org/badge/get-it-on.png"
|
||||||
alt="Get it on F-Droid"
|
alt="Get it on F-Droid"
|
||||||
height="80">](https://f-droid.org/en/packages/com.kunzisoft.keepass.libre/)
|
height="80">](https://f-droid.org/en/packages/com.kunzisoft.keepass.libre/)
|
||||||
@@ -54,15 +51,19 @@ You can contribute in different ways to help us on our work.
|
|||||||
alt="Get it on Google Play"
|
alt="Get it on Google Play"
|
||||||
height="80">](https://play.google.com/store/apps/details?id=com.kunzisoft.keepass.free)
|
height="80">](https://play.google.com/store/apps/details?id=com.kunzisoft.keepass.free)
|
||||||
|
|
||||||
|
## F.A.Q.
|
||||||
|
|
||||||
|
Other questions? You can read the [F.A.Q.](https://github.com/Kunzisoft/KeePassDX/blob/master/FAQ.md)
|
||||||
|
|
||||||
## Other devices
|
## Other devices
|
||||||
|
|
||||||
- [KeePass XC](https://keepassxc.org/) (https://keepassxc.org/) works with **GNU/Linux**, **Mac** and **Windows**, is updated regulary and under the terms of the GNU General Public License. It's the recommended version for computers.
|
- [KeePass XC](https://keepassxc.org/) (https://keepassxc.org/) works with **GNU/Linux**, **Mac** and **Windows**, is updated regularly and under the terms of the GNU General Public License. This is the recommended version for computers.
|
||||||
|
|
||||||
- [KeePass](https://keepass.info/) (https://keepass.info/) is the historical project, with good technical documentation for standardized database files but only running on **Windows**.
|
- [KeePass](https://keepass.info/) (https://keepass.info/) is the historical project, with good technical documentation for standardized database files but only running on **Windows**.
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
Copyright (c) 2017 Jeremy Jamet / Kunzisoft.
|
Copyright (c) 2017 Jeremy Jamet / [Kunzisoft](https://www.kunzisoft.com).
|
||||||
|
|
||||||
This file is part of KeePass DX.
|
This file is part of KeePass DX.
|
||||||
|
|
||||||
@@ -79,4 +80,4 @@ You can contribute in different ways to help us on our work.
|
|||||||
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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
This project is a fork of [KeepassDroid](https://github.com/bpellin/keepassdroid) by bpellin.
|
*This project is a fork of [KeepassDroid](https://github.com/bpellin/keepassdroid) by bpellin.*
|
||||||
|
|||||||
@@ -8,18 +8,13 @@ android {
|
|||||||
applicationId "com.kunzisoft.keepass"
|
applicationId "com.kunzisoft.keepass"
|
||||||
minSdkVersion 14
|
minSdkVersion 14
|
||||||
targetSdkVersion 27
|
targetSdkVersion 27
|
||||||
versionCode = 9
|
versionCode = 15
|
||||||
versionName = "2.5.0.0beta9"
|
versionName = "2.5.0.0beta15"
|
||||||
multiDexEnabled true
|
multiDexEnabled true
|
||||||
|
|
||||||
testApplicationId = "com.kunzisoft.keepass.tests"
|
testApplicationId = "com.kunzisoft.keepass.tests"
|
||||||
testInstrumentationRunner = "android.test.InstrumentationTestRunner"
|
testInstrumentationRunner = "android.test.InstrumentationTestRunner"
|
||||||
|
|
||||||
ndk {
|
|
||||||
abiFilters 'x86', 'x86_64', 'armeabi', 'armeabi-v7a',
|
|
||||||
'arm64-v8a', 'mips', 'mips64'
|
|
||||||
}
|
|
||||||
|
|
||||||
buildConfigField "String[]", "ICON_PACKS", "{\"classic\",\"material\"}"
|
buildConfigField "String[]", "ICON_PACKS", "{\"classic\",\"material\"}"
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -47,7 +42,7 @@ android {
|
|||||||
versionNameSuffix "-libre"
|
versionNameSuffix "-libre"
|
||||||
buildConfigField "boolean", "FULL_VERSION", "true"
|
buildConfigField "boolean", "FULL_VERSION", "true"
|
||||||
buildConfigField "boolean", "CLOSED_STORE", "false"
|
buildConfigField "boolean", "CLOSED_STORE", "false"
|
||||||
buildConfigField "String[]", "STYLES_DISABLED", "{\"KeepassDXStyle_Dark\",\"KeepassDXStyle_Purple\"}"
|
buildConfigField "String[]", "STYLES_DISABLED", "{\"KeepassDXStyle_Dark\",\"KeepassDXStyle_Red\",\"KeepassDXStyle_Purple\"}"
|
||||||
buildConfigField "String[]", "ICON_PACKS_DISABLED", "{}"
|
buildConfigField "String[]", "ICON_PACKS_DISABLED", "{}"
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -64,8 +59,8 @@ android {
|
|||||||
versionNameSuffix "-free"
|
versionNameSuffix "-free"
|
||||||
buildConfigField "boolean", "FULL_VERSION", "false"
|
buildConfigField "boolean", "FULL_VERSION", "false"
|
||||||
buildConfigField "boolean", "CLOSED_STORE", "true"
|
buildConfigField "boolean", "CLOSED_STORE", "true"
|
||||||
buildConfigField "String[]", "STYLES_DISABLED", "{\"KeepassDXStyle_Dark\",\"KeepassDXStyle_Blue\",\"KeepassDXStyle_Purple\"}"
|
buildConfigField "String[]", "STYLES_DISABLED", "{\"KeepassDXStyle_Dark\",\"KeepassDXStyle_Blue\",\"KeepassDXStyle_Red\",\"KeepassDXStyle_Purple\"}"
|
||||||
buildConfigField "String[]", "ICON_PACKS_DISABLED", "{\"material\"}"
|
buildConfigField "String[]", "ICON_PACKS_DISABLED", "{}"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -110,9 +105,11 @@ dependencies {
|
|||||||
implementation 'commons-collections:commons-collections:3.2.1'
|
implementation 'commons-collections:commons-collections:3.2.1'
|
||||||
// Base64
|
// Base64
|
||||||
implementation 'biz.source_code:base64coder:2010-12-19'
|
implementation 'biz.source_code:base64coder:2010-12-19'
|
||||||
implementation 'com.google.code.gson:gson:2.8.1'
|
implementation 'com.google.code.gson:gson:2.8.4'
|
||||||
implementation 'com.google.guava:guava:23.0-android'
|
implementation 'com.google.guava:guava:23.0-android'
|
||||||
// Icon pack, classic for all, material for libre and pro
|
// Icon pack, classic for all, material for libre and pro
|
||||||
implementation project(path: ':icon-pack-classic')
|
implementation project(path: ':icon-pack-classic')
|
||||||
implementation project(path: ':icon-pack-material')
|
implementation project(path: ':icon-pack-material')
|
||||||
|
implementation project(path: ':magikeyboard')
|
||||||
|
implementation project(path: ':keepass-model')
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,8 +19,6 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.tests.database;
|
package com.kunzisoft.keepass.tests.database;
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.test.AndroidTestCase;
|
import android.test.AndroidTestCase;
|
||||||
|
|
||||||
@@ -30,9 +28,11 @@ import com.kunzisoft.keepass.database.PwDatabaseV3;
|
|||||||
import com.kunzisoft.keepass.database.PwEntry;
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
import com.kunzisoft.keepass.database.PwEntryV3;
|
import com.kunzisoft.keepass.database.PwEntryV3;
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
import com.kunzisoft.keepass.database.edit.DeleteGroup;
|
import com.kunzisoft.keepass.database.action.node.DeleteGroupRunnable;
|
||||||
import com.kunzisoft.keepass.search.SearchDbHelper;
|
import com.kunzisoft.keepass.search.SearchDbHelper;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
public class DeleteEntry extends AndroidTestCase {
|
public class DeleteEntry extends AndroidTestCase {
|
||||||
private static final String GROUP1_NAME = "Group1";
|
private static final String GROUP1_NAME = "Group1";
|
||||||
private static final String ENTRY1_NAME = "Test1";
|
private static final String ENTRY1_NAME = "Test1";
|
||||||
@@ -60,7 +60,7 @@ public class DeleteEntry extends AndroidTestCase {
|
|||||||
assertNotNull("Could not find group1", group1);
|
assertNotNull("Could not find group1", group1);
|
||||||
|
|
||||||
// Delete the group
|
// Delete the group
|
||||||
DeleteGroup task = new DeleteGroup(null, db, group1, null, true);
|
DeleteGroupRunnable task = new DeleteGroupRunnable(null, db, group1, null, true);
|
||||||
task.run();
|
task.run();
|
||||||
|
|
||||||
// Verify the entries were deleted
|
// Verify the entries were deleted
|
||||||
|
|||||||
50
app/src/free/res/drawable-v24/ic_launcher_foreground.xml
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:viewportWidth="108"
|
||||||
|
android:viewportHeight="108"
|
||||||
|
android:width="108dp"
|
||||||
|
android:height="108dp">
|
||||||
|
<group
|
||||||
|
android:translateY="-332">
|
||||||
|
<group
|
||||||
|
android:translateY="332">
|
||||||
|
<path
|
||||||
|
android:pathData="M65.728516 32.791016L58.052734 35.904297 56.173828 48.380859 35.306641 69.267578 35.238281 73.759766 69.478516 108 108 108 108 70.810547 73.09375 35.904297 65.728516 32.791016Z"
|
||||||
|
android:fillColor="@color/long_shadow"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeLineCap="round"
|
||||||
|
android:strokeMiterLimit="4" />
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M88.76953 339.91602L4.1718754 424.59766 4.0000004 436 15.400391 435.82813 27.240234 424 40 424l0 -12 12 0 0 -12.73438 34.01172 -33.97656A8 8 0 0 1 84 360a8 8 0 0 1 8 -8 8 8 0 0 1 5.296882 2.01367l2.787098 -2.7832 -11.31445 -11.31445z"
|
||||||
|
android:fillColor="#eaeaea"
|
||||||
|
android:strokeWidth="1"
|
||||||
|
android:strokeColor="#58000000" />
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M4.0000004 340L4.1718754 351.40137 88.59863 435.82812 100 436 99.828122 424.59863 15.401367 340.17188Z"
|
||||||
|
android:fillColor="#81c784" />
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M81.39454 332.00195a27 27 0 0 0 -19.48634 7.90625 27 27 0 0 0 0 38.1836 27 27 0 0 0 38.1836 0 27 27 0 0 0 0 -38.1836 27 27 0 0 0 -18.69726 -7.90625zM92 352a8 8 0 0 1 8 8 8 8 0 0 1 -8 8 8 8 0 0 1 -8 -8 8 8 0 0 1 8 -8z"
|
||||||
|
android:fillColor="#eaeaea"
|
||||||
|
android:strokeWidth="1"
|
||||||
|
android:strokeColor="#58000000" />
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
</vector>
|
||||||
5
app/src/free/res/mipmap-anydpi-v26/ic_launcher.xml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@color/green" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
||||||
5
app/src/free/res/mipmap-anydpi-v26/ic_launcher_round.xml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@color/green" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
||||||
|
Before Width: | Height: | Size: 4.6 KiB After Width: | Height: | Size: 5.1 KiB |
|
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 2.1 KiB |
|
Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 6.4 KiB After Width: | Height: | Size: 7.3 KiB |
|
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 17 KiB |
50
app/src/libre/res/drawable-v24/ic_launcher_foreground.xml
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:viewportWidth="108"
|
||||||
|
android:viewportHeight="108"
|
||||||
|
android:width="108dp"
|
||||||
|
android:height="108dp">
|
||||||
|
<group
|
||||||
|
android:translateY="-332">
|
||||||
|
<group
|
||||||
|
android:translateY="332">
|
||||||
|
<path
|
||||||
|
android:pathData="M65.728516 32.791016L58.052734 35.904297 56.173828 48.380859 35.306641 69.267578 35.238281 73.759766 69.478516 108 108 108 108 70.810547 73.09375 35.904297 65.728516 32.791016Z"
|
||||||
|
android:fillColor="@color/long_shadow"
|
||||||
|
android:strokeLineJoin="round"
|
||||||
|
android:strokeLineCap="round"
|
||||||
|
android:strokeMiterLimit="4" />
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M88.76953 339.91602L4.1718754 424.59766 4.0000004 436 15.400391 435.82813 27.240234 424 40 424l0 -12 12 0 0 -12.73438 34.01172 -33.97656A8 8 0 0 1 84 360a8 8 0 0 1 8 -8 8 8 0 0 1 5.296882 2.01367l2.787098 -2.7832 -11.31445 -11.31445z"
|
||||||
|
android:fillColor="#eaeaea"
|
||||||
|
android:strokeWidth="1"
|
||||||
|
android:strokeColor="#58000000"/>
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M4.0000004 340L4.1718754 351.40137 88.59863 435.82812 100 436 99.828122 424.59863 15.401367 340.17188Z"
|
||||||
|
android:fillColor="#64b5f6" />
|
||||||
|
</group>
|
||||||
|
<group
|
||||||
|
android:scaleX="0.3939503"
|
||||||
|
android:scaleY="0.3939503"
|
||||||
|
android:translateX="33.66343"
|
||||||
|
android:translateY="233.998">
|
||||||
|
<path
|
||||||
|
android:pathData="M81.39454 332.00195a27 27 0 0 0 -19.48634 7.90625 27 27 0 0 0 0 38.1836 27 27 0 0 0 38.1836 0 27 27 0 0 0 0 -38.1836 27 27 0 0 0 -18.69726 -7.90625zM92 352a8 8 0 0 1 8 8 8 8 0 0 1 -8 8 8 8 0 0 1 -8 -8 8 8 0 0 1 8 -8z"
|
||||||
|
android:fillColor="#eaeaea"
|
||||||
|
android:strokeWidth="1"
|
||||||
|
android:strokeColor="#58000000" />
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
</vector>
|
||||||
5
app/src/libre/res/mipmap-anydpi-v26/ic_launcher.xml
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@color/green" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@color/green" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
||||||
@@ -31,7 +31,7 @@
|
|||||||
android:name="com.kunzisoft.keepass.fileselect.FileSelectActivity"
|
android:name="com.kunzisoft.keepass.fileselect.FileSelectActivity"
|
||||||
android:theme="@style/KeepassDXStyle.SplashScreen"
|
android:theme="@style/KeepassDXStyle.SplashScreen"
|
||||||
android:label="@string/app_name"
|
android:label="@string/app_name"
|
||||||
android:launchMode="singleTask"
|
android:launchMode="singleTop"
|
||||||
android:configChanges="orientation|keyboardHidden"
|
android:configChanges="orientation|keyboardHidden"
|
||||||
android:windowSoftInputMode="stateHidden" >
|
android:windowSoftInputMode="stateHidden" >
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
@@ -132,13 +132,15 @@
|
|||||||
<activity android:name="com.kunzisoft.keepass.settings.SettingsActivity" />
|
<activity android:name="com.kunzisoft.keepass.settings.SettingsActivity" />
|
||||||
<activity android:name="com.kunzisoft.keepass.autofill.AutoFillAuthActivity"
|
<activity android:name="com.kunzisoft.keepass.autofill.AutoFillAuthActivity"
|
||||||
android:configChanges="orientation|keyboardHidden" />
|
android:configChanges="orientation|keyboardHidden" />
|
||||||
|
<activity android:name="com.kunzisoft.keepass.selection.EntrySelectionAuthActivity"
|
||||||
|
android:configChanges="orientation|keyboardHidden" />
|
||||||
<activity android:name="com.kunzisoft.keepass.settings.SettingsAutofillActivity" />
|
<activity android:name="com.kunzisoft.keepass.settings.SettingsAutofillActivity" />
|
||||||
|
|
||||||
<service android:name="com.kunzisoft.keepass.timeout.TimeoutService" />
|
|
||||||
<service
|
<service
|
||||||
android:name="com.kunzisoft.keepass.notifications.NotificationCopyingService"
|
android:name="com.kunzisoft.keepass.notifications.NotificationCopyingService"
|
||||||
android:enabled="true"
|
android:enabled="true"
|
||||||
android:exported="false" />
|
android:exported="false" />
|
||||||
|
<!-- Receiver for Autofill -->
|
||||||
<service
|
<service
|
||||||
android:name="com.kunzisoft.keepass.autofill.KeeAutofillService"
|
android:name="com.kunzisoft.keepass.autofill.KeeAutofillService"
|
||||||
android:label="@string/autofill_service_name"
|
android:label="@string/autofill_service_name"
|
||||||
@@ -146,11 +148,11 @@
|
|||||||
<meta-data
|
<meta-data
|
||||||
android:name="android.autofill"
|
android:name="android.autofill"
|
||||||
android:resource="@xml/dataset_service" />
|
android:resource="@xml/dataset_service" />
|
||||||
|
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.service.autofill.AutofillService" />
|
<action android:name="android.service.autofill.AutofillService" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
</service>
|
</service>
|
||||||
|
|
||||||
<meta-data android:name="com.sec.android.support.multiwindow" android:value="true" />
|
<meta-data android:name="com.sec.android.support.multiwindow" android:value="true" />
|
||||||
</application>
|
</application>
|
||||||
</manifest>
|
</manifest>
|
||||||
|
|||||||
@@ -47,6 +47,8 @@ import com.kunzisoft.keepass.database.PwDatabase;
|
|||||||
import com.kunzisoft.keepass.database.PwEntry;
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedString;
|
import com.kunzisoft.keepass.database.security.ProtectedString;
|
||||||
import com.kunzisoft.keepass.icons.IconPackChooser;
|
import com.kunzisoft.keepass.icons.IconPackChooser;
|
||||||
|
import com.kunzisoft.keepass.lock.LockingActivity;
|
||||||
|
import com.kunzisoft.keepass.lock.LockingHideActivity;
|
||||||
import com.kunzisoft.keepass.notifications.NotificationCopyingService;
|
import com.kunzisoft.keepass.notifications.NotificationCopyingService;
|
||||||
import com.kunzisoft.keepass.notifications.NotificationField;
|
import com.kunzisoft.keepass.notifications.NotificationField;
|
||||||
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
||||||
@@ -76,15 +78,15 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
|
|
||||||
protected PwEntry mEntry;
|
protected PwEntry mEntry;
|
||||||
private boolean mShowPassword;
|
private boolean mShowPassword;
|
||||||
protected boolean readOnly = false;
|
|
||||||
|
|
||||||
private ClipboardHelper clipboardHelper;
|
private ClipboardHelper clipboardHelper;
|
||||||
private boolean firstLaunchOfActivity;
|
private boolean firstLaunchOfActivity;
|
||||||
|
|
||||||
public static void launch(Activity act, PwEntry pw) {
|
public static void launch(Activity act, PwEntry pw, boolean readOnly) {
|
||||||
if (LockingActivity.checkTimeIsAllowedOrFinish(act)) {
|
if (LockingActivity.checkTimeIsAllowedOrFinish(act)) {
|
||||||
Intent intent = new Intent(act, EntryActivity.class);
|
Intent intent = new Intent(act, EntryActivity.class);
|
||||||
intent.putExtra(KEY_ENTRY, Types.UUIDtoBytes(pw.getUUID()));
|
intent.putExtra(KEY_ENTRY, Types.UUIDtoBytes(pw.getUUID()));
|
||||||
|
ReadOnlyHelper.putReadOnlyInIntent(intent, readOnly);
|
||||||
act.startActivityForResult(intent, EntryEditActivity.ADD_OR_UPDATE_ENTRY_REQUEST_CODE);
|
act.startActivityForResult(intent, EntryEditActivity.ADD_OR_UPDATE_ENTRY_REQUEST_CODE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -108,7 +110,7 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
finish();
|
finish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
readOnly = db.isReadOnly();
|
readOnly = db.isReadOnly() || readOnly;
|
||||||
|
|
||||||
mShowPassword = !PreferencesUtil.isPasswordMask(this);
|
mShowPassword = !PreferencesUtil.isPasswordMask(this);
|
||||||
|
|
||||||
@@ -150,12 +152,28 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
// Start to manage field reference to copy a value from ref
|
// Start to manage field reference to copy a value from ref
|
||||||
mEntry.startToManageFieldReferences(App.getDB().getPwDatabase());
|
mEntry.startToManageFieldReferences(App.getDB().getPwDatabase());
|
||||||
|
|
||||||
|
boolean containsUsernameToCopy =
|
||||||
|
mEntry.getUsername().length() > 0;
|
||||||
|
boolean containsPasswordToCopy =
|
||||||
|
(mEntry.getPassword().length() > 0
|
||||||
|
&& PreferencesUtil.allowCopyPasswordAndProtectedFields(this));
|
||||||
|
boolean containsExtraFieldToCopy =
|
||||||
|
(mEntry.allowExtraFields()
|
||||||
|
&& ((mEntry.containsCustomFields()
|
||||||
|
&& mEntry.containsCustomFieldsNotProtected())
|
||||||
|
|| (mEntry.containsCustomFields()
|
||||||
|
&& mEntry.containsCustomFieldsProtected()
|
||||||
|
&& PreferencesUtil.allowCopyPasswordAndProtectedFields(this))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
// If notifications enabled in settings
|
// If notifications enabled in settings
|
||||||
// Don't if application timeout
|
// Don't if application timeout
|
||||||
if (firstLaunchOfActivity && !App.isShutdown() && isClipboardNotificationsEnable(getApplicationContext())) {
|
if (firstLaunchOfActivity && !App.isShutdown() && isClipboardNotificationsEnable(getApplicationContext())) {
|
||||||
if (mEntry.getUsername().length() > 0
|
if (containsUsernameToCopy
|
||||||
|| (mEntry.getPassword().length() > 0 && PreferencesUtil.allowCopyPassword(this))
|
|| containsPasswordToCopy
|
||||||
|| mEntry.containsCustomFields()) {
|
|| containsExtraFieldToCopy
|
||||||
|
) {
|
||||||
// username already copied, waiting for user's action before copy password.
|
// username already copied, waiting for user's action before copy password.
|
||||||
Intent intent = new Intent(this, NotificationCopyingService.class);
|
Intent intent = new Intent(this, NotificationCopyingService.class);
|
||||||
intent.setAction(NotificationCopyingService.ACTION_NEW_NOTIFICATION);
|
intent.setAction(NotificationCopyingService.ACTION_NEW_NOTIFICATION);
|
||||||
@@ -164,35 +182,37 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
// Construct notification fields
|
// Construct notification fields
|
||||||
ArrayList<NotificationField> notificationFields = new ArrayList<>();
|
ArrayList<NotificationField> notificationFields = new ArrayList<>();
|
||||||
// Add username if exists to notifications
|
// Add username if exists to notifications
|
||||||
if (mEntry.getUsername().length() > 0)
|
if (containsUsernameToCopy)
|
||||||
notificationFields.add(
|
notificationFields.add(
|
||||||
new NotificationField(
|
new NotificationField(
|
||||||
NotificationField.NotificationFieldId.USERNAME,
|
NotificationField.NotificationFieldId.USERNAME,
|
||||||
mEntry.getUsername(),
|
mEntry.getUsername(),
|
||||||
getResources()));
|
getResources()));
|
||||||
// Add password to notifications
|
// Add password to notifications
|
||||||
if (PreferencesUtil.allowCopyPassword(this)) {
|
if (containsPasswordToCopy) {
|
||||||
if (mEntry.getPassword().length() > 0)
|
notificationFields.add(
|
||||||
notificationFields.add(
|
new NotificationField(
|
||||||
new NotificationField(
|
NotificationField.NotificationFieldId.PASSWORD,
|
||||||
NotificationField.NotificationFieldId.PASSWORD,
|
mEntry.getPassword(),
|
||||||
mEntry.getPassword(),
|
getResources()));
|
||||||
getResources()));
|
|
||||||
}
|
}
|
||||||
// Add extra fields
|
// Add extra fields
|
||||||
if (mEntry.allowExtraFields()) {
|
if (containsExtraFieldToCopy) {
|
||||||
try {
|
try {
|
||||||
mEntry.getFields().doActionToAllCustomProtectedField(new ExtraFields.ActionProtected() {
|
mEntry.getFields().doActionToAllCustomProtectedField(new ExtraFields.ActionProtected() {
|
||||||
private int anonymousFieldNumber = 0;
|
private int anonymousFieldNumber = 0;
|
||||||
@Override
|
@Override
|
||||||
public void doAction(String key, ProtectedString value) {
|
public void doAction(String key, ProtectedString value) {
|
||||||
notificationFields.add(
|
//If value is not protected or allowed
|
||||||
new NotificationField(
|
if (!value.isProtected() || PreferencesUtil.allowCopyPasswordAndProtectedFields(EntryActivity.this)) {
|
||||||
NotificationField.NotificationFieldId.getAnonymousFieldId()[anonymousFieldNumber],
|
notificationFields.add(
|
||||||
value.toString(),
|
new NotificationField(
|
||||||
key,
|
NotificationField.NotificationFieldId.getAnonymousFieldId()[anonymousFieldNumber],
|
||||||
getResources()));
|
value.toString(),
|
||||||
anonymousFieldNumber++;
|
key,
|
||||||
|
getResources()));
|
||||||
|
anonymousFieldNumber++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} catch (ArrayIndexOutOfBoundsException e) {
|
} catch (ArrayIndexOutOfBoundsException e) {
|
||||||
@@ -215,66 +235,70 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
* Displays the explanation for copying a field and editing an entry
|
* Displays the explanation for copying a field and editing an entry
|
||||||
*/
|
*/
|
||||||
private void checkAndPerformedEducation(Menu menu) {
|
private void checkAndPerformedEducation(Menu menu) {
|
||||||
|
if (PreferencesUtil.isEducationScreensEnabled(this)) {
|
||||||
|
|
||||||
if (entryContentsView != null && entryContentsView.isUserNamePresent()
|
if (entryContentsView != null && entryContentsView.isUserNamePresent()
|
||||||
&& !PreferencesUtil.isEducationCopyUsernamePerformed(this)) {
|
&& !PreferencesUtil.isEducationCopyUsernamePerformed(this)) {
|
||||||
TapTargetView.showFor(this,
|
|
||||||
TapTarget.forView(findViewById(R.id.entry_user_name_action_image),
|
|
||||||
getString(R.string.education_field_copy_title),
|
|
||||||
getString(R.string.education_field_copy_summary))
|
|
||||||
.tintTarget(false)
|
|
||||||
.cancelable(true),
|
|
||||||
new TapTargetView.Listener() {
|
|
||||||
@Override
|
|
||||||
public void onTargetClick(TapTargetView view) {
|
|
||||||
super.onTargetClick(view);
|
|
||||||
clipboardHelper.timeoutCopyToClipboard(mEntry.getUsername(),
|
|
||||||
getString(R.string.copy_field, getString(R.string.entry_user_name)));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onOuterCircleClick(TapTargetView view) {
|
|
||||||
super.onOuterCircleClick(view);
|
|
||||||
view.dismiss(false);
|
|
||||||
// Launch autofill settings
|
|
||||||
startActivity(new Intent(EntryActivity.this, SettingsAutofillActivity.class));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
PreferencesUtil.saveEducationPreference(this,
|
|
||||||
R.string.education_copy_username_key);
|
|
||||||
|
|
||||||
} else if (!PreferencesUtil.isEducationEntryEditPerformed(this)) {
|
|
||||||
|
|
||||||
try {
|
|
||||||
TapTargetView.showFor(this,
|
TapTargetView.showFor(this,
|
||||||
TapTarget.forToolbarMenuItem(toolbar, R.id.menu_edit,
|
TapTarget.forView(findViewById(R.id.entry_user_name_action_image),
|
||||||
getString(R.string.education_entry_edit_title),
|
getString(R.string.education_field_copy_title),
|
||||||
getString(R.string.education_entry_edit_summary))
|
getString(R.string.education_field_copy_summary))
|
||||||
.tintTarget(true)
|
.textColorInt(Color.WHITE)
|
||||||
|
.tintTarget(false)
|
||||||
.cancelable(true),
|
.cancelable(true),
|
||||||
new TapTargetView.Listener() {
|
new TapTargetView.Listener() {
|
||||||
@Override
|
@Override
|
||||||
public void onTargetClick(TapTargetView view) {
|
public void onTargetClick(TapTargetView view) {
|
||||||
super.onTargetClick(view);
|
super.onTargetClick(view);
|
||||||
MenuItem editItem = menu.findItem(R.id.menu_edit);
|
clipboardHelper.timeoutCopyToClipboard(mEntry.getUsername(),
|
||||||
onOptionsItemSelected(editItem);
|
getString(R.string.copy_field, getString(R.string.entry_user_name)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onOuterCircleClick(TapTargetView view) {
|
public void onOuterCircleClick(TapTargetView view) {
|
||||||
super.onOuterCircleClick(view);
|
super.onOuterCircleClick(view);
|
||||||
view.dismiss(false);
|
view.dismiss(false);
|
||||||
// Open Keepass doc to create field references
|
// Launch autofill settings
|
||||||
Intent browserIntent = new Intent(Intent.ACTION_VIEW,
|
startActivity(new Intent(EntryActivity.this, SettingsAutofillActivity.class));
|
||||||
Uri.parse(getString(R.string.field_references_url)));
|
|
||||||
startActivity(browserIntent);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
PreferencesUtil.saveEducationPreference(this,
|
PreferencesUtil.saveEducationPreference(this,
|
||||||
R.string.education_entry_edit_key);
|
R.string.education_copy_username_key);
|
||||||
} catch (Exception e) {
|
|
||||||
// If icon not visible
|
} else if (!PreferencesUtil.isEducationEntryEditPerformed(this)) {
|
||||||
Log.w(TAG, "Can't performed education for entry's edition");
|
|
||||||
|
try {
|
||||||
|
TapTargetView.showFor(this,
|
||||||
|
TapTarget.forToolbarMenuItem(toolbar, R.id.menu_edit,
|
||||||
|
getString(R.string.education_entry_edit_title),
|
||||||
|
getString(R.string.education_entry_edit_summary))
|
||||||
|
.textColorInt(Color.WHITE)
|
||||||
|
.tintTarget(true)
|
||||||
|
.cancelable(true),
|
||||||
|
new TapTargetView.Listener() {
|
||||||
|
@Override
|
||||||
|
public void onTargetClick(TapTargetView view) {
|
||||||
|
super.onTargetClick(view);
|
||||||
|
MenuItem editItem = menu.findItem(R.id.menu_edit);
|
||||||
|
onOptionsItemSelected(editItem);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onOuterCircleClick(TapTargetView view) {
|
||||||
|
super.onOuterCircleClick(view);
|
||||||
|
view.dismiss(false);
|
||||||
|
// Open Keepass doc to create field references
|
||||||
|
Intent browserIntent = new Intent(Intent.ACTION_VIEW,
|
||||||
|
Uri.parse(getString(R.string.field_references_url)));
|
||||||
|
startActivity(browserIntent);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
PreferencesUtil.saveEducationPreference(this,
|
||||||
|
R.string.education_entry_edit_key);
|
||||||
|
} catch (Exception e) {
|
||||||
|
// If icon not visible
|
||||||
|
Log.w(TAG, "Can't performed education for entry's edition");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -307,7 +331,7 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
);
|
);
|
||||||
|
|
||||||
entryContentsView.assignPassword(mEntry.getPassword());
|
entryContentsView.assignPassword(mEntry.getPassword());
|
||||||
if (PreferencesUtil.allowCopyPassword(this)) {
|
if (PreferencesUtil.allowCopyPasswordAndProtectedFields(this)) {
|
||||||
entryContentsView.assignPasswordCopyListener(view ->
|
entryContentsView.assignPasswordCopyListener(view ->
|
||||||
clipboardHelper.timeoutCopyToClipboard(mEntry.getPassword(),
|
clipboardHelper.timeoutCopyToClipboard(mEntry.getPassword(),
|
||||||
getString(R.string.copy_field, getString(R.string.entry_password)))
|
getString(R.string.copy_field, getString(R.string.entry_password)))
|
||||||
@@ -323,14 +347,15 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
if (mEntry.allowExtraFields()) {
|
if (mEntry.allowExtraFields()) {
|
||||||
entryContentsView.clearExtraFields();
|
entryContentsView.clearExtraFields();
|
||||||
|
|
||||||
mEntry.getFields().doActionToAllCustomProtectedField((label, value) ->
|
mEntry.getFields().doActionToAllCustomProtectedField((label, value) -> {
|
||||||
|
boolean showAction = (!value.isProtected() || PreferencesUtil.allowCopyPasswordAndProtectedFields(EntryActivity.this));
|
||||||
entryContentsView.addExtraField(label, value, view ->
|
entryContentsView.addExtraField(label, value, showAction, view ->
|
||||||
clipboardHelper.timeoutCopyToClipboard(
|
clipboardHelper.timeoutCopyToClipboard(
|
||||||
value.toString(),
|
value.toString(),
|
||||||
getString(R.string.copy_field, label)
|
getString(R.string.copy_field, label)
|
||||||
)
|
)
|
||||||
));
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Assign dates
|
// Assign dates
|
||||||
@@ -372,7 +397,7 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
super.onCreateOptionsMenu(menu);
|
super.onCreateOptionsMenu(menu);
|
||||||
|
|
||||||
MenuInflater inflater = getMenuInflater();
|
MenuInflater inflater = getMenuInflater();
|
||||||
MenuUtil.donationMenuInflater(inflater, menu);
|
MenuUtil.contributionMenuInflater(inflater, menu);
|
||||||
inflater.inflate(R.menu.entry, menu);
|
inflater.inflate(R.menu.entry, menu);
|
||||||
inflater.inflate(R.menu.database_lock, menu);
|
inflater.inflate(R.menu.database_lock, menu);
|
||||||
|
|
||||||
@@ -415,8 +440,8 @@ public class EntryActivity extends LockingHideActivity {
|
|||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
switch ( item.getItemId() ) {
|
switch ( item.getItemId() ) {
|
||||||
case R.id.menu_donate:
|
case R.id.menu_contribute:
|
||||||
return MenuUtil.onDonationItemSelected(this);
|
return MenuUtil.onContributionItemSelected(this);
|
||||||
|
|
||||||
case R.id.menu_toggle_pass:
|
case R.id.menu_toggle_pass:
|
||||||
mShowPassword = !mShowPassword;
|
mShowPassword = !mShowPassword;
|
||||||
|
|||||||
@@ -24,7 +24,6 @@ import android.content.Intent;
|
|||||||
import android.content.res.TypedArray;
|
import android.content.res.TypedArray;
|
||||||
import android.graphics.Color;
|
import android.graphics.Color;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.os.Handler;
|
|
||||||
import android.support.v7.widget.Toolbar;
|
import android.support.v7.widget.Toolbar;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.view.Menu;
|
import android.view.Menu;
|
||||||
@@ -49,16 +48,20 @@ import com.kunzisoft.keepass.database.PwEntry;
|
|||||||
import com.kunzisoft.keepass.database.PwGroup;
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
import com.kunzisoft.keepass.database.PwGroupId;
|
import com.kunzisoft.keepass.database.PwGroupId;
|
||||||
import com.kunzisoft.keepass.database.PwIconStandard;
|
import com.kunzisoft.keepass.database.PwIconStandard;
|
||||||
import com.kunzisoft.keepass.database.edit.AddEntry;
|
import com.kunzisoft.keepass.database.PwNode;
|
||||||
import com.kunzisoft.keepass.database.edit.OnFinish;
|
import com.kunzisoft.keepass.database.action.RunnableOnFinish;
|
||||||
import com.kunzisoft.keepass.database.edit.RunnableOnFinish;
|
import com.kunzisoft.keepass.database.action.node.AddEntryRunnable;
|
||||||
import com.kunzisoft.keepass.database.edit.UpdateEntry;
|
import com.kunzisoft.keepass.database.action.node.AfterActionNodeOnFinish;
|
||||||
|
import com.kunzisoft.keepass.database.action.node.UpdateEntryRunnable;
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedString;
|
import com.kunzisoft.keepass.database.security.ProtectedString;
|
||||||
import com.kunzisoft.keepass.dialogs.GeneratePasswordDialogFragment;
|
import com.kunzisoft.keepass.dialogs.GeneratePasswordDialogFragment;
|
||||||
import com.kunzisoft.keepass.dialogs.IconPickerDialogFragment;
|
import com.kunzisoft.keepass.dialogs.IconPickerDialogFragment;
|
||||||
import com.kunzisoft.keepass.icons.IconPackChooser;
|
import com.kunzisoft.keepass.icons.IconPackChooser;
|
||||||
|
import com.kunzisoft.keepass.lock.LockingActivity;
|
||||||
|
import com.kunzisoft.keepass.lock.LockingHideActivity;
|
||||||
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
||||||
import com.kunzisoft.keepass.tasks.ProgressTask;
|
import com.kunzisoft.keepass.tasks.SaveDatabaseProgressTaskDialogFragment;
|
||||||
|
import com.kunzisoft.keepass.tasks.UpdateProgressTaskStatus;
|
||||||
import com.kunzisoft.keepass.utils.MenuUtil;
|
import com.kunzisoft.keepass.utils.MenuUtil;
|
||||||
import com.kunzisoft.keepass.utils.Types;
|
import com.kunzisoft.keepass.utils.Types;
|
||||||
import com.kunzisoft.keepass.utils.Util;
|
import com.kunzisoft.keepass.utils.Util;
|
||||||
@@ -66,6 +69,8 @@ import com.kunzisoft.keepass.view.EntryEditCustomField;
|
|||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
import static com.kunzisoft.keepass.dialogs.IconPickerDialogFragment.UNDEFINED_ICON_ID;
|
import static com.kunzisoft.keepass.dialogs.IconPickerDialogFragment.UNDEFINED_ICON_ID;
|
||||||
|
|
||||||
public class EntryEditActivity extends LockingHideActivity
|
public class EntryEditActivity extends LockingHideActivity
|
||||||
@@ -173,7 +178,7 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
|
|
||||||
PwDatabase pm = db.getPwDatabase();
|
PwDatabase pm = db.getPwDatabase();
|
||||||
if ( uuidBytes == null ) {
|
if ( uuidBytes == null ) {
|
||||||
PwGroupId parentId = (PwGroupId) intent.getSerializableExtra(KEY_PARENT);
|
PwGroupId parentId = intent.getParcelableExtra(KEY_PARENT);
|
||||||
PwGroup parent = pm.getGroupByGroupId(parentId);
|
PwGroup parent = pm.getGroupByGroupId(parentId);
|
||||||
mEntry = db.createEntry(parent);
|
mEntry = db.createEntry(parent);
|
||||||
mIsNew = true;
|
mIsNew = true;
|
||||||
@@ -249,16 +254,21 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
}
|
}
|
||||||
mCallbackNewEntry = populateNewEntry();
|
mCallbackNewEntry = populateNewEntry();
|
||||||
|
|
||||||
OnFinish onFinish = new AfterSave();
|
// Open a progress dialog and save entry
|
||||||
|
AfterActionNodeOnFinish onFinish = new AfterSave();
|
||||||
EntryEditActivity act = EntryEditActivity.this;
|
EntryEditActivity act = EntryEditActivity.this;
|
||||||
RunnableOnFinish task;
|
RunnableOnFinish task;
|
||||||
if ( mIsNew ) {
|
if ( mIsNew ) {
|
||||||
task = new AddEntry(act, App.getDB(), mCallbackNewEntry, onFinish);
|
task = new AddEntryRunnable(act, App.getDB(), mCallbackNewEntry, onFinish);
|
||||||
} else {
|
} else {
|
||||||
task = new UpdateEntry(act, App.getDB(), mEntry, mCallbackNewEntry, onFinish);
|
task = new UpdateEntryRunnable(act, App.getDB(), mEntry, mCallbackNewEntry, onFinish);
|
||||||
}
|
}
|
||||||
ProgressTask pt = new ProgressTask(act, task, R.string.saving_database);
|
task.setUpdateProgressTaskStatus(
|
||||||
pt.run();
|
new UpdateProgressTaskStatus(this,
|
||||||
|
SaveDatabaseProgressTaskDialogFragment.start(
|
||||||
|
getSupportFragmentManager())
|
||||||
|
));
|
||||||
|
new Thread(task).start();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -266,57 +276,58 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
* Displays the explanation for the icon selection, the password generator and for a new field
|
* Displays the explanation for the icon selection, the password generator and for a new field
|
||||||
*/
|
*/
|
||||||
private void checkAndPerformedEducation() {
|
private void checkAndPerformedEducation() {
|
||||||
|
if (PreferencesUtil.isEducationScreensEnabled(this)) {
|
||||||
|
// TODO Show icon
|
||||||
|
|
||||||
// TODO Show icon
|
if (!PreferencesUtil.isEducationPasswordGeneratorPerformed(this)) {
|
||||||
|
TapTargetView.showFor(this,
|
||||||
|
TapTarget.forView(generatePasswordView,
|
||||||
|
getString(R.string.education_generate_password_title),
|
||||||
|
getString(R.string.education_generate_password_summary))
|
||||||
|
.textColorInt(Color.WHITE)
|
||||||
|
.tintTarget(false)
|
||||||
|
.cancelable(true),
|
||||||
|
new TapTargetView.Listener() {
|
||||||
|
@Override
|
||||||
|
public void onTargetClick(TapTargetView view) {
|
||||||
|
super.onTargetClick(view);
|
||||||
|
openPasswordGenerator();
|
||||||
|
}
|
||||||
|
|
||||||
if (!PreferencesUtil.isEducationPasswordGeneratorPerformed(this)) {
|
@Override
|
||||||
TapTargetView.showFor(this,
|
public void onOuterCircleClick(TapTargetView view) {
|
||||||
TapTarget.forView(generatePasswordView,
|
super.onOuterCircleClick(view);
|
||||||
getString(R.string.education_generate_password_title),
|
view.dismiss(false);
|
||||||
getString(R.string.education_generate_password_summary))
|
}
|
||||||
.tintTarget(false)
|
});
|
||||||
.cancelable(true),
|
PreferencesUtil.saveEducationPreference(this,
|
||||||
new TapTargetView.Listener() {
|
R.string.education_password_generator_key);
|
||||||
@Override
|
} else if (mEntry.allowExtraFields()
|
||||||
public void onTargetClick(TapTargetView view) {
|
|
||||||
super.onTargetClick(view);
|
|
||||||
openPasswordGenerator();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onOuterCircleClick(TapTargetView view) {
|
|
||||||
super.onOuterCircleClick(view);
|
|
||||||
view.dismiss(false);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
PreferencesUtil.saveEducationPreference(this,
|
|
||||||
R.string.education_password_generator_key);
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (mEntry.allowExtraFields()
|
|
||||||
&& !mEntry.containsCustomFields()
|
&& !mEntry.containsCustomFields()
|
||||||
&& !PreferencesUtil.isEducationEntryNewFieldPerformed(this)) {
|
&& !PreferencesUtil.isEducationEntryNewFieldPerformed(this)) {
|
||||||
TapTargetView.showFor(this,
|
TapTargetView.showFor(this,
|
||||||
TapTarget.forView(addNewFieldView,
|
TapTarget.forView(addNewFieldView,
|
||||||
getString(R.string.education_entry_new_field_title),
|
getString(R.string.education_entry_new_field_title),
|
||||||
getString(R.string.education_entry_new_field_summary))
|
getString(R.string.education_entry_new_field_summary))
|
||||||
.tintTarget(false)
|
.textColorInt(Color.WHITE)
|
||||||
.cancelable(true),
|
.tintTarget(false)
|
||||||
new TapTargetView.Listener() {
|
.cancelable(true),
|
||||||
@Override
|
new TapTargetView.Listener() {
|
||||||
public void onTargetClick(TapTargetView view) {
|
@Override
|
||||||
super.onTargetClick(view);
|
public void onTargetClick(TapTargetView view) {
|
||||||
addNewCustomField();
|
super.onTargetClick(view);
|
||||||
}
|
addNewCustomField();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onOuterCircleClick(TapTargetView view) {
|
public void onOuterCircleClick(TapTargetView view) {
|
||||||
super.onOuterCircleClick(view);
|
super.onOuterCircleClick(view);
|
||||||
view.dismiss(false);
|
view.dismiss(false);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
PreferencesUtil.saveEducationPreference(this,
|
PreferencesUtil.saveEducationPreference(this,
|
||||||
R.string.education_entry_new_field_key);
|
R.string.education_entry_new_field_key);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -432,7 +443,7 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
return App.getDB().getPwDatabase().getIconFactory().getIcon(mSelectedIconID);
|
return App.getDB().getPwDatabase().getIconFactory().getIcon(mSelectedIconID);
|
||||||
else {
|
else {
|
||||||
if (mIsNew) {
|
if (mIsNew) {
|
||||||
return App.getDB().getPwDatabase().getIconFactory().getFirstIcon();
|
return App.getDB().getPwDatabase().getIconFactory().getKeyIcon();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Keep previous icon, if no new one was selected
|
// Keep previous icon, if no new one was selected
|
||||||
@@ -447,15 +458,15 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
super.onCreateOptionsMenu(menu);
|
super.onCreateOptionsMenu(menu);
|
||||||
|
|
||||||
MenuInflater inflater = getMenuInflater();
|
MenuInflater inflater = getMenuInflater();
|
||||||
MenuUtil.donationMenuInflater(inflater, menu);
|
MenuUtil.contributionMenuInflater(inflater, menu);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
switch ( item.getItemId() ) {
|
switch ( item.getItemId() ) {
|
||||||
case R.id.menu_donate:
|
case R.id.menu_contribute:
|
||||||
return MenuUtil.onDonationItemSelected(this);
|
return MenuUtil.onContributionItemSelected(this);
|
||||||
|
|
||||||
case android.R.id.home:
|
case android.R.id.home:
|
||||||
finish();
|
finish();
|
||||||
@@ -537,7 +548,7 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
if (mCallbackNewEntry != null) {
|
if (mCallbackNewEntry != null) {
|
||||||
Bundle bundle = new Bundle();
|
Bundle bundle = new Bundle();
|
||||||
Intent intentEntry = new Intent();
|
Intent intentEntry = new Intent();
|
||||||
bundle.putSerializable(ADD_OR_UPDATE_ENTRY_KEY, mCallbackNewEntry);
|
bundle.putParcelable(ADD_OR_UPDATE_ENTRY_KEY, mCallbackNewEntry);
|
||||||
intentEntry.putExtras(bundle);
|
intentEntry.putExtras(bundle);
|
||||||
if (mIsNew) {
|
if (mIsNew) {
|
||||||
setResult(ADD_ENTRY_RESULT_CODE, intentEntry);
|
setResult(ADD_ENTRY_RESULT_CODE, intentEntry);
|
||||||
@@ -552,20 +563,20 @@ public class EntryEditActivity extends LockingHideActivity
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private final class AfterSave extends OnFinish {
|
private final class AfterSave extends AfterActionNodeOnFinish {
|
||||||
|
|
||||||
AfterSave() {
|
|
||||||
super(new Handler());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run(@Nullable PwNode oldNode, @Nullable PwNode newNode) {
|
||||||
if ( mSuccess ) {
|
runOnUiThread(() -> {
|
||||||
finish();
|
if ( mSuccess ) {
|
||||||
} else {
|
finish();
|
||||||
displayMessage(EntryEditActivity.this);
|
} else {
|
||||||
}
|
displayMessage(EntryEditActivity.this);
|
||||||
|
}
|
||||||
|
|
||||||
|
SaveDatabaseProgressTaskDialogFragment.stop(EntryEditActivity.this);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package com.kunzisoft.keepass.activities;
|
||||||
|
|
||||||
|
import android.content.Intent;
|
||||||
|
|
||||||
|
public interface IntentBuildLauncher {
|
||||||
|
void startActivityForResult(Intent intent);
|
||||||
|
}
|
||||||
@@ -22,52 +22,45 @@ package com.kunzisoft.keepass.activities;
|
|||||||
import android.annotation.SuppressLint;
|
import android.annotation.SuppressLint;
|
||||||
import android.app.assist.AssistStructure;
|
import android.app.assist.AssistStructure;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.SharedPreferences;
|
|
||||||
import android.content.SharedPreferences.Editor;
|
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.os.Handler;
|
import android.support.annotation.Nullable;
|
||||||
import android.preference.PreferenceManager;
|
|
||||||
import android.support.v7.widget.LinearLayoutManager;
|
|
||||||
import android.support.v7.widget.RecyclerView;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.view.Menu;
|
import android.view.Menu;
|
||||||
import android.view.MenuInflater;
|
import android.view.MenuInflater;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
import android.view.View;
|
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.adapters.NodeAdapter;
|
import com.kunzisoft.keepass.adapters.NodeAdapter;
|
||||||
import com.kunzisoft.keepass.app.App;
|
import com.kunzisoft.keepass.app.App;
|
||||||
import com.kunzisoft.keepass.autofill.AutofillHelper;
|
import com.kunzisoft.keepass.autofill.AutofillHelper;
|
||||||
import com.kunzisoft.keepass.compat.EditorCompat;
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
import com.kunzisoft.keepass.database.Database;
|
||||||
import com.kunzisoft.keepass.database.PwDatabase;
|
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
import com.kunzisoft.keepass.database.PwNode;
|
import com.kunzisoft.keepass.database.PwNode;
|
||||||
import com.kunzisoft.keepass.database.SortNodeEnum;
|
import com.kunzisoft.keepass.database.SortNodeEnum;
|
||||||
import com.kunzisoft.keepass.database.edit.AfterActionNodeOnFinish;
|
|
||||||
import com.kunzisoft.keepass.database.edit.OnFinish;
|
|
||||||
import com.kunzisoft.keepass.dialogs.AssignMasterKeyDialogFragment;
|
import com.kunzisoft.keepass.dialogs.AssignMasterKeyDialogFragment;
|
||||||
import com.kunzisoft.keepass.dialogs.SortDialogFragment;
|
import com.kunzisoft.keepass.dialogs.SortDialogFragment;
|
||||||
|
import com.kunzisoft.keepass.lock.LockingActivity;
|
||||||
import com.kunzisoft.keepass.password.AssignPasswordHelper;
|
import com.kunzisoft.keepass.password.AssignPasswordHelper;
|
||||||
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
import com.kunzisoft.keepass.selection.EntrySelectionHelper;
|
||||||
import com.kunzisoft.keepass.tasks.UIToastTask;
|
|
||||||
import com.kunzisoft.keepass.utils.MenuUtil;
|
import com.kunzisoft.keepass.utils.MenuUtil;
|
||||||
|
|
||||||
public abstract class ListNodesActivity extends LockingActivity
|
public abstract class ListNodesActivity extends LockingActivity
|
||||||
implements AssignMasterKeyDialogFragment.AssignPasswordDialogListener,
|
implements AssignMasterKeyDialogFragment.AssignPasswordDialogListener,
|
||||||
NodeAdapter.OnNodeClickCallback,
|
NodeAdapter.NodeClickCallback,
|
||||||
SortDialogFragment.SortSelectionListener {
|
SortDialogFragment.SortSelectionListener {
|
||||||
|
|
||||||
protected PwGroup mCurrentGroup;
|
protected static final String LIST_NODES_FRAGMENT_TAG = "LIST_NODES_FRAGMENT_TAG";
|
||||||
protected NodeAdapter mAdapter;
|
protected ListNodesFragment listNodesFragment;
|
||||||
|
|
||||||
private SharedPreferences prefs;
|
|
||||||
|
|
||||||
|
protected Database database;
|
||||||
|
protected PwGroup rootGroup;
|
||||||
|
protected PwGroup mCurrentGroup;
|
||||||
|
protected TextView groupNameView;
|
||||||
|
|
||||||
|
protected boolean entrySelectionMode;
|
||||||
protected AutofillHelper autofillHelper;
|
protected AutofillHelper autofillHelper;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -77,67 +70,94 @@ public abstract class ListNodesActivity extends LockingActivity
|
|||||||
if ( isFinishing() ) {
|
if ( isFinishing() ) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
database = App.getDB();
|
||||||
|
|
||||||
// Likely the app has been killed exit the activity
|
// Likely the app has been killed exit the activity
|
||||||
if ( ! App.getDB().getLoaded() ) {
|
if ( ! database.getLoaded() ) {
|
||||||
finish();
|
finish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
prefs = PreferenceManager.getDefaultSharedPreferences(this);
|
|
||||||
|
|
||||||
invalidateOptionsMenu();
|
invalidateOptionsMenu();
|
||||||
|
|
||||||
// TODO Move in search
|
rootGroup = database.getPwDatabase().getRootGroup();
|
||||||
setContentView(R.layout.list_nodes);
|
mCurrentGroup = retrieveCurrentGroup(savedInstanceState);
|
||||||
|
|
||||||
mCurrentGroup = initCurrentGroup();
|
initializeListNodesFragment(mCurrentGroup);
|
||||||
|
|
||||||
mAdapter = new NodeAdapter(this);
|
|
||||||
addOptionsToAdapter(mAdapter);
|
|
||||||
|
|
||||||
|
entrySelectionMode = EntrySelectionHelper.isIntentInEntrySelectionMode(getIntent());
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
||||||
autofillHelper = new AutofillHelper();
|
autofillHelper = new AutofillHelper();
|
||||||
autofillHelper.retrieveAssistStructure(getIntent());
|
autofillHelper.retrieveAssistStructure(getIntent());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract PwGroup initCurrentGroup();
|
protected abstract PwGroup retrieveCurrentGroup(@Nullable Bundle savedInstanceState);
|
||||||
|
|
||||||
protected abstract RecyclerView defineNodeList();
|
|
||||||
|
|
||||||
protected void addOptionsToAdapter(NodeAdapter nodeAdapter) {
|
|
||||||
mAdapter.setOnNodeClickListener(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
// Add elements to the list
|
// Refresh the title
|
||||||
mAdapter.rebuildList(mCurrentGroup);
|
assignToolbarElements();
|
||||||
assignListToNodeAdapter(defineNodeList());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void setGroupTitle() {
|
protected void initializeListNodesFragment(PwGroup currentGroup) {
|
||||||
if ( mCurrentGroup != null ) {
|
listNodesFragment = (ListNodesFragment) getSupportFragmentManager()
|
||||||
String name = mCurrentGroup.getName();
|
.findFragmentByTag(LIST_NODES_FRAGMENT_TAG);
|
||||||
TextView tv = findViewById(R.id.group_name);
|
if (listNodesFragment == null)
|
||||||
if ( name != null && name.length() > 0 ) {
|
listNodesFragment = ListNodesFragment.newInstance(currentGroup, readOnly);
|
||||||
if ( tv != null ) {
|
}
|
||||||
tv.setText(name);
|
|
||||||
}
|
/**
|
||||||
} else {
|
* Attach the fragment's list of node.
|
||||||
if ( tv != null ) {
|
* <br />
|
||||||
tv.setText(getText(R.string.root));
|
* <strong>R.id.nodes_list_fragment_container</strong> must be the id of the container
|
||||||
}
|
*/
|
||||||
}
|
protected void attachFragmentToContentView() {
|
||||||
}
|
getSupportFragmentManager().beginTransaction().replace(
|
||||||
|
R.id.nodes_list_fragment_container,
|
||||||
|
listNodesFragment,
|
||||||
|
LIST_NODES_FRAGMENT_TAG)
|
||||||
|
.commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assignToolbarElements() {
|
||||||
|
if (mCurrentGroup != null) {
|
||||||
|
String title = mCurrentGroup.getName();
|
||||||
|
if (title != null && title.length() > 0) {
|
||||||
|
if (groupNameView != null) {
|
||||||
|
groupNameView.setText(title);
|
||||||
|
groupNameView.invalidate();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (groupNameView != null) {
|
||||||
|
groupNameView.setText(getText(R.string.root));
|
||||||
|
groupNameView.invalidate();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
|
super.onCreateOptionsMenu(menu);
|
||||||
|
|
||||||
|
MenuInflater inflater = getMenuInflater();
|
||||||
|
MenuUtil.contributionMenuInflater(inflater, menu);
|
||||||
|
inflater.inflate(R.menu.default_menu, menu);
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void assignListToNodeAdapter(RecyclerView recyclerView) {
|
@Override
|
||||||
recyclerView.setScrollBarStyle(View.SCROLLBARS_INSIDE_INSET);
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
recyclerView.setLayoutManager(new LinearLayoutManager(this));
|
switch ( item.getItemId() ) {
|
||||||
recyclerView.setAdapter(mAdapter);
|
default:
|
||||||
|
// Check the time lock before launching settings
|
||||||
|
MenuUtil.onDefaultMenuOptionsItemSelected(this, item, readOnly, true);
|
||||||
|
return super.onOptionsItemSelected(item);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -150,7 +170,7 @@ public abstract class ListNodesActivity extends LockingActivity
|
|||||||
if (assistStructure != null) {
|
if (assistStructure != null) {
|
||||||
switch (node.getType()) {
|
switch (node.getType()) {
|
||||||
case GROUP:
|
case GROUP:
|
||||||
GroupActivity.launch(this, (PwGroup) node, assistStructure);
|
openGroup((PwGroup) node);
|
||||||
break;
|
break;
|
||||||
case ENTRY:
|
case ENTRY:
|
||||||
// Build response with the entry selected
|
// Build response with the entry selected
|
||||||
@@ -161,80 +181,49 @@ public abstract class ListNodesActivity extends LockingActivity
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( assistStructure == null ){
|
if ( assistStructure == null ){
|
||||||
switch (node.getType()) {
|
if (entrySelectionMode) {
|
||||||
case GROUP:
|
switch (node.getType()) {
|
||||||
GroupActivity.launch(this, (PwGroup) node);
|
case GROUP:
|
||||||
break;
|
openGroup((PwGroup) node);
|
||||||
case ENTRY:
|
break;
|
||||||
EntryActivity.launch(this, (PwEntry) node);
|
case ENTRY:
|
||||||
break;
|
EntrySelectionHelper.buildResponseWhenEntrySelected(this, (PwEntry) node);
|
||||||
|
finish();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
switch (node.getType()) {
|
||||||
|
case GROUP:
|
||||||
|
openGroup((PwGroup) node);
|
||||||
|
break;
|
||||||
|
case ENTRY:
|
||||||
|
EntryActivity.launch(this, (PwEntry) node, readOnly);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
protected void openGroup(PwGroup group) {
|
||||||
public boolean onCreateOptionsMenu(Menu menu) {
|
// Check Timeout
|
||||||
super.onCreateOptionsMenu(menu);
|
if (checkTimeIsAllowedOrFinish(this)) {
|
||||||
|
startRecordTime(this);
|
||||||
MenuInflater inflater = getMenuInflater();
|
|
||||||
MenuUtil.donationMenuInflater(inflater, menu);
|
|
||||||
inflater.inflate(R.menu.tree, menu);
|
|
||||||
inflater.inflate(R.menu.default_menu, menu);
|
|
||||||
|
|
||||||
return true;
|
ListNodesFragment newListNodeFragment = ListNodesFragment.newInstance(group, readOnly);
|
||||||
}
|
getSupportFragmentManager().beginTransaction()
|
||||||
|
.setCustomAnimations(R.anim.slide_in_right, R.anim.slide_out_left,
|
||||||
@Override
|
R.anim.slide_in_left, R.anim.slide_out_right)
|
||||||
public void onSortSelected(SortNodeEnum sortNodeEnum, boolean ascending, boolean groupsBefore, boolean recycleBinBottom) {
|
.replace(R.id.nodes_list_fragment_container,
|
||||||
// Toggle setting
|
newListNodeFragment,
|
||||||
Editor editor = prefs.edit();
|
LIST_NODES_FRAGMENT_TAG)
|
||||||
editor.putString(getString(R.string.sort_node_key), sortNodeEnum.name());
|
.addToBackStack(LIST_NODES_FRAGMENT_TAG)
|
||||||
editor.putBoolean(getString(R.string.sort_ascending_key), ascending);
|
.commit();
|
||||||
editor.putBoolean(getString(R.string.sort_group_before_key), groupsBefore);
|
listNodesFragment = newListNodeFragment;
|
||||||
editor.putBoolean(getString(R.string.sort_recycle_bin_bottom_key), recycleBinBottom);
|
mCurrentGroup = group;
|
||||||
EditorCompat.apply(editor);
|
assignToolbarElements();
|
||||||
|
}
|
||||||
// Tell the adapter to refresh it's list
|
|
||||||
mAdapter.notifyChangeSort(sortNodeEnum, ascending, groupsBefore);
|
|
||||||
mAdapter.rebuildList(mCurrentGroup);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
|
||||||
switch ( item.getItemId() ) {
|
|
||||||
|
|
||||||
case R.id.menu_sort:
|
|
||||||
SortDialogFragment sortDialogFragment;
|
|
||||||
|
|
||||||
PwDatabase database = App.getDB().getPwDatabase();
|
|
||||||
/*
|
|
||||||
// TODO Recycle bin bottom
|
|
||||||
if (database.isRecycleBinAvailable() && database.isRecycleBinEnabled()) {
|
|
||||||
sortDialogFragment =
|
|
||||||
SortDialogFragment.getInstance(
|
|
||||||
PrefsUtil.getListSort(this),
|
|
||||||
PrefsUtil.getAscendingSort(this),
|
|
||||||
PrefsUtil.getGroupsBeforeSort(this),
|
|
||||||
PrefsUtil.getRecycleBinBottomSort(this));
|
|
||||||
} else {
|
|
||||||
*/
|
|
||||||
sortDialogFragment =
|
|
||||||
SortDialogFragment.getInstance(
|
|
||||||
PreferencesUtil.getListSort(this),
|
|
||||||
PreferencesUtil.getAscendingSort(this),
|
|
||||||
PreferencesUtil.getGroupsBeforeSort(this));
|
|
||||||
//}
|
|
||||||
|
|
||||||
sortDialogFragment.show(getSupportFragmentManager(), "sortDialog");
|
|
||||||
return true;
|
|
||||||
|
|
||||||
default:
|
|
||||||
// Check the time lock before launching settings
|
|
||||||
MenuUtil.onDefaultMenuOptionsItemSelected(this, item, true);
|
|
||||||
return super.onOptionsItemSelected(item);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onAssignKeyDialogPositiveClick(
|
public void onAssignKeyDialogPositiveClick(
|
||||||
boolean masterPasswordChecked, String masterPassword,
|
boolean masterPasswordChecked, String masterPassword,
|
||||||
@@ -253,29 +242,17 @@ public abstract class ListNodesActivity extends LockingActivity
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSortSelected(SortNodeEnum sortNodeEnum, boolean ascending, boolean groupsBefore, boolean recycleBinBottom) {
|
||||||
|
if (listNodesFragment != null)
|
||||||
|
listNodesFragment.onSortSelected(sortNodeEnum, ascending, groupsBefore, recycleBinBottom);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
||||||
super.onActivityResult(requestCode, resultCode, data);
|
super.onActivityResult(requestCode, resultCode, data);
|
||||||
|
|
||||||
switch (requestCode) {
|
EntrySelectionHelper.onActivityResultSetResultAndFinish(this, requestCode, resultCode, data);
|
||||||
case EntryEditActivity.ADD_OR_UPDATE_ENTRY_REQUEST_CODE:
|
|
||||||
if (resultCode == EntryEditActivity.ADD_ENTRY_RESULT_CODE ||
|
|
||||||
resultCode == EntryEditActivity.UPDATE_ENTRY_RESULT_CODE) {
|
|
||||||
PwNode newNode = (PwNode) data.getSerializableExtra(EntryEditActivity.ADD_OR_UPDATE_ENTRY_KEY);
|
|
||||||
if (newNode != null) {
|
|
||||||
if (resultCode == EntryEditActivity.ADD_ENTRY_RESULT_CODE)
|
|
||||||
mAdapter.addNode(newNode);
|
|
||||||
if (resultCode == EntryEditActivity.UPDATE_ENTRY_RESULT_CODE) {
|
|
||||||
//mAdapter.updateLastNodeRegister(newNode);
|
|
||||||
mAdapter.rebuildList(mCurrentGroup);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.e(this.getClass().getName(), "New node can be retrieve in Activity Result");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
||||||
AutofillHelper.onActivityResultSetResultAndFinish(this, requestCode, resultCode, data);
|
AutofillHelper.onActivityResultSetResultAndFinish(this, requestCode, resultCode, data);
|
||||||
}
|
}
|
||||||
@@ -300,67 +277,18 @@ public abstract class ListNodesActivity extends LockingActivity
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class AfterAddNode extends AfterActionNodeOnFinish {
|
@Override
|
||||||
AfterAddNode(Handler handler) {
|
public void onBackPressed() {
|
||||||
super(handler);
|
if (checkTimeIsAllowedOrFinish(this)) {
|
||||||
}
|
startRecordTime(this);
|
||||||
|
|
||||||
public void run(PwNode oldNode, PwNode newNode) {
|
super.onBackPressed();
|
||||||
super.run();
|
|
||||||
if (mSuccess) {
|
|
||||||
mAdapter.addNode(newNode);
|
|
||||||
} else {
|
|
||||||
displayMessage(ListNodesActivity.this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class AfterUpdateNode extends AfterActionNodeOnFinish {
|
listNodesFragment = (ListNodesFragment) getSupportFragmentManager().findFragmentByTag(LIST_NODES_FRAGMENT_TAG);
|
||||||
AfterUpdateNode(Handler handler) {
|
// to refresh fragment
|
||||||
super(handler);
|
listNodesFragment.rebuildList();
|
||||||
}
|
mCurrentGroup = listNodesFragment.getMainGroup();
|
||||||
|
assignToolbarElements();
|
||||||
public void run(PwNode oldNode, PwNode newNode) {
|
|
||||||
super.run();
|
|
||||||
if (mSuccess) {
|
|
||||||
mAdapter.updateNode(oldNode, newNode);
|
|
||||||
} else {
|
|
||||||
displayMessage(ListNodesActivity.this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class AfterDeleteNode extends OnFinish {
|
|
||||||
private PwNode pwNode;
|
|
||||||
|
|
||||||
AfterDeleteNode(Handler handler, PwNode pwNode) {
|
|
||||||
super(handler);
|
|
||||||
this.pwNode = pwNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( mSuccess) {
|
|
||||||
mAdapter.removeNode(pwNode);
|
|
||||||
PwGroup parent = pwNode.getParent();
|
|
||||||
Database db = App.getDB();
|
|
||||||
PwDatabase database = db.getPwDatabase();
|
|
||||||
if (db.isRecycleBinAvailable() &&
|
|
||||||
db.isRecycleBinEnabled()) {
|
|
||||||
PwGroup recycleBin = database.getRecycleBin();
|
|
||||||
// Add trash if it doesn't exists
|
|
||||||
if (parent.equals(recycleBin)
|
|
||||||
&& mCurrentGroup != null
|
|
||||||
&& mCurrentGroup.getParent() == null
|
|
||||||
&& !mCurrentGroup.equals(recycleBin)) {
|
|
||||||
mAdapter.addNode(parent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
mHandler.post(new UIToastTask(ListNodesActivity.this, "Unrecoverable error: " + mMessage));
|
|
||||||
App.setShutdown();
|
|
||||||
finish();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,275 @@
|
|||||||
|
package com.kunzisoft.keepass.activities;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.SharedPreferences;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.preference.PreferenceManager;
|
||||||
|
import android.support.annotation.NonNull;
|
||||||
|
import android.support.annotation.Nullable;
|
||||||
|
import android.support.v7.widget.LinearLayoutManager;
|
||||||
|
import android.support.v7.widget.RecyclerView;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.Menu;
|
||||||
|
import android.view.MenuInflater;
|
||||||
|
import android.view.MenuItem;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.R;
|
||||||
|
import com.kunzisoft.keepass.adapters.NodeAdapter;
|
||||||
|
import com.kunzisoft.keepass.app.App;
|
||||||
|
import com.kunzisoft.keepass.database.PwDatabase;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
import com.kunzisoft.keepass.database.PwNode;
|
||||||
|
import com.kunzisoft.keepass.database.SortNodeEnum;
|
||||||
|
import com.kunzisoft.keepass.dialogs.SortDialogFragment;
|
||||||
|
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
||||||
|
import com.kunzisoft.keepass.stylish.StylishFragment;
|
||||||
|
|
||||||
|
public class ListNodesFragment extends StylishFragment implements
|
||||||
|
SortDialogFragment.SortSelectionListener {
|
||||||
|
|
||||||
|
private static final String TAG = ListNodesFragment.class.getName();
|
||||||
|
|
||||||
|
private static final String GROUP_KEY = "GROUP_KEY";
|
||||||
|
|
||||||
|
private NodeAdapter.NodeClickCallback nodeClickCallback;
|
||||||
|
private NodeAdapter.NodeMenuListener nodeMenuListener;
|
||||||
|
private OnScrollListener onScrollListener;
|
||||||
|
|
||||||
|
private RecyclerView listView;
|
||||||
|
protected PwGroup mCurrentGroup;
|
||||||
|
protected NodeAdapter mAdapter;
|
||||||
|
|
||||||
|
// Preferences for sorting
|
||||||
|
private SharedPreferences prefs;
|
||||||
|
|
||||||
|
private boolean readOnly;
|
||||||
|
|
||||||
|
public static ListNodesFragment newInstance(PwGroup group, boolean readOnly) {
|
||||||
|
Bundle bundle = new Bundle();
|
||||||
|
if (group != null) {
|
||||||
|
bundle.putParcelable(GROUP_KEY, group);
|
||||||
|
}
|
||||||
|
ReadOnlyHelper.putReadOnlyInBundle(bundle, readOnly);
|
||||||
|
ListNodesFragment listNodesFragment = new ListNodesFragment();
|
||||||
|
listNodesFragment.setArguments(bundle);
|
||||||
|
return listNodesFragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAttach(Context context) {
|
||||||
|
super.onAttach(context);
|
||||||
|
try {
|
||||||
|
nodeClickCallback = (NodeAdapter.NodeClickCallback) context;
|
||||||
|
} catch (ClassCastException e) {
|
||||||
|
// The activity doesn't implement the interface, throw exception
|
||||||
|
throw new ClassCastException(context.toString()
|
||||||
|
+ " must implement " + NodeAdapter.NodeClickCallback.class.getName());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
nodeMenuListener = (NodeAdapter.NodeMenuListener) context;
|
||||||
|
} catch (ClassCastException e) {
|
||||||
|
nodeMenuListener = null;
|
||||||
|
// Context menu can be omit
|
||||||
|
Log.w(TAG, context.toString()
|
||||||
|
+ " must implement " + NodeAdapter.NodeMenuListener.class.getName());
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
onScrollListener = (OnScrollListener) context;
|
||||||
|
} catch (ClassCastException e) {
|
||||||
|
onScrollListener = null;
|
||||||
|
// Context menu can be omit
|
||||||
|
Log.w(TAG, context.toString()
|
||||||
|
+ " must implement " + RecyclerView.OnScrollListener.class.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreate(@Nullable Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
|
||||||
|
if ( getActivity() != null ) {
|
||||||
|
setHasOptionsMenu(true);
|
||||||
|
|
||||||
|
readOnly = ReadOnlyHelper.retrieveReadOnlyFromInstanceStateOrArguments(savedInstanceState, getArguments());
|
||||||
|
|
||||||
|
if (getArguments() != null) {
|
||||||
|
// Contains all the group in element
|
||||||
|
if (getArguments().containsKey(GROUP_KEY)) {
|
||||||
|
mCurrentGroup = getArguments().getParcelable(GROUP_KEY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mAdapter = new NodeAdapter(getContextThemed(), getActivity().getMenuInflater(), readOnly);
|
||||||
|
mAdapter.setOnNodeClickListener(nodeClickCallback);
|
||||||
|
|
||||||
|
if (nodeMenuListener != null) {
|
||||||
|
mAdapter.setActivateContextMenu(true);
|
||||||
|
mAdapter.setNodeMenuListener(nodeMenuListener);
|
||||||
|
}
|
||||||
|
prefs = PreferenceManager.getDefaultSharedPreferences(getContext());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSaveInstanceState(@NonNull Bundle outState) {
|
||||||
|
ReadOnlyHelper.onSaveInstanceState(outState, readOnly);
|
||||||
|
super.onSaveInstanceState(outState);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
|
||||||
|
super.onCreateView(inflater, container, savedInstanceState);
|
||||||
|
|
||||||
|
// To apply theme
|
||||||
|
View rootView = inflater.cloneInContext(getContextThemed())
|
||||||
|
.inflate(R.layout.list_nodes_fragment, container, false);
|
||||||
|
listView = rootView.findViewById(R.id.nodes_list);
|
||||||
|
|
||||||
|
if (onScrollListener != null) {
|
||||||
|
listView.addOnScrollListener(new RecyclerView.OnScrollListener() {
|
||||||
|
@Override
|
||||||
|
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
|
||||||
|
super.onScrolled(recyclerView, dx, dy);
|
||||||
|
onScrollListener.onScrolled(dy);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return rootView;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onResume() {
|
||||||
|
super.onResume();
|
||||||
|
|
||||||
|
rebuildList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void rebuildList() {
|
||||||
|
// Add elements to the list
|
||||||
|
mAdapter.rebuildList(mCurrentGroup);
|
||||||
|
assignListToNodeAdapter(listView);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void assignListToNodeAdapter(RecyclerView recyclerView) {
|
||||||
|
recyclerView.setScrollBarStyle(View.SCROLLBARS_INSIDE_INSET);
|
||||||
|
recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));
|
||||||
|
recyclerView.setAdapter(mAdapter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSortSelected(SortNodeEnum sortNodeEnum, boolean ascending, boolean groupsBefore, boolean recycleBinBottom) {
|
||||||
|
// Toggle setting
|
||||||
|
SharedPreferences.Editor editor = prefs.edit();
|
||||||
|
editor.putString(getString(R.string.sort_node_key), sortNodeEnum.name());
|
||||||
|
editor.putBoolean(getString(R.string.sort_ascending_key), ascending);
|
||||||
|
editor.putBoolean(getString(R.string.sort_group_before_key), groupsBefore);
|
||||||
|
editor.putBoolean(getString(R.string.sort_recycle_bin_bottom_key), recycleBinBottom);
|
||||||
|
editor.apply();
|
||||||
|
|
||||||
|
// Tell the adapter to refresh it's list
|
||||||
|
mAdapter.notifyChangeSort(sortNodeEnum, ascending, groupsBefore);
|
||||||
|
mAdapter.rebuildList(mCurrentGroup);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
||||||
|
inflater.inflate(R.menu.tree, menu);
|
||||||
|
|
||||||
|
super.onCreateOptionsMenu(menu, inflater);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
|
switch ( item.getItemId() ) {
|
||||||
|
|
||||||
|
case R.id.menu_sort:
|
||||||
|
SortDialogFragment sortDialogFragment;
|
||||||
|
|
||||||
|
PwDatabase database = App.getDB().getPwDatabase();
|
||||||
|
/*
|
||||||
|
// TODO Recycle bin bottom
|
||||||
|
if (database.isRecycleBinAvailable() && database.isRecycleBinEnabled()) {
|
||||||
|
sortDialogFragment =
|
||||||
|
SortDialogFragment.getInstance(
|
||||||
|
PrefsUtil.getListSort(this),
|
||||||
|
PrefsUtil.getAscendingSort(this),
|
||||||
|
PrefsUtil.getGroupsBeforeSort(this),
|
||||||
|
PrefsUtil.getRecycleBinBottomSort(this));
|
||||||
|
} else {
|
||||||
|
*/
|
||||||
|
sortDialogFragment =
|
||||||
|
SortDialogFragment.getInstance(
|
||||||
|
PreferencesUtil.getListSort(getContext()),
|
||||||
|
PreferencesUtil.getAscendingSort(getContext()),
|
||||||
|
PreferencesUtil.getGroupsBeforeSort(getContext()));
|
||||||
|
//}
|
||||||
|
|
||||||
|
sortDialogFragment.show(getChildFragmentManager(), "sortDialog");
|
||||||
|
return true;
|
||||||
|
|
||||||
|
default:
|
||||||
|
return super.onOptionsItemSelected(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onActivityResult(int requestCode, int resultCode, Intent data) {
|
||||||
|
super.onActivityResult(requestCode, resultCode, data);
|
||||||
|
|
||||||
|
switch (requestCode) {
|
||||||
|
case EntryEditActivity.ADD_OR_UPDATE_ENTRY_REQUEST_CODE:
|
||||||
|
if (resultCode == EntryEditActivity.ADD_ENTRY_RESULT_CODE ||
|
||||||
|
resultCode == EntryEditActivity.UPDATE_ENTRY_RESULT_CODE) {
|
||||||
|
PwNode newNode = data.getParcelableExtra(EntryEditActivity.ADD_OR_UPDATE_ENTRY_KEY);
|
||||||
|
if (newNode != null) {
|
||||||
|
if (resultCode == EntryEditActivity.ADD_ENTRY_RESULT_CODE)
|
||||||
|
mAdapter.addNode(newNode);
|
||||||
|
if (resultCode == EntryEditActivity.UPDATE_ENTRY_RESULT_CODE) {
|
||||||
|
//mAdapter.updateLastNodeRegister(newNode);
|
||||||
|
mAdapter.rebuildList(mCurrentGroup);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Log.e(this.getClass().getName(), "New node can be retrieve in Activity Result");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEmpty() {
|
||||||
|
return mAdapter == null || mAdapter.getItemCount() <= 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addNode(PwNode newNode) {
|
||||||
|
mAdapter.addNode(newNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void updateNode(PwNode oldNode, PwNode newNode) {
|
||||||
|
mAdapter.updateNode(oldNode, newNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void removeNode(PwNode pwNode) {
|
||||||
|
mAdapter.removeNode(pwNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PwGroup getMainGroup() {
|
||||||
|
return mCurrentGroup;
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface OnScrollListener {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback method to be invoked when the RecyclerView has been scrolled. This will be
|
||||||
|
* called after the scroll has completed.
|
||||||
|
*
|
||||||
|
* @param dy The amount of vertical scroll.
|
||||||
|
*/
|
||||||
|
void onScrolled(int dy);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
package com.kunzisoft.keepass.activities;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.os.Bundle;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
||||||
|
|
||||||
|
public class ReadOnlyHelper {
|
||||||
|
|
||||||
|
public static final String READ_ONLY_KEY = "READ_ONLY_KEY";
|
||||||
|
|
||||||
|
public static final boolean READ_ONLY_DEFAULT = false;
|
||||||
|
|
||||||
|
public static boolean retrieveReadOnlyFromInstanceStateOrPreference(Context context, Bundle savedInstanceState) {
|
||||||
|
boolean readOnly;
|
||||||
|
if (savedInstanceState != null
|
||||||
|
&& savedInstanceState.containsKey(READ_ONLY_KEY)) {
|
||||||
|
readOnly = savedInstanceState.getBoolean(READ_ONLY_KEY);
|
||||||
|
} else {
|
||||||
|
readOnly = PreferencesUtil.enableReadOnlyDatabase(context);
|
||||||
|
}
|
||||||
|
return readOnly;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean retrieveReadOnlyFromInstanceStateOrArguments(Bundle savedInstanceState, Bundle arguments) {
|
||||||
|
boolean readOnly = READ_ONLY_DEFAULT;
|
||||||
|
if (savedInstanceState != null
|
||||||
|
&& savedInstanceState.containsKey(READ_ONLY_KEY)) {
|
||||||
|
readOnly = savedInstanceState.getBoolean(READ_ONLY_KEY);
|
||||||
|
} else if (arguments != null
|
||||||
|
&& arguments.containsKey(READ_ONLY_KEY)) {
|
||||||
|
readOnly = arguments.getBoolean(READ_ONLY_KEY);
|
||||||
|
}
|
||||||
|
return readOnly;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean retrieveReadOnlyFromInstanceStateOrIntent(Bundle savedInstanceState, Intent intent) {
|
||||||
|
boolean readOnly = READ_ONLY_DEFAULT;
|
||||||
|
if (savedInstanceState != null
|
||||||
|
&& savedInstanceState.containsKey(READ_ONLY_KEY)) {
|
||||||
|
readOnly = savedInstanceState.getBoolean(READ_ONLY_KEY);
|
||||||
|
} else {
|
||||||
|
if (intent != null)
|
||||||
|
readOnly = intent.getBooleanExtra(READ_ONLY_KEY, READ_ONLY_DEFAULT);
|
||||||
|
}
|
||||||
|
return readOnly;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void putReadOnlyInIntent(Intent intent, boolean readOnly) {
|
||||||
|
intent.putExtra(READ_ONLY_KEY, readOnly);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void putReadOnlyInBundle(Bundle bundle, boolean readOnly) {
|
||||||
|
bundle.putBoolean(READ_ONLY_KEY, readOnly);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onSaveInstanceState(Bundle outState, boolean readOnly) {
|
||||||
|
outState.putBoolean(READ_ONLY_KEY, readOnly);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -20,8 +20,6 @@
|
|||||||
package com.kunzisoft.keepass.adapters;
|
package com.kunzisoft.keepass.adapters;
|
||||||
|
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.ImageView;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
|
|
||||||
@@ -30,7 +28,7 @@ class EntryViewHolder extends BasicViewHolder {
|
|||||||
EntryViewHolder(View itemView) {
|
EntryViewHolder(View itemView) {
|
||||||
super(itemView);
|
super(itemView);
|
||||||
container = itemView.findViewById(R.id.entry_container);
|
container = itemView.findViewById(R.id.entry_container);
|
||||||
icon = (ImageView) itemView.findViewById(R.id.entry_icon);
|
icon = itemView.findViewById(R.id.entry_icon);
|
||||||
text = (TextView) itemView.findViewById(R.id.entry_text);
|
text = itemView.findViewById(R.id.entry_text);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,8 +20,6 @@
|
|||||||
package com.kunzisoft.keepass.adapters;
|
package com.kunzisoft.keepass.adapters;
|
||||||
|
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.ImageView;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
|
|
||||||
@@ -30,7 +28,7 @@ class GroupViewHolder extends BasicViewHolder {
|
|||||||
GroupViewHolder(View itemView) {
|
GroupViewHolder(View itemView) {
|
||||||
super(itemView);
|
super(itemView);
|
||||||
container = itemView.findViewById(R.id.group_container);
|
container = itemView.findViewById(R.id.group_container);
|
||||||
icon = (ImageView) itemView.findViewById(R.id.group_icon);
|
icon = itemView.findViewById(R.id.group_icon);
|
||||||
text = (TextView) itemView.findViewById(R.id.group_text);
|
text = itemView.findViewById(R.id.group_text);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,12 +26,15 @@ import android.support.annotation.NonNull;
|
|||||||
import android.support.v7.util.SortedList;
|
import android.support.v7.util.SortedList;
|
||||||
import android.support.v7.widget.RecyclerView;
|
import android.support.v7.widget.RecyclerView;
|
||||||
import android.support.v7.widget.util.SortedListAdapterCallback;
|
import android.support.v7.widget.util.SortedListAdapterCallback;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.util.TypedValue;
|
||||||
import android.view.ContextMenu;
|
import android.view.ContextMenu;
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
import android.view.Menu;
|
import android.view.MenuInflater;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.Toast;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.app.App;
|
import com.kunzisoft.keepass.app.App;
|
||||||
@@ -42,19 +45,23 @@ import com.kunzisoft.keepass.icons.IconPackChooser;
|
|||||||
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
import com.kunzisoft.keepass.settings.PreferencesUtil;
|
||||||
|
|
||||||
public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
||||||
|
private static final String TAG = NodeAdapter.class.getName();
|
||||||
|
|
||||||
private SortedList<PwNode> nodeSortedList;
|
private SortedList<PwNode> nodeSortedList;
|
||||||
|
|
||||||
private Context context;
|
private Context context;
|
||||||
private LayoutInflater inflater;
|
private LayoutInflater inflater;
|
||||||
|
private MenuInflater menuInflater;
|
||||||
private float textSize;
|
private float textSize;
|
||||||
|
private float iconSize;
|
||||||
private SortNodeEnum listSort;
|
private SortNodeEnum listSort;
|
||||||
private boolean groupsBeforeSort;
|
private boolean groupsBeforeSort;
|
||||||
private boolean ascendingSort;
|
private boolean ascendingSort;
|
||||||
|
|
||||||
private OnNodeClickCallback onNodeClickCallback;
|
private NodeClickCallback nodeClickCallback;
|
||||||
private NodeMenuListener nodeMenuListener;
|
private NodeMenuListener nodeMenuListener;
|
||||||
private boolean activateContextMenu;
|
private boolean activateContextMenu;
|
||||||
|
private boolean readOnly;
|
||||||
|
|
||||||
private int iconGroupColor;
|
private int iconGroupColor;
|
||||||
private int iconEntryColor;
|
private int iconEntryColor;
|
||||||
@@ -63,14 +70,13 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
* Create node list adapter with contextMenu or not
|
* Create node list adapter with contextMenu or not
|
||||||
* @param context Context to use
|
* @param context Context to use
|
||||||
*/
|
*/
|
||||||
public NodeAdapter(final Context context) {
|
public NodeAdapter(final Context context, MenuInflater menuInflater, boolean readOnly) {
|
||||||
this.inflater = LayoutInflater.from(context);
|
this.inflater = LayoutInflater.from(context);
|
||||||
|
this.menuInflater = menuInflater;
|
||||||
this.context = context;
|
this.context = context;
|
||||||
this.textSize = PreferencesUtil.getListTextSize(context);
|
assignPreferences();
|
||||||
this.listSort = PreferencesUtil.getListSort(context);
|
|
||||||
this.groupsBeforeSort = PreferencesUtil.getGroupsBeforeSort(context);
|
|
||||||
this.ascendingSort = PreferencesUtil.getAscendingSort(context);
|
|
||||||
this.activateContextMenu = false;
|
this.activateContextMenu = false;
|
||||||
|
this.readOnly = readOnly;
|
||||||
|
|
||||||
this.nodeSortedList = new SortedList<>(PwNode.class, new SortedListAdapterCallback<PwNode>(this) {
|
this.nodeSortedList = new SortedList<>(PwNode.class, new SortedListAdapterCallback<PwNode>(this) {
|
||||||
@Override public int compare(PwNode item1, PwNode item2) {
|
@Override public int compare(PwNode item1, PwNode item2) {
|
||||||
@@ -101,13 +107,32 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
this.activateContextMenu = activate;
|
this.activateContextMenu = activate;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void assignPreferences() {
|
||||||
|
this.textSize = PreferencesUtil.getListTextSize(context);
|
||||||
|
// Retrieve the icon size
|
||||||
|
int iconDefaultSize = (int) TypedValue.applyDimension(
|
||||||
|
TypedValue.COMPLEX_UNIT_DIP,
|
||||||
|
context.getResources().getInteger(R.integer.list_icon_size_default),
|
||||||
|
context.getResources().getDisplayMetrics()
|
||||||
|
);
|
||||||
|
this.iconSize = iconDefaultSize * textSize / Float.parseFloat(context.getString(R.string.list_size_default));
|
||||||
|
this.listSort = PreferencesUtil.getListSort(context);
|
||||||
|
this.groupsBeforeSort = PreferencesUtil.getGroupsBeforeSort(context);
|
||||||
|
this.ascendingSort = PreferencesUtil.getAscendingSort(context);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Rebuild the list by clear and build children from the group
|
* Rebuild the list by clear and build children from the group
|
||||||
*/
|
*/
|
||||||
public void rebuildList(PwGroup group) {
|
public void rebuildList(PwGroup group) {
|
||||||
this.nodeSortedList.clear();
|
this.nodeSortedList.clear();
|
||||||
if (group != null) {
|
assignPreferences();
|
||||||
|
// TODO verify sort
|
||||||
|
try {
|
||||||
this.nodeSortedList.addAll(group.getDirectChildren());
|
this.nodeSortedList.addAll(group.getDirectChildren());
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "Can't add node elements to the list", e);
|
||||||
|
Toast.makeText(context, "Can't add node elements to the list : " + e.getMessage(), Toast.LENGTH_LONG).show();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -194,9 +219,12 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
// Context menu
|
// Context menu
|
||||||
if (activateContextMenu) {
|
if (activateContextMenu) {
|
||||||
holder.container.setOnCreateContextMenuListener(
|
holder.container.setOnCreateContextMenuListener(
|
||||||
new ContextMenuBuilder(subNode, nodeMenuListener));
|
new ContextMenuBuilder(subNode, nodeMenuListener, readOnly));
|
||||||
}
|
}
|
||||||
// Assign text size
|
// Assign image and text size
|
||||||
|
// Relative size of the icon
|
||||||
|
holder.icon.getLayoutParams().height = ((int) iconSize);
|
||||||
|
holder.icon.getLayoutParams().width = ((int) iconSize);
|
||||||
holder.text.setTextSize(textSize);
|
holder.text.setTextSize(textSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -208,8 +236,8 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
/**
|
/**
|
||||||
* Assign a listener when a node is clicked
|
* Assign a listener when a node is clicked
|
||||||
*/
|
*/
|
||||||
public void setOnNodeClickListener(OnNodeClickCallback onNodeClickCallback) {
|
public void setOnNodeClickListener(NodeClickCallback nodeClickCallback) {
|
||||||
this.onNodeClickCallback = onNodeClickCallback;
|
this.nodeClickCallback = nodeClickCallback;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -222,7 +250,7 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
/**
|
/**
|
||||||
* Callback listener to redefine to do an action when a node is click
|
* Callback listener to redefine to do an action when a node is click
|
||||||
*/
|
*/
|
||||||
public interface OnNodeClickCallback {
|
public interface NodeClickCallback {
|
||||||
void onNodeClick(PwNode node);
|
void onNodeClick(PwNode node);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -232,6 +260,8 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
public interface NodeMenuListener {
|
public interface NodeMenuListener {
|
||||||
boolean onOpenMenuClick(PwNode node);
|
boolean onOpenMenuClick(PwNode node);
|
||||||
boolean onEditMenuClick(PwNode node);
|
boolean onEditMenuClick(PwNode node);
|
||||||
|
boolean onCopyMenuClick(PwNode node);
|
||||||
|
boolean onMoveMenuClick(PwNode node);
|
||||||
boolean onDeleteMenuClick(PwNode node);
|
boolean onDeleteMenuClick(PwNode node);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -247,8 +277,8 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onClick(View v) {
|
public void onClick(View v) {
|
||||||
if (onNodeClickCallback != null)
|
if (nodeClickCallback != null)
|
||||||
onNodeClickCallback.onNodeClick(node);
|
nodeClickCallback.onNodeClick(node);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -257,29 +287,45 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
*/
|
*/
|
||||||
private class ContextMenuBuilder implements View.OnCreateContextMenuListener {
|
private class ContextMenuBuilder implements View.OnCreateContextMenuListener {
|
||||||
|
|
||||||
private static final int MENU_OPEN = Menu.FIRST;
|
|
||||||
private static final int MENU_EDIT = MENU_OPEN + 1;
|
|
||||||
private static final int MENU_DELETE = MENU_EDIT + 1;
|
|
||||||
|
|
||||||
private PwNode node;
|
private PwNode node;
|
||||||
private NodeMenuListener menuListener;
|
private NodeMenuListener menuListener;
|
||||||
|
private boolean readOnly;
|
||||||
|
|
||||||
ContextMenuBuilder(PwNode node, NodeMenuListener menuListener) {
|
ContextMenuBuilder(PwNode node, NodeMenuListener menuListener, boolean readOnly) {
|
||||||
this.menuListener = menuListener;
|
this.menuListener = menuListener;
|
||||||
this.node = node;
|
this.node = node;
|
||||||
|
this.readOnly = readOnly;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onCreateContextMenu(ContextMenu contextMenu, View view, ContextMenu.ContextMenuInfo contextMenuInfo) {
|
public void onCreateContextMenu(ContextMenu contextMenu, View view, ContextMenu.ContextMenuInfo contextMenuInfo) {
|
||||||
MenuItem clearMenu = contextMenu.add(Menu.NONE, MENU_OPEN, Menu.NONE, R.string.menu_open);
|
menuInflater.inflate(R.menu.node_menu, contextMenu);
|
||||||
clearMenu.setOnMenuItemClickListener(mOnMyActionClickListener);
|
|
||||||
if (!App.getDB().isReadOnly() && !node.equals(App.getDB().getPwDatabase().getRecycleBin())) {
|
MenuItem menuItem = contextMenu.findItem(R.id.menu_open);
|
||||||
|
menuItem.setOnMenuItemClickListener(mOnMyActionClickListener);
|
||||||
|
if (readOnly || node.equals(App.getDB().getPwDatabase().getRecycleBin())) {
|
||||||
|
contextMenu.removeItem(R.id.menu_edit);
|
||||||
|
contextMenu.removeItem(R.id.menu_copy);
|
||||||
|
contextMenu.removeItem(R.id.menu_move);
|
||||||
|
contextMenu.removeItem(R.id.menu_delete);
|
||||||
|
} else {
|
||||||
// Edition
|
// Edition
|
||||||
clearMenu = contextMenu.add(Menu.NONE, MENU_EDIT, Menu.NONE, R.string.menu_edit);
|
menuItem = contextMenu.findItem(R.id.menu_edit);
|
||||||
clearMenu.setOnMenuItemClickListener(mOnMyActionClickListener);
|
menuItem.setOnMenuItemClickListener(mOnMyActionClickListener);
|
||||||
|
// Copy (not for group)
|
||||||
|
if (node.getType().equals(PwNode.Type.ENTRY)) {
|
||||||
|
menuItem = contextMenu.findItem(R.id.menu_copy);
|
||||||
|
menuItem.setOnMenuItemClickListener(mOnMyActionClickListener);
|
||||||
|
} else {
|
||||||
|
// TODO COPY For Group
|
||||||
|
contextMenu.removeItem(R.id.menu_copy);
|
||||||
|
}
|
||||||
|
// Move
|
||||||
|
menuItem = contextMenu.findItem(R.id.menu_move);
|
||||||
|
menuItem.setOnMenuItemClickListener(mOnMyActionClickListener);
|
||||||
// Deletion
|
// Deletion
|
||||||
clearMenu = contextMenu.add(Menu.NONE, MENU_DELETE, Menu.NONE, R.string.menu_delete);
|
menuItem = contextMenu.findItem(R.id.menu_delete);
|
||||||
clearMenu.setOnMenuItemClickListener(mOnMyActionClickListener);
|
menuItem.setOnMenuItemClickListener(mOnMyActionClickListener);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -289,11 +335,15 @@ public class NodeAdapter extends RecyclerView.Adapter<BasicViewHolder> {
|
|||||||
if (menuListener == null)
|
if (menuListener == null)
|
||||||
return false;
|
return false;
|
||||||
switch ( item.getItemId() ) {
|
switch ( item.getItemId() ) {
|
||||||
case MENU_OPEN:
|
case R.id.menu_open:
|
||||||
return menuListener.onOpenMenuClick(node);
|
return menuListener.onOpenMenuClick(node);
|
||||||
case MENU_EDIT:
|
case R.id.menu_edit:
|
||||||
return menuListener.onEditMenuClick(node);
|
return menuListener.onEditMenuClick(node);
|
||||||
case MENU_DELETE:
|
case R.id.menu_copy:
|
||||||
|
return menuListener.onCopyMenuClick(node);
|
||||||
|
case R.id.menu_move:
|
||||||
|
return menuListener.onMoveMenuClick(node);
|
||||||
|
case R.id.menu_delete:
|
||||||
return menuListener.onDeleteMenuClick(node);
|
return menuListener.onDeleteMenuClick(node);
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ public class AutoFillAuthActivity extends AppCompatActivity {
|
|||||||
// Pass extra for Autofill (EXTRA_ASSIST_STRUCTURE)
|
// Pass extra for Autofill (EXTRA_ASSIST_STRUCTURE)
|
||||||
AssistStructure assistStructure = autofillHelper.retrieveAssistStructure(getIntent());
|
AssistStructure assistStructure = autofillHelper.retrieveAssistStructure(getIntent());
|
||||||
if (assistStructure != null) {
|
if (assistStructure != null) {
|
||||||
FileSelectActivity.launch(this, assistStructure);
|
FileSelectActivity.launchForAutofillResult(this, assistStructure);
|
||||||
} else {
|
} else {
|
||||||
setResult(RESULT_CANCELED);
|
setResult(RESULT_CANCELED);
|
||||||
finish();
|
finish();
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ import android.widget.RemoteViews;
|
|||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
import com.kunzisoft.keepass.selection.EntrySelectionHelper;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -64,6 +65,7 @@ public class AutofillHelper {
|
|||||||
|
|
||||||
public static void addAssistStructureExtraInIntent(Intent intent, AssistStructure assistStructure) {
|
public static void addAssistStructureExtraInIntent(Intent intent, AssistStructure assistStructure) {
|
||||||
if (assistStructure != null) {
|
if (assistStructure != null) {
|
||||||
|
EntrySelectionHelper.addEntrySelectionModeExtraInIntent(intent);
|
||||||
intent.putExtra(android.view.autofill.AutofillManager.EXTRA_ASSIST_STRUCTURE, assistStructure);
|
intent.putExtra(android.view.autofill.AutofillManager.EXTRA_ASSIST_STRUCTURE, assistStructure);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -144,7 +146,7 @@ public class AutofillHelper {
|
|||||||
/**
|
/**
|
||||||
* Utility method to loop and close each activity with return data
|
* Utility method to loop and close each activity with return data
|
||||||
*/
|
*/
|
||||||
public static void onActivityResultSetResult(Activity activity, int requestCode, int resultCode, Intent data) {
|
public static void onActivityResultSetResultAndFinish(Activity activity, int requestCode, int resultCode, Intent data) {
|
||||||
if (requestCode == AUTOFILL_RESPONSE_REQUEST_CODE) {
|
if (requestCode == AUTOFILL_RESPONSE_REQUEST_CODE) {
|
||||||
if (resultCode == Activity.RESULT_OK) {
|
if (resultCode == Activity.RESULT_OK) {
|
||||||
activity.setResult(resultCode, data);
|
activity.setResult(resultCode, data);
|
||||||
@@ -152,15 +154,6 @@ public class AutofillHelper {
|
|||||||
if (resultCode == Activity.RESULT_CANCELED) {
|
if (resultCode == Activity.RESULT_CANCELED) {
|
||||||
activity.setResult(Activity.RESULT_CANCELED);
|
activity.setResult(Activity.RESULT_CANCELED);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Utility method to loop and close each activity with return data
|
|
||||||
*/
|
|
||||||
public static void onActivityResultSetResultAndFinish(Activity activity, int requestCode, int resultCode, Intent data) {
|
|
||||||
if (requestCode == AUTOFILL_RESPONSE_REQUEST_CODE) {
|
|
||||||
onActivityResultSetResult(activity, requestCode, resultCode, data);
|
|
||||||
activity.finish();
|
activity.finish();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.compat;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import java.lang.reflect.Constructor;
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
|
|
||||||
|
|
||||||
@SuppressWarnings({"unchecked", "rawtypes"})
|
|
||||||
public class BackupManagerCompat {
|
|
||||||
private static Class classBackupManager;
|
|
||||||
private static Constructor constructorBackupManager;
|
|
||||||
private static Method dataChanged;
|
|
||||||
|
|
||||||
private Object backupManager;
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
classBackupManager = Class.forName("android.app.backup.BackupManager");
|
|
||||||
constructorBackupManager = classBackupManager.getConstructor(Context.class);
|
|
||||||
dataChanged = classBackupManager.getMethod("dataChanged", (Class[]) null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Do nothing, class does not exist
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public BackupManagerCompat(Context ctx) {
|
|
||||||
if (constructorBackupManager != null) {
|
|
||||||
try {
|
|
||||||
backupManager = constructorBackupManager.newInstance(ctx);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Do nothing
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void dataChanged() {
|
|
||||||
if (backupManager != null && dataChanged != null) {
|
|
||||||
try {
|
|
||||||
dataChanged.invoke(backupManager, (Object[]) null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Do nothing
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.compat;
|
|
||||||
|
|
||||||
import android.content.res.Resources;
|
|
||||||
import android.graphics.Bitmap;
|
|
||||||
import android.graphics.drawable.BitmapDrawable;
|
|
||||||
|
|
||||||
import java.lang.reflect.Constructor;
|
|
||||||
|
|
||||||
// This compatiblity hack can go away when support for Android 1.5 api level 3 is dropped
|
|
||||||
public class BitmapDrawableCompat {
|
|
||||||
private static Constructor<BitmapDrawable> constResBitmap;
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
constResBitmap = BitmapDrawable.class.getConstructor(Resources.class, Bitmap.class);
|
|
||||||
// This constructor is support in this api version
|
|
||||||
} catch (Exception e) {
|
|
||||||
// This constructor is not supported
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static BitmapDrawable getBitmapDrawable(Resources res, Bitmap bitmap) {
|
|
||||||
if (constResBitmap != null) {
|
|
||||||
try {
|
|
||||||
return constResBitmap.newInstance(res, bitmap);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Do nothing, fall through to the safe constructor
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new BitmapDrawable(bitmap);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.compat;
|
|
||||||
|
|
||||||
import android.os.Build;
|
|
||||||
|
|
||||||
import java.lang.reflect.Field;
|
|
||||||
|
|
||||||
public class BuildCompat {
|
|
||||||
private static Field manufacturer;
|
|
||||||
private static String manuText;
|
|
||||||
|
|
||||||
public static final int VERSION_CODE_GINGERBREAD = 9;
|
|
||||||
public static final int VERSION_CODE_ICE_CREAM_SANDWICH = 14;
|
|
||||||
public static final int VERSION_CODE_JELLY_BEAN = 16;
|
|
||||||
public static final int VERSION_CODE_JELLY_BEAN_MR2 = 18;
|
|
||||||
public static final int VERSION_KITKAT = 19;
|
|
||||||
public static final int VERSION_CODE_M = 23;
|
|
||||||
|
|
||||||
private static Field versionSDK;
|
|
||||||
private static int versionInt;
|
|
||||||
|
|
||||||
static {
|
|
||||||
// MANUFACTURER is only available in API version 4 and later
|
|
||||||
try {
|
|
||||||
manufacturer = Build.class.getField("MANUFACTURER");
|
|
||||||
manuText = (String) manufacturer.get(null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
manuText = "";
|
|
||||||
}
|
|
||||||
|
|
||||||
// SDK is only available in API version 4 and later
|
|
||||||
try {
|
|
||||||
versionSDK = Build.VERSION.class.getField("SDK_INT");
|
|
||||||
versionInt = versionSDK.getInt(null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
try {
|
|
||||||
versionInt = Integer.parseInt(Build.VERSION.SDK);
|
|
||||||
} catch (Exception nfe) {
|
|
||||||
versionInt = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getManufacturer() {
|
|
||||||
return manuText;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static int getSdkVersion() {
|
|
||||||
return versionInt;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -25,9 +25,6 @@ import android.net.Uri;
|
|||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
|
|
||||||
public class ClipDataCompat {
|
public class ClipDataCompat {
|
||||||
private static Class clipData;
|
|
||||||
private static Class clipDataItem;
|
|
||||||
private static Class clipDescription;
|
|
||||||
private static Method getClipDataFromIntent;
|
private static Method getClipDataFromIntent;
|
||||||
private static Method getDescription;
|
private static Method getDescription;
|
||||||
private static Method getItemCount;
|
private static Method getItemCount;
|
||||||
@@ -39,14 +36,14 @@ public class ClipDataCompat {
|
|||||||
|
|
||||||
static {
|
static {
|
||||||
try {
|
try {
|
||||||
clipData = Class.forName("android.content.ClipData");
|
Class clipData = Class.forName("android.content.ClipData");
|
||||||
getDescription = clipData.getMethod("getDescription", (Class[])null);
|
getDescription = clipData.getMethod("getDescription", (Class[])null);
|
||||||
getItemCount = clipData.getMethod("getItemCount", (Class[])null);
|
getItemCount = clipData.getMethod("getItemCount", (Class[])null);
|
||||||
getItemAt = clipData.getMethod("getItemAt", new Class[]{int.class});
|
getItemAt = clipData.getMethod("getItemAt", new Class[]{int.class});
|
||||||
clipDescription = Class.forName("android.content.ClipDescription");
|
Class clipDescription = Class.forName("android.content.ClipDescription");
|
||||||
getLabel = clipDescription.getMethod("getLabel", (Class[])null);
|
getLabel = clipDescription.getMethod("getLabel", (Class[])null);
|
||||||
|
|
||||||
clipDataItem = Class.forName("android.content.ClipData$Item");
|
Class clipDataItem = Class.forName("android.content.ClipData$Item");
|
||||||
getUri = clipDataItem.getMethod("getUri", (Class[])null);
|
getUri = clipDataItem.getMethod("getUri", (Class[])null);
|
||||||
|
|
||||||
getClipDataFromIntent = Intent.class.getMethod("getClipData", (Class[])null);
|
getClipDataFromIntent = Intent.class.getMethod("getClipData", (Class[])null);
|
||||||
@@ -58,7 +55,6 @@ public class ClipDataCompat {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static Uri getUriFromIntent(Intent i, String key) {
|
public static Uri getUriFromIntent(Intent i, String key) {
|
||||||
boolean clipDataSucceeded = false;
|
|
||||||
if (initSucceded) {
|
if (initSucceded) {
|
||||||
try {
|
try {
|
||||||
Object clip = getClipDataFromIntent.invoke(i);
|
Object clip = getClipDataFromIntent.invoke(i);
|
||||||
|
|||||||
@@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.compat;
|
|
||||||
|
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.net.Uri;
|
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
|
|
||||||
public class ContentResolverCompat {
|
|
||||||
public static boolean available;
|
|
||||||
private static Class contentResolver;
|
|
||||||
private static Method takePersistableUriPermission;
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
contentResolver = ContentResolver.class;
|
|
||||||
takePersistableUriPermission = contentResolver.getMethod("takePersistableUriPermission", new Class[]{Uri.class, int.class});
|
|
||||||
|
|
||||||
available = true;
|
|
||||||
} catch (Exception e) {
|
|
||||||
available = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void takePersistableUriPermission(ContentResolver resolver, Uri uri, int modeFlags) {
|
|
||||||
if (available) {
|
|
||||||
try {
|
|
||||||
takePersistableUriPermission.invoke(resolver, new Object[]{uri, modeFlags});
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Fail silently
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.compat;
|
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.content.SharedPreferences;
|
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
|
|
||||||
public class EditorCompat {
|
|
||||||
private static Method apply;
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
apply = Activity.class.getMethod("apply", (Class<SharedPreferences.Editor>[]) null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Substitute commit for apply when not available (API level < 9)
|
|
||||||
try {
|
|
||||||
apply = Activity.class.getMethod("commit", (Class<SharedPreferences.Editor>[]) null);
|
|
||||||
} catch (Exception f) {
|
|
||||||
// Should be impossible, but leave apply null in this case
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void apply(SharedPreferences.Editor edit) {
|
|
||||||
try {
|
|
||||||
apply.invoke(edit, (Object[]) null);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// Shouldn't be possible, but call commit directly if this happens
|
|
||||||
edit.commit();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -44,7 +44,6 @@ public final class PRNGFixes {
|
|||||||
|
|
||||||
private static final byte[] BUILD_FINGERPRINT_AND_DEVICE_SERIAL =
|
private static final byte[] BUILD_FINGERPRINT_AND_DEVICE_SERIAL =
|
||||||
getBuildFingerprintAndDeviceSerial();
|
getBuildFingerprintAndDeviceSerial();
|
||||||
private static int sdkVersion = BuildCompat.getSdkVersion();
|
|
||||||
|
|
||||||
/** Hidden constructor to prevent instantiation. */
|
/** Hidden constructor to prevent instantiation. */
|
||||||
private PRNGFixes() {}
|
private PRNGFixes() {}
|
||||||
@@ -67,11 +66,11 @@ public final class PRNGFixes {
|
|||||||
|
|
||||||
private static boolean supportedOnThisDevice() {
|
private static boolean supportedOnThisDevice() {
|
||||||
// Blacklist on samsung devices
|
// Blacklist on samsung devices
|
||||||
if (StrUtil.indexOfIgnoreCase(BuildCompat.getManufacturer(), "samsung", Locale.ENGLISH) >= 0) {
|
if (StrUtil.indexOfIgnoreCase(Build.MANUFACTURER, "samsung", Locale.ENGLISH) >= 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sdkVersion > BuildCompat.VERSION_CODE_JELLY_BEAN_MR2) {
|
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN_MR2) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -126,8 +125,8 @@ public final class PRNGFixes {
|
|||||||
* @throws SecurityException if the fix is needed but could not be applied.
|
* @throws SecurityException if the fix is needed but could not be applied.
|
||||||
*/
|
*/
|
||||||
private static void applyOpenSSLFix() throws SecurityException {
|
private static void applyOpenSSLFix() throws SecurityException {
|
||||||
if ((sdkVersion < BuildCompat.VERSION_CODE_JELLY_BEAN)
|
if ((Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN)
|
||||||
|| (sdkVersion > BuildCompat.VERSION_CODE_JELLY_BEAN_MR2)) {
|
|| (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN_MR2)) {
|
||||||
// No need to apply the fix
|
// No need to apply the fix
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -162,7 +161,7 @@ public final class PRNGFixes {
|
|||||||
*/
|
*/
|
||||||
private static void installLinuxPRNGSecureRandom()
|
private static void installLinuxPRNGSecureRandom()
|
||||||
throws SecurityException {
|
throws SecurityException {
|
||||||
if (sdkVersion > BuildCompat.VERSION_CODE_JELLY_BEAN_MR2) {
|
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN_MR2) {
|
||||||
// No need to apply the fix
|
// No need to apply the fix
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ package com.kunzisoft.keepass.compat;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.SharedPreferences;
|
import android.content.SharedPreferences;
|
||||||
|
import android.os.Build;
|
||||||
import android.preference.PreferenceManager;
|
import android.preference.PreferenceManager;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
@@ -45,7 +46,7 @@ public class StorageAF {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean supportsStorageFramework() { return BuildCompat.getSdkVersion() >= BuildCompat.VERSION_KITKAT; }
|
public static boolean supportsStorageFramework() { return Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT; }
|
||||||
|
|
||||||
public static boolean useStorageFramework(Context ctx) {
|
public static boolean useStorageFramework(Context ctx) {
|
||||||
if (!supportsStorageFramework()) { return false; }
|
if (!supportsStorageFramework()) { return false; }
|
||||||
|
|||||||
@@ -19,6 +19,9 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.crypto.keyDerivation;
|
package com.kunzisoft.keepass.crypto.keyDerivation;
|
||||||
|
|
||||||
|
import android.content.res.Resources;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.crypto.CryptoUtil;
|
import com.kunzisoft.keepass.crypto.CryptoUtil;
|
||||||
import com.kunzisoft.keepass.crypto.finalkey.FinalKey;
|
import com.kunzisoft.keepass.crypto.finalkey.FinalKey;
|
||||||
import com.kunzisoft.keepass.crypto.finalkey.FinalKeyFactory;
|
import com.kunzisoft.keepass.crypto.finalkey.FinalKeyFactory;
|
||||||
@@ -30,8 +33,8 @@ import java.util.UUID;
|
|||||||
|
|
||||||
public class AesKdf extends KdfEngine {
|
public class AesKdf extends KdfEngine {
|
||||||
|
|
||||||
public static final int DEFAULT_ROUNDS = 6000;
|
private static final int DEFAULT_ROUNDS = 6000;
|
||||||
public static final String DEFAULT_NAME = "AES-KDF";
|
private static final String DEFAULT_NAME = "AES-KDF";
|
||||||
|
|
||||||
public static final UUID CIPHER_UUID = Types.bytestoUUID(
|
public static final UUID CIPHER_UUID = Types.bytestoUUID(
|
||||||
new byte[]{(byte) 0xC9, (byte) 0xD9, (byte) 0xF3, (byte) 0x9A, (byte) 0x62, (byte) 0x8A, (byte) 0x44, (byte) 0x60,
|
new byte[]{(byte) 0xC9, (byte) 0xD9, (byte) 0xF3, (byte) 0x9A, (byte) 0x62, (byte) 0x8A, (byte) 0x44, (byte) 0x60,
|
||||||
@@ -41,17 +44,22 @@ public class AesKdf extends KdfEngine {
|
|||||||
public static final String ParamRounds = "R";
|
public static final String ParamRounds = "R";
|
||||||
public static final String ParamSeed = "S";
|
public static final String ParamSeed = "S";
|
||||||
|
|
||||||
public AesKdf() {
|
AesKdf() {
|
||||||
uuid = CIPHER_UUID;
|
setUUID(CIPHER_UUID);
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getName() {
|
@Override
|
||||||
return DEFAULT_NAME;
|
public String getName(Resources resources) {
|
||||||
|
if (resources == null)
|
||||||
|
return DEFAULT_NAME;
|
||||||
|
return resources.getString(R.string.kdf_AES);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public KdfParameters getDefaultParameters() {
|
public KdfParameters getDefaultParameters() {
|
||||||
KdfParameters p = super.getDefaultParameters();
|
KdfParameters p = new KdfParameters(uuid);
|
||||||
|
|
||||||
|
p.setParamUUID();
|
||||||
p.setUInt32(ParamRounds, DEFAULT_ROUNDS);
|
p.setUInt32(ParamRounds, DEFAULT_ROUNDS);
|
||||||
|
|
||||||
return p;
|
return p;
|
||||||
@@ -93,4 +101,9 @@ public class AesKdf extends KdfEngine {
|
|||||||
public void setKeyRounds(KdfParameters p, long keyRounds) {
|
public void setKeyRounds(KdfParameters p, long keyRounds) {
|
||||||
p.setUInt64(ParamRounds, keyRounds);
|
p.setUInt64(ParamRounds, keyRounds);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getDefaultKeyRounds() {
|
||||||
|
return DEFAULT_ROUNDS;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,9 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.crypto.keyDerivation;
|
package com.kunzisoft.keepass.crypto.keyDerivation;
|
||||||
|
|
||||||
|
import android.content.res.Resources;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.utils.Types;
|
import com.kunzisoft.keepass.utils.Types;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
@@ -26,21 +29,22 @@ import java.security.SecureRandom;
|
|||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
public class Argon2Kdf extends KdfEngine {
|
public class Argon2Kdf extends KdfEngine {
|
||||||
|
|
||||||
public static final UUID CIPHER_UUID = Types.bytestoUUID(
|
public static final UUID CIPHER_UUID = Types.bytestoUUID(
|
||||||
new byte[]{(byte) 0xEF, (byte) 0x63, (byte) 0x6D, (byte) 0xDF, (byte) 0x8C, (byte) 0x29, (byte) 0x44, (byte) 0x4B,
|
new byte[]{(byte) 0xEF, (byte) 0x63, (byte) 0x6D, (byte) 0xDF, (byte) 0x8C, (byte) 0x29, (byte) 0x44, (byte) 0x4B,
|
||||||
(byte) 0x91, (byte) 0xF7, (byte) 0xA9, (byte) 0xA4, (byte)0x03, (byte) 0xE3, (byte) 0x0A, (byte) 0x0C
|
(byte) 0x91, (byte) 0xF7, (byte) 0xA9, (byte) 0xA4, (byte)0x03, (byte) 0xE3, (byte) 0x0A, (byte) 0x0C
|
||||||
});
|
});
|
||||||
|
|
||||||
public static final String ParamSalt = "S"; // byte[]
|
private static final String ParamSalt = "S"; // byte[]
|
||||||
public static final String ParamParallelism = "P"; // UInt32
|
private static final String ParamParallelism = "P"; // UInt32
|
||||||
public static final String ParamMemory = "M"; // UInt64
|
private static final String ParamMemory = "M"; // UInt64
|
||||||
public static final String ParamIterations = "I"; // UInt64
|
private static final String ParamIterations = "I"; // UInt64
|
||||||
public static final String ParamVersion = "V"; // UInt32
|
private static final String ParamVersion = "V"; // UInt32
|
||||||
public static final String ParamSecretKey = "K"; // byte[]
|
private static final String ParamSecretKey = "K"; // byte[]
|
||||||
public static final String ParamAssocData = "A"; // byte[]
|
private static final String ParamAssocData = "A"; // byte[]
|
||||||
|
|
||||||
public static final long MinVersion = 0x10;
|
private static final long MinVersion = 0x10;
|
||||||
public static final long MaxVersion = 0x13;
|
private static final long MaxVersion = 0x13;
|
||||||
|
|
||||||
private static final int MinSalt = 8;
|
private static final int MinSalt = 8;
|
||||||
private static final int MaxSalt = Integer.MAX_VALUE;
|
private static final int MaxSalt = Integer.MAX_VALUE;
|
||||||
@@ -58,22 +62,28 @@ public class Argon2Kdf extends KdfEngine {
|
|||||||
private static final long DefaultMemory = 1024 * 1024;
|
private static final long DefaultMemory = 1024 * 1024;
|
||||||
private static final long DefaultParallelism = 2;
|
private static final long DefaultParallelism = 2;
|
||||||
|
|
||||||
public Argon2Kdf() {
|
private static final String DEFAULT_NAME = "Argon2";
|
||||||
uuid = CIPHER_UUID;
|
|
||||||
|
Argon2Kdf() {
|
||||||
|
setUUID(CIPHER_UUID);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getName() {
|
public String getName(Resources resources) {
|
||||||
return "Argon2";
|
if (resources == null)
|
||||||
|
return DEFAULT_NAME;
|
||||||
|
return resources.getString(R.string.kdf_Argon2);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public KdfParameters getDefaultParameters() {
|
public KdfParameters getDefaultParameters() {
|
||||||
KdfParameters p = super.getDefaultParameters();
|
KdfParameters p = new KdfParameters(uuid);
|
||||||
|
|
||||||
p.setUInt32(ParamVersion, MaxVersion);
|
p.setParamUUID();
|
||||||
p.setUInt64(ParamMemory, DefaultMemory);
|
|
||||||
p.setUInt32(ParamParallelism, DefaultParallelism);
|
p.setUInt32(ParamParallelism, DefaultParallelism);
|
||||||
|
p.setUInt64(ParamMemory, DefaultMemory);
|
||||||
|
p.setUInt64(ParamIterations, DefaultIterations);
|
||||||
|
p.setUInt32(ParamVersion, MaxVersion);
|
||||||
|
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
@@ -113,4 +123,35 @@ public class Argon2Kdf extends KdfEngine {
|
|||||||
p.setUInt64(ParamIterations, keyRounds);
|
p.setUInt64(ParamIterations, keyRounds);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getDefaultKeyRounds() {
|
||||||
|
return DefaultIterations;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getMemoryUsage(KdfParameters p) {
|
||||||
|
return p.getUInt64(ParamMemory);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMemoryUsage(KdfParameters p, long memory) {
|
||||||
|
p.setUInt64(ParamMemory, memory);
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getDefaultMemoryUsage() {
|
||||||
|
return DefaultMemory;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getParallelism(KdfParameters p) {
|
||||||
|
return (int) p.getUInt32(ParamParallelism); // TODO Verify
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setParallelism(KdfParameters p, int parallelism) {
|
||||||
|
p.setUInt32(ParamParallelism, parallelism);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getDefaultParallelism() {
|
||||||
|
return (int) DefaultParallelism; // TODO Verify
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,24 +19,59 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.crypto.keyDerivation;
|
package com.kunzisoft.keepass.crypto.keyDerivation;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.ObjectNameResource;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
public abstract class KdfEngine {
|
public abstract class KdfEngine implements ObjectNameResource{
|
||||||
public UUID uuid;
|
|
||||||
|
|
||||||
public KdfParameters getDefaultParameters() {
|
public static final int UNKNOW_VALUE = -1;
|
||||||
return new KdfParameters(uuid);
|
public static final String UNKNOW_VALUE_STRING = String.valueOf(-1);
|
||||||
|
|
||||||
|
protected UUID uuid;
|
||||||
|
|
||||||
|
public UUID getUUID() {
|
||||||
|
return uuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setUUID(UUID uuid) {
|
||||||
|
this.uuid = uuid;
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract KdfParameters getDefaultParameters();
|
||||||
|
|
||||||
public abstract byte[] transform(byte[] masterKey, KdfParameters p) throws IOException;
|
public abstract byte[] transform(byte[] masterKey, KdfParameters p) throws IOException;
|
||||||
|
|
||||||
public abstract void randomize(KdfParameters p);
|
public abstract void randomize(KdfParameters p);
|
||||||
|
|
||||||
public abstract String getName();
|
|
||||||
|
|
||||||
public abstract long getKeyRounds(KdfParameters p);
|
public abstract long getKeyRounds(KdfParameters p);
|
||||||
|
|
||||||
public abstract void setKeyRounds(KdfParameters p, long keyRounds);
|
public abstract void setKeyRounds(KdfParameters p, long keyRounds);
|
||||||
|
|
||||||
|
public abstract long getDefaultKeyRounds();
|
||||||
|
|
||||||
|
public long getMemoryUsage(KdfParameters p) {
|
||||||
|
return UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMemoryUsage(KdfParameters p, long memory) {
|
||||||
|
// Do nothing by default
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getDefaultMemoryUsage() {
|
||||||
|
return UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getParallelism(KdfParameters p) {
|
||||||
|
return UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setParallelism(KdfParameters p, int parallelism) {
|
||||||
|
// Do nothing by default
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getDefaultParallelism() {
|
||||||
|
return UNKNOW_VALUE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,31 +19,37 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.crypto.keyDerivation;
|
package com.kunzisoft.keepass.crypto.keyDerivation;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.exception.UnknownKDF;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
public class KdfFactory {
|
public class KdfFactory {
|
||||||
|
|
||||||
public static List<KdfEngine> kdfList = new ArrayList<>();
|
public static AesKdf aesKdf = new AesKdf();
|
||||||
|
public static Argon2Kdf argon2Kdf = new Argon2Kdf();
|
||||||
|
|
||||||
|
public static List<KdfEngine> kdfListV3 = new ArrayList<>();
|
||||||
|
public static List<KdfEngine> kdfListV4 = new ArrayList<>();
|
||||||
|
|
||||||
static {
|
static {
|
||||||
kdfList.add(new AesKdf());
|
kdfListV3.add(aesKdf);
|
||||||
kdfList.add(new Argon2Kdf());
|
|
||||||
|
kdfListV4.add(aesKdf);
|
||||||
|
kdfListV4.add(argon2Kdf);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static KdfParameters getDefaultParameters() {
|
public static KdfEngine getEngineV4(KdfParameters kdfParameters) throws UnknownKDF {
|
||||||
return kdfList.get(0).getDefaultParameters();
|
UnknownKDF unknownKDFException = new UnknownKDF();
|
||||||
}
|
if (kdfParameters == null) {
|
||||||
|
throw unknownKDFException;
|
||||||
public static KdfEngine get(UUID uuid) {
|
}
|
||||||
for (KdfEngine engine: kdfList) {
|
for (KdfEngine engine: kdfListV4) {
|
||||||
if (engine.uuid.equals(uuid)) {
|
if (engine.getUUID().equals(kdfParameters.getUUID())) {
|
||||||
return engine;
|
return engine;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
throw unknownKDFException;
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,34 +31,36 @@ import java.util.UUID;
|
|||||||
|
|
||||||
public class KdfParameters extends VariantDictionary {
|
public class KdfParameters extends VariantDictionary {
|
||||||
|
|
||||||
public UUID kdfUUID;
|
private UUID kdfUUID;
|
||||||
|
|
||||||
private static final String ParamUUID = "$UUID";
|
private static final String ParamUUID = "$UUID";
|
||||||
|
|
||||||
public KdfParameters(UUID uuid) {
|
KdfParameters(UUID uuid) {
|
||||||
kdfUUID = uuid;
|
kdfUUID = uuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public UUID getUUID() {
|
||||||
|
return kdfUUID;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void setParamUUID() {
|
||||||
|
setByteArray(ParamUUID, Types.UUIDtoBytes(kdfUUID));
|
||||||
|
}
|
||||||
|
|
||||||
public static KdfParameters deserialize(byte[] data) throws IOException {
|
public static KdfParameters deserialize(byte[] data) throws IOException {
|
||||||
ByteArrayInputStream bis = new ByteArrayInputStream(data);
|
ByteArrayInputStream bis = new ByteArrayInputStream(data);
|
||||||
LEDataInputStream lis = new LEDataInputStream(bis);
|
LEDataInputStream lis = new LEDataInputStream(bis);
|
||||||
|
|
||||||
VariantDictionary d = VariantDictionary.deserialize(lis);
|
VariantDictionary d = VariantDictionary.deserialize(lis);
|
||||||
if (d == null) {
|
if (d == null) {
|
||||||
assert(false);
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
UUID uuid = Types.bytestoUUID(d.getByteArray(ParamUUID));
|
UUID uuid = Types.bytestoUUID(d.getByteArray(ParamUUID));
|
||||||
if (uuid == null) {
|
|
||||||
assert(false);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
KdfParameters kdfP = new KdfParameters(uuid);
|
KdfParameters kdfP = new KdfParameters(uuid);
|
||||||
kdfP.copyTo(d);
|
kdfP.copyTo(d);
|
||||||
return kdfP;
|
return kdfP;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static byte[] serialize(KdfParameters kdf) throws IOException {
|
public static byte[] serialize(KdfParameters kdf) throws IOException {
|
||||||
|
|||||||
@@ -19,20 +19,57 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.utils.MemUtil;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
public class AutoType implements Cloneable, Serializable {
|
public class AutoType implements Cloneable, Parcelable {
|
||||||
private static final long OBF_OPT_NONE = 0;
|
private static final long OBF_OPT_NONE = 0;
|
||||||
|
|
||||||
public boolean enabled = true;
|
public boolean enabled = true;
|
||||||
public long obfuscationOptions = OBF_OPT_NONE;
|
public long obfuscationOptions = OBF_OPT_NONE;
|
||||||
public String defaultSequence = "";
|
public String defaultSequence = "";
|
||||||
|
|
||||||
private HashMap<String, String> windowSeqPairs = new HashMap<>();
|
private HashMap<String, String> windowSeqPairs = new HashMap<>();
|
||||||
|
|
||||||
|
public AutoType() {}
|
||||||
|
|
||||||
|
public AutoType(Parcel in) {
|
||||||
|
enabled = in.readByte() != 0;
|
||||||
|
obfuscationOptions = in.readLong();
|
||||||
|
defaultSequence = in.readString();
|
||||||
|
windowSeqPairs = MemUtil.readStringParcelableMap(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
dest.writeByte((byte) (enabled ? 1 : 0));
|
||||||
|
dest.writeLong(obfuscationOptions);
|
||||||
|
dest.writeString(defaultSequence);
|
||||||
|
MemUtil.writeStringParcelableMap(dest, windowSeqPairs);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Parcelable.Creator<AutoType> CREATOR = new Parcelable.Creator<AutoType>() {
|
||||||
|
@Override
|
||||||
|
public AutoType createFromParcel(Parcel in) {
|
||||||
|
return new AutoType(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public AutoType[] newArray(int size) {
|
||||||
|
return new AutoType[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public AutoType clone() {
|
public AutoType clone() {
|
||||||
AutoType auto;
|
AutoType auto;
|
||||||
|
|||||||
@@ -20,23 +20,21 @@
|
|||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.SharedPreferences;
|
|
||||||
import android.content.res.Resources;
|
import android.content.res.Resources;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.preference.PreferenceManager;
|
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfEngine;
|
||||||
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfFactory;
|
||||||
import com.kunzisoft.keepass.database.exception.ContentFileNotFoundException;
|
import com.kunzisoft.keepass.database.exception.ContentFileNotFoundException;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidPasswordException;
|
|
||||||
import com.kunzisoft.keepass.database.exception.PwDbOutputException;
|
import com.kunzisoft.keepass.database.exception.PwDbOutputException;
|
||||||
import com.kunzisoft.keepass.database.load.Importer;
|
import com.kunzisoft.keepass.database.load.Importer;
|
||||||
import com.kunzisoft.keepass.database.load.ImporterFactory;
|
import com.kunzisoft.keepass.database.load.ImporterFactory;
|
||||||
import com.kunzisoft.keepass.database.save.PwDbOutput;
|
import com.kunzisoft.keepass.database.save.PwDbOutput;
|
||||||
import com.kunzisoft.keepass.icons.IconDrawableFactory;
|
import com.kunzisoft.keepass.icons.IconDrawableFactory;
|
||||||
import com.kunzisoft.keepass.search.SearchDbHelper;
|
import com.kunzisoft.keepass.search.SearchDbHelper;
|
||||||
import com.kunzisoft.keepass.tasks.UpdateStatus;
|
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater;
|
||||||
import com.kunzisoft.keepass.utils.UriUtil;
|
import com.kunzisoft.keepass.utils.UriUtil;
|
||||||
|
|
||||||
import java.io.BufferedInputStream;
|
import java.io.BufferedInputStream;
|
||||||
@@ -47,6 +45,11 @@ import java.io.IOException;
|
|||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
import java.io.SyncFailedException;
|
import java.io.SyncFailedException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
|
||||||
public class Database {
|
public class Database {
|
||||||
@@ -99,19 +102,11 @@ public class Database {
|
|||||||
loaded = true;
|
loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void loadData(Context ctx, InputStream is, String password, InputStream keyInputStream) throws IOException, InvalidDBException {
|
public void loadData(Context ctx, Uri uri, String password, Uri keyfile, ProgressTaskUpdater status) throws IOException, FileNotFoundException, InvalidDBException {
|
||||||
loadData(ctx, is, password, keyInputStream, new UpdateStatus(), !Importer.DEBUG);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void loadData(Context ctx, Uri uri, String password, Uri keyfile) throws IOException, FileNotFoundException, InvalidDBException {
|
|
||||||
loadData(ctx, uri, password, keyfile, new UpdateStatus(), !Importer.DEBUG);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void loadData(Context ctx, Uri uri, String password, Uri keyfile, UpdateStatus status) throws IOException, FileNotFoundException, InvalidDBException {
|
|
||||||
loadData(ctx, uri, password, keyfile, status, !Importer.DEBUG);
|
loadData(ctx, uri, password, keyfile, status, !Importer.DEBUG);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void loadData(Context ctx, Uri uri, String password, Uri keyfile, UpdateStatus status, boolean debug) throws IOException, FileNotFoundException, InvalidDBException {
|
private void loadData(Context ctx, Uri uri, String password, Uri keyfile, ProgressTaskUpdater status, boolean debug) throws IOException, FileNotFoundException, InvalidDBException {
|
||||||
mUri = uri;
|
mUri = uri;
|
||||||
readOnly = false;
|
readOnly = false;
|
||||||
if (uri.getScheme().equals("file")) {
|
if (uri.getScheme().equals("file")) {
|
||||||
@@ -119,26 +114,10 @@ public class Database {
|
|||||||
readOnly = !file.canWrite();
|
readOnly = !file.canWrite();
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
passUrisAsInputStreams(ctx, uri, password, keyfile, status, debug);
|
||||||
passUrisAsInputStreams(ctx, uri, password, keyfile, status, debug, 0);
|
|
||||||
} catch (InvalidPasswordException e) {
|
|
||||||
// Retry with rounds fix
|
|
||||||
try {
|
|
||||||
passUrisAsInputStreams(ctx, uri, password, keyfile, status, debug, getFixRounds(ctx));
|
|
||||||
} catch (Exception e2) {
|
|
||||||
// Rethrow original exception
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private long getFixRounds(Context ctx) {
|
private void passUrisAsInputStreams(Context ctx, Uri uri, String password, Uri keyfile, ProgressTaskUpdater status, boolean debug) throws IOException, FileNotFoundException, InvalidDBException {
|
||||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ctx);
|
|
||||||
return prefs.getLong(ctx.getString(R.string.roundsFix_key), ctx.getResources().getInteger(R.integer.roundsFix_default));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
private void passUrisAsInputStreams(Context ctx, Uri uri, String password, Uri keyfile, UpdateStatus status, boolean debug, long roundsFix) throws IOException, FileNotFoundException, InvalidDBException {
|
|
||||||
InputStream is, kfIs;
|
InputStream is, kfIs;
|
||||||
try {
|
try {
|
||||||
is = UriUtil.getUriInputStream(ctx, uri);
|
is = UriUtil.getUriInputStream(ctx, uri);
|
||||||
@@ -153,18 +132,14 @@ public class Database {
|
|||||||
Log.e("KPD", "Database::loadData", e);
|
Log.e("KPD", "Database::loadData", e);
|
||||||
throw ContentFileNotFoundException.getInstance(keyfile);
|
throw ContentFileNotFoundException.getInstance(keyfile);
|
||||||
}
|
}
|
||||||
loadData(ctx, is, password, kfIs, status, debug, roundsFix);
|
loadData(ctx, is, password, kfIs, status, debug);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void loadData(Context ctx, InputStream is, String password, InputStream kfIs, boolean debug) throws IOException, InvalidDBException {
|
public void loadData(Context ctx, InputStream is, String password, InputStream keyFileInputStream, boolean debug) throws IOException, InvalidDBException {
|
||||||
loadData(ctx, is, password, kfIs, new UpdateStatus(), debug);
|
loadData(ctx, is, password, keyFileInputStream, null, debug);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void loadData(Context ctx, InputStream is, String password, InputStream kfIs, UpdateStatus status, boolean debug) throws IOException, InvalidDBException {
|
private void loadData(Context ctx, InputStream is, String password, InputStream keyFileInputStream, ProgressTaskUpdater progressTaskUpdater, boolean debug) throws IOException, InvalidDBException {
|
||||||
loadData(ctx, is, password, kfIs, status, debug, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void loadData(Context ctx, InputStream is, String password, InputStream kfIs, UpdateStatus status, boolean debug, long roundsFix) throws IOException, InvalidDBException {
|
|
||||||
BufferedInputStream bis = new BufferedInputStream(is);
|
BufferedInputStream bis = new BufferedInputStream(is);
|
||||||
|
|
||||||
if ( ! bis.markSupported() ) {
|
if ( ! bis.markSupported() ) {
|
||||||
@@ -174,11 +149,11 @@ public class Database {
|
|||||||
// We'll end up reading 8 bytes to identify the header. Might as well use two extra.
|
// We'll end up reading 8 bytes to identify the header. Might as well use two extra.
|
||||||
bis.mark(10);
|
bis.mark(10);
|
||||||
|
|
||||||
Importer imp = ImporterFactory.createImporter(bis, debug);
|
Importer databaseImporter = ImporterFactory.createImporter(bis, debug);
|
||||||
|
|
||||||
bis.reset(); // Return to the start
|
bis.reset(); // Return to the start
|
||||||
|
|
||||||
pm = imp.openDatabase(bis, password, kfIs, status, roundsFix);
|
pm = databaseImporter.openDatabase(bis, password, keyFileInputStream, progressTaskUpdater);
|
||||||
if ( pm != null ) {
|
if ( pm != null ) {
|
||||||
try {
|
try {
|
||||||
switch (pm.getVersion()) {
|
switch (pm.getVersion()) {
|
||||||
@@ -222,19 +197,26 @@ public class Database {
|
|||||||
saveData(ctx, mUri);
|
saveData(ctx, mUri);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void saveData(Context ctx, Uri uri) throws IOException, PwDbOutputException {
|
private void saveData(Context ctx, Uri uri) throws IOException, PwDbOutputException {
|
||||||
|
String errorMessage = "Failed to store database.";
|
||||||
|
|
||||||
if (uri.getScheme().equals("file")) {
|
if (uri.getScheme().equals("file")) {
|
||||||
String filename = uri.getPath();
|
String filename = uri.getPath();
|
||||||
File tempFile = new File(filename + ".tmp");
|
File tempFile = new File(filename + ".tmp");
|
||||||
FileOutputStream fos = new FileOutputStream(tempFile);
|
|
||||||
//BufferedOutputStream bos = new BufferedOutputStream(fos);
|
|
||||||
|
|
||||||
//PwDbV3Output pmo = new PwDbV3Output(pm, bos, App.getCalendar());
|
FileOutputStream fos = null;
|
||||||
PwDbOutput pmo = PwDbOutput.getInstance(pm, fos);
|
try {
|
||||||
pmo.output();
|
fos = new FileOutputStream(tempFile);
|
||||||
//bos.flush();
|
PwDbOutput pmo = PwDbOutput.getInstance(pm, fos);
|
||||||
//bos.close();
|
if (pmo != null)
|
||||||
fos.close();
|
pmo.output();
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, errorMessage, e);
|
||||||
|
throw new IOException(errorMessage, e);
|
||||||
|
} finally {
|
||||||
|
if (fos != null)
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
// Force data to disk before continuing
|
// Force data to disk before continuing
|
||||||
try {
|
try {
|
||||||
@@ -246,24 +228,28 @@ public class Database {
|
|||||||
File orig = new File(filename);
|
File orig = new File(filename);
|
||||||
|
|
||||||
if (!tempFile.renameTo(orig)) {
|
if (!tempFile.renameTo(orig)) {
|
||||||
throw new IOException("Failed to store database.");
|
throw new IOException(errorMessage);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
OutputStream os;
|
OutputStream os = null;
|
||||||
try {
|
try {
|
||||||
os = ctx.getContentResolver().openOutputStream(uri);
|
os = ctx.getContentResolver().openOutputStream(uri);
|
||||||
|
PwDbOutput pmo = PwDbOutput.getInstance(pm, os);
|
||||||
|
if (pmo != null)
|
||||||
|
pmo.output();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new IOException("Failed to store database.");
|
Log.e(TAG, errorMessage, e);
|
||||||
|
throw new IOException(errorMessage, e);
|
||||||
|
} finally {
|
||||||
|
if (os != null)
|
||||||
|
os.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
PwDbOutput pmo = PwDbOutput.getInstance(pm, os);
|
|
||||||
pmo.output();
|
|
||||||
os.close();
|
|
||||||
}
|
}
|
||||||
mUri = uri;
|
mUri = uri;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO Clear database when lock broadcast is receive in backstage
|
||||||
public void clear() {
|
public void clear() {
|
||||||
drawFactory.clearCache();
|
drawFactory.clearCache();
|
||||||
|
|
||||||
@@ -348,16 +334,88 @@ public class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwEncryptionAlgorithm getEncryptionAlgorithm() {
|
||||||
|
return getPwDatabase().getEncryptionAlgorithm();
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<PwEncryptionAlgorithm> getAvailableEncryptionAlgorithms() {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
return ((PwDatabaseV4) getPwDatabase()).getAvailableEncryptionAlgorithms();
|
||||||
|
case V3:
|
||||||
|
return ((PwDatabaseV3) getPwDatabase()).getAvailableEncryptionAlgorithms();
|
||||||
|
}
|
||||||
|
return new ArrayList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean allowEncryptionAlgorithmModification() {
|
||||||
|
return getAvailableEncryptionAlgorithms().size() > 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assignEncryptionAlgorithm(PwEncryptionAlgorithm algorithm) {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
((PwDatabaseV4) getPwDatabase()).setEncryptionAlgorithm(algorithm);
|
||||||
|
((PwDatabaseV4) getPwDatabase()).setDataEngine(algorithm.getCipherEngine());
|
||||||
|
((PwDatabaseV4) getPwDatabase()).setDataCipher(algorithm.getDataCipher());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public String getEncryptionAlgorithmName(Resources resources) {
|
public String getEncryptionAlgorithmName(Resources resources) {
|
||||||
return getPwDatabase().getEncryptionAlgorithm().getName(resources);
|
return getPwDatabase().getEncryptionAlgorithm().getName(resources);
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getKeyDerivationName() {
|
public List<KdfEngine> getAvailableKdfEngines() {
|
||||||
return getPwDatabase().getKeyDerivationName();
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
return KdfFactory.kdfListV4;
|
||||||
|
case V3:
|
||||||
|
return KdfFactory.kdfListV3;
|
||||||
|
}
|
||||||
|
return new ArrayList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean allowKdfModification() {
|
||||||
|
return getAvailableKdfEngines().size() > 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public KdfEngine getKdfEngine() {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
KdfEngine kdfEngine = ((PwDatabaseV4) getPwDatabase()).getKdfEngine();
|
||||||
|
if (kdfEngine == null)
|
||||||
|
return KdfFactory.aesKdf;
|
||||||
|
return kdfEngine;
|
||||||
|
default:
|
||||||
|
case V3:
|
||||||
|
return KdfFactory.aesKdf;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assignKdfEngine(KdfEngine kdfEngine) {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
PwDatabaseV4 db = ((PwDatabaseV4) getPwDatabase());
|
||||||
|
if (db.getKdfParameters() == null
|
||||||
|
|| !db.getKdfParameters().getUUID().equals(kdfEngine.getDefaultParameters().getUUID()))
|
||||||
|
db.setKdfParameters(kdfEngine.getDefaultParameters());
|
||||||
|
setNumberKeyEncryptionRounds(kdfEngine.getDefaultKeyRounds());
|
||||||
|
setMemoryUsage(kdfEngine.getDefaultMemoryUsage());
|
||||||
|
setParallelism(kdfEngine.getDefaultParallelism());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getKeyDerivationName(Resources resources) {
|
||||||
|
KdfEngine kdfEngine = getKdfEngine();
|
||||||
|
if (kdfEngine != null) {
|
||||||
|
return kdfEngine.getName(resources);
|
||||||
|
}
|
||||||
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getNumberKeyEncryptionRoundsAsString() {
|
public String getNumberKeyEncryptionRoundsAsString() {
|
||||||
return Long.toString(getPwDatabase().getNumberKeyEncryptionRounds());
|
return Long.toString(getNumberKeyEncryptionRounds());
|
||||||
}
|
}
|
||||||
|
|
||||||
public long getNumberKeyEncryptionRounds() {
|
public long getNumberKeyEncryptionRounds() {
|
||||||
@@ -368,6 +426,44 @@ public class Database {
|
|||||||
getPwDatabase().setNumberKeyEncryptionRounds(numberRounds);
|
getPwDatabase().setNumberKeyEncryptionRounds(numberRounds);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getMemoryUsageAsString() {
|
||||||
|
return Long.toString(getMemoryUsage());
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getMemoryUsage() {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
return ((PwDatabaseV4) getPwDatabase()).getMemoryUsage();
|
||||||
|
}
|
||||||
|
return KdfEngine.UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMemoryUsage(long memory) {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
((PwDatabaseV4) getPwDatabase()).setMemoryUsage(memory);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getParallelismAsString() {
|
||||||
|
return Integer.toString(getParallelism());
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getParallelism() {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
return ((PwDatabaseV4) getPwDatabase()).getParallelism();
|
||||||
|
}
|
||||||
|
return KdfEngine.UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setParallelism(int parallelism) {
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V4:
|
||||||
|
((PwDatabaseV4) getPwDatabase()).setParallelism(parallelism);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public PwEntry createEntry(PwGroup parent) {
|
public PwEntry createEntry(PwGroup parent) {
|
||||||
PwEntry newPwEntry = null;
|
PwEntry newPwEntry = null;
|
||||||
try {
|
try {
|
||||||
@@ -547,6 +643,44 @@ public class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return A duplicate entry with the same values, a new UUID,
|
||||||
|
* @param entryToCopy
|
||||||
|
* @param newParent
|
||||||
|
*/
|
||||||
|
public @Nullable PwEntry copyEntry(PwEntry entryToCopy, PwGroup newParent) {
|
||||||
|
try {
|
||||||
|
// TODO encapsulate
|
||||||
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
case V3:
|
||||||
|
PwEntryV3 entryV3Copied = ((PwEntryV3) entryToCopy).clone();
|
||||||
|
entryV3Copied.setUUID(UUID.randomUUID());
|
||||||
|
entryV3Copied.setParent((PwGroupV3) newParent);
|
||||||
|
addEntryTo(entryV3Copied, newParent);
|
||||||
|
return entryV3Copied;
|
||||||
|
case V4:
|
||||||
|
PwEntryV4 entryV4Copied = ((PwEntryV4) entryToCopy).clone();
|
||||||
|
entryV4Copied.setUUID(UUID.randomUUID());
|
||||||
|
entryV4Copied.setParent((PwGroupV4) newParent);
|
||||||
|
addEntryTo(entryV4Copied, newParent);
|
||||||
|
return entryV4Copied;
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "This version of PwEntry can't be updated", e);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void moveEntry(PwEntry entryToMove, PwGroup newParent) {
|
||||||
|
removeEntryFrom(entryToMove, entryToMove.parent);
|
||||||
|
addEntryTo(entryToMove, newParent);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void moveGroup(PwGroup groupToMove, PwGroup newParent) {
|
||||||
|
removeGroupFrom(groupToMove, groupToMove.parent);
|
||||||
|
addGroupTo(groupToMove, newParent);
|
||||||
|
}
|
||||||
|
|
||||||
public void deleteEntry(PwEntry entry) {
|
public void deleteEntry(PwEntry entry) {
|
||||||
try {
|
try {
|
||||||
switch (getPwDatabase().getVersion()) {
|
switch (getPwDatabase().getVersion()) {
|
||||||
|
|||||||
@@ -57,7 +57,7 @@ public class EntrySearchV4 {
|
|||||||
|
|
||||||
boolean negate = false;
|
boolean negate = false;
|
||||||
if (sp.searchString.startsWith("-")) {
|
if (sp.searchString.startsWith("-")) {
|
||||||
sp.searchString.substring(1); // TODO Verify bug or no need ?
|
sp.searchString = sp.searchString.substring(1);
|
||||||
negate = sp.searchString.length() > 0;
|
negate = sp.searchString.length() > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,9 +19,12 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedString;
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.security.ProtectedString;
|
||||||
|
import com.kunzisoft.keepass.utils.MemUtil;
|
||||||
|
|
||||||
import java.io.Serializable;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
@@ -32,7 +35,7 @@ import static com.kunzisoft.keepass.database.PwEntryV4.STR_TITLE;
|
|||||||
import static com.kunzisoft.keepass.database.PwEntryV4.STR_URL;
|
import static com.kunzisoft.keepass.database.PwEntryV4.STR_URL;
|
||||||
import static com.kunzisoft.keepass.database.PwEntryV4.STR_USERNAME;
|
import static com.kunzisoft.keepass.database.PwEntryV4.STR_USERNAME;
|
||||||
|
|
||||||
public class ExtraFields implements Serializable, Cloneable {
|
public class ExtraFields implements Parcelable, Cloneable {
|
||||||
|
|
||||||
private Map<String, ProtectedString> fields;
|
private Map<String, ProtectedString> fields;
|
||||||
|
|
||||||
@@ -40,10 +43,52 @@ public class ExtraFields implements Serializable, Cloneable {
|
|||||||
fields = new HashMap<>();
|
fields = new HashMap<>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ExtraFields(Parcel in) {
|
||||||
|
fields = MemUtil.readStringParcelableMap(in, ProtectedString.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
MemUtil.writeStringParcelableMap(dest, flags, fields);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Parcelable.Creator<ExtraFields> CREATOR = new Parcelable.Creator<ExtraFields>() {
|
||||||
|
@Override
|
||||||
|
public ExtraFields createFromParcel(Parcel in) {
|
||||||
|
return new ExtraFields(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ExtraFields[] newArray(int size) {
|
||||||
|
return new ExtraFields[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
public boolean containsCustomFields() {
|
public boolean containsCustomFields() {
|
||||||
return !getCustomProtectedFields().keySet().isEmpty();
|
return !getCustomProtectedFields().keySet().isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean containsCustomFieldsProtected() {
|
||||||
|
for (Map.Entry<String, ProtectedString> field : getCustomProtectedFields().entrySet()) {
|
||||||
|
if (field.getValue().isProtected())
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean containsCustomFieldsNotProtected() {
|
||||||
|
for (Map.Entry<String, ProtectedString> field : getCustomProtectedFields().entrySet()) {
|
||||||
|
if (!field.getValue().isProtected())
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
public String getProtectedStringValue(String key) {
|
public String getProtectedStringValue(String key) {
|
||||||
ProtectedString value = fields.get(key);
|
ProtectedString value = fields.get(key);
|
||||||
if ( value == null ) return "";
|
if ( value == null ) return "";
|
||||||
|
|||||||
@@ -17,16 +17,13 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import android.os.Handler;
|
import android.content.res.Resources;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.PwNode;
|
/**
|
||||||
|
* Interface to generify items with a name resource, that can be (for example) visible in a list
|
||||||
public abstract class AfterActionNodeOnFinish extends OnFinish {
|
*/
|
||||||
public AfterActionNodeOnFinish(Handler handler) {
|
public interface ObjectNameResource {
|
||||||
super(handler);
|
String getName(Resources resources);
|
||||||
}
|
|
||||||
|
|
||||||
public abstract void run(PwNode oldNode, PwNode newNode);
|
|
||||||
}
|
}
|
||||||
@@ -19,11 +19,8 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.crypto.finalkey.FinalKey;
|
|
||||||
import com.kunzisoft.keepass.crypto.finalkey.FinalKeyFactory;
|
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
||||||
import com.kunzisoft.keepass.database.exception.KeyFileEmptyException;
|
import com.kunzisoft.keepass.database.exception.KeyFileEmptyException;
|
||||||
import com.kunzisoft.keepass.stream.NullOutputStream;
|
|
||||||
import com.kunzisoft.keepass.utils.Util;
|
import com.kunzisoft.keepass.utils.Util;
|
||||||
|
|
||||||
import java.io.ByteArrayInputStream;
|
import java.io.ByteArrayInputStream;
|
||||||
@@ -31,7 +28,6 @@ import java.io.ByteArrayOutputStream;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
import java.security.DigestOutputStream;
|
|
||||||
import java.security.MessageDigest;
|
import java.security.MessageDigest;
|
||||||
import java.security.NoSuchAlgorithmException;
|
import java.security.NoSuchAlgorithmException;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@@ -100,39 +96,9 @@ public abstract class PwDatabase<PwGroupDB extends PwGroup<PwGroupDB, PwGroupDB,
|
|||||||
return finalKey;
|
return finalKey;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void makeFinalKey(byte[] masterSeed, byte[] masterSeed2, long numRounds) throws IOException {
|
|
||||||
|
|
||||||
// Write checksum Checksum
|
|
||||||
MessageDigest md;
|
|
||||||
try {
|
|
||||||
md = MessageDigest.getInstance("SHA-256");
|
|
||||||
} catch (NoSuchAlgorithmException e) {
|
|
||||||
throw new IOException("SHA-256 not implemented here.");
|
|
||||||
}
|
|
||||||
NullOutputStream nos = new NullOutputStream();
|
|
||||||
DigestOutputStream dos = new DigestOutputStream(nos, md);
|
|
||||||
|
|
||||||
byte[] transformedMasterKey = transformMasterKey(masterSeed2, masterKey, numRounds);
|
|
||||||
dos.write(masterSeed);
|
|
||||||
dos.write(transformedMasterKey);
|
|
||||||
|
|
||||||
finalKey = md.digest();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Encrypt the master key a few times to make brute-force key-search harder
|
|
||||||
* @throws IOException
|
|
||||||
*/
|
|
||||||
protected static byte[] transformMasterKey( byte[] pKeySeed, byte[] pKey, long rounds ) throws IOException {
|
|
||||||
FinalKey key = FinalKeyFactory.createFinalKey();
|
|
||||||
|
|
||||||
return key.transformMasterKey(pKeySeed, pKey, rounds);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public abstract byte[] getMasterKey(String key, InputStream keyInputStream) throws InvalidKeyFileException, IOException;
|
public abstract byte[] getMasterKey(String key, InputStream keyInputStream) throws InvalidKeyFileException, IOException;
|
||||||
|
|
||||||
public void setMasterKey(String key, InputStream keyInputStream)
|
public void retrieveMasterKey(String key, InputStream keyInputStream)
|
||||||
throws InvalidKeyFileException, IOException {
|
throws InvalidKeyFileException, IOException {
|
||||||
masterKey = getMasterKey(key, keyInputStream);
|
masterKey = getMasterKey(key, keyInputStream);
|
||||||
}
|
}
|
||||||
@@ -280,7 +246,7 @@ public abstract class PwDatabase<PwGroupDB extends PwGroup<PwGroupDB, PwGroupDB,
|
|||||||
this.algorithm = algorithm;
|
this.algorithm = algorithm;
|
||||||
}
|
}
|
||||||
|
|
||||||
public abstract String getKeyDerivationName();
|
public abstract List<PwEncryptionAlgorithm> getAvailableEncryptionAlgorithms();
|
||||||
|
|
||||||
public abstract List<PwGroupDB> getGrpRoots();
|
public abstract List<PwGroupDB> getGrpRoots();
|
||||||
|
|
||||||
|
|||||||
@@ -45,11 +45,16 @@ Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|||||||
|
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.AesKdf;
|
import com.kunzisoft.keepass.crypto.finalkey.FinalKey;
|
||||||
|
import com.kunzisoft.keepass.crypto.finalkey.FinalKeyFactory;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
||||||
|
import com.kunzisoft.keepass.stream.NullOutputStream;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
import java.security.DigestOutputStream;
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
@@ -95,10 +100,12 @@ public class PwDatabaseV3 extends PwDatabase<PwGroupV3, PwEntryV3> {
|
|||||||
return PwVersion.V3;
|
return PwVersion.V3;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getKeyDerivationName() {
|
public List<PwEncryptionAlgorithm> getAvailableEncryptionAlgorithms() {
|
||||||
return AesKdf.DEFAULT_NAME;
|
List<PwEncryptionAlgorithm> list = new ArrayList<>();
|
||||||
}
|
list.add(PwEncryptionAlgorithm.AES_Rijndael);
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<PwGroupV3> getGroups() {
|
public List<PwGroupV3> getGroups() {
|
||||||
@@ -250,6 +257,35 @@ public class PwDatabaseV3 extends PwDatabase<PwGroupV3, PwEntryV3> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encrypt the master key a few times to make brute-force key-search harder
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
private static byte[] transformMasterKey( byte[] pKeySeed, byte[] pKey, long rounds ) throws IOException {
|
||||||
|
FinalKey key = FinalKeyFactory.createFinalKey();
|
||||||
|
|
||||||
|
return key.transformMasterKey(pKeySeed, pKey, rounds);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void makeFinalKey(byte[] masterSeed, byte[] masterSeed2, long numRounds) throws IOException {
|
||||||
|
|
||||||
|
// Write checksum Checksum
|
||||||
|
MessageDigest md;
|
||||||
|
try {
|
||||||
|
md = MessageDigest.getInstance("SHA-256");
|
||||||
|
} catch (NoSuchAlgorithmException e) {
|
||||||
|
throw new IOException("SHA-256 not implemented here.");
|
||||||
|
}
|
||||||
|
NullOutputStream nos = new NullOutputStream();
|
||||||
|
DigestOutputStream dos = new DigestOutputStream(nos, md);
|
||||||
|
|
||||||
|
byte[] transformedMasterKey = transformMasterKey(masterSeed2, masterKey, numRounds);
|
||||||
|
dos.write(masterSeed);
|
||||||
|
dos.write(transformedMasterKey);
|
||||||
|
|
||||||
|
finalKey = md.digest();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected String getPasswordEncoding() {
|
protected String getPasswordEncoding() {
|
||||||
return "ISO-8859-1";
|
return "ISO-8859-1";
|
||||||
@@ -288,7 +324,6 @@ public class PwDatabaseV3 extends PwDatabase<PwGroupV3, PwEntryV3> {
|
|||||||
|
|
||||||
// Add tree to root groups
|
// Add tree to root groups
|
||||||
groups.add(newGroup);
|
groups.add(newGroup);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -19,17 +19,18 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.util.Log;
|
||||||
import android.webkit.URLUtil;
|
import android.webkit.URLUtil;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.collections.VariantDictionary;
|
import com.kunzisoft.keepass.collections.VariantDictionary;
|
||||||
import com.kunzisoft.keepass.crypto.CryptoUtil;
|
import com.kunzisoft.keepass.crypto.CryptoUtil;
|
||||||
import com.kunzisoft.keepass.crypto.engine.AesEngine;
|
import com.kunzisoft.keepass.crypto.engine.AesEngine;
|
||||||
import com.kunzisoft.keepass.crypto.engine.CipherEngine;
|
import com.kunzisoft.keepass.crypto.engine.CipherEngine;
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.AesKdf;
|
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.KdfEngine;
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfEngine;
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.KdfFactory;
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfFactory;
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.KdfParameters;
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfParameters;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
||||||
|
import com.kunzisoft.keepass.database.exception.UnknownKDF;
|
||||||
import com.kunzisoft.keepass.utils.EmptyUtils;
|
import com.kunzisoft.keepass.utils.EmptyUtils;
|
||||||
|
|
||||||
import org.w3c.dom.Document;
|
import org.w3c.dom.Document;
|
||||||
@@ -50,6 +51,7 @@ import java.util.List;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
import javax.xml.parsers.DocumentBuilder;
|
import javax.xml.parsers.DocumentBuilder;
|
||||||
import javax.xml.parsers.DocumentBuilderFactory;
|
import javax.xml.parsers.DocumentBuilderFactory;
|
||||||
|
|
||||||
@@ -57,6 +59,7 @@ import biz.source_code.base64Coder.Base64Coder;
|
|||||||
|
|
||||||
|
|
||||||
public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
||||||
|
private static final String TAG = PwDatabaseV4.class.getName();
|
||||||
|
|
||||||
private static final int DEFAULT_HISTORY_MAX_ITEMS = 10; // -1 unlimited
|
private static final int DEFAULT_HISTORY_MAX_ITEMS = 10; // -1 unlimited
|
||||||
private static final long DEFAULT_HISTORY_MAX_SIZE = 6 * 1024 * 1024; // -1 unlimited
|
private static final long DEFAULT_HISTORY_MAX_SIZE = 6 * 1024 * 1024; // -1 unlimited
|
||||||
@@ -66,9 +69,10 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
private UUID dataCipher = AesEngine.CIPHER_UUID;
|
private UUID dataCipher = AesEngine.CIPHER_UUID;
|
||||||
private CipherEngine dataEngine = new AesEngine();
|
private CipherEngine dataEngine = new AesEngine();
|
||||||
private PwCompressionAlgorithm compressionAlgorithm = PwCompressionAlgorithm.Gzip;
|
private PwCompressionAlgorithm compressionAlgorithm = PwCompressionAlgorithm.Gzip;
|
||||||
private KdfEngine kdfEngine;
|
private KdfParameters kdfParameters;
|
||||||
|
private long numKeyEncRounds;
|
||||||
|
private VariantDictionary publicCustomData = new VariantDictionary();
|
||||||
|
|
||||||
private long numKeyEncRounds = AesKdf.DEFAULT_ROUNDS; // By default take the AES rounds
|
|
||||||
protected String name = "KeePass DX database";
|
protected String name = "KeePass DX database";
|
||||||
private PwDate nameChanged = new PwDate();
|
private PwDate nameChanged = new PwDate();
|
||||||
private PwDate settingsChanged = new PwDate();
|
private PwDate settingsChanged = new PwDate();
|
||||||
@@ -98,9 +102,7 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
private List<PwIconCustom> customIcons = new ArrayList<>();
|
private List<PwIconCustom> customIcons = new ArrayList<>();
|
||||||
private Map<String, String> customData = new HashMap<>();
|
private Map<String, String> customData = new HashMap<>();
|
||||||
|
|
||||||
private KdfParameters kdfParameters = KdfFactory.getDefaultParameters();
|
private BinaryPool binPool = new BinaryPool();
|
||||||
private VariantDictionary publicCustomData = new VariantDictionary();
|
|
||||||
private BinaryPool binPool = new BinaryPool();
|
|
||||||
|
|
||||||
public String localizedAppName = "KeePassDX"; // TODO resource
|
public String localizedAppName = "KeePassDX"; // TODO resource
|
||||||
|
|
||||||
@@ -131,9 +133,17 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
|
|
||||||
public void setDataEngine(CipherEngine dataEngine) {
|
public void setDataEngine(CipherEngine dataEngine) {
|
||||||
this.dataEngine = dataEngine;
|
this.dataEngine = dataEngine;
|
||||||
this.algorithm = dataEngine.getPwEncryptionAlgorithm();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<PwEncryptionAlgorithm> getAvailableEncryptionAlgorithms() {
|
||||||
|
List<PwEncryptionAlgorithm> list = new ArrayList<>();
|
||||||
|
list.add(PwEncryptionAlgorithm.AES_Rijndael);
|
||||||
|
list.add(PwEncryptionAlgorithm.Twofish);
|
||||||
|
list.add(PwEncryptionAlgorithm.ChaCha20);
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
public PwCompressionAlgorithm getCompressionAlgorithm() {
|
public PwCompressionAlgorithm getCompressionAlgorithm() {
|
||||||
return compressionAlgorithm;
|
return compressionAlgorithm;
|
||||||
}
|
}
|
||||||
@@ -142,6 +152,23 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
this.compressionAlgorithm = compressionAlgorithm;
|
this.compressionAlgorithm = compressionAlgorithm;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public @Nullable KdfEngine getKdfEngine() {
|
||||||
|
try {
|
||||||
|
return KdfFactory.getEngineV4(kdfParameters);
|
||||||
|
} catch (UnknownKDF unknownKDF) {
|
||||||
|
Log.i(TAG, "Unable to retrieve KDF engine", unknownKDF);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public KdfParameters getKdfParameters() {
|
||||||
|
return kdfParameters;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setKdfParameters(KdfParameters kdfParameters) {
|
||||||
|
this.kdfParameters = kdfParameters;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long getNumberKeyEncryptionRounds() {
|
public long getNumberKeyEncryptionRounds() {
|
||||||
if (getKdfEngine() != null && getKdfParameters() != null)
|
if (getKdfEngine() != null && getKdfParameters() != null)
|
||||||
@@ -156,7 +183,31 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
numKeyEncRounds = rounds;
|
numKeyEncRounds = rounds;
|
||||||
}
|
}
|
||||||
|
|
||||||
public PwDate getNameChanged() {
|
public long getMemoryUsage() {
|
||||||
|
if (getKdfEngine() != null && getKdfParameters() != null) {
|
||||||
|
return getKdfEngine().getMemoryUsage(getKdfParameters());
|
||||||
|
}
|
||||||
|
return KdfEngine.UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMemoryUsage(long memory) {
|
||||||
|
if (getKdfEngine() != null && getKdfParameters() != null)
|
||||||
|
getKdfEngine().setMemoryUsage(getKdfParameters(), memory);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getParallelism() {
|
||||||
|
if (getKdfEngine() != null && getKdfParameters() != null) {
|
||||||
|
return getKdfEngine().getParallelism(getKdfParameters());
|
||||||
|
}
|
||||||
|
return KdfEngine.UNKNOW_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setParallelism(int parallelism) {
|
||||||
|
if (getKdfEngine() != null && getKdfParameters() != null)
|
||||||
|
getKdfEngine().setParallelism(getKdfParameters(), parallelism);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PwDate getNameChanged() {
|
||||||
return nameChanged;
|
return nameChanged;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -326,15 +377,6 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
this.customData.put(label, value);
|
this.customData.put(label, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
public KdfEngine getKdfEngine() {
|
|
||||||
return kdfEngine;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getKeyDerivationName() {
|
|
||||||
return kdfEngine.getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public byte[] getMasterKey(String key, InputStream keyInputStream)
|
public byte[] getMasterKey(String key, InputStream keyInputStream)
|
||||||
throws InvalidKeyFileException, IOException {
|
throws InvalidKeyFileException, IOException {
|
||||||
@@ -359,47 +401,11 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
return md.digest(fKey);
|
return md.digest(fKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public void makeFinalKey(byte[] masterSeed) throws IOException {
|
||||||
public void makeFinalKey(byte[] masterSeed, byte[] masterSeed2, long numRounds) throws IOException {
|
|
||||||
|
|
||||||
byte[] transformedMasterKey = transformMasterKey(masterSeed2, masterKey, numRounds);
|
KdfEngine kdfEngine = KdfFactory.getEngineV4(kdfParameters);
|
||||||
|
|
||||||
|
byte[] transformedMasterKey = kdfEngine.transform(masterKey, kdfParameters);
|
||||||
byte[] cmpKey = new byte[65];
|
|
||||||
System.arraycopy(masterSeed, 0, cmpKey, 0, 32);
|
|
||||||
System.arraycopy(transformedMasterKey, 0, cmpKey, 32, 32);
|
|
||||||
finalKey = CryptoUtil.resizeKey(cmpKey, 0, 64, dataEngine.keyLength());
|
|
||||||
|
|
||||||
MessageDigest md;
|
|
||||||
try {
|
|
||||||
md = MessageDigest.getInstance("SHA-512");
|
|
||||||
cmpKey[64] = 1;
|
|
||||||
hmacKey = md.digest(cmpKey);
|
|
||||||
} catch (NoSuchAlgorithmException e) {
|
|
||||||
throw new IOException("No SHA-512 implementation");
|
|
||||||
} finally {
|
|
||||||
Arrays.fill(cmpKey, (byte)0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
public void makeFinalKey(byte[] masterSeed, KdfParameters kdfP) throws IOException {
|
|
||||||
makeFinalKey(masterSeed, kdfP, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void makeFinalKey(byte[] masterSeed, KdfParameters kdfP, long roundsFix)
|
|
||||||
throws IOException {
|
|
||||||
|
|
||||||
kdfEngine = KdfFactory.get(kdfP.kdfUUID);
|
|
||||||
if (kdfEngine == null) {
|
|
||||||
throw new IOException("Unknown key derivation function");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set to 6000 rounds to open corrupted database
|
|
||||||
if (roundsFix > 0 && kdfP.kdfUUID.equals(AesKdf.CIPHER_UUID)) {
|
|
||||||
kdfP.setUInt32(AesKdf.ParamRounds, roundsFix);
|
|
||||||
numKeyEncRounds = roundsFix;
|
|
||||||
}
|
|
||||||
|
|
||||||
byte[] transformedMasterKey = kdfEngine.transform(masterKey, kdfP);
|
|
||||||
if (transformedMasterKey.length != 32) {
|
if (transformedMasterKey.length != 32) {
|
||||||
transformedMasterKey = CryptoUtil.hashSha256(transformedMasterKey);
|
transformedMasterKey = CryptoUtil.hashSha256(transformedMasterKey);
|
||||||
}
|
}
|
||||||
@@ -549,15 +555,15 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
super.populateGlobals(currentGroup);
|
super.populateGlobals(currentGroup);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Ensure that the recycle bin tree exists, if enabled and create it
|
/**
|
||||||
* if it doesn't exist
|
* Ensure that the recycle bin tree exists, if enabled and create it
|
||||||
*
|
* if it doesn't exist
|
||||||
*/
|
*/
|
||||||
private void ensureRecycleBin() {
|
private void ensureRecycleBin() {
|
||||||
if (getRecycleBin() == null) {
|
if (getRecycleBin() == null) {
|
||||||
// Create recycle bin
|
// Create recycle bin
|
||||||
|
|
||||||
PwGroupV4 recycleBin = new PwGroupV4(RECYCLEBIN_NAME, iconFactory.getIcon(PwIconStandard.TRASH_BIN));
|
PwGroupV4 recycleBin = new PwGroupV4(RECYCLEBIN_NAME, iconFactory.getTrashIcon());
|
||||||
recycleBin.setEnableAutoType(false);
|
recycleBin.setEnableAutoType(false);
|
||||||
recycleBin.setEnableSearching(false);
|
recycleBin.setEnableSearching(false);
|
||||||
recycleBin.setExpanded(false);
|
recycleBin.setExpanded(false);
|
||||||
@@ -681,8 +687,7 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
@Override
|
@Override
|
||||||
public void undoDeleteEntry(PwEntryV4 entry, PwGroupV4 origParent) {
|
public void undoDeleteEntry(PwEntryV4 entry, PwGroupV4 origParent) {
|
||||||
super.undoDeleteEntry(entry, origParent);
|
super.undoDeleteEntry(entry, origParent);
|
||||||
// TODO undo delete entry
|
deletedObjects.remove(new PwDeletedObject(entry.getUUID()));
|
||||||
deletedObjects.remove(entry);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -695,14 +700,6 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
return groups.get(recycleId);
|
return groups.get(recycleId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public KdfParameters getKdfParameters() {
|
|
||||||
return kdfParameters;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setKdfParameters(KdfParameters kdfParameters) {
|
|
||||||
this.kdfParameters = kdfParameters;
|
|
||||||
}
|
|
||||||
|
|
||||||
public VariantDictionary getPublicCustomData() {
|
public VariantDictionary getPublicCustomData() {
|
||||||
return publicCustomData;
|
return publicCustomData;
|
||||||
}
|
}
|
||||||
@@ -739,9 +736,7 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void initNew(String dbPath) {
|
public void initNew(String dbPath) {
|
||||||
String filename = URLUtil.guessFileName(dbPath, null, null);
|
rootGroup = new PwGroupV4(dbNameFromPath(dbPath), iconFactory.getFolderIcon());
|
||||||
|
|
||||||
rootGroup = new PwGroupV4(dbNameFromPath(dbPath), iconFactory.getIcon(PwIconStandard.FOLDER));
|
|
||||||
groups.put(rootGroup.getId(), rootGroup);
|
groups.put(rootGroup.getId(), rootGroup);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -759,64 +754,4 @@ public class PwDatabaseV4 extends PwDatabase<PwGroupV4, PwEntryV4> {
|
|||||||
return filename.substring(0, lastExtDot);
|
return filename.substring(0, lastExtDot);
|
||||||
}
|
}
|
||||||
|
|
||||||
private class GroupHasCustomData extends GroupHandler<PwGroupV4> {
|
|
||||||
|
|
||||||
public boolean hasCustomData = false;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean operate(PwGroupV4 group) {
|
|
||||||
if (group == null) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (group.containsCustomData()) {
|
|
||||||
hasCustomData = true;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class EntryHasCustomData extends EntryHandler<PwEntryV4> {
|
|
||||||
|
|
||||||
public boolean hasCustomData = false;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean operate(PwEntryV4 entry) {
|
|
||||||
if (entry == null) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (entry.containsCustomData()) {
|
|
||||||
hasCustomData = true;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getMinKdbxVersion() {
|
|
||||||
if (!AesKdf.CIPHER_UUID.equals(kdfParameters.kdfUUID)) {
|
|
||||||
return PwDbHeaderV4.FILE_VERSION_32;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (publicCustomData.size() > 0) {
|
|
||||||
return PwDbHeaderV4.FILE_VERSION_32;
|
|
||||||
}
|
|
||||||
|
|
||||||
EntryHasCustomData entryHandler = new EntryHasCustomData();
|
|
||||||
GroupHasCustomData groupHandler = new GroupHasCustomData();
|
|
||||||
|
|
||||||
if (rootGroup == null ) {
|
|
||||||
return PwDbHeaderV4.FILE_VERSION_32_3;
|
|
||||||
}
|
|
||||||
rootGroup.preOrderTraverseTree(groupHandler, entryHandler);
|
|
||||||
if (groupHandler.hasCustomData || entryHandler.hasCustomData) {
|
|
||||||
return PwDbHeaderV4.FILE_VERSION_32;
|
|
||||||
}
|
|
||||||
|
|
||||||
return PwDbHeaderV4.FILE_VERSION_32_3;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -19,10 +19,12 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.app.App;
|
import com.kunzisoft.keepass.app.App;
|
||||||
import com.kunzisoft.keepass.utils.Types;
|
import com.kunzisoft.keepass.utils.Types;
|
||||||
|
|
||||||
import java.io.Serializable;
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
@@ -33,15 +35,14 @@ import java.util.Date;
|
|||||||
* @author bpellin
|
* @author bpellin
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class PwDate implements Cloneable, Serializable {
|
public class PwDate implements Cloneable, Parcelable {
|
||||||
|
|
||||||
private static final int DATE_SIZE = 5;
|
private static final int DATE_SIZE = 5;
|
||||||
|
|
||||||
private boolean cDateBuilt = false;
|
private Date jDate;
|
||||||
private boolean jDateBuilt = false;
|
private boolean jDateBuilt = false;
|
||||||
|
|
||||||
private Date jDate;
|
|
||||||
private byte[] cDate;
|
private byte[] cDate;
|
||||||
|
private boolean cDateBuilt = false;
|
||||||
|
|
||||||
public static final Date NEVER_EXPIRE = getNeverExpire();
|
public static final Date NEVER_EXPIRE = getNeverExpire();
|
||||||
public static final Date DEFAULT_DATE = getDefaultDate();
|
public static final Date DEFAULT_DATE = getDefaultDate();
|
||||||
@@ -93,6 +94,42 @@ public class PwDate implements Cloneable, Serializable {
|
|||||||
jDate = new Date();
|
jDate = new Date();
|
||||||
jDateBuilt = true;
|
jDateBuilt = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected PwDate(Parcel in) {
|
||||||
|
try {
|
||||||
|
jDate = (Date) in.readSerializable();
|
||||||
|
jDateBuilt = in.readByte() != 0;
|
||||||
|
in.readByteArray(cDate);
|
||||||
|
cDateBuilt = in.readByte() != 0;
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
dest.writeSerializable(jDate);
|
||||||
|
dest.writeByte((byte) (jDateBuilt ? 1 : 0));
|
||||||
|
dest.writeByteArray(cDate);
|
||||||
|
dest.writeByte((byte) (cDateBuilt ? 1 : 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwDate> CREATOR = new Creator<PwDate>() {
|
||||||
|
@Override
|
||||||
|
public PwDate createFromParcel(Parcel in) {
|
||||||
|
return new PwDate(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwDate[] newArray(int size) {
|
||||||
|
return new PwDate[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PwDate clone() {
|
public PwDate clone() {
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.AesKdf;
|
import com.kunzisoft.keepass.crypto.keyDerivation.AesKdf;
|
||||||
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfFactory;
|
||||||
import com.kunzisoft.keepass.crypto.keyDerivation.KdfParameters;
|
import com.kunzisoft.keepass.crypto.keyDerivation.KdfParameters;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidDBVersionException;
|
import com.kunzisoft.keepass.database.exception.InvalidDBVersionException;
|
||||||
import com.kunzisoft.keepass.stream.CopyInputStream;
|
import com.kunzisoft.keepass.stream.CopyInputStream;
|
||||||
@@ -41,11 +42,10 @@ import javax.crypto.spec.SecretKeySpec;
|
|||||||
public class PwDbHeaderV4 extends PwDbHeader {
|
public class PwDbHeaderV4 extends PwDbHeader {
|
||||||
public static final int DBSIG_PRE2 = 0xB54BFB66;
|
public static final int DBSIG_PRE2 = 0xB54BFB66;
|
||||||
public static final int DBSIG_2 = 0xB54BFB67;
|
public static final int DBSIG_2 = 0xB54BFB67;
|
||||||
|
|
||||||
private static final int FILE_VERSION_CRITICAL_MASK = 0xFFFF0000;
|
private static final int FILE_VERSION_CRITICAL_MASK = 0xFFFF0000;
|
||||||
public static final int FILE_VERSION_32_3 = 0x00030001;
|
public static final int FILE_VERSION_32_3 = 0x00030001;
|
||||||
public static final int FILE_VERSION_32_4 = 0x00040000;
|
public static final int FILE_VERSION_32_4 = 0x00040000;
|
||||||
public static final int FILE_VERSION_32 = FILE_VERSION_32_4;
|
|
||||||
|
|
||||||
public class PwDbHeaderV4Fields {
|
public class PwDbHeaderV4Fields {
|
||||||
public static final byte EndOfHeader = 0;
|
public static final byte EndOfHeader = 0;
|
||||||
@@ -91,12 +91,83 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
public CrsAlgorithm innerRandomStream;
|
public CrsAlgorithm innerRandomStream;
|
||||||
public long version;
|
public long version;
|
||||||
|
|
||||||
public PwDbHeaderV4(PwDatabaseV4 d) {
|
public PwDbHeaderV4(PwDatabaseV4 databaseV4) {
|
||||||
db = d;
|
this.db = databaseV4;
|
||||||
version = d.getMinKdbxVersion();
|
this.version = getMinKdbxVersion(databaseV4); // Only for writing
|
||||||
masterSeed = new byte[32];
|
this.masterSeed = new byte[32];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long getVersion() {
|
||||||
|
return version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setVersion(long version) {
|
||||||
|
this.version = version;
|
||||||
|
}
|
||||||
|
|
||||||
|
private class GroupHasCustomData extends GroupHandler<PwGroupV4> {
|
||||||
|
|
||||||
|
boolean hasCustomData = false;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean operate(PwGroupV4 group) {
|
||||||
|
if (group == null) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (group.containsCustomData()) {
|
||||||
|
hasCustomData = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class EntryHasCustomData extends EntryHandler<PwEntryV4> {
|
||||||
|
|
||||||
|
boolean hasCustomData = false;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean operate(PwEntryV4 entry) {
|
||||||
|
if (entry == null) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entry.containsCustomData()) {
|
||||||
|
hasCustomData = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private int getMinKdbxVersion(PwDatabaseV4 databaseV4) {
|
||||||
|
// Return v4 if AES is not use
|
||||||
|
if (databaseV4.getKdfParameters() != null
|
||||||
|
&& !databaseV4.getKdfParameters().getUUID().equals(AesKdf.CIPHER_UUID)) {
|
||||||
|
return PwDbHeaderV4.FILE_VERSION_32_4;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return V4 if custom data are present
|
||||||
|
if (databaseV4.containsPublicCustomData()) {
|
||||||
|
return PwDbHeaderV4.FILE_VERSION_32_4;
|
||||||
|
}
|
||||||
|
|
||||||
|
EntryHasCustomData entryHandler = new EntryHasCustomData();
|
||||||
|
GroupHasCustomData groupHandler = new GroupHasCustomData();
|
||||||
|
|
||||||
|
if (databaseV4.getRootGroup() == null ) {
|
||||||
|
return PwDbHeaderV4.FILE_VERSION_32_3;
|
||||||
|
}
|
||||||
|
databaseV4.getRootGroup().preOrderTraverseTree(groupHandler, entryHandler);
|
||||||
|
if (groupHandler.hasCustomData || entryHandler.hasCustomData) {
|
||||||
|
return PwDbHeaderV4.FILE_VERSION_32_4;
|
||||||
|
}
|
||||||
|
|
||||||
|
return PwDbHeaderV4.FILE_VERSION_32_3;
|
||||||
|
}
|
||||||
|
|
||||||
/** Assumes the input stream is at the beginning of the .kdbx file
|
/** Assumes the input stream is at the beginning of the .kdbx file
|
||||||
* @param is
|
* @param is
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
@@ -122,7 +193,7 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
throw new InvalidDBVersionException();
|
throw new InvalidDBVersionException();
|
||||||
}
|
}
|
||||||
|
|
||||||
version = lis.readUInt();
|
version = lis.readUInt(); // Erase previous value
|
||||||
if ( ! validVersion(version) ) {
|
if ( ! validVersion(version) ) {
|
||||||
throw new InvalidDBVersionException();
|
throw new InvalidDBVersionException();
|
||||||
}
|
}
|
||||||
@@ -138,7 +209,7 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
|
|
||||||
private boolean readHeaderField(LEDataInputStream dis) throws IOException {
|
private boolean readHeaderField(LEDataInputStream dis) throws IOException {
|
||||||
byte fieldID = (byte) dis.read();
|
byte fieldID = (byte) dis.read();
|
||||||
|
|
||||||
int fieldSize;
|
int fieldSize;
|
||||||
if (version < FILE_VERSION_32_4) {
|
if (version < FILE_VERSION_32_4) {
|
||||||
fieldSize = dis.readUShort();
|
fieldSize = dis.readUShort();
|
||||||
@@ -149,13 +220,13 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
byte[] fieldData = null;
|
byte[] fieldData = null;
|
||||||
if ( fieldSize > 0 ) {
|
if ( fieldSize > 0 ) {
|
||||||
fieldData = new byte[fieldSize];
|
fieldData = new byte[fieldSize];
|
||||||
|
|
||||||
int readSize = dis.read(fieldData);
|
int readSize = dis.read(fieldData);
|
||||||
if ( readSize != fieldSize ) {
|
if ( readSize != fieldSize ) {
|
||||||
throw new IOException("Header ended early.");
|
throw new IOException("Header ended early.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ( fieldID ) {
|
switch ( fieldID ) {
|
||||||
case PwDbHeaderV4Fields.EndOfHeader:
|
case PwDbHeaderV4Fields.EndOfHeader:
|
||||||
return true;
|
return true;
|
||||||
@@ -173,24 +244,13 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.TransformSeed:
|
case PwDbHeaderV4Fields.TransformSeed:
|
||||||
assert(version < PwDbHeaderV4.FILE_VERSION_32_4); // TODO file > FILEVERSION
|
if(version < PwDbHeaderV4.FILE_VERSION_32_4)
|
||||||
AesKdf kdfS = new AesKdf();
|
setTransformSeed(fieldData);
|
||||||
if (!db.getKdfParameters().kdfUUID.equals(kdfS.uuid)) {
|
|
||||||
db.setKdfParameters(kdfS.getDefaultParameters());
|
|
||||||
}
|
|
||||||
|
|
||||||
db.getKdfParameters().setByteArray(AesKdf.ParamSeed, fieldData);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.TransformRounds:
|
case PwDbHeaderV4Fields.TransformRounds:
|
||||||
assert(version < PwDbHeaderV4.FILE_VERSION_32_4);
|
if(version < PwDbHeaderV4.FILE_VERSION_32_4)
|
||||||
AesKdf kdfR = new AesKdf();
|
setTransformRound(fieldData);
|
||||||
if (!db.getKdfParameters().kdfUUID.equals(kdfR.uuid)) {
|
|
||||||
db.setKdfParameters(kdfR.getDefaultParameters());
|
|
||||||
}
|
|
||||||
long rounds = LEDataInputStream.readLong(fieldData, 0);
|
|
||||||
db.getKdfParameters().setUInt64(AesKdf.ParamRounds, rounds);
|
|
||||||
db.setNumberKeyEncryptionRounds(rounds);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.EncryptionIV:
|
case PwDbHeaderV4Fields.EncryptionIV:
|
||||||
@@ -198,8 +258,8 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.InnerRandomstreamKey:
|
case PwDbHeaderV4Fields.InnerRandomstreamKey:
|
||||||
assert(version < PwDbHeaderV4.FILE_VERSION_32_4);
|
if(version < PwDbHeaderV4.FILE_VERSION_32_4)
|
||||||
innerRandomStreamKey = fieldData;
|
innerRandomStreamKey = fieldData;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.StreamStartBytes:
|
case PwDbHeaderV4Fields.StreamStartBytes:
|
||||||
@@ -207,8 +267,8 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.InnerRandomStreamID:
|
case PwDbHeaderV4Fields.InnerRandomStreamID:
|
||||||
assert(version < PwDbHeaderV4.FILE_VERSION_32_4);
|
if(version < PwDbHeaderV4.FILE_VERSION_32_4)
|
||||||
setRandomStreamID(fieldData);
|
setRandomStreamID(fieldData);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PwDbHeaderV4Fields.KdfParameters:
|
case PwDbHeaderV4Fields.KdfParameters:
|
||||||
@@ -224,6 +284,12 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void assignAesKdfEngineIfNotExists() {
|
||||||
|
if (db.getKdfParameters() == null || !db.getKdfParameters().getUUID().equals(KdfFactory.aesKdf.getUUID())) {
|
||||||
|
db.setKdfParameters(KdfFactory.aesKdf.getDefaultParameters());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void setCipher(byte[] pbId) throws IOException {
|
private void setCipher(byte[] pbId) throws IOException {
|
||||||
if ( pbId == null || pbId.length != 16 ) {
|
if ( pbId == null || pbId.length != 16 ) {
|
||||||
@@ -232,6 +298,18 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
|
|
||||||
db.setDataCipher(Types.bytestoUUID(pbId));
|
db.setDataCipher(Types.bytestoUUID(pbId));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void setTransformSeed(byte[] seed) {
|
||||||
|
assignAesKdfEngineIfNotExists();
|
||||||
|
db.getKdfParameters().setByteArray(AesKdf.ParamSeed, seed);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setTransformRound(byte[] roundsByte) {
|
||||||
|
assignAesKdfEngineIfNotExists();
|
||||||
|
long rounds = LEDataInputStream.readLong(roundsByte, 0);
|
||||||
|
db.getKdfParameters().setUInt64(AesKdf.ParamRounds, rounds);
|
||||||
|
db.setNumberKeyEncryptionRounds(rounds);
|
||||||
|
}
|
||||||
|
|
||||||
private void setCompressionFlags(byte[] pbFlags) throws IOException {
|
private void setCompressionFlags(byte[] pbFlags) throws IOException {
|
||||||
if ( pbFlags == null || pbFlags.length != 4 ) {
|
if ( pbFlags == null || pbFlags.length != 4 ) {
|
||||||
@@ -244,23 +322,6 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
db.setCompressionAlgorithm(PwCompressionAlgorithm.fromId(flag));
|
db.setCompressionAlgorithm(PwCompressionAlgorithm.fromId(flag));
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setTransformRounds(byte[] rounds) throws IOException {
|
|
||||||
if ( rounds == null || rounds.length != 8 ) {
|
|
||||||
throw new IOException("Invalid rounds.");
|
|
||||||
}
|
|
||||||
|
|
||||||
long rnd = LEDataInputStream.readLong(rounds, 0);
|
|
||||||
|
|
||||||
if ( rnd < 0 || rnd > Integer.MAX_VALUE ) {
|
|
||||||
//TODO: Actually support really large numbers
|
|
||||||
throw new IOException("Rounds higher than " + Integer.MAX_VALUE + " are not currently supported.");
|
|
||||||
}
|
|
||||||
|
|
||||||
db.setNumberKeyEncryptionRounds(rnd);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setRandomStreamID(byte[] streamID) throws IOException {
|
public void setRandomStreamID(byte[] streamID) throws IOException {
|
||||||
@@ -276,25 +337,22 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
innerRandomStream = CrsAlgorithm.fromId(id);
|
innerRandomStream = CrsAlgorithm.fromId(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Determines if this is a supported version.
|
/**
|
||||||
|
* Determines if this is a supported version.
|
||||||
*
|
*
|
||||||
* A long is needed here to represent the unsigned int since we perform
|
* A long is needed here to represent the unsigned int since we perform arithmetic on it.
|
||||||
* arithmetic on it.
|
* @param version Database version
|
||||||
* @param version
|
* @return true if it's a supported version
|
||||||
* @return
|
|
||||||
*/
|
*/
|
||||||
private boolean validVersion(long version) {
|
private boolean validVersion(long version) {
|
||||||
|
return ! ((version & FILE_VERSION_CRITICAL_MASK) > (FILE_VERSION_32_4 & FILE_VERSION_CRITICAL_MASK));
|
||||||
return ! ((version & FILE_VERSION_CRITICAL_MASK) > (FILE_VERSION_32 & FILE_VERSION_CRITICAL_MASK));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean matchesHeader(int sig1, int sig2) {
|
public static boolean matchesHeader(int sig1, int sig2) {
|
||||||
return (sig1 == PWM_DBSIG_1) && ( (sig2 == DBSIG_2) || (sig2 == DBSIG_2) );
|
return (sig1 == PWM_DBSIG_1) && ( (sig2 == DBSIG_PRE2) || (sig2 == DBSIG_2) );
|
||||||
}
|
}
|
||||||
|
|
||||||
public static byte[] computeHeaderHmac(byte[] header, byte[] key) throws IOException{
|
public static byte[] computeHeaderHmac(byte[] header, byte[] key) throws IOException{
|
||||||
byte[] headerHmac;
|
|
||||||
byte[] blockKey = HmacBlockStream.GetHmacKey64(key, Types.ULONG_MAX_VALUE);
|
byte[] blockKey = HmacBlockStream.GetHmacKey64(key, Types.ULONG_MAX_VALUE);
|
||||||
|
|
||||||
Mac hmac;
|
Mac hmac;
|
||||||
@@ -312,8 +370,7 @@ public class PwDbHeaderV4 extends PwDbHeader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public byte[] getTransformSeed() {
|
public byte[] getTransformSeed() {
|
||||||
assert(version < FILE_VERSION_32_4);
|
// version < FILE_VERSION_32_4)
|
||||||
|
return db.getKdfParameters().getByteArray(AesKdf.ParamSeed);
|
||||||
return db.getKdfParameters().getByteArray(AesKdf.ParamSeed);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,8 +22,14 @@ package com.kunzisoft.keepass.database;
|
|||||||
import android.content.res.Resources;
|
import android.content.res.Resources;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
|
import com.kunzisoft.keepass.crypto.engine.AesEngine;
|
||||||
|
import com.kunzisoft.keepass.crypto.engine.ChaCha20Engine;
|
||||||
|
import com.kunzisoft.keepass.crypto.engine.CipherEngine;
|
||||||
|
import com.kunzisoft.keepass.crypto.engine.TwofishEngine;
|
||||||
|
|
||||||
public enum PwEncryptionAlgorithm {
|
import java.util.UUID;
|
||||||
|
|
||||||
|
public enum PwEncryptionAlgorithm implements ObjectNameResource {
|
||||||
|
|
||||||
AES_Rijndael,
|
AES_Rijndael,
|
||||||
Twofish,
|
Twofish,
|
||||||
@@ -33,11 +39,35 @@ public enum PwEncryptionAlgorithm {
|
|||||||
switch (this) {
|
switch (this) {
|
||||||
default:
|
default:
|
||||||
case AES_Rijndael:
|
case AES_Rijndael:
|
||||||
return resources.getString(R.string.rijndael);
|
return resources.getString(R.string.encryption_rijndael);
|
||||||
case Twofish:
|
case Twofish:
|
||||||
return resources.getString(R.string.twofish);
|
return resources.getString(R.string.encryption_twofish);
|
||||||
case ChaCha20:
|
case ChaCha20:
|
||||||
return resources.getString(R.string.chacha20);
|
return resources.getString(R.string.encryption_chacha20);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public CipherEngine getCipherEngine() {
|
||||||
|
switch (this) {
|
||||||
|
default:
|
||||||
|
case AES_Rijndael:
|
||||||
|
return new AesEngine();
|
||||||
|
case Twofish:
|
||||||
|
return new TwofishEngine();
|
||||||
|
case ChaCha20:
|
||||||
|
return new ChaCha20Engine();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public UUID getDataCipher() {
|
||||||
|
switch (this) {
|
||||||
|
default:
|
||||||
|
case AES_Rijndael:
|
||||||
|
return AesEngine.CIPHER_UUID;
|
||||||
|
case Twofish:
|
||||||
|
return TwofishEngine.CIPHER_UUID;
|
||||||
|
case ChaCha20:
|
||||||
|
return ChaCha20Engine.CIPHER_UUID;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,8 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.iterator.EntrySearchStringIterator;
|
import com.kunzisoft.keepass.database.iterator.EntrySearchStringIterator;
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedString;
|
import com.kunzisoft.keepass.database.security.ProtectedString;
|
||||||
|
|
||||||
@@ -30,6 +32,19 @@ public abstract class PwEntry<Parent extends PwGroup> extends PwNode<Parent> {
|
|||||||
|
|
||||||
protected UUID uuid = PwDatabase.UUID_ZERO;
|
protected UUID uuid = PwDatabase.UUID_ZERO;
|
||||||
|
|
||||||
|
public PwEntry() {}
|
||||||
|
|
||||||
|
public PwEntry(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
uuid = (UUID) in.readSerializable();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeSerializable(uuid);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void construct(Parent parent) {
|
protected void construct(Parent parent) {
|
||||||
super.construct(parent);
|
super.construct(parent);
|
||||||
@@ -117,6 +132,22 @@ public abstract class PwEntry<Parent extends PwGroup> extends PwNode<Parent> {
|
|||||||
return getFields().containsCustomFields();
|
return getFields().containsCustomFields();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If entry contains extra fields that are protected
|
||||||
|
* @return true if there is extra fields protected
|
||||||
|
*/
|
||||||
|
public boolean containsCustomFieldsProtected() {
|
||||||
|
return getFields().containsCustomFieldsProtected();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If entry contains extra fields that are not protected
|
||||||
|
* @return true if there is extra fields not protected
|
||||||
|
*/
|
||||||
|
public boolean containsCustomFieldsNotProtected() {
|
||||||
|
return getFields().containsCustomFieldsNotProtected();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add an extra field to the list (standard or custom)
|
* Add an extra field to the list (standard or custom)
|
||||||
* @param label Label of field, must be unique
|
* @param label Label of field, must be unique
|
||||||
|
|||||||
@@ -42,6 +42,8 @@ Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|||||||
|
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import java.io.UnsupportedEncodingException;
|
import java.io.UnsupportedEncodingException;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
@@ -77,13 +79,11 @@ public class PwEntryV3 extends PwEntry<PwGroupV3> {
|
|||||||
|
|
||||||
// TODO Parent ID to remove
|
// TODO Parent ID to remove
|
||||||
private int groupId;
|
private int groupId;
|
||||||
|
|
||||||
private String title;
|
private String title;
|
||||||
private String username;
|
private String username;
|
||||||
private byte[] password;
|
private byte[] password;
|
||||||
private String url;
|
private String url;
|
||||||
private String additional;
|
private String additional;
|
||||||
|
|
||||||
/** A string describing what is in pBinaryData */
|
/** A string describing what is in pBinaryData */
|
||||||
private String binaryDesc;
|
private String binaryDesc;
|
||||||
private byte[] binaryData;
|
private byte[] binaryData;
|
||||||
@@ -97,6 +97,43 @@ public class PwEntryV3 extends PwEntry<PwGroupV3> {
|
|||||||
groupId = ((PwGroupIdV3) this.parent.getId()).getId(); // TODO remove
|
groupId = ((PwGroupIdV3) this.parent.getId()).getId(); // TODO remove
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwEntryV3(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
groupId = in.readInt();
|
||||||
|
title = in.readString();
|
||||||
|
username = in.readString();
|
||||||
|
in.readByteArray(password);
|
||||||
|
url = in.readString();
|
||||||
|
additional = in.readString();
|
||||||
|
binaryDesc = in.readString();
|
||||||
|
in.readByteArray(binaryData);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeInt(groupId);
|
||||||
|
dest.writeString(title);
|
||||||
|
dest.writeString(username);
|
||||||
|
dest.writeByteArray(password);
|
||||||
|
dest.writeString(url);
|
||||||
|
dest.writeString(additional);
|
||||||
|
dest.writeString(binaryDesc);
|
||||||
|
dest.writeByteArray(binaryData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwEntryV3> CREATOR = new Creator<PwEntryV3>() {
|
||||||
|
@Override
|
||||||
|
public PwEntryV3 createFromParcel(Parcel in) {
|
||||||
|
return new PwEntryV3(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwEntryV3[] newArray(int size) {
|
||||||
|
return new PwEntryV3[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
protected void updateWith(PwEntryV3 source) {
|
protected void updateWith(PwEntryV3 source) {
|
||||||
super.assign(source);
|
super.assign(source);
|
||||||
groupId = source.groupId;
|
groupId = source.groupId;
|
||||||
@@ -199,7 +236,7 @@ public class PwEntryV3 extends PwEntry<PwGroupV3> {
|
|||||||
public void populateBlankFields(PwDatabaseV3 db) {
|
public void populateBlankFields(PwDatabaseV3 db) {
|
||||||
// TODO verify and remove
|
// TODO verify and remove
|
||||||
if (icon == null) {
|
if (icon == null) {
|
||||||
icon = db.getIconFactory().getFirstIcon();
|
icon = db.getIconFactory().getKeyIcon();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (username == null) {
|
if (username == null) {
|
||||||
|
|||||||
@@ -19,8 +19,11 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedBinary;
|
import com.kunzisoft.keepass.database.security.ProtectedBinary;
|
||||||
import com.kunzisoft.keepass.database.security.ProtectedString;
|
import com.kunzisoft.keepass.database.security.ProtectedString;
|
||||||
|
import com.kunzisoft.keepass.utils.MemUtil;
|
||||||
import com.kunzisoft.keepass.utils.SprEngineV4;
|
import com.kunzisoft.keepass.utils.SprEngineV4;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@@ -37,7 +40,7 @@ public class PwEntryV4 extends PwEntry<PwGroupV4> implements ITimeLogger {
|
|||||||
public static final String STR_URL = "URL";
|
public static final String STR_URL = "URL";
|
||||||
public static final String STR_NOTES = "Notes";
|
public static final String STR_NOTES = "Notes";
|
||||||
|
|
||||||
// To decode each field not serializable
|
// To decode each field not parcelable
|
||||||
private transient PwDatabaseV4 mDatabase = null;
|
private transient PwDatabaseV4 mDatabase = null;
|
||||||
private transient boolean mDecodeRef = false;
|
private transient boolean mDecodeRef = false;
|
||||||
|
|
||||||
@@ -45,7 +48,6 @@ public class PwEntryV4 extends PwEntry<PwGroupV4> implements ITimeLogger {
|
|||||||
private long usageCount = 0;
|
private long usageCount = 0;
|
||||||
private PwDate parentGroupLastMod = new PwDate();
|
private PwDate parentGroupLastMod = new PwDate();
|
||||||
private Map<String, String> customData = new HashMap<>();
|
private Map<String, String> customData = new HashMap<>();
|
||||||
|
|
||||||
private ExtraFields fields = new ExtraFields();
|
private ExtraFields fields = new ExtraFields();
|
||||||
private HashMap<String, ProtectedBinary> binaries = new HashMap<>();
|
private HashMap<String, ProtectedBinary> binaries = new HashMap<>();
|
||||||
private String foregroundColor = "";
|
private String foregroundColor = "";
|
||||||
@@ -53,7 +55,6 @@ public class PwEntryV4 extends PwEntry<PwGroupV4> implements ITimeLogger {
|
|||||||
private String overrideURL = "";
|
private String overrideURL = "";
|
||||||
private AutoType autoType = new AutoType();
|
private AutoType autoType = new AutoType();
|
||||||
private ArrayList<PwEntryV4> history = new ArrayList<>();
|
private ArrayList<PwEntryV4> history = new ArrayList<>();
|
||||||
|
|
||||||
private String url = "";
|
private String url = "";
|
||||||
private String additional = "";
|
private String additional = "";
|
||||||
private String tags = "";
|
private String tags = "";
|
||||||
@@ -71,8 +72,8 @@ public class PwEntryV4 extends PwEntry<PwGroupV4> implements ITimeLogger {
|
|||||||
customIcon = source.customIcon;
|
customIcon = source.customIcon;
|
||||||
usageCount = source.usageCount;
|
usageCount = source.usageCount;
|
||||||
parentGroupLastMod = source.parentGroupLastMod;
|
parentGroupLastMod = source.parentGroupLastMod;
|
||||||
// TODO customData
|
customData.clear();
|
||||||
|
customData.putAll(source.customData); // Add all custom elements in map
|
||||||
fields = source.fields;
|
fields = source.fields;
|
||||||
binaries = source.binaries;
|
binaries = source.binaries;
|
||||||
foregroundColor = source.foregroundColor;
|
foregroundColor = source.foregroundColor;
|
||||||
@@ -80,12 +81,60 @@ public class PwEntryV4 extends PwEntry<PwGroupV4> implements ITimeLogger {
|
|||||||
overrideURL = source.overrideURL;
|
overrideURL = source.overrideURL;
|
||||||
autoType = source.autoType;
|
autoType = source.autoType;
|
||||||
history = source.history;
|
history = source.history;
|
||||||
|
|
||||||
url = source.url;
|
url = source.url;
|
||||||
additional = source.additional;
|
additional = source.additional;
|
||||||
tags = source.tags;
|
tags = source.tags;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwEntryV4(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
customIcon = in.readParcelable(PwIconCustom.class.getClassLoader());
|
||||||
|
usageCount = in.readLong();
|
||||||
|
parentGroupLastMod = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
customData = MemUtil.readStringParcelableMap(in);
|
||||||
|
fields = in.readParcelable(ExtraFields.class.getClassLoader());
|
||||||
|
binaries = MemUtil.readStringParcelableMap(in, ProtectedBinary.class);
|
||||||
|
foregroundColor = in.readString();
|
||||||
|
backgroupColor = in.readString();
|
||||||
|
overrideURL = in.readString();
|
||||||
|
autoType = in.readParcelable(AutoType.class.getClassLoader());
|
||||||
|
history = in.readArrayList(PwEntryV4.class.getClassLoader()); // TODO verify
|
||||||
|
url = in.readString();
|
||||||
|
additional = in.readString();
|
||||||
|
tags = in.readString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeParcelable(customIcon, flags);
|
||||||
|
dest.writeLong(usageCount);
|
||||||
|
dest.writeParcelable(parentGroupLastMod, flags);
|
||||||
|
MemUtil.writeStringParcelableMap(dest, customData);
|
||||||
|
dest.writeParcelable(fields, flags);
|
||||||
|
MemUtil.writeStringParcelableMap(dest, flags, binaries);
|
||||||
|
dest.writeString(foregroundColor);
|
||||||
|
dest.writeString(backgroupColor);
|
||||||
|
dest.writeString(overrideURL);
|
||||||
|
dest.writeParcelable(autoType, flags);
|
||||||
|
dest.writeList(history);
|
||||||
|
dest.writeString(url);
|
||||||
|
dest.writeString(additional);
|
||||||
|
dest.writeString(tags);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwEntryV4> CREATOR = new Creator<PwEntryV4>() {
|
||||||
|
@Override
|
||||||
|
public PwEntryV4 createFromParcel(Parcel in) {
|
||||||
|
return new PwEntryV4(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwEntryV4[] newArray(int size) {
|
||||||
|
return new PwEntryV4[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
public PwEntryV4 clone() {
|
public PwEntryV4 clone() {
|
||||||
|
|||||||
@@ -19,6 +19,8 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
@@ -27,8 +29,24 @@ public abstract class PwGroup<Parent extends PwGroup, ChildGroup extends PwGroup
|
|||||||
|
|
||||||
protected String name = "";
|
protected String name = "";
|
||||||
|
|
||||||
protected List<ChildGroup> childGroups = new ArrayList<>();
|
// TODO verify children not needed
|
||||||
protected List<ChildEntry> childEntries = new ArrayList<>();
|
transient protected List<ChildGroup> childGroups = new ArrayList<>();
|
||||||
|
transient protected List<ChildEntry> childEntries = new ArrayList<>();
|
||||||
|
|
||||||
|
protected PwGroup() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected PwGroup(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
name = in.readString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeString(name);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PwGroup clone() {
|
public PwGroup clone() {
|
||||||
|
|||||||
@@ -19,8 +19,20 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
public abstract class PwGroupId implements Serializable {
|
public abstract class PwGroupId implements Parcelable {
|
||||||
|
|
||||||
|
public PwGroupId() {}
|
||||||
|
|
||||||
|
public PwGroupId(Parcel in) {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,13 +19,39 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
public class PwGroupIdV3 extends PwGroupId {
|
public class PwGroupIdV3 extends PwGroupId {
|
||||||
|
|
||||||
private int id;
|
private int id;
|
||||||
|
|
||||||
public PwGroupIdV3(int i) {
|
public PwGroupIdV3(int groupId) {
|
||||||
id = i;
|
super();
|
||||||
|
this.id = groupId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwGroupIdV3(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
id = in.readInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeInt(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwGroupIdV3> CREATOR = new Creator<PwGroupIdV3>() {
|
||||||
|
@Override
|
||||||
|
public PwGroupIdV3 createFromParcel(Parcel in) {
|
||||||
|
return new PwGroupIdV3(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwGroupIdV3[] newArray(int size) {
|
||||||
|
return new PwGroupIdV3[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object compare) {
|
public boolean equals(Object compare) {
|
||||||
|
|||||||
@@ -19,15 +19,42 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
public class PwGroupIdV4 extends PwGroupId {
|
public class PwGroupIdV4 extends PwGroupId {
|
||||||
|
|
||||||
private UUID uuid;
|
private UUID uuid;
|
||||||
|
|
||||||
public PwGroupIdV4(UUID u) {
|
public PwGroupIdV4(UUID uuid) {
|
||||||
uuid = u;
|
super();
|
||||||
|
this.uuid = uuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwGroupIdV4(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
uuid = (UUID) in.readSerializable();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeSerializable(uuid);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwGroupIdV4> CREATOR = new Creator<PwGroupIdV4>() {
|
||||||
|
@Override
|
||||||
|
public PwGroupIdV4 createFromParcel(Parcel in) {
|
||||||
|
return new PwGroupIdV4(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwGroupIdV4[] newArray(int size) {
|
||||||
|
return new PwGroupIdV4[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object id) {
|
public boolean equals(Object id) {
|
||||||
if ( ! (id instanceof PwGroupIdV4) ) {
|
if ( ! (id instanceof PwGroupIdV4) ) {
|
||||||
@@ -36,12 +63,12 @@ public class PwGroupIdV4 extends PwGroupId {
|
|||||||
PwGroupIdV4 v4 = (PwGroupIdV4) id;
|
PwGroupIdV4 v4 = (PwGroupIdV4) id;
|
||||||
return uuid.equals(v4.uuid);
|
return uuid.equals(v4.uuid);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return uuid.hashCode();
|
return uuid.hashCode();
|
||||||
}
|
}
|
||||||
|
|
||||||
public UUID getId() {
|
public UUID getId() {
|
||||||
return uuid;
|
return uuid;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,19 +20,13 @@
|
|||||||
|
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
/**
|
import android.os.Parcel;
|
||||||
* @author Brian Pellin <bpellin@gmail.com>
|
|
||||||
* @author Naomaru Itoi <nao@phoneid.org>
|
|
||||||
* @author Bill Zwicky <wrzwicky@pobox.com>
|
|
||||||
* @author Dominik Reichl <dominik.reichl@t-online.de>
|
|
||||||
*/
|
|
||||||
public class PwGroupV3 extends PwGroup<PwGroupV3, PwGroupV3, PwEntryV3> {
|
public class PwGroupV3 extends PwGroup<PwGroupV3, PwGroupV3, PwEntryV3> {
|
||||||
|
|
||||||
// for tree traversing
|
// for tree traversing
|
||||||
private int groupId;
|
private int groupId;
|
||||||
|
|
||||||
private int level = 0; // short
|
private int level = 0; // short
|
||||||
|
|
||||||
/** Used by KeePass internally, don't use */
|
/** Used by KeePass internally, don't use */
|
||||||
private int flags;
|
private int flags;
|
||||||
|
|
||||||
@@ -40,6 +34,33 @@ public class PwGroupV3 extends PwGroup<PwGroupV3, PwGroupV3, PwEntryV3> {
|
|||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwGroupV3(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
groupId = in.readInt();
|
||||||
|
level = in.readInt();
|
||||||
|
flags = in.readInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeInt(groupId);
|
||||||
|
dest.writeInt(level);
|
||||||
|
dest.writeInt(flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwGroupV3> CREATOR = new Creator<PwGroupV3>() {
|
||||||
|
@Override
|
||||||
|
public PwGroupV3 createFromParcel(Parcel in) {
|
||||||
|
return new PwGroupV3(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwGroupV3[] newArray(int size) {
|
||||||
|
return new PwGroupV3[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
public PwGroupV3(PwGroupV3 p) {
|
public PwGroupV3(PwGroupV3 p) {
|
||||||
construct(p);
|
construct(p);
|
||||||
}
|
}
|
||||||
@@ -109,7 +130,7 @@ public class PwGroupV3 extends PwGroup<PwGroupV3, PwGroupV3, PwEntryV3> {
|
|||||||
public void populateBlankFields(PwDatabaseV3 db) {
|
public void populateBlankFields(PwDatabaseV3 db) {
|
||||||
// TODO populate blanck field
|
// TODO populate blanck field
|
||||||
if (icon == null) {
|
if (icon == null) {
|
||||||
icon = db.getIconFactory().getFirstIcon();
|
icon = db.getIconFactory().getFolderIcon();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (name == null) {
|
if (name == null) {
|
||||||
|
|||||||
@@ -19,6 +19,10 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.utils.MemUtil;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
@@ -32,9 +36,7 @@ public class PwGroupV4 extends PwGroup<PwGroupV4, PwGroupV4, PwEntryV4> implemen
|
|||||||
private long usageCount = 0;
|
private long usageCount = 0;
|
||||||
private PwDate parentGroupLastMod = new PwDate();
|
private PwDate parentGroupLastMod = new PwDate();
|
||||||
private Map<String, String> customData = new HashMap<>();
|
private Map<String, String> customData = new HashMap<>();
|
||||||
|
|
||||||
private boolean expires = false;
|
private boolean expires = false;
|
||||||
|
|
||||||
private String notes = "";
|
private String notes = "";
|
||||||
private boolean isExpanded = true;
|
private boolean isExpanded = true;
|
||||||
private String defaultAutoTypeSequence = "";
|
private String defaultAutoTypeSequence = "";
|
||||||
@@ -57,6 +59,53 @@ public class PwGroupV4 extends PwGroup<PwGroupV4, PwGroupV4, PwEntryV4> implemen
|
|||||||
this.icon = icon;
|
this.icon = icon;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwGroupV4(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
uuid = (UUID) in.readSerializable();
|
||||||
|
customIcon = in.readParcelable(PwIconCustom.class.getClassLoader());
|
||||||
|
usageCount = in.readLong();
|
||||||
|
parentGroupLastMod = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
customData = MemUtil.readStringParcelableMap(in);
|
||||||
|
expires = in.readByte() != 0;
|
||||||
|
notes = in.readString();
|
||||||
|
isExpanded = in.readByte() != 0;
|
||||||
|
defaultAutoTypeSequence = in.readString();
|
||||||
|
byte autoTypeByte = in.readByte();
|
||||||
|
enableAutoType = (autoTypeByte == -1) ? null : autoTypeByte != 0;
|
||||||
|
byte enableSearchingByte = in.readByte();
|
||||||
|
enableSearching = (enableSearchingByte == -1) ? null : enableSearchingByte != 0;
|
||||||
|
lastTopVisibleEntry = (UUID) in.readSerializable();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
super.writeToParcel(dest, flags);
|
||||||
|
dest.writeSerializable(uuid);
|
||||||
|
dest.writeParcelable(customIcon, flags);
|
||||||
|
dest.writeLong(usageCount);
|
||||||
|
dest.writeParcelable(parentGroupLastMod, flags);
|
||||||
|
MemUtil.writeStringParcelableMap(dest, customData);
|
||||||
|
dest.writeByte((byte) (expires ? 1 : 0));
|
||||||
|
dest.writeString(notes);
|
||||||
|
dest.writeByte((byte) (isExpanded ? 1 : 0));
|
||||||
|
dest.writeString(defaultAutoTypeSequence);
|
||||||
|
dest.writeByte((byte) (enableAutoType == null ? -1 : (enableAutoType ? 1 : 0)));
|
||||||
|
dest.writeByte((byte) (enableAutoType == null ? -1 : (enableAutoType ? 1 : 0)));
|
||||||
|
dest.writeSerializable(lastTopVisibleEntry);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwGroupV4> CREATOR = new Creator<PwGroupV4>() {
|
||||||
|
@Override
|
||||||
|
public PwGroupV4 createFromParcel(Parcel in) {
|
||||||
|
return new PwGroupV4(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwGroupV4[] newArray(int size) {
|
||||||
|
return new PwGroupV4[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
protected void updateWith(PwGroupV4 source) {
|
protected void updateWith(PwGroupV4 source) {
|
||||||
super.assign(source);
|
super.assign(source);
|
||||||
uuid = source.uuid;
|
uuid = source.uuid;
|
||||||
|
|||||||
@@ -19,11 +19,21 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
public abstract class PwIcon implements Serializable {
|
public abstract class PwIcon implements Parcelable {
|
||||||
|
|
||||||
public boolean isMetaStreamIcon() {
|
public boolean isMetaStreamIcon() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected PwIcon() {}
|
||||||
|
|
||||||
|
protected PwIcon(Parcel in) {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,8 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
public class PwIconCustom extends PwIcon {
|
public class PwIconCustom extends PwIcon {
|
||||||
@@ -27,16 +29,42 @@ public class PwIconCustom extends PwIcon {
|
|||||||
public final UUID uuid;
|
public final UUID uuid;
|
||||||
public byte[] imageData;
|
public byte[] imageData;
|
||||||
|
|
||||||
public PwIconCustom(UUID u, byte[] data) {
|
public PwIconCustom(UUID uuid, byte[] data) {
|
||||||
uuid = u;
|
super();
|
||||||
imageData = data;
|
this.uuid = uuid;
|
||||||
|
this.imageData = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
public PwIconCustom(PwIconCustom icon) {
|
public PwIconCustom(PwIconCustom icon) {
|
||||||
|
super();
|
||||||
uuid = icon.uuid;
|
uuid = icon.uuid;
|
||||||
imageData = icon.imageData;
|
imageData = icon.imageData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected PwIconCustom(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
uuid = (UUID) in.readSerializable();
|
||||||
|
in.readByteArray(imageData);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
dest.writeSerializable(uuid);
|
||||||
|
dest.writeByteArray(imageData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwIconCustom> CREATOR = new Creator<PwIconCustom>() {
|
||||||
|
@Override
|
||||||
|
public PwIconCustom createFromParcel(Parcel in) {
|
||||||
|
return new PwIconCustom(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwIconCustom[] newArray(int size) {
|
||||||
|
return new PwIconCustom[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
final int prime = 31;
|
final int prime = 31;
|
||||||
@@ -55,10 +83,7 @@ public class PwIconCustom extends PwIcon {
|
|||||||
return false;
|
return false;
|
||||||
PwIconCustom other = (PwIconCustom) obj;
|
PwIconCustom other = (PwIconCustom) obj;
|
||||||
if (uuid == null) {
|
if (uuid == null) {
|
||||||
if (other.uuid != null)
|
return other.uuid == null;
|
||||||
return false;
|
} else return uuid.equals(other.uuid);
|
||||||
} else if (!uuid.equals(other.uuid))
|
|
||||||
return false;
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,20 +37,23 @@ public class PwIconFactory {
|
|||||||
*/
|
*/
|
||||||
private ReferenceMap customCache = new ReferenceMap(AbstractReferenceMap.HARD, AbstractReferenceMap.WEAK);
|
private ReferenceMap customCache = new ReferenceMap(AbstractReferenceMap.HARD, AbstractReferenceMap.WEAK);
|
||||||
|
|
||||||
public PwIconStandard getFirstIcon() {
|
public PwIconStandard getKeyIcon() {
|
||||||
return getIcon(0);
|
return getIcon(PwIconStandard.KEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PwIconStandard getTrashIcon() {
|
||||||
|
return getIcon(PwIconStandard.TRASH);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PwIconStandard getFolderIcon() {
|
||||||
|
return getIcon(PwIconStandard.FOLDER);
|
||||||
|
}
|
||||||
|
|
||||||
public PwIconStandard getIcon(int iconId) {
|
public PwIconStandard getIcon(int iconId) {
|
||||||
PwIconStandard icon = (PwIconStandard) cache.get(iconId);
|
PwIconStandard icon = (PwIconStandard) cache.get(iconId);
|
||||||
|
|
||||||
if (icon == null) {
|
if (icon == null) {
|
||||||
if (iconId == 1) {
|
icon = new PwIconStandard(iconId);
|
||||||
icon = PwIconStandard.FIRST;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
icon = new PwIconStandard(iconId);
|
|
||||||
}
|
|
||||||
cache.put(iconId, icon);
|
cache.put(iconId, icon);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,15 +19,15 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
public class PwIconStandard extends PwIcon {
|
public class PwIconStandard extends PwIcon {
|
||||||
public final int iconId;
|
public final int iconId;
|
||||||
|
|
||||||
// The first is number 0
|
public static final int KEY = 0;
|
||||||
public static PwIconStandard FIRST = new PwIconStandard(0);
|
public static final int TRASH = 43;
|
||||||
|
|
||||||
public static final int TRASH_BIN = 43;
|
|
||||||
public static final int FOLDER = 48;
|
public static final int FOLDER = 48;
|
||||||
|
|
||||||
public PwIconStandard(int iconId) {
|
public PwIconStandard(int iconId) {
|
||||||
this.iconId = iconId;
|
this.iconId = iconId;
|
||||||
}
|
}
|
||||||
@@ -36,6 +36,28 @@ public class PwIconStandard extends PwIcon {
|
|||||||
this.iconId = icon.iconId;
|
this.iconId = icon.iconId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected PwIconStandard(Parcel in) {
|
||||||
|
super(in);
|
||||||
|
iconId = in.readInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
dest.writeInt(iconId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Creator<PwIconStandard> CREATOR = new Creator<PwIconStandard>() {
|
||||||
|
@Override
|
||||||
|
public PwIconStandard createFromParcel(Parcel in) {
|
||||||
|
return new PwIconStandard(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PwIconStandard[] newArray(int size) {
|
||||||
|
return new PwIconStandard[size];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isMetaStreamIcon() {
|
public boolean isMetaStreamIcon() {
|
||||||
return iconId == 0;
|
return iconId == 0;
|
||||||
|
|||||||
@@ -20,33 +20,69 @@
|
|||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database;
|
package com.kunzisoft.keepass.database;
|
||||||
|
|
||||||
import org.joda.time.LocalDate;
|
import android.os.Parcel;
|
||||||
|
import android.os.Parcelable;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import com.kunzisoft.keepass.app.App;
|
||||||
|
|
||||||
|
import org.joda.time.LocalDate;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract class who manage Groups and Entries
|
* Abstract class who manage Groups and Entries
|
||||||
*/
|
*/
|
||||||
public abstract class PwNode<Parent extends PwGroup> implements ISmallTimeLogger, Serializable, Cloneable {
|
public abstract class PwNode<Parent extends PwGroup> implements ISmallTimeLogger, Parcelable, Cloneable {
|
||||||
|
|
||||||
protected Parent parent = null;
|
protected Parent parent = null;
|
||||||
|
protected PwIconStandard icon = new PwIconStandard(0);
|
||||||
protected PwIconStandard icon = PwIconStandard.FIRST;
|
|
||||||
|
|
||||||
protected PwDate creation = new PwDate();
|
protected PwDate creation = new PwDate();
|
||||||
protected PwDate lastMod = new PwDate();
|
protected PwDate lastMod = new PwDate();
|
||||||
protected PwDate lastAccess = new PwDate();
|
protected PwDate lastAccess = new PwDate();
|
||||||
protected PwDate expireDate = PwDate.PW_NEVER_EXPIRE;
|
protected PwDate expireDate = PwDate.PW_NEVER_EXPIRE;
|
||||||
|
|
||||||
|
protected PwNode() {}
|
||||||
|
|
||||||
|
protected PwNode(Parcel in) {
|
||||||
|
// TODO better technique ?
|
||||||
|
try {
|
||||||
|
PwGroupId pwGroupId = in.readParcelable(PwGroupId.class.getClassLoader());
|
||||||
|
parent = (Parent) App.getDB().getPwDatabase().getGroupByGroupId(pwGroupId);
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
icon = in.readParcelable(PwIconStandard.class.getClassLoader());
|
||||||
|
creation = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
lastMod = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
lastAccess = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
expireDate = in.readParcelable(PwDate.class.getClassLoader());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeToParcel(Parcel dest, int flags) {
|
||||||
|
PwGroupId parentId = null;
|
||||||
|
if (parent != null)
|
||||||
|
parentId = parent.getId();
|
||||||
|
dest.writeParcelable(parentId, flags);
|
||||||
|
|
||||||
|
dest.writeParcelable(icon, flags);
|
||||||
|
dest.writeParcelable(creation, flags);
|
||||||
|
dest.writeParcelable(lastMod, flags);
|
||||||
|
dest.writeParcelable(lastAccess, flags);
|
||||||
|
dest.writeParcelable(expireDate, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int describeContents() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
protected void construct(Parent parent) {
|
protected void construct(Parent parent) {
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void assign(PwNode<Parent> source) {
|
protected void assign(PwNode<Parent> source) {
|
||||||
this.parent = source.parent;
|
this.parent = source.parent;
|
||||||
|
|
||||||
this.icon = source.icon;
|
this.icon = source.icon;
|
||||||
|
|
||||||
this.creation = source.creation;
|
this.creation = source.creation;
|
||||||
this.lastMod = source.lastMod;
|
this.lastMod = source.lastMod;
|
||||||
this.lastAccess = source.lastAccess;
|
this.lastAccess = source.lastAccess;
|
||||||
@@ -59,9 +95,7 @@ public abstract class PwNode<Parent extends PwGroup> implements ISmallTimeLogger
|
|||||||
try {
|
try {
|
||||||
newNode = (PwNode) super.clone();
|
newNode = (PwNode) super.clone();
|
||||||
// newNode.parent stay the same in copy
|
// newNode.parent stay the same in copy
|
||||||
|
|
||||||
newNode.icon = new PwIconStandard(this.icon);
|
newNode.icon = new PwIconStandard(this.icon);
|
||||||
|
|
||||||
newNode.creation = creation.clone();
|
newNode.creation = creation.clone();
|
||||||
newNode.lastMod = lastMod.clone();
|
newNode.lastMod = lastMod.clone();
|
||||||
newNode.lastAccess = lastAccess.clone();
|
newNode.lastAccess = lastAccess.clone();
|
||||||
@@ -119,6 +153,13 @@ public abstract class PwNode<Parent extends PwGroup> implements ISmallTimeLogger
|
|||||||
parent = prt;
|
parent = prt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return true if parent is present (can be a root or a detach element)
|
||||||
|
*/
|
||||||
|
public boolean containsParent() {
|
||||||
|
return getParent() != null;
|
||||||
|
}
|
||||||
|
|
||||||
public PwDate getCreationTime() {
|
public PwDate getCreationTime() {
|
||||||
return creation;
|
return creation;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,47 @@
|
|||||||
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
|
||||||
|
public abstract class ActionWithSaveDatabaseRunnable extends RunnableOnFinish {
|
||||||
|
|
||||||
|
protected Context mContext;
|
||||||
|
protected boolean mDontSave;
|
||||||
|
protected Database mDatabase;
|
||||||
|
|
||||||
|
public ActionWithSaveDatabaseRunnable(Context context, Database database, OnFinishRunnable finish, boolean dontSave) {
|
||||||
|
super(finish);
|
||||||
|
|
||||||
|
this.mDatabase = database;
|
||||||
|
this.mContext = context;
|
||||||
|
this.mDontSave = dontSave;
|
||||||
|
this.mFinish = new AfterActionRunnable(finish);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
// Commit to disk
|
||||||
|
SaveDatabaseRunnable save = new SaveDatabaseRunnable(mContext, mDatabase, mFinish, mDontSave);
|
||||||
|
save.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void runWithoutSaveDatabase() {
|
||||||
|
mFinish.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract protected void onFinish(boolean success, String message);
|
||||||
|
|
||||||
|
private class AfterActionRunnable extends OnFinishRunnable {
|
||||||
|
|
||||||
|
AfterActionRunnable(OnFinishRunnable finish) {
|
||||||
|
super(finish);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
onFinish(mSuccess, mMessage);
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.net.Uri;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwDatabase;
|
||||||
|
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
||||||
|
import com.kunzisoft.keepass.utils.UriUtil;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
|
||||||
|
public class AssignPasswordInDatabaseRunnable extends ActionWithSaveDatabaseRunnable {
|
||||||
|
|
||||||
|
private String mPassword;
|
||||||
|
private Uri mKeyfile;
|
||||||
|
private byte[] mBackupKey;
|
||||||
|
|
||||||
|
public AssignPasswordInDatabaseRunnable(Context ctx, Database db, String password, Uri keyfile, OnFinishRunnable finish) {
|
||||||
|
this(ctx, db, password, keyfile, finish, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public AssignPasswordInDatabaseRunnable(Context ctx, Database db, String password, Uri keyfile, OnFinishRunnable finish, boolean dontSave) {
|
||||||
|
super(ctx, db, finish, dontSave);
|
||||||
|
|
||||||
|
this.mPassword = password;
|
||||||
|
this.mKeyfile = keyfile;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
PwDatabase pm = mDatabase.getPwDatabase();
|
||||||
|
|
||||||
|
mBackupKey = new byte[pm.getMasterKey().length];
|
||||||
|
System.arraycopy(pm.getMasterKey(), 0, mBackupKey, 0, mBackupKey.length);
|
||||||
|
|
||||||
|
// Set key
|
||||||
|
try {
|
||||||
|
InputStream is = UriUtil.getUriInputStream(mContext, mKeyfile);
|
||||||
|
pm.retrieveMasterKey(mPassword, is);
|
||||||
|
// Save Database
|
||||||
|
super.run();
|
||||||
|
} catch (InvalidKeyFileException|IOException e) {
|
||||||
|
erase(mBackupKey);
|
||||||
|
finish(false, e.getMessage());
|
||||||
|
super.runWithoutSaveDatabase();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if (!success) {
|
||||||
|
// Erase the current master key
|
||||||
|
erase(mDatabase.getPwDatabase().getMasterKey());
|
||||||
|
mDatabase.getPwDatabase().setMasterKey(mBackupKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Overwrite the array as soon as we don't need it to avoid keeping the extra data in memory
|
||||||
|
*/
|
||||||
|
private void erase(byte[] array) {
|
||||||
|
if ( array == null ) return;
|
||||||
|
|
||||||
|
for ( int i = 0; i < array.length; i++ ) {
|
||||||
|
array[i] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
|
||||||
@@ -26,18 +26,18 @@ import com.kunzisoft.keepass.database.Database;
|
|||||||
import com.kunzisoft.keepass.database.PwDatabase;
|
import com.kunzisoft.keepass.database.PwDatabase;
|
||||||
import com.kunzisoft.keepass.utils.UriUtil;
|
import com.kunzisoft.keepass.utils.UriUtil;
|
||||||
|
|
||||||
public class CreateDB extends RunnableOnFinish {
|
public class CreateDatabaseRunnable extends RunnableOnFinish {
|
||||||
|
|
||||||
private String mFilename;
|
|
||||||
private boolean mDontSave;
|
|
||||||
private Context ctx;
|
|
||||||
|
|
||||||
public CreateDB(Context ctx, String filename, OnFinish finish, boolean dontSave) {
|
private Context mContext;
|
||||||
|
private boolean mDontSave;
|
||||||
|
private String mFilename;
|
||||||
|
|
||||||
|
public CreateDatabaseRunnable(Context mContext, String filename, OnFinishRunnable finish, boolean dontSave) {
|
||||||
super(finish);
|
super(finish);
|
||||||
|
|
||||||
mFilename = filename;
|
this.mContext = mContext;
|
||||||
mDontSave = dontSave;
|
this.mDontSave = dontSave;
|
||||||
this.ctx = ctx;
|
this.mFilename = filename;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -56,8 +56,9 @@ public class CreateDB extends RunnableOnFinish {
|
|||||||
App.clearShutdown();
|
App.clearShutdown();
|
||||||
|
|
||||||
// Commit changes
|
// Commit changes
|
||||||
SaveDB save = new SaveDB(ctx, db, mFinish, mDontSave);
|
SaveDatabaseRunnable save = new SaveDatabaseRunnable(mContext, db, mFinish, mDontSave);
|
||||||
mFinish = null;
|
mFinish = null;
|
||||||
|
|
||||||
save.run();
|
save.run();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -17,17 +17,18 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
|
||||||
public class FileOnFinish extends OnFinish implements Serializable {
|
public class FileOnFinishRunnable extends OnFinishRunnable implements Serializable {
|
||||||
|
|
||||||
private Uri mFilename = null;
|
private Uri mFilename = null;
|
||||||
protected FileOnFinish mOnFinish;
|
protected FileOnFinishRunnable mOnFinish;
|
||||||
|
|
||||||
public FileOnFinish(FileOnFinish finish) {
|
public FileOnFinishRunnable(FileOnFinishRunnable finish) {
|
||||||
super(finish);
|
super(finish);
|
||||||
|
|
||||||
mOnFinish = finish;
|
mOnFinish = finish;
|
||||||
@@ -17,12 +17,14 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.SharedPreferences;
|
import android.content.SharedPreferences;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.preference.PreferenceManager;
|
import android.preference.PreferenceManager;
|
||||||
|
import android.support.annotation.StringRes;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.R;
|
import com.kunzisoft.keepass.R;
|
||||||
import com.kunzisoft.keepass.app.App;
|
import com.kunzisoft.keepass.app.App;
|
||||||
@@ -40,71 +42,77 @@ import com.kunzisoft.keepass.database.exception.KeyFileEmptyException;
|
|||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
public class LoadDB extends RunnableOnFinish {
|
public class LoadDatabaseRunnable extends RunnableOnFinish {
|
||||||
|
private static final String TAG = LoadDatabaseRunnable.class.getName();
|
||||||
|
|
||||||
|
private Context mContext;
|
||||||
|
private Database mDatabase;
|
||||||
private Uri mUri;
|
private Uri mUri;
|
||||||
private String mPass;
|
private String mPass;
|
||||||
private Uri mKey;
|
private Uri mKey;
|
||||||
private Database mDb;
|
|
||||||
private Context mCtx;
|
|
||||||
private boolean mRememberKeyfile;
|
private boolean mRememberKeyfile;
|
||||||
|
|
||||||
public LoadDB(Database db, Context ctx, Uri uri, String pass, Uri key, OnFinish finish) {
|
public LoadDatabaseRunnable(Context context, Database database, Uri uri, String pass, Uri key, OnFinishRunnable finish) {
|
||||||
super(finish);
|
super(finish);
|
||||||
|
|
||||||
mDb = db;
|
this.mContext = context;
|
||||||
mCtx = ctx;
|
this.mDatabase = database;
|
||||||
mUri = uri;
|
this.mUri = uri;
|
||||||
mPass = pass;
|
this.mPass = pass;
|
||||||
mKey = key;
|
this.mKey = key;
|
||||||
|
|
||||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ctx);
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
||||||
mRememberKeyfile = prefs.getBoolean(ctx.getString(R.string.keyfile_key), ctx.getResources().getBoolean(R.bool.keyfile_default));
|
this.mRememberKeyfile = prefs.getBoolean(context.getString(R.string.keyfile_key), context.getResources().getBoolean(R.bool.keyfile_default));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
try {
|
try {
|
||||||
mDb.loadData(mCtx, mUri, mPass, mKey, mStatus);
|
mDatabase.loadData(mContext, mUri, mPass, mKey, mStatus);
|
||||||
|
|
||||||
saveFileData(mUri, mKey);
|
saveFileData(mUri, mKey);
|
||||||
|
|
||||||
} catch (ArcFourException e) {
|
} catch (ArcFourException e) {
|
||||||
finish(false, mCtx.getString(R.string.error_arc4));
|
catchError(e, R.string.error_arc4);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidPasswordException e) {
|
} catch (InvalidPasswordException e) {
|
||||||
finish(false, mCtx.getString(R.string.InvalidPassword));
|
catchError(e, R.string.InvalidPassword);
|
||||||
return;
|
return;
|
||||||
} catch (ContentFileNotFoundException e) {
|
} catch (ContentFileNotFoundException e) {
|
||||||
finish(false, mCtx.getString(R.string.file_not_found_content));
|
catchError(e, R.string.file_not_found_content);
|
||||||
return;
|
return;
|
||||||
} catch (FileNotFoundException e) {
|
} catch (FileNotFoundException e) {
|
||||||
finish(false, mCtx.getString(R.string.file_not_found));
|
catchError(e, R.string.file_not_found);
|
||||||
return;
|
return;
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
Log.e(TAG, "Database can't be read", e);
|
||||||
finish(false, e.getMessage());
|
finish(false, e.getMessage());
|
||||||
return;
|
return;
|
||||||
} catch (KeyFileEmptyException e) {
|
} catch (KeyFileEmptyException e) {
|
||||||
finish(false, mCtx.getString(R.string.keyfile_is_empty));
|
catchError(e, R.string.keyfile_is_empty);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidAlgorithmException e) {
|
} catch (InvalidAlgorithmException e) {
|
||||||
finish(false, mCtx.getString(R.string.invalid_algorithm));
|
catchError(e, R.string.invalid_algorithm);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidKeyFileException e) {
|
} catch (InvalidKeyFileException e) {
|
||||||
finish(false, mCtx.getString(R.string.keyfile_does_not_exist));
|
catchError(e, R.string.keyfile_does_not_exist);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidDBSignatureException e) {
|
} catch (InvalidDBSignatureException e) {
|
||||||
finish(false, mCtx.getString(R.string.invalid_db_sig));
|
catchError(e, R.string.invalid_db_sig);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidDBVersionException e) {
|
} catch (InvalidDBVersionException e) {
|
||||||
finish(false, mCtx.getString(R.string.unsupported_db_version));
|
catchError(e, R.string.unsupported_db_version);
|
||||||
return;
|
return;
|
||||||
} catch (InvalidDBException e) {
|
} catch (InvalidDBException e) {
|
||||||
finish(false, mCtx.getString(R.string.error_invalid_db));
|
catchError(e, R.string.error_invalid_db);
|
||||||
return;
|
return;
|
||||||
} catch (OutOfMemoryError e) {
|
} catch (OutOfMemoryError e) {
|
||||||
finish(false, mCtx.getString(R.string.error_out_of_memory));
|
String errorMessage = mContext.getString(R.string.error_out_of_memory);
|
||||||
|
Log.e(TAG, errorMessage, e);
|
||||||
|
finish(false, errorMessage);
|
||||||
return;
|
return;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "Database can't be load", e);
|
||||||
finish(false, e.getMessage());
|
finish(false, e.getMessage());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -112,6 +120,12 @@ public class LoadDB extends RunnableOnFinish {
|
|||||||
finish(true);
|
finish(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void catchError(Exception e, @StringRes int messageId) {
|
||||||
|
String errorMessage = mContext.getString(messageId);
|
||||||
|
Log.e(TAG, errorMessage, e);
|
||||||
|
finish(false, errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
private void saveFileData(Uri uri, Uri key) {
|
private void saveFileData(Uri uri, Uri key) {
|
||||||
if ( ! mRememberKeyfile ) {
|
if ( ! mRememberKeyfile ) {
|
||||||
key = null;
|
key = null;
|
||||||
@@ -120,6 +134,4 @@ public class LoadDB extends RunnableOnFinish {
|
|||||||
App.getFileHistory().createFile(uri, key);
|
App.getFileHistory().createFile(uri, key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
@@ -29,27 +29,27 @@ import android.widget.Toast;
|
|||||||
* @author bpellin
|
* @author bpellin
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class OnFinish implements Runnable {
|
public class OnFinishRunnable implements Runnable {
|
||||||
protected boolean mSuccess;
|
protected boolean mSuccess;
|
||||||
protected String mMessage;
|
protected String mMessage;
|
||||||
|
|
||||||
protected OnFinish mOnFinish;
|
protected OnFinishRunnable mOnFinish;
|
||||||
protected Handler mHandler;
|
protected Handler mHandler;
|
||||||
|
|
||||||
public OnFinish() {
|
public OnFinishRunnable() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public OnFinish(Handler handler) {
|
public OnFinishRunnable(Handler handler) {
|
||||||
mOnFinish = null;
|
mOnFinish = null;
|
||||||
mHandler = handler;
|
mHandler = handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
public OnFinish(OnFinish finish, Handler handler) {
|
public OnFinishRunnable(OnFinishRunnable finish, Handler handler) {
|
||||||
mOnFinish = finish;
|
mOnFinish = finish;
|
||||||
mHandler = handler;
|
mHandler = handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
public OnFinish(OnFinish finish) {
|
public OnFinishRunnable(OnFinishRunnable finish) {
|
||||||
mOnFinish = finish;
|
mOnFinish = finish;
|
||||||
mHandler = null;
|
mHandler = null;
|
||||||
}
|
}
|
||||||
@@ -75,11 +75,30 @@ public class OnFinish implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ONLY to use in UIThread, typically in an Activity, Fragment or a Service
|
||||||
|
* @param ctx Context to show the message
|
||||||
|
*/
|
||||||
protected void displayMessage(Context ctx) {
|
protected void displayMessage(Context ctx) {
|
||||||
if ( mMessage != null && mMessage.length() > 0 ) {
|
if ( mMessage != null && mMessage.length() > 0 ) {
|
||||||
Toast.makeText(ctx, mMessage, Toast.LENGTH_LONG).show();
|
Toast.makeText(ctx, mMessage, Toast.LENGTH_LONG).show();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isSuccess() {
|
||||||
|
return mSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSuccess(boolean mSuccess) {
|
||||||
|
this.mSuccess = mSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getMessage() {
|
||||||
|
return mMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMessage(String mMessage) {
|
||||||
|
this.mMessage = mMessage;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -17,17 +17,17 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.tasks.UpdateStatus;
|
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater;
|
||||||
|
|
||||||
|
|
||||||
public abstract class RunnableOnFinish implements Runnable {
|
public abstract class RunnableOnFinish implements Runnable {
|
||||||
|
|
||||||
public OnFinish mFinish;
|
public OnFinishRunnable mFinish;
|
||||||
public UpdateStatus mStatus;
|
public ProgressTaskUpdater mStatus;
|
||||||
|
|
||||||
public RunnableOnFinish(OnFinish finish) {
|
public RunnableOnFinish(OnFinishRunnable finish) {
|
||||||
mFinish = finish;
|
mFinish = finish;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,7 +45,7 @@ public abstract class RunnableOnFinish implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setStatus(UpdateStatus status) {
|
public void setUpdateProgressTaskStatus(ProgressTaskUpdater status) {
|
||||||
mStatus = status;
|
mStatus = status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
|
||||||
@@ -26,25 +26,22 @@ import com.kunzisoft.keepass.database.exception.PwDbOutputException;
|
|||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
public class SaveDB extends RunnableOnFinish {
|
public class SaveDatabaseRunnable extends RunnableOnFinish {
|
||||||
private Database mDb;
|
|
||||||
private boolean mDontSave;
|
|
||||||
private Context mCtx;
|
|
||||||
|
|
||||||
public SaveDB(Context ctx, Database db, OnFinish finish, boolean dontSave) {
|
private Context mContext;
|
||||||
|
private Database mDatabase;
|
||||||
|
private boolean mDontSave;
|
||||||
|
|
||||||
|
public SaveDatabaseRunnable(Context context, Database database, OnFinishRunnable finish, boolean dontSave) {
|
||||||
super(finish);
|
super(finish);
|
||||||
|
|
||||||
mDb = db;
|
this.mContext = context;
|
||||||
mDontSave = dontSave;
|
this.mDatabase = database;
|
||||||
mCtx = ctx;
|
this.mDontSave = dontSave;
|
||||||
}
|
}
|
||||||
|
|
||||||
public SaveDB(Context ctx, Database db, OnFinish finish) {
|
public SaveDatabaseRunnable(Context ctx, Database db, OnFinishRunnable finish) {
|
||||||
super(finish);
|
this(ctx, db, finish, false);
|
||||||
|
|
||||||
mDb = db;
|
|
||||||
mDontSave = false;
|
|
||||||
mCtx = ctx;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -52,7 +49,7 @@ public class SaveDB extends RunnableOnFinish {
|
|||||||
|
|
||||||
if ( ! mDontSave ) {
|
if ( ! mDontSave ) {
|
||||||
try {
|
try {
|
||||||
mDb.saveData(mCtx);
|
mDatabase.saveData(mContext);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
finish(false, e.getMessage());
|
finish(false, e.getMessage());
|
||||||
return;
|
return;
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwNode;
|
||||||
|
import com.kunzisoft.keepass.database.action.ActionWithSaveDatabaseRunnable;
|
||||||
|
|
||||||
|
abstract class ActionNodeDatabaseRunnable extends ActionWithSaveDatabaseRunnable {
|
||||||
|
|
||||||
|
private AfterActionNodeOnFinish callbackRunnable;
|
||||||
|
|
||||||
|
public ActionNodeDatabaseRunnable(Context context, Database db, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
|
super(context, db, finish, dontSave);
|
||||||
|
this.callbackRunnable = finish;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback method who return the node(s) modified after an action
|
||||||
|
* - Add : @param oldNode NULL, @param newNode CreatedNode
|
||||||
|
* - Copy : @param oldNode NodeToCopy, @param newNode NodeCopied
|
||||||
|
* - Delete : @param oldNode NodeToDelete, @param NULL
|
||||||
|
* - Move : @param oldNode NULL, @param NodeToMove
|
||||||
|
* - Update : @param oldNode NodeToUpdate, @param NodeUpdated
|
||||||
|
*/
|
||||||
|
protected void callbackNodeAction(boolean success, String message, PwNode oldNode, PwNode newNode) {
|
||||||
|
if (callbackRunnable != null) {
|
||||||
|
callbackRunnable.setSuccess(success);
|
||||||
|
callbackRunnable.setMessage(message);
|
||||||
|
callbackRunnable.run(oldNode, newNode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -17,57 +17,40 @@
|
|||||||
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
* along with KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
package com.kunzisoft.keepass.database.edit;
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
import com.kunzisoft.keepass.database.Database;
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
|
||||||
public class AddEntry extends RunnableOnFinish {
|
public class AddEntryRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
protected Database mDb;
|
private PwEntry mNewEntry;
|
||||||
private PwEntry mEntry;
|
|
||||||
private Context ctx;
|
|
||||||
private boolean mDontSave;
|
|
||||||
|
|
||||||
public AddEntry(Context ctx, Database db, PwEntry entry, OnFinish finish) {
|
public AddEntryRunnable(Context ctx, Database db, PwEntry entryToAdd, AfterActionNodeOnFinish finish) {
|
||||||
this(ctx, db, entry, finish, false);
|
this(ctx, db, entryToAdd, finish, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public AddEntry(Context ctx, Database db, PwEntry entry, OnFinish finish, boolean dontSave) {
|
public AddEntryRunnable(Context ctx, Database db, PwEntry entryToAdd, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
super(finish);
|
super(ctx, db, finish, dontSave);
|
||||||
|
|
||||||
this.mDb = db;
|
this.mNewEntry = entryToAdd;
|
||||||
this.mEntry = entry;
|
|
||||||
this.ctx = ctx;
|
|
||||||
this.mDontSave = dontSave;
|
|
||||||
|
|
||||||
this.mFinish = new AfterAdd(mFinish);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
mDb.addEntryTo(mEntry, mEntry.getParent());
|
mDatabase.addEntryTo(mNewEntry, mNewEntry.getParent());
|
||||||
|
|
||||||
// Commit to disk
|
// Commit to disk
|
||||||
SaveDB save = new SaveDB(ctx, mDb, mFinish, mDontSave);
|
super.run();
|
||||||
save.run();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private class AfterAdd extends OnFinish {
|
|
||||||
|
|
||||||
AfterAdd(OnFinish finish) {
|
@Override
|
||||||
super(finish);
|
protected void onFinish(boolean success, String message) {
|
||||||
}
|
if ( !success ) {
|
||||||
|
mDatabase.removeEntryFrom(mNewEntry, mNewEntry.getParent());
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
mDb.removeEntryFrom(mEntry, mEntry.getParent());
|
|
||||||
}
|
|
||||||
// TODO if add entry callback
|
|
||||||
super.run();
|
|
||||||
}
|
}
|
||||||
|
callbackNodeAction(success, message, null, mNewEntry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class AddGroupRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private PwGroup mNewGroup;
|
||||||
|
|
||||||
|
public AddGroupRunnable(Context ctx, Database db, PwGroup newGroup, AfterActionNodeOnFinish afterAddNode) {
|
||||||
|
this(ctx, db, newGroup, afterAddNode, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public AddGroupRunnable(Context ctx, Database db, PwGroup newGroup, AfterActionNodeOnFinish afterAddNode, boolean dontSave) {
|
||||||
|
super(ctx, db, afterAddNode, dontSave);
|
||||||
|
|
||||||
|
this.mNewGroup = newGroup;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
mDatabase.addGroupTo(mNewGroup, mNewGroup.getParent());
|
||||||
|
|
||||||
|
// Commit to disk
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
mDatabase.removeGroupFrom(mNewGroup, mNewGroup.getParent());
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, null, mNewGroup);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2018 Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.os.Handler;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.PwNode;
|
||||||
|
import com.kunzisoft.keepass.database.action.OnFinishRunnable;
|
||||||
|
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
|
public abstract class AfterActionNodeOnFinish extends OnFinishRunnable {
|
||||||
|
|
||||||
|
public AfterActionNodeOnFinish() {
|
||||||
|
super(new Handler());
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract void run(@Nullable PwNode oldNode, @Nullable PwNode newNode);
|
||||||
|
}
|
||||||
@@ -0,0 +1,75 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2018 Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class CopyEntryRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private static final String TAG = CopyEntryRunnable.class.getName();
|
||||||
|
|
||||||
|
private PwEntry mEntryToCopy;
|
||||||
|
private PwEntry mEntryCopied;
|
||||||
|
private PwGroup mNewParent;
|
||||||
|
|
||||||
|
public CopyEntryRunnable(Context context, Database db, PwEntry oldE, PwGroup newParent, AfterActionNodeOnFinish afterAddNode) {
|
||||||
|
this(context, db, oldE, newParent, afterAddNode, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public CopyEntryRunnable(Context context, Database db, PwEntry oldE, PwGroup newParent, AfterActionNodeOnFinish afterAddNode, boolean dontSave) {
|
||||||
|
super(context, db, afterAddNode, dontSave);
|
||||||
|
|
||||||
|
this.mEntryToCopy = oldE;
|
||||||
|
this.mNewParent = newParent;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
// Update entry with new values
|
||||||
|
mEntryCopied = mDatabase.copyEntry(mEntryToCopy, mNewParent);
|
||||||
|
|
||||||
|
if (mEntryCopied != null) {
|
||||||
|
mEntryCopied.touch(true, true);
|
||||||
|
|
||||||
|
// Commit to disk
|
||||||
|
super.run();
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "Unable to create a copy of the entry");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
// If we fail to save, try to delete the copy
|
||||||
|
try {
|
||||||
|
mDatabase.deleteEntry(mEntryCopied);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.i(TAG, "Unable to delete the copied entry");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, mEntryToCopy, mEntryCopied);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class DeleteEntryRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private PwEntry mEntryToDelete;
|
||||||
|
private PwGroup mParent;
|
||||||
|
private boolean mRecycle;
|
||||||
|
|
||||||
|
public DeleteEntryRunnable(Context ctx, Database db, PwEntry entry, AfterActionNodeOnFinish finish) {
|
||||||
|
this(ctx, db, entry, finish, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public DeleteEntryRunnable(Context ctx, Database db, PwEntry entry, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
|
super(ctx, db, finish, dontSave);
|
||||||
|
|
||||||
|
this.mEntryToDelete = entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
mParent = mEntryToDelete.getParent();
|
||||||
|
|
||||||
|
// Remove Entry from parent
|
||||||
|
mRecycle = mDatabase.canRecycle(mEntryToDelete);
|
||||||
|
if (mRecycle) {
|
||||||
|
mDatabase.recycle(mEntryToDelete);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
mDatabase.deleteEntry(mEntryToDelete);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Commit database
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
if (mRecycle) {
|
||||||
|
mDatabase.undoRecycle(mEntryToDelete, mParent);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
mDatabase.undoDeleteEntry(mEntryToDelete, mParent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, mEntryToDelete, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,96 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.app.App;
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class DeleteGroupRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private PwGroup<PwGroup, PwGroup, PwEntry> mGroupToDelete;
|
||||||
|
private PwGroup mParent;
|
||||||
|
private boolean mRecycle;
|
||||||
|
|
||||||
|
public DeleteGroupRunnable(Context ctx, Database db, PwGroup<PwGroup, PwGroup, PwEntry> group, AfterActionNodeOnFinish finish) {
|
||||||
|
this(ctx, db, group, finish, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public DeleteGroupRunnable(Context ctx, Database db, PwGroup<PwGroup, PwGroup, PwEntry> group, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
|
super(ctx, db, finish, dontSave);
|
||||||
|
mGroupToDelete = group;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
mParent = mGroupToDelete.getParent();
|
||||||
|
|
||||||
|
// Remove Group from parent
|
||||||
|
mRecycle = mDatabase.canRecycle(mGroupToDelete);
|
||||||
|
if (mRecycle) {
|
||||||
|
mDatabase.recycle(mGroupToDelete);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// TODO tests
|
||||||
|
// Remove child entries
|
||||||
|
List<PwEntry> childEnt = new ArrayList<>(mGroupToDelete.getChildEntries()); // TODO new Methods
|
||||||
|
for ( int i = 0; i < childEnt.size(); i++ ) {
|
||||||
|
DeleteEntryRunnable task = new DeleteEntryRunnable(mContext, mDatabase, childEnt.get(i), null, true);
|
||||||
|
task.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove child groups
|
||||||
|
List<PwGroup> childGrp = new ArrayList<>(mGroupToDelete.getChildGroups());
|
||||||
|
for ( int i = 0; i < childGrp.size(); i++ ) {
|
||||||
|
DeleteGroupRunnable task = new DeleteGroupRunnable(mContext, mDatabase, childGrp.get(i), null, true);
|
||||||
|
task.run();
|
||||||
|
}
|
||||||
|
mDatabase.deleteGroup(mGroupToDelete);
|
||||||
|
|
||||||
|
// Remove from PwDatabaseV3
|
||||||
|
// TODO ENcapsulate
|
||||||
|
mDatabase.getPwDatabase().getGroups().remove(mGroupToDelete);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Commit Database
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
if (mRecycle) {
|
||||||
|
mDatabase.undoRecycle(mGroupToDelete, mParent);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Let's not bother recovering from a failure to save a deleted tree. It is too much work.
|
||||||
|
App.setShutdown();
|
||||||
|
// TODO TEST pm.undoDeleteGroup(mGroup, mParent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, mGroupToDelete, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2018 Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class MoveEntryRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private static final String TAG = MoveEntryRunnable.class.getName();
|
||||||
|
|
||||||
|
private PwEntry mEntryToMove;
|
||||||
|
private PwGroup mOldParent;
|
||||||
|
private PwGroup mNewParent;
|
||||||
|
|
||||||
|
public MoveEntryRunnable(Context context, Database db, PwEntry oldE, PwGroup newParent, AfterActionNodeOnFinish afterAddNode) {
|
||||||
|
this(context, db, oldE, newParent, afterAddNode, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public MoveEntryRunnable(Context context, Database db, PwEntry oldE, PwGroup newParent, AfterActionNodeOnFinish afterAddNode, boolean dontSave) {
|
||||||
|
super(context, db, afterAddNode, dontSave);
|
||||||
|
|
||||||
|
this.mEntryToMove = oldE;
|
||||||
|
this.mNewParent = newParent;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
// Move entry in new parent
|
||||||
|
|
||||||
|
mOldParent = mEntryToMove.getParent();
|
||||||
|
mDatabase.moveEntry(mEntryToMove, mNewParent);
|
||||||
|
|
||||||
|
if (mEntryToMove != null) {
|
||||||
|
mEntryToMove.touch(true, true);
|
||||||
|
|
||||||
|
// Commit to disk
|
||||||
|
super.run();
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "Unable to create a copy of the entry");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
// If we fail to save, try to remove in the first place
|
||||||
|
try {
|
||||||
|
mDatabase.moveEntry(mEntryToMove, mOldParent);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.i(TAG, "Unable to replace the entry");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, null, mEntryToMove);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2018 Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.R;
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class MoveGroupRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private static final String TAG = MoveGroupRunnable.class.getName();
|
||||||
|
|
||||||
|
private PwGroup mGroupToMove;
|
||||||
|
private PwGroup mOldParent;
|
||||||
|
private PwGroup mNewParent;
|
||||||
|
|
||||||
|
public MoveGroupRunnable(Context context, Database db, PwGroup groupToMove, PwGroup newParent, AfterActionNodeOnFinish afterAddNode) {
|
||||||
|
this(context, db, groupToMove, newParent, afterAddNode, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public MoveGroupRunnable(Context context, Database db, PwGroup groupToMove, PwGroup newParent, AfterActionNodeOnFinish afterAddNode, boolean dontSave) {
|
||||||
|
super(context, db, afterAddNode, dontSave);
|
||||||
|
|
||||||
|
this.mGroupToMove = groupToMove;
|
||||||
|
this.mNewParent = newParent;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
mOldParent = mGroupToMove.getParent();
|
||||||
|
// Move group in new parent if not in the current group
|
||||||
|
if (!mGroupToMove.equals(mNewParent)
|
||||||
|
&& !mNewParent.isContainedIn(mGroupToMove)) {
|
||||||
|
mDatabase.moveGroup(mGroupToMove, mNewParent);
|
||||||
|
|
||||||
|
if (mGroupToMove != null) {
|
||||||
|
mGroupToMove.touch(true, true);
|
||||||
|
|
||||||
|
// Commit to disk
|
||||||
|
super.run();
|
||||||
|
} else {
|
||||||
|
Log.e(TAG, "Unable to create a copy of the group");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Only finish thread
|
||||||
|
mFinish.setResult(false);
|
||||||
|
String message = mContext.getString(R.string.error_move_folder_in_itself);
|
||||||
|
Log.e(TAG, message);
|
||||||
|
mFinish.setMessage(message);
|
||||||
|
mFinish.run();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
// If we fail to save, try to move in the first place
|
||||||
|
try {
|
||||||
|
mDatabase.moveGroup(mGroupToMove, mOldParent);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.i(TAG, "Unable to replace the group");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, null, mGroupToMove);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,63 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwEntry;
|
||||||
|
|
||||||
|
public class UpdateEntryRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private PwEntry mOldEntry;
|
||||||
|
private PwEntry mNewEntry;
|
||||||
|
private PwEntry mBackupEntry;
|
||||||
|
|
||||||
|
public UpdateEntryRunnable(Context ctx, Database db, PwEntry oldE, PwEntry newE, AfterActionNodeOnFinish finish) {
|
||||||
|
this(ctx, db, oldE, newE, finish, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UpdateEntryRunnable(Context ctx, Database db, PwEntry oldE, PwEntry newE, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
|
super(ctx, db, finish, dontSave);
|
||||||
|
|
||||||
|
this.mOldEntry = oldE;
|
||||||
|
this.mNewEntry = newE;
|
||||||
|
// Keep backup of original values in case save fails
|
||||||
|
this.mBackupEntry = mOldEntry.clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
// Update entry with new values
|
||||||
|
mDatabase.updateEntry(mOldEntry, mNewEntry);
|
||||||
|
mOldEntry.touch(true, true);
|
||||||
|
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
// If we fail to save, back out changes to global structure
|
||||||
|
mDatabase.updateEntry(mOldEntry, mBackupEntry);
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, mOldEntry, mNewEntry);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
||||||
|
*
|
||||||
|
* This file is part of KeePass DX.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
package com.kunzisoft.keepass.database.action.node;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.kunzisoft.keepass.database.Database;
|
||||||
|
import com.kunzisoft.keepass.database.PwGroup;
|
||||||
|
|
||||||
|
public class UpdateGroupRunnable extends ActionNodeDatabaseRunnable {
|
||||||
|
|
||||||
|
private PwGroup mOldGroup;
|
||||||
|
private PwGroup mNewGroup;
|
||||||
|
private PwGroup mBackupGroup;
|
||||||
|
|
||||||
|
public UpdateGroupRunnable(Context ctx, Database db, PwGroup oldGroup, PwGroup newGroup, AfterActionNodeOnFinish finish) {
|
||||||
|
this(ctx, db, oldGroup, newGroup, finish, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UpdateGroupRunnable(Context ctx, Database db, PwGroup oldGroup, PwGroup newGroup, AfterActionNodeOnFinish finish, boolean dontSave) {
|
||||||
|
super(ctx, db, finish, dontSave);
|
||||||
|
|
||||||
|
this.mOldGroup = oldGroup;
|
||||||
|
this.mNewGroup = newGroup;
|
||||||
|
// Keep backup of original values in case save fails
|
||||||
|
this.mBackupGroup = mOldGroup.clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
// Update group with new values
|
||||||
|
mDatabase.updateGroup(mOldGroup, mNewGroup);
|
||||||
|
mOldGroup.touch(true, true);
|
||||||
|
|
||||||
|
// Commit to disk
|
||||||
|
super.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onFinish(boolean success, String message) {
|
||||||
|
if ( !success ) {
|
||||||
|
// If we fail to save, back out changes to global structure
|
||||||
|
mDatabase.updateGroup(mOldGroup, mBackupGroup);
|
||||||
|
}
|
||||||
|
callbackNodeAction(success, message, mOldGroup, mNewGroup);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,79 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
|
||||||
|
|
||||||
public class AddGroup extends RunnableOnFinish {
|
|
||||||
|
|
||||||
protected Database mDb;
|
|
||||||
private PwGroup mNewGroup;
|
|
||||||
private Context ctx;
|
|
||||||
private boolean mDontSave;
|
|
||||||
|
|
||||||
public AddGroup(Context ctx, Database db, PwGroup newGroup, AfterActionNodeOnFinish afterAddNode) {
|
|
||||||
this(ctx, db, newGroup, afterAddNode, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public AddGroup(Context ctx, Database db, PwGroup newGroup, AfterActionNodeOnFinish afterAddNode,
|
|
||||||
boolean dontSave) {
|
|
||||||
super(afterAddNode);
|
|
||||||
|
|
||||||
this.mDb = db;
|
|
||||||
this.mNewGroup = newGroup;
|
|
||||||
this.mDontSave = dontSave;
|
|
||||||
this.ctx = ctx;
|
|
||||||
|
|
||||||
this.mFinish = new AfterAdd(mFinish);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
mDb.addGroupTo(mNewGroup, mNewGroup.getParent());
|
|
||||||
|
|
||||||
// Commit to disk
|
|
||||||
SaveDB save = new SaveDB(ctx, mDb, mFinish, mDontSave);
|
|
||||||
save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterAdd extends OnFinish {
|
|
||||||
|
|
||||||
AfterAdd(OnFinish finish) {
|
|
||||||
super(finish);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
mDb.removeGroupFrom(mNewGroup, mNewGroup.getParent());
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO Better callback
|
|
||||||
AfterActionNodeOnFinish afterAddNode =
|
|
||||||
(AfterActionNodeOnFinish) super.mOnFinish;
|
|
||||||
afterAddNode.mSuccess = mSuccess;
|
|
||||||
afterAddNode.mMessage = mMessage;
|
|
||||||
afterAddNode.run(null, mNewGroup);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,103 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
|
||||||
|
|
||||||
/** Task to delete entries
|
|
||||||
* @author bpellin
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
public class DeleteEntry extends RunnableOnFinish {
|
|
||||||
|
|
||||||
private Database mDb;
|
|
||||||
private PwEntry mEntry;
|
|
||||||
private boolean mDontSave;
|
|
||||||
private Context ctx;
|
|
||||||
|
|
||||||
public DeleteEntry(Context ctx, Database db, PwEntry entry, OnFinish finish) {
|
|
||||||
this(ctx, db, entry, finish, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public DeleteEntry(Context ctx, Database db, PwEntry entry, OnFinish finish, boolean dontSave) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
this.mDb = db;
|
|
||||||
this.mEntry = entry;
|
|
||||||
this.mDontSave = dontSave;
|
|
||||||
this.ctx = ctx;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
PwGroup parent = mEntry.getParent();
|
|
||||||
|
|
||||||
// Remove Entry from parent
|
|
||||||
boolean recycle = mDb.canRecycle(mEntry);
|
|
||||||
if (recycle) {
|
|
||||||
mDb.recycle(mEntry);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
mDb.deleteEntry(mEntry);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save
|
|
||||||
mFinish = new AfterDelete(mFinish, parent, mEntry, recycle);
|
|
||||||
|
|
||||||
// Commit database
|
|
||||||
SaveDB save = new SaveDB(ctx, mDb, mFinish, mDontSave);
|
|
||||||
save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterDelete extends OnFinish {
|
|
||||||
|
|
||||||
private PwGroup mParent;
|
|
||||||
private PwEntry mEntry;
|
|
||||||
private boolean recycled;
|
|
||||||
|
|
||||||
AfterDelete(OnFinish finish, PwGroup parent, PwEntry entry, boolean r) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
mParent = parent;
|
|
||||||
mEntry = entry;
|
|
||||||
recycled = r;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
if (recycled) {
|
|
||||||
mDb.undoRecycle(mEntry, mParent);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
mDb.undoDeleteEntry(mEntry, mParent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO Callback after delete entry
|
|
||||||
|
|
||||||
super.run();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.app.App;
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class DeleteGroup extends RunnableOnFinish {
|
|
||||||
|
|
||||||
private Context mContext;
|
|
||||||
private Database mDb;
|
|
||||||
private PwGroup<PwGroup, PwGroup, PwEntry> mGroup;
|
|
||||||
private boolean mDontSave;
|
|
||||||
|
|
||||||
public DeleteGroup(Context ctx, Database db, PwGroup<PwGroup, PwGroup, PwEntry> group, OnFinish finish) {
|
|
||||||
super(finish);
|
|
||||||
setMembers(ctx, db, group, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public DeleteGroup(Context ctx, Database db, PwGroup<PwGroup, PwGroup, PwEntry> group, OnFinish finish, boolean dontSave) {
|
|
||||||
super(finish);
|
|
||||||
setMembers(ctx, db, group, dontSave);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setMembers(Context ctx, Database db, PwGroup<PwGroup, PwGroup, PwEntry> group, boolean dontSave) {
|
|
||||||
mDb = db;
|
|
||||||
mGroup = group;
|
|
||||||
mContext = ctx;
|
|
||||||
mDontSave = dontSave;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
PwGroup parent = mGroup.getParent();
|
|
||||||
|
|
||||||
// Remove Group from parent
|
|
||||||
boolean recycle = mDb.canRecycle(mGroup);
|
|
||||||
if (recycle) {
|
|
||||||
mDb.recycle(mGroup);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// TODO tests
|
|
||||||
// Remove child entries
|
|
||||||
List<PwEntry> childEnt = new ArrayList<>(mGroup.getChildEntries()); // TODO new Methods
|
|
||||||
for ( int i = 0; i < childEnt.size(); i++ ) {
|
|
||||||
DeleteEntry task = new DeleteEntry(mContext, mDb, childEnt.get(i), null, true);
|
|
||||||
task.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove child groups
|
|
||||||
List<PwGroup> childGrp = new ArrayList<>(mGroup.getChildGroups());
|
|
||||||
for ( int i = 0; i < childGrp.size(); i++ ) {
|
|
||||||
DeleteGroup task = new DeleteGroup(mContext, mDb, childGrp.get(i), null, true);
|
|
||||||
task.run();
|
|
||||||
}
|
|
||||||
mDb.deleteGroup(mGroup);
|
|
||||||
|
|
||||||
// Remove from PwDatabaseV3
|
|
||||||
// TODO ENcapsulate
|
|
||||||
mDb.getPwDatabase().getGroups().remove(mGroup);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save
|
|
||||||
mFinish = new AfterDelete(mFinish, parent, mGroup, recycle);
|
|
||||||
|
|
||||||
// Commit Database
|
|
||||||
SaveDB save = new SaveDB(mContext, mDb, mFinish, mDontSave);
|
|
||||||
save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterDelete extends OnFinish {
|
|
||||||
|
|
||||||
private PwGroup mParent;
|
|
||||||
private PwGroup mGroup;
|
|
||||||
private boolean recycled;
|
|
||||||
|
|
||||||
AfterDelete(OnFinish finish, PwGroup parent, PwGroup mGroup, boolean recycle) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
this.mParent = parent;
|
|
||||||
this.mGroup = mGroup;
|
|
||||||
this.recycled = recycle;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
if (recycled) {
|
|
||||||
mDb.undoRecycle(mGroup, mParent);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// Let's not bother recovering from a failure to save a deleted tree. It is too much work.
|
|
||||||
App.setShutdown();
|
|
||||||
// TODO TEST pm.undoDeleteGroup(mGroup, mParent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO Callback after delete group
|
|
||||||
|
|
||||||
super.run();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.DialogInterface;
|
|
||||||
import android.net.Uri;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwDatabase;
|
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
|
||||||
import com.kunzisoft.keepass.dialogs.PasswordEncodingDialogHelper;
|
|
||||||
import com.kunzisoft.keepass.utils.UriUtil;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
|
|
||||||
public class SetPassword extends RunnableOnFinish {
|
|
||||||
|
|
||||||
private String mPassword;
|
|
||||||
private Uri mKeyfile;
|
|
||||||
private Database mDb;
|
|
||||||
private boolean mDontSave;
|
|
||||||
private Context ctx;
|
|
||||||
|
|
||||||
public SetPassword(Context ctx, Database db, String password, Uri keyfile, OnFinish finish) {
|
|
||||||
this(ctx, db, password, keyfile, finish, false);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public SetPassword(Context ctx, Database db, String password, Uri keyfile, OnFinish finish, boolean dontSave) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
mDb = db;
|
|
||||||
mPassword = password;
|
|
||||||
mKeyfile = keyfile;
|
|
||||||
mDontSave = dontSave;
|
|
||||||
this.ctx = ctx;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean validatePassword(Context ctx, DialogInterface.OnClickListener onclick) {
|
|
||||||
if (!mDb.getPwDatabase().validatePasswordEncoding(mPassword)) {
|
|
||||||
PasswordEncodingDialogHelper dialog = new PasswordEncodingDialogHelper();
|
|
||||||
dialog.show(ctx, onclick, true);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
PwDatabase pm = mDb.getPwDatabase();
|
|
||||||
|
|
||||||
byte[] backupKey = new byte[pm.getMasterKey().length];
|
|
||||||
System.arraycopy(pm.getMasterKey(), 0, backupKey, 0, backupKey.length);
|
|
||||||
|
|
||||||
// Set key
|
|
||||||
try {
|
|
||||||
InputStream is = UriUtil.getUriInputStream(ctx, mKeyfile);
|
|
||||||
pm.setMasterKey(mPassword, is);
|
|
||||||
} catch (InvalidKeyFileException e) {
|
|
||||||
erase(backupKey);
|
|
||||||
finish(false, e.getMessage());
|
|
||||||
return;
|
|
||||||
} catch (IOException e) {
|
|
||||||
erase(backupKey);
|
|
||||||
finish(false, e.getMessage());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save Database
|
|
||||||
mFinish = new AfterSave(backupKey, mFinish);
|
|
||||||
SaveDB save = new SaveDB(ctx, mDb, mFinish, mDontSave);
|
|
||||||
save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterSave extends OnFinish {
|
|
||||||
private byte[] mBackup;
|
|
||||||
|
|
||||||
public AfterSave(byte[] backup, OnFinish finish) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
mBackup = backup;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( ! mSuccess ) {
|
|
||||||
// Erase the current master key
|
|
||||||
erase(mDb.getPwDatabase().getMasterKey());
|
|
||||||
mDb.getPwDatabase().setMasterKey(mBackup);
|
|
||||||
}
|
|
||||||
|
|
||||||
super.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Overwrite the array as soon as we don't need it to avoid keeping the extra data in memory
|
|
||||||
* @param array
|
|
||||||
*/
|
|
||||||
private void erase(byte[] array) {
|
|
||||||
if ( array == null ) return;
|
|
||||||
|
|
||||||
for ( int i = 0; i < array.length; i++ ) {
|
|
||||||
array[i] = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,84 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwEntry;
|
|
||||||
|
|
||||||
public class UpdateEntry extends RunnableOnFinish {
|
|
||||||
|
|
||||||
private Database mDb;
|
|
||||||
private PwEntry mOldE;
|
|
||||||
private PwEntry mNewE;
|
|
||||||
private Context ctx;
|
|
||||||
private boolean mDontSave;
|
|
||||||
|
|
||||||
public UpdateEntry(Context ctx, Database db, PwEntry oldE, PwEntry newE, OnFinish finish) {
|
|
||||||
this(ctx, db, oldE, newE, finish, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UpdateEntry(Context ctx, Database db, PwEntry oldE, PwEntry newE, OnFinish finish, boolean dontSave) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
this.mDb = db;
|
|
||||||
this.mOldE = oldE;
|
|
||||||
this.mNewE = newE;
|
|
||||||
this.ctx = ctx;
|
|
||||||
this.mDontSave = dontSave;
|
|
||||||
|
|
||||||
// Keep backup of original values in case save fails
|
|
||||||
PwEntry backup;
|
|
||||||
backup = mOldE.clone();
|
|
||||||
|
|
||||||
mFinish = new AfterUpdate(backup, finish);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
// Update entry with new values
|
|
||||||
mDb.updateEntry(mOldE, mNewE);
|
|
||||||
mOldE.touch(true, true);
|
|
||||||
|
|
||||||
// Commit to disk
|
|
||||||
SaveDB save = new SaveDB(ctx, mDb, mFinish, mDontSave);
|
|
||||||
save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterUpdate extends OnFinish {
|
|
||||||
private PwEntry mBackup;
|
|
||||||
|
|
||||||
AfterUpdate(PwEntry backup, OnFinish finish) {
|
|
||||||
super(finish);
|
|
||||||
mBackup = backup;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
// If we fail to save, back out changes to global structure
|
|
||||||
mDb.updateEntry(mOldE, mBackup);
|
|
||||||
}
|
|
||||||
// TODO Callback for update entry
|
|
||||||
super.run();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 Brian Pellin, Jeremy Jamet / Kunzisoft.
|
|
||||||
*
|
|
||||||
* This file is part of KeePass DX.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* KeePass DX 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 KeePass DX. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
package com.kunzisoft.keepass.database.edit;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import com.kunzisoft.keepass.database.Database;
|
|
||||||
import com.kunzisoft.keepass.database.PwGroup;
|
|
||||||
|
|
||||||
public class UpdateGroup extends RunnableOnFinish {
|
|
||||||
|
|
||||||
private Database mDb;
|
|
||||||
private PwGroup mOldGroup;
|
|
||||||
private PwGroup mNewGroup;
|
|
||||||
private Context ctx;
|
|
||||||
private boolean mDontSave;
|
|
||||||
|
|
||||||
public UpdateGroup(Context ctx, Database db, PwGroup oldGroup, PwGroup newGroup, AfterActionNodeOnFinish finish) {
|
|
||||||
this(ctx, db, oldGroup, newGroup, finish, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public UpdateGroup(Context ctx, Database db, PwGroup oldGroup, PwGroup newGroup, AfterActionNodeOnFinish finish, boolean dontSave) {
|
|
||||||
super(finish);
|
|
||||||
|
|
||||||
this.mDb = db;
|
|
||||||
this.mOldGroup = oldGroup;
|
|
||||||
this.mNewGroup = newGroup;
|
|
||||||
this.ctx = ctx;
|
|
||||||
this.mDontSave = dontSave;
|
|
||||||
|
|
||||||
// Keep backup of original values in case save fails
|
|
||||||
PwGroup backup;
|
|
||||||
backup = mOldGroup.clone();
|
|
||||||
|
|
||||||
this.mFinish = new AfterUpdate(backup, finish);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
// Update group with new values
|
|
||||||
mDb.updateGroup(mOldGroup, mNewGroup);
|
|
||||||
mOldGroup.touch(true, true);
|
|
||||||
|
|
||||||
// Commit to disk
|
|
||||||
new SaveDB(ctx, mDb, mFinish, mDontSave).run();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class AfterUpdate extends OnFinish {
|
|
||||||
private PwGroup mBackup;
|
|
||||||
|
|
||||||
AfterUpdate(PwGroup backup, OnFinish finish) {
|
|
||||||
super(finish);
|
|
||||||
mBackup = backup;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
if ( !mSuccess ) {
|
|
||||||
// If we fail to save, back out changes to global structure
|
|
||||||
mDb.updateGroup(mOldGroup, mBackup);
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO Better callback
|
|
||||||
AfterActionNodeOnFinish afterActionNodeOnFinish =
|
|
||||||
(AfterActionNodeOnFinish) super.mOnFinish;
|
|
||||||
afterActionNodeOnFinish.mSuccess = mSuccess;
|
|
||||||
afterActionNodeOnFinish.mMessage = mMessage;
|
|
||||||
afterActionNodeOnFinish.run(mOldGroup, mNewGroup);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
package com.kunzisoft.keepass.database.exception;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
public class UnknownKDF extends IOException {
|
||||||
|
|
||||||
|
private static String message = "Unknown key derivation function";
|
||||||
|
|
||||||
|
public UnknownKDF() {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public UnknownKDF(Exception e) {
|
||||||
|
super(message, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -76,15 +76,19 @@ public class EntrySearchStringIteratorV3 extends EntrySearchStringIterator {
|
|||||||
switch (current) {
|
switch (current) {
|
||||||
case title:
|
case title:
|
||||||
found = sp.searchInTitles;
|
found = sp.searchInTitles;
|
||||||
|
break;
|
||||||
|
|
||||||
case url:
|
case url:
|
||||||
found = sp.searchInUrls;
|
found = sp.searchInUrls;
|
||||||
|
break;
|
||||||
|
|
||||||
case username:
|
case username:
|
||||||
found = sp.searchInUserNames;
|
found = sp.searchInUserNames;
|
||||||
|
break;
|
||||||
|
|
||||||
case comment:
|
case comment:
|
||||||
found = sp.searchInNotes;
|
found = sp.searchInNotes;
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
found = true;
|
found = true;
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ package com.kunzisoft.keepass.database.load;
|
|||||||
|
|
||||||
import com.kunzisoft.keepass.database.PwDatabase;
|
import com.kunzisoft.keepass.database.PwDatabase;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
||||||
import com.kunzisoft.keepass.tasks.UpdateStatus;
|
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
@@ -33,8 +33,7 @@ public abstract class Importer {
|
|||||||
public abstract PwDatabase openDatabase(InputStream inStream, String password, InputStream keyInputStream)
|
public abstract PwDatabase openDatabase(InputStream inStream, String password, InputStream keyInputStream)
|
||||||
throws IOException, InvalidDBException;
|
throws IOException, InvalidDBException;
|
||||||
|
|
||||||
public abstract PwDatabase openDatabase(InputStream inStream, String password, InputStream keyInputStream, UpdateStatus status, long roundsFix)
|
public abstract PwDatabase openDatabase(InputStream inStream, String password, InputStream keyInputStream, ProgressTaskUpdater updater)
|
||||||
throws IOException, InvalidDBException;
|
throws IOException, InvalidDBException;
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,13 +28,11 @@ import java.io.IOException;
|
|||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
|
||||||
public class ImporterFactory {
|
public class ImporterFactory {
|
||||||
public static Importer createImporter(InputStream is) throws InvalidDBSignatureException, IOException
|
public static Importer createImporter(InputStream is) throws InvalidDBSignatureException, IOException {
|
||||||
{
|
|
||||||
return createImporter(is, false);
|
return createImporter(is, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Importer createImporter(InputStream is, boolean debug) throws InvalidDBSignatureException, IOException
|
public static Importer createImporter(InputStream is, boolean debug) throws InvalidDBSignatureException, IOException {
|
||||||
{
|
|
||||||
int sig1 = LEDataInputStream.readInt(is);
|
int sig1 = LEDataInputStream.readInt(is);
|
||||||
int sig2 = LEDataInputStream.readInt(is);
|
int sig2 = LEDataInputStream.readInt(is);
|
||||||
|
|
||||||
@@ -49,6 +47,5 @@ public class ImporterFactory {
|
|||||||
}
|
}
|
||||||
|
|
||||||
throw new InvalidDBSignatureException();
|
throw new InvalidDBSignatureException();
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -63,9 +63,8 @@ import com.kunzisoft.keepass.database.exception.InvalidDBVersionException;
|
|||||||
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
import com.kunzisoft.keepass.database.exception.InvalidKeyFileException;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidPasswordException;
|
import com.kunzisoft.keepass.database.exception.InvalidPasswordException;
|
||||||
import com.kunzisoft.keepass.stream.LEDataInputStream;
|
import com.kunzisoft.keepass.stream.LEDataInputStream;
|
||||||
import com.kunzisoft.keepass.stream.LEDataOutputStream;
|
|
||||||
import com.kunzisoft.keepass.stream.NullOutputStream;
|
import com.kunzisoft.keepass.stream.NullOutputStream;
|
||||||
import com.kunzisoft.keepass.tasks.UpdateStatus;
|
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater;
|
||||||
import com.kunzisoft.keepass.utils.Types;
|
import com.kunzisoft.keepass.utils.Types;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
@@ -124,22 +123,22 @@ public class ImporterV3 extends Importer {
|
|||||||
* @throws InvalidAlgorithmParameterException if error decrypting main file body.
|
* @throws InvalidAlgorithmParameterException if error decrypting main file body.
|
||||||
* @throws ShortBufferException if error decrypting main file body.
|
* @throws ShortBufferException if error decrypting main file body.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public PwDatabaseV3 openDatabase( InputStream inStream, String password, InputStream kfIs)
|
public PwDatabaseV3 openDatabase( InputStream inStream, String password, InputStream kfIs)
|
||||||
throws IOException, InvalidDBException
|
throws IOException, InvalidDBException {
|
||||||
{
|
return openDatabase(inStream, password, kfIs, null);
|
||||||
return openDatabase(inStream, password, kfIs, new UpdateStatus(), 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public PwDatabaseV3 openDatabase( InputStream inStream, String password, InputStream kfIs, UpdateStatus status, long roundsFix)
|
@Override
|
||||||
throws IOException, InvalidDBException
|
public PwDatabaseV3 openDatabase(InputStream inStream, String password, InputStream kfIs, ProgressTaskUpdater progressTaskUpdater)
|
||||||
{
|
throws IOException, InvalidDBException {
|
||||||
PwDatabaseV3 newManager;
|
|
||||||
|
|
||||||
|
PwDatabaseV3 databaseToOpen;
|
||||||
|
|
||||||
// Load entire file, most of it's encrypted.
|
// Load entire file, most of it's encrypted.
|
||||||
int fileSize = inStream.available();
|
int fileSize = inStream.available();
|
||||||
byte[] filebuf = new byte[fileSize + 16]; // Pad with a blocksize (Twofish uses 128 bits), since Android 4.3 tries to write more to the buffer
|
byte[] filebuf = new byte[fileSize + 16]; // Pad with a blocksize (Twofish uses 128 bits), since Android 4.3 tries to write more to the buffer
|
||||||
inStream.read(filebuf, 0, fileSize);
|
inStream.read(filebuf, 0, fileSize); // TODO remove
|
||||||
inStream.close();
|
inStream.close();
|
||||||
|
|
||||||
// Parse header (unencrypted)
|
// Parse header (unencrypted)
|
||||||
@@ -156,34 +155,36 @@ public class ImporterV3 extends Importer {
|
|||||||
throw new InvalidDBVersionException();
|
throw new InvalidDBVersionException();
|
||||||
}
|
}
|
||||||
|
|
||||||
status.updateMessage(R.string.creating_db_key);
|
if (progressTaskUpdater != null)
|
||||||
newManager = createDB();
|
progressTaskUpdater.updateMessage(R.string.creating_db_key);
|
||||||
newManager.setMasterKey(password, kfIs);
|
databaseToOpen = createDB();
|
||||||
|
databaseToOpen.retrieveMasterKey(password, kfIs);
|
||||||
|
|
||||||
// Select algorithm
|
// Select algorithm
|
||||||
if( (hdr.flags & PwDbHeaderV3.FLAG_RIJNDAEL) != 0 ) {
|
if( (hdr.flags & PwDbHeaderV3.FLAG_RIJNDAEL) != 0 ) {
|
||||||
newManager.setEncryptionAlgorithm(PwEncryptionAlgorithm.AES_Rijndael);
|
databaseToOpen.setEncryptionAlgorithm(PwEncryptionAlgorithm.AES_Rijndael);
|
||||||
} else if( (hdr.flags & PwDbHeaderV3.FLAG_TWOFISH) != 0 ) {
|
} else if( (hdr.flags & PwDbHeaderV3.FLAG_TWOFISH) != 0 ) {
|
||||||
newManager.setEncryptionAlgorithm(PwEncryptionAlgorithm.Twofish);
|
databaseToOpen.setEncryptionAlgorithm(PwEncryptionAlgorithm.Twofish);
|
||||||
} else {
|
} else {
|
||||||
throw new InvalidAlgorithmException();
|
throw new InvalidAlgorithmException();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy for testing
|
// Copy for testing
|
||||||
newManager.copyHeader(hdr);
|
databaseToOpen.copyHeader(hdr);
|
||||||
|
|
||||||
newManager.setNumberKeyEncryptionRounds(hdr.numKeyEncRounds);
|
databaseToOpen.setNumberKeyEncryptionRounds(hdr.numKeyEncRounds);
|
||||||
|
|
||||||
// Generate transformedMasterKey from masterKey
|
// Generate transformedMasterKey from masterKey
|
||||||
newManager.makeFinalKey(hdr.masterSeed, hdr.transformSeed, newManager.getNumberKeyEncryptionRounds());
|
databaseToOpen.makeFinalKey(hdr.masterSeed, hdr.transformSeed, databaseToOpen.getNumberKeyEncryptionRounds());
|
||||||
|
|
||||||
status.updateMessage(R.string.decrypting_db);
|
if (progressTaskUpdater != null)
|
||||||
|
progressTaskUpdater.updateMessage(R.string.decrypting_db);
|
||||||
// Initialize Rijndael algorithm
|
// Initialize Rijndael algorithm
|
||||||
Cipher cipher;
|
Cipher cipher;
|
||||||
try {
|
try {
|
||||||
if ( newManager.getEncryptionAlgorithm() == PwEncryptionAlgorithm.AES_Rijndael) {
|
if ( databaseToOpen.getEncryptionAlgorithm() == PwEncryptionAlgorithm.AES_Rijndael) {
|
||||||
cipher = CipherFactory.getInstance("AES/CBC/PKCS5Padding");
|
cipher = CipherFactory.getInstance("AES/CBC/PKCS5Padding");
|
||||||
} else if ( newManager.getEncryptionAlgorithm() == PwEncryptionAlgorithm.Twofish ) {
|
} else if ( databaseToOpen.getEncryptionAlgorithm() == PwEncryptionAlgorithm.Twofish ) {
|
||||||
cipher = CipherFactory.getInstance("Twofish/CBC/PKCS7PADDING");
|
cipher = CipherFactory.getInstance("Twofish/CBC/PKCS7PADDING");
|
||||||
} else {
|
} else {
|
||||||
throw new IOException( "Encryption algorithm is not supported" );
|
throw new IOException( "Encryption algorithm is not supported" );
|
||||||
@@ -196,7 +197,7 @@ public class ImporterV3 extends Importer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
cipher.init( Cipher.DECRYPT_MODE, new SecretKeySpec( newManager.getFinalKey(), "AES" ), new IvParameterSpec( hdr.encryptionIV ) );
|
cipher.init( Cipher.DECRYPT_MODE, new SecretKeySpec( databaseToOpen.getFinalKey(), "AES" ), new IvParameterSpec( hdr.encryptionIV ) );
|
||||||
} catch (InvalidKeyException e1) {
|
} catch (InvalidKeyException e1) {
|
||||||
throw new IOException("Invalid key");
|
throw new IOException("Invalid key");
|
||||||
} catch (InvalidAlgorithmParameterException e1) {
|
} catch (InvalidAlgorithmParameterException e1) {
|
||||||
@@ -216,7 +217,7 @@ public class ImporterV3 extends Importer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Copy decrypted data for testing
|
// Copy decrypted data for testing
|
||||||
newManager.copyEncrypted(filebuf, PwDbHeaderV3.BUF_SIZE, encryptedPartSize);
|
databaseToOpen.copyEncrypted(filebuf, PwDbHeaderV3.BUF_SIZE, encryptedPartSize);
|
||||||
|
|
||||||
MessageDigest md = null;
|
MessageDigest md = null;
|
||||||
try {
|
try {
|
||||||
@@ -249,13 +250,13 @@ public class ImporterV3 extends Importer {
|
|||||||
if( fieldType == 0xFFFF ) {
|
if( fieldType == 0xFFFF ) {
|
||||||
|
|
||||||
// End-Group record. Save group and count it.
|
// End-Group record. Save group and count it.
|
||||||
newGrp.populateBlankFields(newManager);
|
newGrp.populateBlankFields(databaseToOpen);
|
||||||
newManager.addGroup(newGrp);
|
databaseToOpen.addGroup(newGrp);
|
||||||
newGrp = new PwGroupV3();
|
newGrp = new PwGroupV3();
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
readGroupField(newManager, newGrp, fieldType, filebuf, pos);
|
readGroupField(databaseToOpen, newGrp, fieldType, filebuf, pos);
|
||||||
}
|
}
|
||||||
pos += fieldSize;
|
pos += fieldSize;
|
||||||
}
|
}
|
||||||
@@ -268,65 +269,22 @@ public class ImporterV3 extends Importer {
|
|||||||
|
|
||||||
if( fieldType == 0xFFFF ) {
|
if( fieldType == 0xFFFF ) {
|
||||||
// End-Group record. Save group and count it.
|
// End-Group record. Save group and count it.
|
||||||
newEnt.populateBlankFields(newManager);
|
newEnt.populateBlankFields(databaseToOpen);
|
||||||
newManager.addEntry(newEnt);
|
databaseToOpen.addEntry(newEnt);
|
||||||
newEnt = new PwEntryV3();
|
newEnt = new PwEntryV3();
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
readEntryField(newManager, newEnt, filebuf, pos);
|
readEntryField(databaseToOpen, newEnt, filebuf, pos);
|
||||||
}
|
}
|
||||||
pos += 2 + 4 + fieldSize;
|
pos += 2 + 4 + fieldSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
newManager.constructTree(null);
|
databaseToOpen.constructTree(null);
|
||||||
|
|
||||||
return newManager;
|
return databaseToOpen;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* KeePass's custom pad style.
|
|
||||||
*
|
|
||||||
* @param data buffer to pad.
|
|
||||||
* @return addtional bytes to append to data[] to make
|
|
||||||
* a properly padded array.
|
|
||||||
*/
|
|
||||||
public static byte[] makePad( byte[] data ) {
|
|
||||||
//custom pad method
|
|
||||||
|
|
||||||
// append 0x80 plus zeros to a multiple of 4 bytes
|
|
||||||
int thisblk = 32 - data.length % 32; // bytes needed to finish blk
|
|
||||||
int nextblk = 0; // 32 if we need another block
|
|
||||||
// need 9 bytes; add new block if no room
|
|
||||||
if( thisblk < 9 ) {
|
|
||||||
nextblk = 32;
|
|
||||||
}
|
|
||||||
|
|
||||||
// all bytes are zeroed for free
|
|
||||||
byte[] pad = new byte[ thisblk + nextblk ];
|
|
||||||
pad[0] = (byte)0x80;
|
|
||||||
|
|
||||||
// write length*8 to end of final block
|
|
||||||
int ix = thisblk + nextblk - 8;
|
|
||||||
LEDataOutputStream.writeInt( data.length>>29, pad, ix );
|
|
||||||
bsw32( pad, ix );
|
|
||||||
ix += 4;
|
|
||||||
LEDataOutputStream.writeInt( data.length<<3, pad, ix );
|
|
||||||
bsw32( pad, ix );
|
|
||||||
|
|
||||||
return pad;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void bsw32( byte[] ary, int offset ) {
|
|
||||||
byte t = ary[offset];
|
|
||||||
ary[offset] = ary[offset+3];
|
|
||||||
ary[offset+3] = t;
|
|
||||||
t = ary[offset+1];
|
|
||||||
ary[offset+1] = ary[offset+2];
|
|
||||||
ary[offset+2] = t;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Parse and save one record from binary file.
|
* Parse and save one record from binary file.
|
||||||
* @param buf
|
* @param buf
|
||||||
@@ -334,7 +292,7 @@ public class ImporterV3 extends Importer {
|
|||||||
* @return If >0,
|
* @return If >0,
|
||||||
* @throws UnsupportedEncodingException
|
* @throws UnsupportedEncodingException
|
||||||
*/
|
*/
|
||||||
void readGroupField(PwDatabaseV3 db, PwGroupV3 grp, int fieldType, byte[] buf, int offset) throws UnsupportedEncodingException {
|
private void readGroupField(PwDatabaseV3 db, PwGroupV3 grp, int fieldType, byte[] buf, int offset) throws UnsupportedEncodingException {
|
||||||
switch( fieldType ) {
|
switch( fieldType ) {
|
||||||
case 0x0000 :
|
case 0x0000 :
|
||||||
// Ignore field
|
// Ignore field
|
||||||
@@ -371,9 +329,7 @@ public class ImporterV3 extends Importer {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
void readEntryField(PwDatabaseV3 db, PwEntryV3 ent, byte[] buf, int offset)
|
private void readEntryField(PwDatabaseV3 db, PwEntryV3 ent, byte[] buf, int offset) throws UnsupportedEncodingException {
|
||||||
throws UnsupportedEncodingException
|
|
||||||
{
|
|
||||||
int fieldType = LEDataInputStream.readUShort(buf, offset);
|
int fieldType = LEDataInputStream.readUShort(buf, offset);
|
||||||
offset += 2;
|
offset += 2;
|
||||||
int fieldSize = LEDataInputStream.readInt(buf, offset);
|
int fieldSize = LEDataInputStream.readInt(buf, offset);
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ package com.kunzisoft.keepass.database.load;
|
|||||||
|
|
||||||
import com.kunzisoft.keepass.database.PwDatabaseV3Debug;
|
import com.kunzisoft.keepass.database.PwDatabaseV3Debug;
|
||||||
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
import com.kunzisoft.keepass.database.exception.InvalidDBException;
|
||||||
import com.kunzisoft.keepass.tasks.UpdateStatus;
|
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
@@ -35,10 +35,9 @@ public class ImporterV3Debug extends ImporterV3 {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PwDatabaseV3Debug openDatabase(InputStream inStream, String password,
|
public PwDatabaseV3Debug openDatabase(InputStream inStream, String password,
|
||||||
InputStream keyInputStream, UpdateStatus status, long roundsFix) throws IOException,
|
InputStream keyInputStream, ProgressTaskUpdater status) throws IOException,
|
||||||
InvalidDBException {
|
InvalidDBException {
|
||||||
return (PwDatabaseV3Debug) super.openDatabase(inStream, password, keyInputStream, status,
|
return (PwDatabaseV3Debug) super.openDatabase(inStream, password, keyInputStream, status);
|
||||||
roundsFix);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||