Compare commits

..

153 Commits

Author SHA1 Message Date
Zane Schepke 85316bec3f build: change flavor name to improve clarity 2025-04-29 04:59:22 -04:00
Zane Schepke 1935653309 chore(deps): bump compose, datastore 2025-04-29 04:45:11 -04:00
Zane Schepke e3e24b4a06 fix: cleanup logs on update 2025-04-29 04:37:32 -04:00
Zane Schepke 7af53dcc18 fix: skip ping job for static configured tunnels
#741
2025-04-28 17:35:31 -04:00
Zane Schepke 2eb0ab0f19 fix: vpn permission bug
closes #754
2025-04-28 16:07:01 -04:00
Zane Schepke 07857a53c2 fix: regenerate icon to also trigger auth
closes #757
2025-04-28 15:17:55 -04:00
Zane Schepke 25fd31e252 fix: tunnel lock (#765)
fix: start up logger bug
refactor: switch to bound services
refactor: expose resolved peer endpoint
2025-04-28 15:06:43 -04:00
Zane Schepke 0c90b33813 feat: display Wi-Fi security type for Android 12 and greater
refactor: deprecated clipboard manager
2025-04-25 19:25:06 -04:00
Zane Schepke e6671fd3b4 fix: switching APs or Wi-Fi bands with same SSID bug
#741
closes #154
2025-04-25 16:11:37 -04:00
Zane Schepke 735e38e989 feat: add darker theme options
closes #706
2025-04-25 01:59:57 -04:00
Zane Schepke 90698c2b17 fix: select split tunnel apps should appear at top of list
#662
closes #640
2025-04-25 01:17:14 -04:00
Zane Schepke 245b8ee3e7 ci: sort primary to always be first 2025-04-25 00:19:35 -04:00
dependabot[bot] 343554407a chore(deps): bump androidx.datastore:datastore-preferences from 1.1.4 to 1.1.5 (#748)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-25 00:12:22 -04:00
dependabot[bot] b493d83730 chore(deps): bump androidx.compose:compose-bom from 2025.04.00 to 2025.04.01 (#747)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-25 00:12:14 -04:00
dependabot[bot] 53cd717340 chore(deps): bump ClementTsang/delete-tag-and-release from 0.3.1 to 0.4.0 (#738)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-25 00:11:52 -04:00
dependabot[bot] 76574e3dd2 chore(deps): bump androidx.work:work-runtime-ktx from 2.10.0 to 2.10.1 (#746)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-25 00:11:16 -04:00
dependabot[bot] 282a752389 chore(deps): bump roomVersion from 2.7.0 to 2.7.1 (#745)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-25 00:11:06 -04:00
Zane Schepke 5aa9145361 fix: single click in select mode
closes #739
2025-04-25 00:04:30 -04:00
Zane Schepke 586726c848 ci: fix multiple artifacts 2025-04-23 06:32:04 -04:00
Zane Schepke af759a3909 ci: fix removed publish actions 2025-04-23 06:10:18 -04:00
Zane Schepke b467d66554 chore: bump version with notes 2025-04-23 05:59:03 -04:00
Zane Schepke c833e15c8f fix: disable version checker for google, for now 2025-04-23 05:58:42 -04:00
Zane Schepke eec1bbd2f6 ci: fix publish (#737) 2025-04-23 05:27:57 -04:00
Zane Schepke 969e9dfe03 fix: support screen padding 2025-04-23 01:41:50 -04:00
Zane Schepke aeb590db8c refactor: version code generation 2025-04-23 01:32:30 -04:00
Zane Schepke 312062aa36 refactor: app versioning and flavors 2025-04-23 01:23:01 -04:00
dependabot[bot] 287732dfb8 chore(deps): bump ktorClientCore from 3.1.1 to 3.1.2 (#734)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-22 19:42:39 -04:00
dependabot[bot] dca72a70e8 chore(deps): bump hiltAndroid from 2.56.1 to 2.56.2 (#703)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-22 19:42:30 -04:00
dependabot[bot] 1c6543554f chore(deps): bump app.cash.licensee from 1.12.0 to 1.13.0 (#735)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-22 19:42:20 -04:00
dependabot[bot] 8c01f5bea4 chore(deps): bump androidGradlePlugin from 8.9.1 to 8.9.2 (#733)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-22 19:42:12 -04:00
Zane Schepke dd9f329721 fix: banner foreground 2025-04-22 16:51:03 -04:00
Zane Schepke f30c48a90a fix: android tv banner 2025-04-22 16:47:36 -04:00
Zane Schepke 4707d3eb95 fix: app versioning 2025-04-22 03:03:24 -04:00
Zane Schepke cedc2db326 feat: add app licenses screen 2025-04-21 15:33:14 -04:00
Zane Schepke 256e3f7951 fix: version changed while testing 2025-04-21 11:52:40 -04:00
Zane Schepke 9e797b24d6 feat: add in-app updater for release versions
closes #636
2025-04-21 11:51:18 -04:00
Zane Schepke f2b9eb526e fix: amnezia compatibility action
closes #711
2025-04-21 06:56:16 -04:00
Zane Schepke abb29607d3 refactor: ui section divider padding 2025-04-21 06:45:26 -04:00
Zane Schepke f6d7cbc032 fix: recomposition bug, improve cancel scenario
#704
2025-04-20 22:00:26 -04:00
Zane Schepke 9304d79775 feat: variable number tunnel export with file explorer support
feat: listen for user present AndroidTV
#606
closes #704
2025-04-20 21:30:20 -04:00
Zane Schepke 4d18decbf7 fix: simplify bottom nav
closes #716
closes #705
2025-04-19 18:01:09 -04:00
Zane Schepke 76186c092f feat: export variable number of tunnels 2025-04-18 22:41:45 -04:00
Zane Schepke c90a7bbaf5 feat: add multi-select support
closes #332
2025-04-18 18:32:12 -04:00
Zane Schepke d3d70ab2e7 build: change from debug signing 2025-04-17 05:04:29 -04:00
Zane Schepke 9b2d4a3fb5 build: fix release building 2025-04-17 05:02:43 -04:00
Zane Schepke d7741c37c5 chore: bump version, release notes 2025-04-17 04:43:31 -04:00
Zane Schepke 6046e4131f fix: android tv sleep restore
#606
2025-04-17 04:38:15 -04:00
Zane Schepke 4b2d2d20db fix: split tunnel search and select
closes #696
2025-04-17 04:28:20 -04:00
Zane Schepke a09501aaf5 feat(lang): weblate langauge updates (#701)
Co-authored-by: Hosted Weblate <hosted@weblate.org>
Co-authored-by: Matthaiks <kitynska@gmail.com>
Co-authored-by: Languages add-on <noreply-addon-languages@weblate.org>
Co-authored-by: solokot <solokot@gmail.com>
Co-authored-by: Kachelkaiser <kachelkaiser@htpst.de>
Co-authored-by: ssantos <ssantos@web.de>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
Co-authored-by: mak7im01 <mak7im02@gmail.com>
Co-authored-by: nware-lab <nware.labs@gmail.com>
Co-authored-by: 翻譯得真好下次別翻了 <x86_64-pc-linux-gnu@proton.me>
Co-authored-by: Faisal Gull <mail.faisalrehman.345@gmail.com>
2025-04-17 04:08:19 -04:00
Zane Schepke d46a0653f1 ci: remove old workflows 2025-04-17 01:37:32 -04:00
Zane Schepke 49ee2431c2 refactor: revert banner for the google monopoly 2025-04-16 07:26:50 -04:00
Zane Schepke dfcc022257 refactor: minor change 2025-04-16 07:18:21 -04:00
Zane Schepke bc08069a64 ci: fix release token 2025-04-15 16:19:36 -04:00
Zane Schepke fb97adca4f fix: latest tagging 2025-04-15 16:05:55 -04:00
Zane Schepke 41540db9b7 ci: fix org changes (#698) 2025-04-15 15:44:01 -04:00
Zane Schepke a1c663233d fix: space 2025-04-15 06:00:06 -04:00
Zane Schepke c520fa5ed2 fix: ci 2025-04-15 05:59:41 -04:00
Zane Schepke 120bde2939 ci: remove old r8 rules 2025-04-15 05:20:52 -04:00
Zane Schepke 58fcc358ce build: gradle checksum update 2025-04-15 04:57:43 -04:00
Zane Schepke 72722a0be5 fix: android dns issue
closes #687
2025-04-15 03:53:17 -04:00
Zane Schepke 29aba65690 ci: change to org ci vars 2025-04-14 15:52:14 -04:00
Tobias Wienkoop 5d9a534e1c feat: add missing monochrome app icon (#689) 2025-04-12 11:56:56 -04:00
Zane Schepke f5dafa6bf7 ci: fix fastlane 2025-04-11 22:28:59 -04:00
Zane Schepke 4d64d058de chore: bump version with notes 2025-04-11 21:19:11 -04:00
Weblate (bot) 7e9687aeb9 feat(lang): Translations update from Hosted Weblate (#671)
Co-authored-by: Matthaiks <kitynska@gmail.com>
Co-authored-by: Languages add-on <noreply-addon-languages@weblate.org>
Co-authored-by: solokot <solokot@gmail.com>
Co-authored-by: Kachelkaiser <kachelkaiser@htpst.de>
Co-authored-by: ssantos <ssantos@web.de>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
2025-04-11 21:16:41 -04:00
Zane Schepke a6e559ecec fix: logger toggle bug
closes #669
2025-04-11 21:06:47 -04:00
Zane Schepke c6bacf8e15 style: latest logo 2025-04-11 20:03:32 -04:00
Zane Schepke fdfc348e76 fix: kill switch launch on start
closes #686
2025-04-11 19:07:36 -04:00
Zane Schepke 77b83ea569 fix: AndroidTV language selection ui bug
fix: AndroidTV restart on boot

closes #673
closes #606
2025-04-11 18:42:11 -04:00
Zane Schepke 5ded556647 fix: ping job start by default
fix: kernel dns resolution on stop bug

closes #674
2025-04-11 17:54:40 -04:00
Zane Schepke b62e592ee9 chore: bump deps 2025-04-11 04:51:06 -04:00
dependabot[bot] 869e1ebf0d chore(deps): bump com.google.devtools.ksp from 2.1.20-1.0.32 to 2.1.20-2.0.0 (#677)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-11 04:50:04 -04:00
dependabot[bot] 352eae0b28 chore(deps): bump actions/checkout from 3 to 4 (#675)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-11 04:49:08 -04:00
dependabot[bot] 7cb91ecd94 chore(deps): bump androidx.compose.material3:material3 from 1.3.1 to 1.3.2 (#682)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-11 04:47:54 -04:00
dependabot[bot] 3291bb0718 chore(deps): bump roomVersion from 2.6.1 to 2.7.0 (#681)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2025-04-11 04:47:09 -04:00
Zane Schepke ce3f0b85c1 fix: toggle resolution issue
#669
2025-04-11 04:29:03 -04:00
Zane Schepke f9768fc9f0 chore: update license 2025-04-11 02:36:17 -04:00
Zane Schepke 64db37648a style: new app icon 2025-04-11 01:29:21 -04:00
Zane Schepke cc5a2a972b feat(lang): weblate changes (#670)
Co-authored-by: solokot <solokot@gmail.com>
Co-authored-by: lateweb <weblate@techkoala.net>
Co-authored-by: Matthaiks <kitynska@gmail.com>
Co-authored-by: Kachelkaiser <kachelkaiser@outlook.com>
Co-authored-by: CyanWolf <hydemr@pm.me>
Co-authored-by: Henrik Sozzi <henrik_sozzi@hotmail.com>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
Co-authored-by: x86_64-pc-linux-gnu <x86_64-pc-linux-gnu@proton.me>
Co-authored-by: mak7im01 <mak7im02@gmail.com>
Co-authored-by: heykanspor <meingithub@heykan.de>
Co-authored-by: Jan-Pascal van Best <janpascal@vanbest.org>
Co-authored-by: Faisal Gull <mail.faisalrehman.345@gmail.com>
Co-authored-by: தமிழ்நேரம் <anishprabu.t@gmail.com>
Co-authored-by: catelixor <catelixor+weblate@proton.me>
Co-authored-by: Deleted User <noreply+48943@weblate.org>
Co-authored-by: kometchtech <kometch@gmail.com>
2025-04-09 03:41:58 -04:00
Zane Schepke 6294c7372a ci: fix matrix 2025-04-08 22:19:00 -04:00
Zane Schepke d562f36652 ci: fix notification workflow on release 2025-04-08 22:02:10 -04:00
Zane Schepke e77966d70a ci: fix notification workflow on release 2025-04-08 21:39:41 -04:00
Zane Schepke dcf213b63c fix: signing 2025-04-08 21:30:11 -04:00
Zane Schepke ca10586604 chore: bump version with notes 2025-04-08 21:14:39 -04:00
Zane Schepke 53480b0233 fix: back gesture issues on some devices 2025-04-08 21:04:22 -04:00
Zane Schepke 84de3a3991 fix: default to phone preferred dns server
closes #663
2025-04-08 20:51:10 -04:00
Zane Schepke 820ff8a9ad ci: add matrix, fix release notifications 2025-04-08 20:45:57 -04:00
Zane Schepke 1c0b54a8e4 feat: copy wifi name to clipboard
closes #65
2025-04-08 19:11:24 -04:00
Zane Schepke 75364f323c fix: tv navigation bug
closes #666
2025-04-08 18:57:09 -04:00
Zane Schepke b87aa75bf0 feat: add custom intent app control 2025-04-08 18:49:27 -04:00
Zane Schepke c59e7d7637 fix: service shutdown on abrupt shutdown 2025-04-07 13:56:40 -04:00
Zane Schepke 28ef1a7683 fix: tun start bug after bad shutdown 2025-04-07 05:18:26 -04:00
Zane Schepke a5aadb42ed fix: ipv6 static regex bug 2025-04-06 18:33:09 -04:00
Zane Schepke 9e0e17787d chore: bump version, update screenshots 2025-04-06 02:44:04 -04:00
Zane Schepke b4c5b51644 fix: tunnel service handling of multiple tunnels and tunnel job restarts 2025-04-06 01:46:22 -04:00
Zane Schepke 76191c46f3 fix: pop backstack on config save 2025-04-05 21:10:47 -04:00
Zane Schepke ecf5036f56 fix: nav backstack bug 2025-04-05 20:58:20 -04:00
Zane Schepke 1c0d968cfb fix: auto tunnel tile state bug 2025-04-05 20:17:16 -04:00
Zane Schepke bda1a2080a fix: tunnel tile default to previously active tunnel 2025-04-05 19:58:39 -04:00
Zane Schepke 14a71e3118 fix: stop kill switch on trusted auto tunnel bug 2025-04-05 19:12:55 -04:00
Zane Schepke 57391290c5 refactor: major ui refactor and bug fixes (#660)
refactor: improved error handling
fix: vpn kill switch with domain endpoints resolution
fix: bugs in ping restarts and config change restarts
feat: new nav bar
feat: expose network status info
feat: ui enhancements
2025-04-05 18:11:21 -04:00
Zane Schepke cd623c0c0c fix: ui bug 2025-04-02 00:54:44 -04:00
Zane Schepke 212c6cf088 feat: add screen security to config screen 2025-04-01 22:45:36 -04:00
Zane Schepke ca47127bff refactor: state management (#656) 2025-04-01 22:18:38 -04:00
Zane Schepke e63733286c refactor: advanced screen 2025-03-31 17:52:14 -04:00
Zane Schepke 36c76565f7 refactor: optimize log toggle 2025-03-31 17:08:39 -04:00
Zane Schepke 47f8de8c57 refactor: support screen, add matrix link 2025-03-31 16:40:40 -04:00
Zane Schepke 5740012101 chore: fmt and bump hilt 2025-03-31 14:30:51 -04:00
Zane Schepke 6f5bb24cfa fix: dns64 and ip version detection improvements 2025-03-31 13:13:44 -04:00
Zane Schepke 5f791ffda1 chore: bump ksp 2025-03-30 18:48:47 -04:00
Zane Schepke ec244eeda3 chore: bump deps 2025-03-30 18:46:26 -04:00
Hendrik Volkmer ff2a2cc082 feat: Add option to add config via URL (#623)
Co-authored-by: Zane Schepke <zanecschepke@gmail.com>
2025-03-30 18:37:07 -04:00
Zane Schepke a873546e9e fix: bugs in config changes and ping tunnel jobs (#650) 2025-03-30 18:31:26 -04:00
Zane Schepke 757669ddbe docs: update matrix link 2025-03-23 15:35:18 -04:00
Zane Schepke c71c4e5b29 chore: bump version and notes 2025-03-19 22:35:21 -04:00
Zane Schepke 7f0fea3766 fix: improve wifi monitoring to better handle permission changes 2025-03-19 21:51:54 -04:00
Zane Schepke 53c19762ef fix: attempt to improve tile sync 2025-03-16 23:12:39 -04:00
Zane Schepke c98fa04f73 fix: auto tunnel and tunnel regressions 2025-03-16 20:10:44 -04:00
Zane Schepke aba0f7d4d3 chore: bump deps 2025-03-16 02:05:55 -04:00
Zane Schepke fa517b2124 fix: race conditions (#621) 2025-03-16 02:04:09 -04:00
Zane Schepke d7e2648393 docs: update readme links 2025-03-15 19:22:34 -04:00
Zane Schepke 53ff3bb1e5 chore: bump version with notes 2025-03-15 13:10:53 -04:00
Zane Schepke 97ede3d5b4 fix: set prefer ipv4 as default to mimic old wg behavior 2025-03-15 12:44:56 -04:00
Zane Schepke dcd15f7bd8 fix: adding new peer bug
Closes #612
2025-03-15 11:30:10 -04:00
Zane Schepke 6031d85edd fix: re-enable shortcuts
Closes #616
2025-03-15 00:16:02 -04:00
Weblate (bot) a71f8f86b1 feat: Translations update from Hosted Weblate (#556)
Co-authored-by: Zane Schepke <zanecschepke@gmail.com>
Co-authored-by: solokot <solokot@gmail.com>
Co-authored-by: lateweb <weblate@techkoala.net>
Co-authored-by: Matthaiks <kitynska@gmail.com>
Co-authored-by: Kachelkaiser <kachelkaiser@outlook.com>
Co-authored-by: CyanWolf <hydemr@pm.me>
Co-authored-by: Henrik Sozzi <henrik_sozzi@hotmail.com>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
Co-authored-by: x86_64-pc-linux-gnu <x86_64-pc-linux-gnu@proton.me>
Co-authored-by: mak7im01 <mak7im02@gmail.com>
Co-authored-by: heykanspor <meingithub@heykan.de>
2025-03-14 22:51:26 -04:00
Zane Schepke 007c9f4c5d fix: static ipv6 endpoint bug 2025-03-14 22:49:08 -04:00
Zane Schepke e32a99db77 chore: fix fi 2025-03-08 23:37:16 -05:00
Zane Schepke 6670a62e2f chore: fix fastlane 2025-03-08 23:07:02 -05:00
Zane Schepke 34b20bd7f7 fix: pt region localization 2025-03-08 22:52:53 -05:00
Zane Schepke 01e15099ca chore: bump version with notes 2025-03-08 22:27:17 -05:00
Zane Schepke 8f2fd93e77 fix: improve local logging 2025-03-08 22:11:23 -05:00
𝗛𝗼𝗹𝗶 94197c9943 feat: Update Turkish Translation (#605) 2025-03-08 21:32:41 -05:00
Zane Schepke 39fc9014d8 fix: bump deps for dns fix 2025-03-08 21:31:31 -05:00
Zane Schepke 8860b45230 fix: service manager bug 2025-03-02 17:51:48 -05:00
Zane Schepke e220b26d88 fix: temporarily disallow start/stop of pinger while tunnel running 2025-03-02 17:25:29 -05:00
Zane Schepke 2302b473b4 refactor: tunnel toggling 2025-03-02 16:47:55 -05:00
Zane Schepke b01473073f fix: add restart support htc quickboot 2025-03-02 12:09:57 -05:00
Zane Schepke 3ae9a24ca4 refactor: restore on restart/update 2025-03-02 12:01:28 -05:00
Zane Schepke dc3f7fa736 refactor: improve service manager 2025-03-02 11:38:36 -05:00
Zane Schepke 93d6f8aa45 fix: module flavors 2025-03-01 21:07:47 -05:00
Zane Schepke 3ea4aea5cf fix: improve network status monitoring 2025-03-01 20:17:19 -05:00
Zane Schepke 68b41c8925 fix: startup/nav bugs 2025-03-01 12:15:37 -05:00
Zane Schepke 06de1f24c2 fix: tunnel race condition 2025-03-01 11:21:07 -05:00
Zane Schepke a39feeeea6 fix: proguard rules 2025-02-23 23:21:58 -05:00
Zane Schepke c1619ff012 temp disable proguard 2025-02-23 23:15:40 -05:00
Zane Schepke 2534b86005 fix: add back rules 2025-02-23 22:50:20 -05:00
Zane Schepke 15c550737c bump deps, remove redudant proguard rules 2025-02-23 22:39:15 -05:00
Zane Schepke e1e7e27bb5 fix: proguard (#590) 2025-02-23 16:41:51 -05:00
Zane Schepke 8021c133a5 fix: proguard 2025-02-23 15:58:08 -05:00
Zane Schepke 6009445a15 bump tunnel deps 2025-02-23 15:16:12 -05:00
Zane Schepke f80af9dd5e fix: tunnel state bug 2025-02-22 23:21:25 -05:00
Zane Schepke 3f912ed532 fix: tunnel change restart logic 2025-02-22 13:07:15 -05:00
786 changed files with 16412 additions and 13875 deletions
-97
View File
@@ -1,97 +0,0 @@
root = true
[*]
charset = utf-8
indent_size = 4
indent_style = tab
max_line_length = 150
trim_trailing_whitespace = true
insert_final_newline = true
[{*.kt,*.kts}]
ij_continuation_indent_size = 4
ij_java_names_count_to_use_import_on_demand = 9999
ij_kotlin_align_in_columns_case_branch = false
ij_kotlin_align_multiline_binary_operation = false
ij_kotlin_align_multiline_extends_list = false
ij_kotlin_align_multiline_method_parentheses = false
ij_kotlin_align_multiline_parameters = true
ij_kotlin_align_multiline_parameters_in_calls = false
ij_kotlin_assignment_wrap = normal
ij_kotlin_blank_lines_after_class_header = 0
ij_kotlin_blank_lines_around_block_when_branches = 0
ij_kotlin_blank_lines_before_declaration_with_comment_or_annotation_on_separate_line = 1
ij_kotlin_block_comment_at_first_column = true
ij_kotlin_call_parameters_new_line_after_left_paren = true
ij_kotlin_call_parameters_right_paren_on_new_line = false
ij_kotlin_catch_on_new_line = false
ij_kotlin_continuation_indent_for_chained_calls = true
ij_kotlin_continuation_indent_for_expression_bodies = true
ij_kotlin_continuation_indent_in_argument_lists = true
ij_kotlin_continuation_indent_in_elvis = false
ij_kotlin_continuation_indent_in_if_conditions = false
ij_kotlin_continuation_indent_in_parameter_lists = false
ij_kotlin_continuation_indent_in_supertype_lists = false
ij_kotlin_else_on_new_line = false
ij_kotlin_enum_constants_wrap = off
ij_kotlin_extends_list_wrap = normal
ij_kotlin_field_annotation_wrap = split_into_lines
ij_kotlin_finally_on_new_line = false
ij_kotlin_if_rparen_on_new_line = false
ij_kotlin_import_nested_classes = false
ij_kotlin_insert_whitespaces_in_simple_one_line_method = true
ij_kotlin_keep_blank_lines_before_right_brace = 2
ij_kotlin_keep_blank_lines_in_code = 2
ij_kotlin_keep_blank_lines_in_declarations = 2
ij_kotlin_keep_first_column_comment = true
ij_kotlin_keep_indents_on_empty_lines = false
ij_kotlin_keep_line_breaks = true
ij_kotlin_lbrace_on_next_line = false
ij_kotlin_line_comment_add_space = false
ij_kotlin_line_comment_at_first_column = true
ij_kotlin_method_annotation_wrap = split_into_lines
ij_kotlin_method_call_chain_wrap = normal
ij_kotlin_method_parameters_new_line_after_left_paren = true
ij_kotlin_method_parameters_right_paren_on_new_line = true
ij_kotlin_name_count_to_use_star_import = 9999
ij_kotlin_name_count_to_use_star_import_for_members = 9999
ij_kotlin_parameter_annotation_wrap = off
ij_kotlin_space_after_comma = true
ij_kotlin_space_after_extend_colon = true
ij_kotlin_space_after_type_colon = true
ij_kotlin_space_before_catch_parentheses = true
ij_kotlin_space_before_comma = false
ij_kotlin_space_before_extend_colon = true
ij_kotlin_space_before_for_parentheses = true
ij_kotlin_space_before_if_parentheses = true
ij_kotlin_space_before_lambda_arrow = true
ij_kotlin_space_before_type_colon = false
ij_kotlin_space_before_when_parentheses = true
ij_kotlin_space_before_while_parentheses = true
ij_kotlin_spaces_around_additive_operators = true
ij_kotlin_spaces_around_assignment_operators = true
ij_kotlin_spaces_around_equality_operators = true
ij_kotlin_spaces_around_function_type_arrow = true
ij_kotlin_spaces_around_logical_operators = true
ij_kotlin_spaces_around_multiplicative_operators = true
ij_kotlin_spaces_around_range = false
ij_kotlin_spaces_around_relational_operators = true
ij_kotlin_spaces_around_unary_operator = false
ij_kotlin_spaces_around_when_arrow = true
ij_kotlin_variable_annotation_wrap = off
ij_kotlin_while_on_new_line = false
ij_kotlin_wrap_elvis_expressions = 1
ij_kotlin_wrap_expression_body_functions = 1
ij_kotlin_wrap_first_method_in_call_chain = false
#compose
ktlint_standard_filename = disabled
ktlint_standard_no-wildcard-imports = disabled
ktlint_standard_function-naming = disabled
ktlint_standard_property-naming = disabled
ktlint_standard_package-naming = disabled
ktlint_function_naming_ignore_when_annotated_with = Composable
ktlint_code_style = android_studio
ktlint_standard_import-ordering = disabled
ktlint_standard_package-naming = disabled
ij_kotlin_allow_trailing_comma = true
ij_kotlin_allow_trailing_comma_on_call_site = true
+41 -38
View File
@@ -1,4 +1,5 @@
name: build name: Build
on: on:
workflow_dispatch: workflow_dispatch:
inputs: inputs:
@@ -12,6 +13,14 @@ on:
- prerelease - prerelease
- nightly - nightly
- release - release
flavor:
type: choice
description: "Product flavor"
required: true
default: fdroid
options:
- fdroid
- standalone
secrets: secrets:
SIGNING_KEY_ALIAS: SIGNING_KEY_ALIAS:
required: false required: false
@@ -30,6 +39,11 @@ on:
description: "Build type" description: "Build type"
required: true required: true
default: debug default: debug
flavor:
type: string
description: "Product flavor"
required: false
default: fdroid
secrets: secrets:
SIGNING_KEY_ALIAS: SIGNING_KEY_ALIAS:
required: false required: false
@@ -41,6 +55,7 @@ on:
required: false required: false
KEYSTORE: KEYSTORE:
required: false required: false
env: env:
UPLOAD_DIR_ANDROID: android_artifacts UPLOAD_DIR_ANDROID: android_artifacts
@@ -57,6 +72,8 @@ jobs:
UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }} UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }}
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Set up JDK 17 - name: Set up JDK 17
uses: actions/setup-java@v4 uses: actions/setup-java@v4
with: with:
@@ -65,9 +82,6 @@ jobs:
cache: gradle cache: gradle
- name: Grant execute permission for gradlew - name: Grant execute permission for gradlew
run: chmod +x gradlew run: chmod +x gradlew
# Here we need to decode keystore.jks from base64 string and place it
# in the folder specified in the release signing configuration
- name: Decode Keystore - name: Decode Keystore
id: decode_keystore id: decode_keystore
uses: timheuer/base64-to-file@v1.2 uses: timheuer/base64-to-file@v1.2
@@ -75,51 +89,40 @@ jobs:
fileName: ${{ env.KEY_STORE_FILE }} fileName: ${{ env.KEY_STORE_FILE }}
fileDir: ${{ env.KEY_STORE_LOCATION }} fileDir: ${{ env.KEY_STORE_LOCATION }}
encodedString: ${{ secrets.KEYSTORE }} encodedString: ${{ secrets.KEYSTORE }}
# create keystore path for gradle to read
- name: Create keystore path env var - name: Create keystore path env var
if: ${{ inputs.build_type != 'debug' }} if: ${{ inputs.build_type != 'debug' }}
run: | run: |
store_path=${{ env.KEY_STORE_LOCATION }}${{ env.KEY_STORE_FILE }} store_path=${{ env.KEY_STORE_LOCATION }}${{ env.KEY_STORE_FILE }}
echo "KEY_STORE_PATH=$store_path" >> $GITHUB_ENV echo "KEY_STORE_PATH=$store_path" >> $GITHUB_ENV
- name: Create service_account.json - name: Create service_account.json
if: ${{ inputs.build_type != 'debug' }} if: ${{ inputs.build_type != 'debug' }}
id: createServiceAccount id: createServiceAccount
run: echo '${{ secrets.ANDROID_SERVICE_ACCOUNT_JSON }}' > service_account.json run: echo '${{ secrets.SERVICE_ACCOUNT_JSON }}' > service_account.json
- name: Build APK
- name: Build Fdroid Release APK
if: ${{ inputs.build_type == 'release' }}
run: ./gradlew :app:assembleFdroidRelease --info
- name: Build Fdroid Prerelease APK
if: ${{ inputs.build_type == 'prerelease' }}
run: ./gradlew :app:assembleFdroidPrerelease --info
- name: Build Fdroid Nightly APK
if: ${{ inputs.build_type == 'nightly' }}
run: ./gradlew :app:assembleFdroidNightly --info
- name: Build Debug APK
if: ${{ inputs.build_type == 'debug' }}
run: ./gradlew :app:assembleFdroidDebug --stacktrace
# bump versionCode for nightly and prerelease builds
- name: Commit and push versionCode changes
if: ${{ inputs.build_type == 'nightly' || inputs.build_type == 'prerelease' }}
run: | run: |
git config --global user.name 'GitHub Actions' flavor=${{ inputs.flavor }}
git config --global user.email 'actions@github.com' build_type=${{ inputs.build_type }}
git add versionCode.txt case $build_type in
git commit -m "Automated build update" "release")
./gradlew :app:assemble${flavor^}Release --info
;;
"prerelease")
./gradlew :app:assemble${flavor^}Prerelease --info
;;
"nightly")
./gradlew :app:assemble${flavor^}Nightly --info
;;
"debug")
./gradlew :app:assemble${flavor^}Debug --stacktrace
;;
esac
- name: Get release apk path - name: Get release apk path
id: apk-path id: apk-path
run: echo "path=$(find . -regex '^.*/build/outputs/apk/fdroid/${{ inputs.build_type }}/.*\.apk$' -type f | head -1 | tail -c+2)" >> $GITHUB_OUTPUT run: echo "path=$(find . -regex '^.*/build/outputs/apk/${{ inputs.flavor }}/${{ inputs.build_type }}/.*\.apk$' -type f | head -1 | tail -c+2)" >> $GITHUB_OUTPUT
- name: Upload APK
- name: Upload release apk
uses: actions/upload-artifact@v4 uses: actions/upload-artifact@v4
with: with:
name: ${{ env.UPLOAD_DIR_ANDROID }} name: android_artifacts_${{ inputs.flavor }}
path: ${{github.workspace}}/${{ steps.apk-path.outputs.path }} path: app/build/outputs/apk/${{ inputs.flavor }}/release/wgtunnel-${{ inputs.flavor }}-release-*.apk
retention-days: 1 retention-days: 1
if-no-files-found: warn
-20
View File
@@ -1,20 +0,0 @@
name: on-issue
on:
issues:
types: [ opened, closed, reopened ]
jobs:
on-issue:
name: On new issue
runs-on: ubuntu-latest
steps:
- name: Send Telegram Message
run: |
msg_text='${{ github.actor }} updated an issue:
status: ${{ github.event.issue.state }} - #${{ github.event.issue.number }} ${{ github.event.issue.title }}
https://github.com/zaneschepke/wgtunnel/issues/${{ github.event.issue.number }}'
curl -s -X POST 'https://api.telegram.org/bot${{ secrets.TELEGRAM_TOKEN }}/sendMessage' \
-d "chat_id=${{ secrets.TELEGRAM_TO }}&text=${msg_text}&message_thread_id=${{ secrets.TELEGRAM_TOPIC }}"
-21
View File
@@ -1,21 +0,0 @@
name: on-publish
on:
release:
types: [ published ]
jobs:
on-publish:
name: On publish
runs-on: ubuntu-latest
steps:
- name: Send Telegram Message
run: |
msg_text='${{ github.actor }} published a new release:
Release: ${{ github.event.release.tag_name }}
${{ github.event.release.body }}
https://github.com/zaneschepke/wgtunnel/releases/tag/${{ github.event.release.tag_name }}'
curl -s -X POST 'https://api.telegram.org/bot${{ secrets.TELEGRAM_TOKEN }}/sendMessage' \
-d "chat_id=${{ secrets.TELEGRAM_TO }}&text=${msg_text}&message_thread_id=${{ secrets.TELEGRAM_TOPIC }}"
@@ -19,5 +19,5 @@ jobs:
- name: Grant execute permission for gradlew - name: Grant execute permission for gradlew
run: chmod +x gradlew run: chmod +x gradlew
- name: Run ktlint - name: Run ktfmt
run: ./gradlew ktlintCheck run: ./gradlew ktfmtCheck
+83 -69
View File
@@ -2,12 +2,12 @@ name: publish
on: on:
schedule: schedule:
- cron: "4 3 * * *" - cron: "4 3 * * *"
workflow_dispatch: workflow_dispatch:
inputs: inputs:
track: track:
type: choice type: choice
description: "Google play release track" description: "Google Play release track"
options: options:
- none - none
- internal - internal
@@ -30,75 +30,102 @@ on:
description: "Tag name for release" description: "Tag name for release"
required: false required: false
default: nightly default: nightly
flavor:
type: choice
description: "Product flavor"
required: true
default: standalone
options:
- fdroid
- standalone
workflow_call: workflow_call:
inputs:
flavor:
type: string
description: "Product flavor"
required: false
default: standalone
env: env:
UPLOAD_DIR_ANDROID: android_artifacts UPLOAD_DIR_ANDROID: android_artifacts
permissions:
contents: write
packages: write
jobs: jobs:
check_commits: check_commits:
name: Check for New Commits name: Check for New Commits
runs-on: ubuntu-latest runs-on: ubuntu-latest
outputs: outputs:
has_new_commits: ${{ steps.check.outputs.new_commits }} has_new_commits: ${{ steps.check.outputs.new_commits }}
steps: steps:
- name: Checkout Repository - name: Checkout Repository
uses: actions/checkout@v3 uses: actions/checkout@v4
with: with:
fetch-depth: 0 # This fetches all history so we can check commits fetch-depth: 0
- name: Check for new commits - name: Check for new commits
id: check id: check
env: env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_TOKEN: ${{ secrets.PAT }}
run: | run: |
# This script checks for commits newer than 23 hours ago
NEW_COMMITS=$(git rev-list --count --after="$(date -Iseconds -d '23 hours ago')" ${{ github.sha }}) NEW_COMMITS=$(git rev-list --count --after="$(date -Iseconds -d '23 hours ago')" ${{ github.sha }})
echo "new_commits=$NEW_COMMITS" >> $GITHUB_OUTPUT echo "new_commits=$NEW_COMMITS" >> $GITHUB_OUTPUT
build: build-fdroid:
if: ${{ inputs.release_type != 'none' }} if: ${{ inputs.release_type == 'release' || inputs.flavor == 'fdroid' }}
uses: ./.github/workflows/build.yml uses: ./.github/workflows/build.yml
secrets: inherit secrets: inherit
with: with:
build_type: ${{ inputs.release_type == '' && 'nightly' || inputs.release_type }} build_type: ${{ inputs.release_type == '' && 'nightly' || inputs.release_type }}
flavor: fdroid
build-standalone:
if: ${{ inputs.release_type == 'release' || inputs.release_type == 'nightly' || inputs.release_type == 'prerelease' || inputs.flavor == 'standalone' }}
uses: ./.github/workflows/build.yml
secrets: inherit
with:
build_type: ${{ inputs.release_type == '' && 'nightly' || inputs.release_type }}
flavor: standalone
publish: publish:
needs: needs:
- check_commits - check_commits
- build - build-standalone
if: ${{ needs.check_commits.outputs.has_new_commits > 0 && inputs.release_type != 'none' }} if: ${{ needs.check_commits.outputs.has_new_commits > 0 && inputs.release_type != 'none' }}
name: publish-github name: publish-github
runs-on: ubuntu-latest runs-on: ubuntu-latest
env:
GH_USER: ${{ secrets.GH_USER }}
# GH needed for gh cli
GH_TOKEN: ${{ secrets.GH_TOKEN }}
GH_REPO: ${{ github.repository }}
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
with:
fetch-depth: 0
ref: main
- name: Install system dependencies - name: Install system dependencies
run: | run: |
sudo apt update && sudo apt install -y gh apksigner sudo apt update && sudo apt install -y gh apksigner
- name: Set TAG_NAME
# update latest tag run: |
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
echo "TAG_NAME=${{ github.event.inputs.tag_name }}" >> $GITHUB_ENV
elif [ "${{ github.event_name }}" = "schedule" ]; then
echo "TAG_NAME=nightly" >> $GITHUB_ENV
echo "RELEASE_TYPE=nightly" >> $GITHUB_ENV
fi
- name: Set latest tag - name: Set latest tag
uses: rickstaa/action-create-tag@v1 uses: rickstaa/action-create-tag@v1
id: tag_creation id: tag_creation
with: with:
tag: "latest" # or any tag name you wish to use tag: "latest"
message: "Automated tag for HEAD commit" message: "Automated tag for HEAD commit"
force_push_tag: true force_push_tag: true
github_token: ${{ secrets.GITHUB_TOKEN }}
tag_exists_error: false tag_exists_error: false
- name: Get latest release - name: Get latest release
id: latest_release id: latest_release
uses: kaliber5/action-get-release@v1 uses: kaliber5/action-get-release@v1
with: with:
token: ${{ secrets.GITHUB_TOKEN }} token: ${{ secrets.GITHUB_TOKEN }}
latest: true latest: true
- name: Generate Changelog - name: Generate Changelog
id: changelog id: changelog
uses: requarks/changelog-action@v1 uses: requarks/changelog-action@v1
@@ -106,40 +133,20 @@ jobs:
token: ${{ secrets.GITHUB_TOKEN }} token: ${{ secrets.GITHUB_TOKEN }}
toTag: ${{ github.event_name == 'schedule' && 'nightly' || steps.latest_release.outputs.tag_name }} toTag: ${{ github.event_name == 'schedule' && 'nightly' || steps.latest_release.outputs.tag_name }}
fromTag: "latest" fromTag: "latest"
writeToFile: false # we won't write to file, just output writeToFile: false
- name: Get version code
if: ${{ inputs.release_type == 'release' }}
run: |
version_code=$(grep "VERSION_CODE" buildSrc/src/main/kotlin/Constants.kt | awk '{print $5}' | tr -d '\n')
echo "VERSION_CODE=$version_code" >> $GITHUB_ENV
- name: Push changes
if: ${{ inputs.release_type == '' || inputs.release_type == 'nightly' || inputs.release_type == 'prerelease' }}
uses: ad-m/github-push-action@master
with:
github_token: ${{ secrets.GITHUB_TOKEN }}
branch: ${{ github.ref }}
- name: Make download dir - name: Make download dir
run: mkdir ${{ github.workspace }}/temp run: mkdir ${{ github.workspace }}/temp
- name: Download artifacts - name: Download artifacts
uses: actions/download-artifact@v4 uses: actions/download-artifact@v4
with: with:
name: ${{ env.UPLOAD_DIR_ANDROID }} pattern: android_artifacts_*
path: ${{ github.workspace }}/temp path: ${{ github.workspace }}/temp
merge-multiple: true
# Setup TAG_NAME, which is used as a general "name"
- if: github.event_name == 'workflow_dispatch'
run: echo "TAG_NAME=${{ github.event.inputs.tag_name }}" >> $GITHUB_ENV
- if: github.event_name == 'schedule'
run: echo "TAG_NAME=nightly" >> $GITHUB_ENV
- name: Set version release notes - name: Set version release notes
if: ${{ inputs.release_type == 'release' }} if: ${{ inputs.release_type == 'release' }}
run: | run: |
RELEASE_NOTES="$(cat ${{ github.workspace }}/fastlane/metadata/android/en-US/changelogs/${{ env.VERSION_CODE }}.txt)" VERSION_NAME=$(grep "const val VERSION_NAME" buildSrc/src/main/kotlin/Constants.kt | awk -F'"' '{print $2}')
RELEASE_NOTES="$(cat ${{ github.workspace }}/fastlane/metadata/android/en-US/changelogs/${VERSION_NAME}.txt || echo "No changelog found for ${VERSION_NAME}")"
echo "RELEASE_NOTES<<EOF" >> $GITHUB_ENV echo "RELEASE_NOTES<<EOF" >> $GITHUB_ENV
echo "$RELEASE_NOTES" >> $GITHUB_ENV echo "$RELEASE_NOTES" >> $GITHUB_ENV
echo "EOF" >> $GITHUB_ENV echo "EOF" >> $GITHUB_ENV
@@ -148,32 +155,40 @@ jobs:
if: ${{ contains(env.TAG_NAME, 'nightly') }} if: ${{ contains(env.TAG_NAME, 'nightly') }}
run: | run: |
echo "RELEASE_NOTES=Nightly build for the latest development version of the app." >> $GITHUB_ENV echo "RELEASE_NOTES=Nightly build for the latest development version of the app." >> $GITHUB_ENV
gh release delete nightly --yes || true
git push origin :nightly || true
- name: On prerelease release notes - name: On prerelease release notes
if: ${{ inputs.release_type == 'prerelease' }} if: ${{ inputs.release_type == 'prerelease' }}
run: | run: |
echo "RELEASE_NOTES=Testing version of app for specific feature." >> $GITHUB_ENV echo "RELEASE_NOTES=Testing version of app for specific feature." >> $GITHUB_ENV
gh release delete ${{ github.event.inputs.tag_name }} --yes || true
- name: Get checksum - name: Delete previous release
id: checksum if: ${{ contains(env.TAG_NAME, 'nightly') || inputs.release_type == 'prerelease' }}
run: | uses: ClementTsang/delete-tag-and-release@v0.4.0
file_path=$(find ${{ github.workspace }}/temp -type f -iname "*.apk" | tail -n1) with:
echo "checksum=$(apksigner verify -print-certs $file_path | grep -Po "(?<=SHA-256 digest:) .*" | tr -d "[:blank:]")" >> $GITHUB_OUTPUT tag_name: ${{ env.TAG_NAME }}
delete_release: true
- name: Create Release with Fastlane changelog notes
id: create_release
uses: softprops/action-gh-release@v2
env: env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- name: Get checksums
id: checksum
run: |
checksums=""
for file_path in $(find ${{ github.workspace }}/temp -type f -iname "*.apk"); do
checksum=$(apksigner verify -print-certs $file_path | grep -Po "(?<=SHA-256 digest:) .*" | tr -d "[:blank:]")
checksums="$checksums\n$file_path: $checksum"
done
echo "checksum<<EOF" >> $GITHUB_OUTPUT
echo -e "$checksums" >> $GITHUB_OUTPUT
echo "EOF" >> $GITHUB_OUTPUT
- name: Create Release
id: create_release
uses: softprops/action-gh-release@v2
with: with:
body: | body: |
${{ env.RELEASE_NOTES }} ${{ env.RELEASE_NOTES }}
SHA-256 fingerprint for the 4096-bit signing certificate: SHA-256 fingerprints for the 4096-bit signing certificate:
```sh ```sh
${{ steps.checksum.outputs.checksum }} ${{ steps.checksum.outputs.checksum }}
``` ```
@@ -192,18 +207,20 @@ jobs:
make_latest: ${{ inputs.release_type == 'release' }} make_latest: ${{ inputs.release_type == 'release' }}
files: | files: |
${{ github.workspace }}/temp/* ${{ github.workspace }}/temp/*
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
publish-fdroid: publish-fdroid-public:
runs-on: ubuntu-latest runs-on: ubuntu-latest
needs: needs:
- build - build-fdroid
if: inputs.release_type == 'release' if: inputs.release_type == 'release'
steps: steps:
- name: Dispatch update for fdroid repo - name: Dispatch update for fdroid repo
uses: peter-evans/repository-dispatch@v3 uses: peter-evans/repository-dispatch@v3
with: with:
token: ${{ secrets.PAT }} token: ${{ secrets.GITHUB_TOKEN }}
repository: zaneschepke/fdroid repository: wgtunnel/fdroid
event-type: fdroid-update event-type: fdroid-update
publish-play: publish-play:
@@ -217,8 +234,6 @@ jobs:
SIGNING_STORE_PASSWORD: ${{ secrets.SIGNING_STORE_PASSWORD }} SIGNING_STORE_PASSWORD: ${{ secrets.SIGNING_STORE_PASSWORD }}
KEY_STORE_FILE: 'android_keystore.jks' KEY_STORE_FILE: 'android_keystore.jks'
KEY_STORE_LOCATION: ${{ github.workspace }}/app/keystore/ KEY_STORE_LOCATION: ${{ github.workspace }}/app/keystore/
GH_USER: ${{ secrets.GH_USER }}
GH_TOKEN: ${{ secrets.GH_TOKEN }}
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
@@ -259,5 +274,4 @@ jobs:
bundler-cache: true bundler-cache: true
- name: Distribute app to Prod track 🚀 - name: Distribute app to Prod track 🚀
run: (cd ${{ github.workspace }} && bundle install && bundle exec fastlane ${{ inputs.track }}) run: (cd ${{ github.workspace }} && bundle install && bundle exec fastlane ${{ inputs.track }})
+1 -1
View File
@@ -70,5 +70,5 @@ lint/tmp/
app/release/output.json app/release/output.json
.idea/codeStyles/ .idea/codeStyles/
# where we keep our signing secrets locally # where we keep our signing secrets locally
app/signing.properties
/.kotlin/ /.kotlin/
/app/keystore/
+1 -1
View File
@@ -1,6 +1,6 @@
MIT License MIT License
Copyright (c) 2023 WG Auto Tunnel Copyright © 2023-2025 Zane Schepke
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal
+7 -7
View File
@@ -4,7 +4,7 @@ WG Tunnel
<div align="center"> <div align="center">
An alternative Android client app for [WireGuard®](https://www.wireguard.com/) An alternative Android client app for [WireGuard](https://www.wireguard.com/)
and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/) and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/)
<br /> <br />
<br /> <br />
@@ -23,14 +23,14 @@ and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/)
[![Google Play](https://img.shields.io/badge/Google_Play-414141?style=for-the-badge&logo=google-play&logoColor=white)](https://play.google.com/store/apps/details?id=com.zaneschepke.wireguardautotunnel) [![Google Play](https://img.shields.io/badge/Google_Play-414141?style=for-the-badge&logo=google-play&logoColor=white)](https://play.google.com/store/apps/details?id=com.zaneschepke.wireguardautotunnel)
[![F-Droid](https://img.shields.io/static/v1?style=for-the-badge&message=F-Droid&color=1976D2&logo=F-Droid&logoColor=FFFFFF&label=)](https://f-droid.org/packages/com.zaneschepke.wireguardautotunnel/) [![F-Droid](https://img.shields.io/static/v1?style=for-the-badge&message=F-Droid&color=1976D2&logo=F-Droid&logoColor=FFFFFF&label=)](https://f-droid.org/packages/com.zaneschepke.wireguardautotunnel/)
[![Personal](https://img.shields.io/static/v1?style=for-the-badge&message=Personal&color=1976D2&logo=F-Droid&logoColor=FFFFFF&label=)](https://github.com/zaneschepke/fdroid) [![Personal](https://img.shields.io/static/v1?style=for-the-badge&message=Personal&color=1976D2&logo=F-Droid&logoColor=FFFFFF&label=)](https://github.com/zaneschepke/fdroid)
[![Obtainium](https://img.shields.io/badge/Obtainium-414141?style=for-the-badge&logo=Obtainium&logoColor=white)](https://apps.obtainium.imranr.dev/redirect?r=obtainium://app/%7B%22id%22%3A%22com.zaneschepke.wireguardautotunnel%22%2C%22url%22%3A%22https%3A%2F%2Fgithub.com%2Fzaneschepke%2Fwgtunnel%22%2C%22author%22%3A%22zaneschepke%22%2C%22name%22%3A%22WG%20Tunnel%22%2C%22preferredApkIndex%22%3A0%2C%22additionalSettings%22%3A%22%7B%5C%22includePrereleases%5C%22%3Afalse%2C%5C%22fallbackToOlderReleases%5C%22%3Atrue%2C%5C%22filterReleaseTitlesByRegEx%5C%22%3A%5C%22%5C%22%2C%5C%22filterReleaseNotesByRegEx%5C%22%3A%5C%22%5C%22%2C%5C%22verifyLatestTag%5C%22%3Atrue%2C%5C%22sortMethodChoice%5C%22%3A%5C%22date%5C%22%2C%5C%22useLatestAssetDateAsReleaseDate%5C%22%3Afalse%2C%5C%22releaseTitleAsVersion%5C%22%3Afalse%2C%5C%22trackOnly%5C%22%3Afalse%2C%5C%22versionExtractionRegEx%5C%22%3A%5C%22%5C%22%2C%5C%22matchGroupToUse%5C%22%3A%5C%22%5C%22%2C%5C%22versionDetection%5C%22%3Atrue%2C%5C%22releaseDateAsVersion%5C%22%3Afalse%2C%5C%22useVersionCodeAsOSVersion%5C%22%3Afalse%2C%5C%22apkFilterRegEx%5C%22%3A%5C%22%5C%22%2C%5C%22invertAPKFilter%5C%22%3Afalse%2C%5C%22autoApkFilterByArch%5C%22%3Atrue%2C%5C%22appName%5C%22%3A%5C%22WG%20Tunnel%5C%22%2C%5C%22appAuthor%5C%22%3A%5C%22Zane%20Schepke%5C%22%2C%5C%22shizukuPretendToBeGooglePlay%5C%22%3Afalse%2C%5C%22allowInsecure%5C%22%3Afalse%2C%5C%22exemptFromBackgroundUpdates%5C%22%3Afalse%2C%5C%22skipUpdateNotifications%5C%22%3Afalse%2C%5C%22about%5C%22%3A%5C%22%5C%22%2C%5C%22refreshBeforeDownload%5C%22%3Afalse%7D%22%2C%22overrideSource%22%3Anull%7D)
</div> </div>
<div align="center"> <div align="center">
[![Discord](https://img.shields.io/badge/Discord-%235865F2.svg?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/rbRRNh6H7V) [<img src="https://img.shields.io/badge/Telegram-26A5E4.svg?style=for-the-badge&logo=Telegram&logoColor=white">](https://t.me/wgtunnel)
[![Telegram](https://img.shields.io/badge/Telegram-2CA5E0?style=for-the-badge&logo=telegram&logoColor=white)](https://t.me/wgtunnel) [<img src="https://img.shields.io/badge/Matrix-000000.svg?style=for-the-badge&logo=Matrix&logoColor=white">](https://matrix.to/#/#wg-tunnel-space:matrix.org)
</div> </div>
<details open="open"> <details open="open">
@@ -49,7 +49,7 @@ and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/)
<div style="text-align: left;"> <div style="text-align: left;">
## About ## About
Inspired by the official [wireguard-android](https://github.com/WireGuard/wireguard-android) app, WG Tunnel was created to address features and support missing from the official app. This app combines support for both [WireGuard®](https://www.wireguard.com/) Inspired by the official [wireguard-android](https://github.com/WireGuard/wireguard-android) app, WG Tunnel was created to address features and support missing from the official app. This app combines support for both [WireGuard](https://www.wireguard.com/)
and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/), with its primary feature of auto-tunneling (on-demand tunneling). and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/), with its primary feature of auto-tunneling (on-demand tunneling).
</div> </div>
@@ -61,14 +61,14 @@ and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/), with its pr
Thank you to the following: Thank you to the following:
- All of the users that have helped contribute to the project with ideas, translations, feedback, bug reports, testing, and donations. - All of the users that have helped contribute to the project with ideas, translations, feedback, bug reports, testing, and donations.
- [WireGuard®](https://www.wireguard.com/) - © Jason A. Donenfeld (https://github.com/WireGuard/wireguard-android) - [WireGuard](https://www.wireguard.com/) - Jason A. Donenfeld (https://github.com/WireGuard/wireguard-android)
- [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/) - Amnezia Team (https://github.com/amnezia-vpn/amneziawg-android) - [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/) - Amnezia Team (https://github.com/amnezia-vpn/amneziawg-android)
## Screenshots ## Screenshots
</div> </div>
<div style="display: flex; flex-wrap: wrap; justify-content: center; gap: 10px;"> <div style="display: flex; flex-wrap: wrap; justify-content: left; gap: 10px;">
<img label="Main" src="fastlane/metadata/android/en-US/images/phoneScreenshots/main_screen.png" width="200" /> <img label="Main" src="fastlane/metadata/android/en-US/images/phoneScreenshots/main_screen.png" width="200" />
<img label="Settings" src="fastlane/metadata/android/en-US/images/phoneScreenshots/settings_screen.png" width="200" /> <img label="Settings" src="fastlane/metadata/android/en-US/images/phoneScreenshots/settings_screen.png" width="200" />
<img label="Auto" src="fastlane/metadata/android/en-US/images/phoneScreenshots/auto_screen.png" width="200" /> <img label="Auto" src="fastlane/metadata/android/en-US/images/phoneScreenshots/auto_screen.png" width="200" />
+2 -1
View File
@@ -1,2 +1,3 @@
/build /build
/release /release
/src/main/assets/licenses.json
+192 -214
View File
@@ -1,255 +1,233 @@
plugins { plugins {
alias(libs.plugins.android.application) alias(libs.plugins.android.application)
alias(libs.plugins.kotlin.android) alias(libs.plugins.kotlin.android)
alias(libs.plugins.hilt.android) alias(libs.plugins.hilt.android)
alias(libs.plugins.kotlinxSerialization) alias(libs.plugins.kotlinxSerialization)
alias(libs.plugins.ksp) alias(libs.plugins.ksp)
alias(libs.plugins.compose.compiler) alias(libs.plugins.compose.compiler)
alias(libs.plugins.grgit) alias(libs.plugins.grgit)
} alias(libs.plugins.licensee)
val versionFile = file("$rootDir/versionCode.txt")
val versionCodeIncrement = with(getBuildTaskName().lowercase()) {
when {
this.contains(Constants.NIGHTLY) || this.contains(Constants.PRERELEASE) -> {
if (versionFile.exists()) {
versionFile.readText().trim().toInt() + 1
} else {
1
}
}
else -> 0
}
} }
android { android {
namespace = Constants.APP_ID namespace = Constants.APP_ID
compileSdk = Constants.TARGET_SDK compileSdk = Constants.TARGET_SDK
androidResources { androidResources { generateLocaleConfig = true }
generateLocaleConfig = true
}
// reproducibility dependenciesInfo {
dependenciesInfo { includeInApk = false
// Disables dependency metadata when building APKs. includeInBundle = false
includeInApk = false }
// Disables dependency metadata when building Android App Bundles.
includeInBundle = false
}
defaultConfig { defaultConfig {
applicationId = Constants.APP_ID applicationId = Constants.APP_ID
minSdk = Constants.MIN_SDK minSdk = Constants.MIN_SDK
targetSdk = Constants.TARGET_SDK targetSdk = Constants.TARGET_SDK
versionCode = Constants.VERSION_CODE + versionCodeIncrement versionCode = computeVersionCode()
versionName = determineVersionName() versionName = computeVersionName()
ksp { arg("room.schemaLocation", "$projectDir/schemas") } ksp { arg("room.schemaLocation", "$projectDir/schemas") }
sourceSets { sourceSets { getByName("debug").assets.srcDirs(files("$projectDir/schemas")) }
getByName("debug").assets.srcDirs(files("$projectDir/schemas")) // Room
}
buildConfigField("String[]", "LANGUAGES", "new String[]{ ${languageList().joinToString(separator = ", ") { "\"$it\"" }} }") buildConfigField(
"String[]",
"LANGUAGES",
"new String[]{ ${languageList().joinToString(separator = ", ") { "\"$it\"" }} }",
)
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
vectorDrawables { useSupportLibrary = true } vectorDrawables { useSupportLibrary = true }
} }
signingConfigs { signingConfigs {
create(Constants.RELEASE) { create(Constants.RELEASE) {
storeFile = getStoreFile() storeFile = file(System.getenv("KEY_STORE_PATH") ?: "keystore/android_keystore.jks")
storePassword = getSigningProperty(Constants.STORE_PASS_VAR) storePassword =
keyAlias = getSigningProperty(Constants.KEY_ALIAS_VAR) LocalProperties.get("SIGNING_STORE_PASSWORD")
keyPassword = getSigningProperty(Constants.KEY_PASS_VAR) ?: System.getenv("SIGNING_STORE_PASSWORD")
} keyAlias =
} LocalProperties.get("SIGNING_KEY_ALIAS") ?: System.getenv("SIGNING_KEY_ALIAS")
keyPassword =
LocalProperties.get("SIGNING_KEY_PASSWORD") ?: System.getenv("SIGNING_KEY_PASSWORD")
}
}
buildTypes { buildTypes {
// don't strip packaging.jniLibs.keepDebugSymbols.addAll(
packaging.jniLibs.keepDebugSymbols.addAll( listOf("libwg-go.so", "libwg-quick.so", "libwg.so")
listOf("libwg-go.so", "libwg-quick.so", "libwg.so"), )
)
release { release {
isDebuggable = false isDebuggable = false
isMinifyEnabled = true isMinifyEnabled = true
isShrinkResources = true isShrinkResources = true
proguardFiles( proguardFiles(
getDefaultProguardFile("proguard-android-optimize.txt"), getDefaultProguardFile("proguard-android-optimize.txt"),
"proguard-rules.pro", "proguard-rules.pro",
) )
signingConfig = signingConfigs.getByName(Constants.RELEASE) signingConfig = signingConfigs.getByName(Constants.RELEASE)
resValue("string", "provider", "\"${Constants.APP_NAME}.provider\"") resValue("string", "provider", "\"${Constants.APP_NAME}.provider\"")
} }
debug {
applicationIdSuffix = ".debug"
versionNameSuffix = "-debug"
resValue("string", "app_name", "WG Tunnel - Debug")
isDebuggable = true
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.debug\"")
}
create(Constants.PRERELEASE) { debug {
initWith(buildTypes.getByName(Constants.RELEASE)) applicationIdSuffix = ".debug"
applicationIdSuffix = ".prerelease" resValue("string", "app_name", "WG Tunnel - Debug")
versionNameSuffix = "-pre" isDebuggable = true
resValue("string", "app_name", "WG Tunnel - Pre") resValue("string", "provider", "\"${Constants.APP_NAME}.provider.debug\"")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.pre\"") }
}
create(Constants.NIGHTLY) { create(Constants.PRERELEASE) {
initWith(buildTypes.getByName(Constants.RELEASE)) initWith(buildTypes.getByName(Constants.RELEASE))
applicationIdSuffix = ".nightly" applicationIdSuffix = ".prerelease"
versionNameSuffix = "-nightly" resValue("string", "app_name", "WG Tunnel - Pre")
resValue("string", "app_name", "WG Tunnel - Nightly") resValue("string", "provider", "\"${Constants.APP_NAME}.provider.pre\"")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.nightly\"") }
}
applicationVariants.all { create(Constants.NIGHTLY) {
val variant = this initWith(buildTypes.getByName(Constants.RELEASE))
variant.outputs applicationIdSuffix = ".nightly"
.map { it as com.android.build.gradle.internal.api.BaseVariantOutputImpl } resValue("string", "app_name", "WG Tunnel - Nightly")
.forEach { output -> resValue("string", "provider", "\"${Constants.APP_NAME}.provider.nightly\"")
val outputFileName = }
"${Constants.APP_NAME}-${variant.flavorName}-" + }
"${variant.buildType.name}-${variant.versionName}.apk"
output.outputFileName = outputFileName flavorDimensions.add("type")
} productFlavors {
} create("fdroid") {
} dimension = "type"
flavorDimensions.add(Constants.TYPE) buildConfigField("String", "FLAVOR", "\"fdroid\"")
productFlavors { }
create("fdroid") { create("google") {
dimension = Constants.TYPE dimension = "type"
proguardFile("fdroid-rules.pro") buildConfigField("String", "FLAVOR", "\"google\"")
} }
create("general") { create("standalone") {
dimension = Constants.TYPE dimension = "type"
} buildConfigField("String", "FLAVOR", "\"standalone\"")
} }
compileOptions { }
sourceCompatibility = JavaVersion.VERSION_17
targetCompatibility = JavaVersion.VERSION_17 compileOptions {
isCoreLibraryDesugaringEnabled = true sourceCompatibility = JavaVersion.VERSION_17
} targetCompatibility = JavaVersion.VERSION_17
kotlinOptions { jvmTarget = Constants.JVM_TARGET } isCoreLibraryDesugaringEnabled = true
buildFeatures { }
compose = true kotlinOptions { jvmTarget = Constants.JVM_TARGET }
buildConfig = true buildFeatures {
} compose = true
packaging { resources { excludes += "/META-INF/{AL2.0,LGPL2.1}" } } buildConfig = true
}
packaging { resources { excludes += "/META-INF/{AL2.0,LGPL2.1}" } }
licensee {
Constants.allowedLicenses.forEach { allow(it) }
allowUrl(Constants.XZING_LICENSE_URL)
allowUrl("https://rafaellins.mit-license.org/2021/")
}
applicationVariants.all {
val variant = this
variant.outputs
.map { it as com.android.build.gradle.internal.api.BaseVariantOutputImpl }
.forEach { output ->
val outputFileName =
if (variant.flavorName == "fdroid" && variant.buildType.name == "release") {
"${Constants.APP_NAME}-fdroid-release-${variant.versionName}.apk"
} else {
"${Constants.APP_NAME}-${variant.flavorName}-v${variant.versionName}.apk"
}
output.outputFileName = outputFileName
}
}
} }
dependencies { dependencies {
implementation(project(":logcatter"))
implementation(project(":networkmonitor"))
implementation(project(":logcatter")) implementation(libs.androidx.core.ktx)
implementation(libs.androidx.lifecycle.runtime.ktx)
implementation(libs.androidx.lifecycle.service)
implementation(libs.androidx.activity.compose)
implementation(platform(libs.androidx.compose.bom))
implementation(libs.androidx.compose.ui)
implementation(libs.androidx.compose.ui.graphics)
implementation(libs.androidx.compose.ui.tooling.preview)
implementation(libs.androidx.material3)
implementation(libs.androidx.appcompat)
implementation(libs.material)
implementation(libs.androidx.storage)
implementation(libs.androidx.core.ktx) testImplementation(libs.junit)
implementation(libs.androidx.lifecycle.runtime.ktx) testImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.espresso.core)
androidTestImplementation(platform(libs.androidx.compose.bom))
androidTestImplementation(libs.androidx.compose.ui.test)
androidTestImplementation(libs.androidx.room.testing)
debugImplementation(libs.androidx.compose.ui.tooling)
debugImplementation(libs.androidx.compose.manifest)
// helpers for implementing LifecycleOwner in a Service implementation(libs.tunnel)
implementation(libs.androidx.lifecycle.service) implementation(libs.amneziawg.android)
implementation(libs.androidx.activity.compose) coreLibraryDesugaring(libs.desugar.jdk.libs)
implementation(platform(libs.androidx.compose.bom))
implementation(libs.androidx.compose.ui)
implementation(libs.androidx.compose.ui.graphics)
implementation(libs.androidx.compose.ui.tooling.preview)
implementation(libs.androidx.material3)
implementation(libs.androidx.appcompat)
implementation(libs.material)
// test implementation(libs.timber)
testImplementation(libs.junit)
testImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.junit)
androidTestImplementation(libs.androidx.espresso.core)
androidTestImplementation(platform(libs.androidx.compose.bom))
androidTestImplementation(libs.androidx.compose.ui.test)
androidTestImplementation(libs.androidx.room.testing)
debugImplementation(libs.androidx.compose.ui.tooling)
debugImplementation(libs.androidx.compose.manifest)
// tunnel implementation(libs.androidx.navigation.compose)
implementation(libs.tunnel) implementation(libs.androidx.hilt.navigation.compose)
implementation(libs.amneziawg.android)
coreLibraryDesugaring(libs.desugar.jdk.libs)
// logging implementation(libs.hilt.android)
implementation(libs.timber) ksp(libs.hilt.android.compiler)
ksp(libs.androidx.hilt.compiler)
// compose navigation implementation(libs.accompanist.permissions)
implementation(libs.androidx.navigation.compose) implementation(libs.accompanist.drawablepainter)
implementation(libs.androidx.hilt.navigation.compose)
// hilt implementation(libs.androidx.room.runtime)
implementation(libs.hilt.android) ksp(libs.androidx.room.compiler)
ksp(libs.hilt.android.compiler) implementation(libs.androidx.room.ktx)
ksp(libs.androidx.hilt.compiler) implementation(libs.androidx.datastore.preferences)
// accompanist implementation(libs.lifecycle.runtime.compose)
implementation(libs.accompanist.permissions) implementation(libs.androidx.lifecycle.runtime.ktx)
implementation(libs.accompanist.drawablepainter) implementation(libs.androidx.lifecycle.process)
// storage implementation(libs.kotlinx.serialization.json)
implementation(libs.androidx.room.runtime)
ksp(libs.androidx.room.compiler)
implementation(libs.androidx.room.ktx)
implementation(libs.androidx.datastore.preferences)
// lifecycle implementation(libs.zxing.android.embedded)
implementation(libs.lifecycle.runtime.compose) implementation(libs.material.icons.extended)
implementation(libs.androidx.lifecycle.runtime.ktx)
implementation(libs.androidx.lifecycle.process)
// icons implementation(libs.androidx.biometric.ktx)
implementation(libs.material.icons.extended) implementation(libs.pin.lock.compose)
// serialization
implementation(libs.kotlinx.serialization.json)
// barcode scanning implementation(libs.androidx.core)
implementation(libs.zxing.android.embedded)
// bio implementation(libs.androidx.core.splashscreen)
implementation(libs.androidx.biometric.ktx)
implementation(libs.pin.lock.compose)
// shortcuts implementation(libs.androidx.work.runtime)
implementation(libs.androidx.core) implementation(libs.androidx.hilt.work)
// splash implementation(libs.qrcode.kotlin)
implementation(libs.androidx.core.splashscreen) implementation(libs.semver4j)
// worker implementation(libs.ktor.client.core)
implementation(libs.androidx.work.runtime) implementation(libs.ktor.client.okhttp)
implementation(libs.androidx.hilt.work) implementation(libs.ktor.client.cio)
implementation(libs.ktor.client.content.negotiation)
implementation(libs.ktor.serialization.kotlinx.json)
implementation(libs.slf4j.android)
} }
fun determineVersionName(): String { tasks.register<Copy>("copyLicenseeJsonToAssets") {
return with(getBuildTaskName().lowercase()) { dependsOn("licensee")
when { val outputAssets = layout.projectDirectory.dir("src/main/assets")
contains(Constants.NIGHTLY) || contains(Constants.PRERELEASE) -> from(layout.buildDirectory.file("reports/licensee/androidFdroidRelease/artifacts.json")) {
Constants.VERSION_NAME + rename("artifacts.json", "licenses.json")
"-${grgitService.service.get().grgit.head().abbreviatedId}" }
else -> Constants.VERSION_NAME into(outputAssets)
}
}
} }
val incrementVersionCode by tasks.registering { tasks.named("preBuild") { dependsOn("copyLicenseeJsonToAssets") }
doLast {
val versionFile = file("$rootDir/versionCode.txt")
if (versionFile.exists()) {
versionFile.writeText(versionCodeIncrement.toString())
println("Incremented versionCode to $versionCodeIncrement")
}
}
}
tasks.whenTaskAdded {
if (name.startsWith("assemble") && !name.lowercase().contains("debug")) {
dependsOn(incrementVersionCode)
}
}
-5
View File
@@ -1,5 +0,0 @@
-dontwarn com.google.errorprone.annotations.**
-keepclassmembers class * extends androidx.datastore.preferences.protobuf.GeneratedMessageLite {
<fields>;
}
-24
View File
@@ -1,24 +0,0 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable
# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile
-keepclassmembers class * extends androidx.datastore.preferences.protobuf.GeneratedMessageLite {
<fields>;
}
@@ -13,10 +13,10 @@ import org.junit.runner.RunWith
*/ */
@RunWith(AndroidJUnit4::class) @RunWith(AndroidJUnit4::class)
class ExampleInstrumentedTest { class ExampleInstrumentedTest {
@Test @Test
fun useAppContext() { fun useAppContext() {
// Context of the app under test. // Context of the app under test.
val appContext = InstrumentationRegistry.getInstrumentation().targetContext val appContext = InstrumentationRegistry.getInstrumentation().targetContext
assertEquals("com.zaneschepke.wireguardautotunnel", appContext.packageName) assertEquals("com.zaneschepke.wireguardautotunnel", appContext.packageName)
} }
} }
@@ -5,40 +5,35 @@ import androidx.test.ext.junit.runners.AndroidJUnit4
import androidx.test.platform.app.InstrumentationRegistry import androidx.test.platform.app.InstrumentationRegistry
import com.zaneschepke.wireguardautotunnel.data.AppDatabase import com.zaneschepke.wireguardautotunnel.data.AppDatabase
import com.zaneschepke.wireguardautotunnel.data.Queries import com.zaneschepke.wireguardautotunnel.data.Queries
import java.io.IOException
import org.junit.Rule import org.junit.Rule
import org.junit.Test import org.junit.Test
import org.junit.runner.RunWith import org.junit.runner.RunWith
import java.io.IOException
@RunWith(AndroidJUnit4::class) @RunWith(AndroidJUnit4::class)
class MigrationTest { class MigrationTest {
private val dbName = "migration-test" private val dbName = "migration-test"
@get:Rule @get:Rule
val helper: MigrationTestHelper = val helper: MigrationTestHelper =
MigrationTestHelper( MigrationTestHelper(InstrumentationRegistry.getInstrumentation(), AppDatabase::class.java)
InstrumentationRegistry.getInstrumentation(),
AppDatabase::class.java,
)
@Test @Test
@Throws(IOException::class) @Throws(IOException::class)
fun migrate6To7() { fun migrate6To7() {
helper.createDatabase(dbName, 6).apply { helper.createDatabase(dbName, 6).apply {
// Database has schema version 1. Insert some data using SQL queries. // Database has schema version 1. Insert some data using SQL queries.
// You can't use DAO classes because they expect the latest schema. // You can't use DAO classes because they expect the latest schema.
execSQL(Queries.createDefaultSettings()) execSQL(Queries.createDefaultSettings())
execSQL( execSQL(Queries.createTunnelConfig())
Queries.createTunnelConfig(), // Prepare for the next version.
) close()
// Prepare for the next version. }
close()
}
// Re-open the database with version 2 and provide // Re-open the database with version 2 and provide
// MIGRATION_1_2 as the migration process. // MIGRATION_1_2 as the migration process.
helper.runMigrationsAndValidate(dbName, 7, true) helper.runMigrationsAndValidate(dbName, 7, true)
// MigrationTestHelper automatically verifies the schema changes, // MigrationTestHelper automatically verifies the schema changes,
// but you need to validate that the data was migrated properly. // but you need to validate that the data was migrated properly.
} }
} }
+5
View File
@@ -0,0 +1,5 @@
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
<!-- Permissions specific to full -->
<!--updater-->
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />
</manifest>
+27 -26
View File
@@ -2,14 +2,9 @@
<manifest xmlns:android="http://schemas.android.com/apk/res/android" <manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"> xmlns:tools="http://schemas.android.com/tools">
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" /> <uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION" />
<!--foreground service exempt android 14--> <!--foreground service exempt android 14-->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_SYSTEM_EXEMPTED" /> <uses-permission android:name="android.permission.FOREGROUND_SERVICE_SYSTEM_EXEMPTED" />
<uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM" <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"
@@ -21,10 +16,11 @@
<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS" /> <uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS" />
<!--start service on boot permission--> <!--start service on boot permission-->
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /> <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<!--android tv support-->
<permission <permission
android:name="${applicationId}.permission.CONTROL_TUNNELS" android:name="${applicationId}.permission.CONTROL_TUNNELS"
android:label="@string/app_permission_title"
android:description="@string/app_permission_description"
android:icon="@mipmap/ic_launcher" android:icon="@mipmap/ic_launcher"
android:protectionLevel="dangerous" /> android:protectionLevel="dangerous" />
@@ -50,12 +46,13 @@
<queries> <queries>
<intent> <intent>
<action android:name="android.intent.action.MAIN" /> <action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent> </intent>
</queries> </queries>
<application <application
android:name=".WireGuardAutoTunnel" android:name=".WireGuardAutoTunnel"
android:allowBackup="false" android:allowBackup="false"
android:banner="@drawable/ic_banner" android:banner="@mipmap/ic_banner"
android:dataExtractionRules="@xml/data_extraction_rules" android:dataExtractionRules="@xml/data_extraction_rules"
android:enableOnBackInvokedCallback="true" android:enableOnBackInvokedCallback="true"
android:fullBackupContent="@xml/backup_rules" android:fullBackupContent="@xml/backup_rules"
@@ -68,7 +65,7 @@
<activity <activity
android:name=".MainActivity" android:name=".MainActivity"
android:exported="true" android:exported="true"
android:windowSoftInputMode="adjustResize" android:windowSoftInputMode="adjustNothing"
android:theme="@style/Theme.WireguardAutoTunnel" android:theme="@style/Theme.WireguardAutoTunnel"
android:configChanges="orientation|screenSize|keyboardHidden" android:configChanges="orientation|screenSize|keyboardHidden"
> >
@@ -89,7 +86,7 @@
<activity <activity
android:name=".core.shortcut.ShortcutsActivity" android:name=".core.shortcut.ShortcutsActivity"
android:enabled="true" android:enabled="true"
android:exported="true" android:exported="false"
android:noHistory="true" android:noHistory="true"
android:excludeFromRecents="true" android:excludeFromRecents="true"
android:finishOnTaskLaunch="true" android:finishOnTaskLaunch="true"
@@ -116,7 +113,7 @@
<service <service
android:name=".core.service.tile.TunnelControlTile" android:name=".core.service.tile.TunnelControlTile"
android:exported="true" android:exported="true"
android:icon="@drawable/ic_launcher" android:icon="@drawable/ic_notification"
android:label="@string/tunnel_control" android:label="@string/tunnel_control"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE"> android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data <meta-data
@@ -133,7 +130,7 @@
<service <service
android:name=".core.service.tile.AutoTunnelControlTile" android:name=".core.service.tile.AutoTunnelControlTile"
android:exported="true" android:exported="true"
android:icon="@drawable/ic_launcher" android:icon="@drawable/ic_notification"
android:label="@string/auto_tunnel" android:label="@string/auto_tunnel"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE"> android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data <meta-data
@@ -166,27 +163,17 @@
<action android:name="android.net.VpnService" /> <action android:name="android.net.VpnService" />
</intent-filter> </intent-filter>
</service> </service>
<receiver <receiver
android:name=".core.broadcast.BootReceiver" android:name=".core.broadcast.RestartReceiver"
android:enabled="true" android:enabled="true"
android:exported="false"> android:exported="false">
<intent-filter> <intent-filter>
<category android:name="android.intent.category.DEFAULT" />
<action android:name="android.intent.action.BOOT_COMPLETED" /> <action android:name="android.intent.action.BOOT_COMPLETED" />
<action android:name="android.intent.action.ACTION_BOOT_COMPLETED" />
<action android:name="android.intent.action.QUICKBOOT_POWERON" /> <action android:name="android.intent.action.QUICKBOOT_POWERON" />
<action android:name="com.htc.intent.action.QUICKBOOT_POWERON" /> <action android:name="com.htc.intent.action.QUICKBOOT_POWERON" />
<action android:name="android.intent.action.MY_PACKAGE_REPLACED" />
</intent-filter> </intent-filter>
</receiver> </receiver>
<receiver
android:name=".core.broadcast.AppUpdateReceiver"
android:exported="false">
<intent-filter>
<action android:name="android.intent.action.MY_PACKAGE_REPLACED" />
</intent-filter>
</receiver>
<receiver <receiver
android:name=".core.broadcast.KernelReceiver" android:name=".core.broadcast.KernelReceiver"
android:exported="false" android:exported="false"
@@ -195,6 +182,20 @@
<action android:name="com.wireguard.android.action.REFRESH_TUNNEL_STATES" /> <action android:name="com.wireguard.android.action.REFRESH_TUNNEL_STATES" />
</intent-filter> </intent-filter>
</receiver> </receiver>
<!--custom security solution for easier user integration-->
<receiver
android:name=".core.broadcast.RemoteControlReceiver"
android:enabled="true"
android:exported="true" tools:ignore="ExportedReceiver">
<intent-filter>
<action android:name="com.zaneschepke.wireguardautotunnel.START_TUNNEL" />
<action android:name="com.zaneschepke.wireguardautotunnel.STOP_TUNNEL" />
<action android:name="com.zaneschepke.wireguardautotunnel.START_AUTO_TUNNEL" />
<action android:name="com.zaneschepke.wireguardautotunnel.STOP_AUTO_TUNNEL" />
<action android:name="com.zaneschepke.wireguardautotunnel.START_KILL_SWITCH" />
<action android:name="com.zaneschepke.wireguardautotunnel.STOP_KILL_SWITCH" />
</intent-filter>
</receiver>
<receiver <receiver
android:name=".core.broadcast.NotificationActionReceiver" android:name=".core.broadcast.NotificationActionReceiver"
android:exported="false" android:exported="false"
Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

After

Width:  |  Height:  |  Size: 21 KiB

@@ -1,267 +1,348 @@
package com.zaneschepke.wireguardautotunnel package com.zaneschepke.wireguardautotunnel
import android.Manifest
import android.annotation.SuppressLint
import android.content.Intent import android.content.Intent
import android.content.pm.PackageManager
import android.graphics.Color import android.graphics.Color
import android.os.Build import android.os.Build
import android.os.Bundle import android.os.Bundle
import android.provider.Settings
import androidx.activity.SystemBarStyle import androidx.activity.SystemBarStyle
import androidx.activity.compose.BackHandler import androidx.activity.compose.rememberLauncherForActivityResult
import androidx.activity.compose.setContent import androidx.activity.compose.setContent
import androidx.activity.enableEdgeToEdge import androidx.activity.enableEdgeToEdge
import androidx.activity.result.ActivityResult
import androidx.activity.result.contract.ActivityResultContracts
import androidx.activity.viewModels import androidx.activity.viewModels
import androidx.appcompat.app.AppCompatActivity import androidx.appcompat.app.AppCompatActivity
import androidx.compose.animation.core.tween import androidx.compose.animation.AnimatedVisibility
import androidx.compose.animation.fadeIn import androidx.compose.animation.slideInVertically
import androidx.compose.animation.fadeOut import androidx.compose.animation.slideOutVertically
import androidx.compose.foundation.layout.Box import androidx.compose.foundation.background
import androidx.compose.foundation.layout.WindowInsets import androidx.compose.foundation.gestures.detectTapGestures
import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.*
import androidx.compose.foundation.layout.padding
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.rounded.Home
import androidx.compose.material.icons.rounded.QuestionMark
import androidx.compose.material.icons.rounded.Settings import androidx.compose.material.icons.rounded.Settings
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.*
import androidx.compose.material3.Scaffold import androidx.compose.runtime.*
import androidx.compose.material3.SnackbarData
import androidx.compose.material3.SnackbarHost
import androidx.compose.material3.surfaceColorAtElevation
import androidx.compose.runtime.CompositionLocalProvider
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource import androidx.compose.ui.input.pointer.pointerInput
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.core.content.ContextCompat
import androidx.core.net.toUri
import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen
import androidx.lifecycle.compose.collectAsStateWithLifecycle import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.compose.NavHost import androidx.navigation.compose.NavHost
import androidx.navigation.compose.composable import androidx.navigation.compose.composable
import androidx.navigation.compose.currentBackStackEntryAsState
import androidx.navigation.compose.rememberNavController import androidx.navigation.compose.rememberNavController
import androidx.navigation.toRoute import androidx.navigation.toRoute
import com.zaneschepke.wireguardautotunnel.core.shortcut.ShortcutManager import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.core.worker.ServiceWorker
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.ui.Route import com.zaneschepke.wireguardautotunnel.ui.Route
import com.zaneschepke.wireguardautotunnel.ui.common.navigation.BottomNavBar import com.zaneschepke.wireguardautotunnel.ui.common.dialog.VpnDeniedDialog
import com.zaneschepke.wireguardautotunnel.ui.common.navigation.BottomNavItem
import com.zaneschepke.wireguardautotunnel.ui.common.navigation.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.CustomSnackBar import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.CustomSnackBar
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarControllerProvider import com.zaneschepke.wireguardautotunnel.ui.navigation.LocalIsAndroidTV
import com.zaneschepke.wireguardautotunnel.ui.screens.main.ConfigScreen import com.zaneschepke.wireguardautotunnel.ui.navigation.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.navigation.components.BottomNavbar
import com.zaneschepke.wireguardautotunnel.ui.navigation.components.DynamicTopAppBar
import com.zaneschepke.wireguardautotunnel.ui.navigation.components.currentNavBackStackEntryAsNavBarState
import com.zaneschepke.wireguardautotunnel.ui.screens.autotunnel.AutoTunnelScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.autotunnel.advanced.AutoTunnelAdvancedScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.autotunnel.disclosure.LocationDisclosureScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.MainScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.MainScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.OptionsScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.autotunnel.TunnelAutoTunnelScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.PinLockScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.config.ConfigScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.ScannerScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.scanner.ScannerScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.SplitTunnelScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.splittunnel.SplitTunnelScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.main.TunnelAutoTunnelScreen import com.zaneschepke.wireguardautotunnel.ui.screens.main.tunneloptions.TunnelOptionsScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.AdvancedScreen import com.zaneschepke.wireguardautotunnel.ui.screens.pin.PinLockScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.AppearanceScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.DisplayScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.KillSwitchScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.LanguageScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.LocationDisclosureScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.SettingsScreen import com.zaneschepke.wireguardautotunnel.ui.screens.settings.SettingsScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.autotunnel.AutoTunnelScreen import com.zaneschepke.wireguardautotunnel.ui.screens.settings.advanced.SettingsAdvancedScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.support.LogsScreen import com.zaneschepke.wireguardautotunnel.ui.screens.settings.appearance.AppearanceScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.appearance.display.DisplayScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.appearance.language.LanguageScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.killswitch.KillSwitchScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.LogsScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.support.SupportScreen import com.zaneschepke.wireguardautotunnel.ui.screens.support.SupportScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.support.license.LicenseScreen
import com.zaneschepke.wireguardautotunnel.ui.theme.WireguardAutoTunnelTheme import com.zaneschepke.wireguardautotunnel.ui.theme.WireguardAutoTunnelTheme
import com.zaneschepke.wireguardautotunnel.util.Constants import com.zaneschepke.wireguardautotunnel.util.extensions.isRunningOnTv
import com.zaneschepke.wireguardautotunnel.util.extensions.requestAutoTunnelTileServiceUpdate
import com.zaneschepke.wireguardautotunnel.viewmodel.AppViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.AppViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.event.AppEvent
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject import javax.inject.Inject
import kotlin.system.exitProcess import kotlin.system.exitProcess
import org.amnezia.awg.backend.GoBackend.VpnService
import timber.log.Timber
@AndroidEntryPoint @AndroidEntryPoint
class MainActivity : AppCompatActivity() { class MainActivity : AppCompatActivity() {
@Inject @Inject lateinit var appStateRepository: AppStateRepository
lateinit var appStateRepository: AppStateRepository
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var tunnelManager: TunnelManager
@Inject @Inject lateinit var networkMonitor: NetworkMonitor
lateinit var shortcutManager: ShortcutManager
override fun onCreate(savedInstanceState: Bundle?) { private var lastLocationPermissionState: Boolean? = null
enableEdgeToEdge(
statusBarStyle = SystemBarStyle.Companion.auto(Color.TRANSPARENT, Color.TRANSPARENT),
navigationBarStyle = SystemBarStyle.Companion.auto(Color.TRANSPARENT, Color.TRANSPARENT),
)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
window.isNavigationBarContrastEnforced = false
}
super.onCreate(savedInstanceState)
val viewModel by viewModels<AppViewModel>() @SuppressLint("BatteryLife")
override fun onCreate(savedInstanceState: Bundle?) {
enableEdgeToEdge(
statusBarStyle = SystemBarStyle.Companion.auto(Color.TRANSPARENT, Color.TRANSPARENT),
navigationBarStyle = SystemBarStyle.Companion.auto(Color.TRANSPARENT, Color.TRANSPARENT),
)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
window.isNavigationBarContrastEnforced = false
}
super.onCreate(savedInstanceState)
installSplashScreen().apply { val viewModel by viewModels<AppViewModel>()
setKeepOnScreenCondition {
!viewModel.isAppReady.value
}
}
setContent { installSplashScreen().apply {
val appUiState by viewModel.uiState.collectAsStateWithLifecycle() setKeepOnScreenCondition { !viewModel.appViewState.value.isAppReady }
val configurationChange by viewModel.configurationChange.collectAsStateWithLifecycle() }
val navController = rememberNavController()
LaunchedEffect(configurationChange) { setContent {
if (configurationChange) { val isTv = isRunningOnTv()
Intent(this@MainActivity, MainActivity::class.java).also { val appUiState by viewModel.uiState.collectAsStateWithLifecycle()
startActivity(it) val appViewState by viewModel.appViewState.collectAsStateWithLifecycle()
exitProcess(0) val tunnelError by viewModel.tunnelManager.errorEvents.collectAsStateWithLifecycle(null)
}
}
}
LaunchedEffect(Unit) { val navController = rememberNavController()
viewModel.getEmitSplitTunnelApps(this@MainActivity) val backStackEntry by navController.currentBackStackEntryAsState()
} val navBarState by
currentNavBackStackEntryAsNavBarState(
navController,
backStackEntry,
viewModel,
appUiState,
appViewState,
)
val snackbar = remember { SnackbarHostState() }
var showVpnPermissionDialog by remember { mutableStateOf(false) }
var vpnPermissionDenied by remember { mutableStateOf(false) }
LaunchedEffect(appUiState.autoTunnelActive) { val vpnActivity =
requestAutoTunnelTileServiceUpdate() rememberLauncherForActivityResult(
} ActivityResultContracts.StartActivityForResult(),
onResult = {
if (it.resultCode != RESULT_OK) {
showVpnPermissionDialog = true
vpnPermissionDenied = true
} else {
vpnPermissionDenied = false
showVpnPermissionDialog = false
}
},
)
with(appUiState.appSettings) { LaunchedEffect(appUiState.tunnels) {
LaunchedEffect(isAutoTunnelEnabled) { if (!appViewState.isAppReady) {
this@MainActivity.requestAutoTunnelTileServiceUpdate() viewModel.handleEvent(AppEvent.AppReadyCheck(appUiState.tunnels))
} }
LaunchedEffect(isShortcutsEnabled) { }
if (!isShortcutsEnabled) return@LaunchedEffect shortcutManager.removeShortcuts()
shortcutManager.addShortcuts()
}
}
ServiceWorker.start(this) val batteryActivity =
rememberLauncherForActivityResult(
ActivityResultContracts.StartActivityForResult()
) { _: ActivityResult ->
viewModel.handleEvent(AppEvent.SetBatteryOptimizeDisableShown)
}
CompositionLocalProvider(LocalNavController provides navController) { LaunchedEffect(tunnelError) {
SnackbarControllerProvider { host -> if (tunnelError == null) return@LaunchedEffect
WireguardAutoTunnelTheme(theme = appUiState.generalState.theme) { val message = tunnelError!!.second.toStringRes()
Scaffold( val context = this@MainActivity
contentWindowInsets = WindowInsets(0), snackbar.showSnackbar(
snackbarHost = { context.getString(R.string.tunnel_error_template, context.getString(message))
SnackbarHost(host) { snackbarData: SnackbarData -> )
CustomSnackBar( }
snackbarData.visuals.message,
isRtl = false, with(appViewState) {
containerColor = LaunchedEffect(isConfigChanged) {
MaterialTheme.colorScheme.surfaceColorAtElevation( if (isConfigChanged) {
2.dp, Intent(this@MainActivity, MainActivity::class.java).also {
), startActivity(it)
) exitProcess(0)
} }
}, }
bottomBar = { }
BottomNavBar( LaunchedEffect(errorMessage) {
navController, errorMessage?.let {
listOf( snackbar.showSnackbar(it.asString(this@MainActivity))
BottomNavItem( viewModel.handleEvent(AppEvent.MessageShown)
name = stringResource(R.string.tunnels), }
route = Route.Main, }
icon = Icons.Rounded.Home, LaunchedEffect(popBackStack) {
), if (popBackStack) {
BottomNavItem( navController.popBackStack()
name = stringResource(R.string.settings), viewModel.handleEvent(AppEvent.PopBackStack(false))
route = Route.Settings, }
icon = Icons.Rounded.Settings, }
), LaunchedEffect(requestVpnPermission) {
BottomNavItem( if (requestVpnPermission) {
name = stringResource(R.string.support), if (!vpnPermissionDenied) {
route = Route.Support, vpnActivity.launch(VpnService.prepare(this@MainActivity))
icon = Icons.Rounded.QuestionMark, } else {
), showVpnPermissionDialog = true
), }
) viewModel.handleEvent(AppEvent.VpnPermissionRequested)
}, }
) { padding -> }
Box(modifier = Modifier.Companion.fillMaxSize().padding(padding)) { LaunchedEffect(requestBatteryPermission) {
NavHost( if (requestBatteryPermission) {
navController, batteryActivity.launch(
enterTransition = { fadeIn(tween(Constants.TRANSITION_ANIMATION_TIME)) }, Intent().apply {
exitTransition = { fadeOut(tween(Constants.TRANSITION_ANIMATION_TIME)) }, action = Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS
startDestination = (if (appUiState.generalState.isPinLockEnabled) Route.Lock else Route.Main), data = "package:${this@MainActivity.packageName}".toUri()
) { }
composable<Route.Main> { )
MainScreen( }
uiState = appUiState, }
) }
}
composable<Route.Settings> { CompositionLocalProvider(LocalIsAndroidTV provides isTv) {
SettingsScreen( CompositionLocalProvider(LocalNavController provides navController) {
appViewModel = viewModel, WireguardAutoTunnelTheme(theme = appUiState.appState.theme) {
uiState = appUiState, VpnDeniedDialog(
) showVpnPermissionDialog,
} onDismiss = {
composable<Route.LocationDisclosure> { showVpnPermissionDialog = false
LocationDisclosureScreen(viewModel, appUiState) vpnPermissionDenied = false
} },
composable<Route.AutoTunnel> { )
AutoTunnelScreen(
appUiState.appSettings, Scaffold(
) modifier =
} Modifier.pointerInput(Unit) {
composable<Route.Appearance> { detectTapGestures {
AppearanceScreen() viewModel.handleEvent(AppEvent.ClearSelectedTunnels)
} }
composable<Route.Language> { },
LanguageScreen(appUiState, viewModel) snackbarHost = {
} SnackbarHost(snackbar) { snackbarData: SnackbarData ->
composable<Route.Display> { CustomSnackBar(
DisplayScreen(appUiState) snackbarData.visuals.message,
} isRtl = false,
composable<Route.Support> { containerColor =
SupportScreen(appUiState, viewModel) MaterialTheme.colorScheme.surfaceColorAtElevation(2.dp),
} )
composable<Route.AutoTunnelAdvanced> { }
AdvancedScreen(appUiState.appSettings, viewModel) },
} topBar = { DynamicTopAppBar(navBarState) },
composable<Route.Logs> { bottomBar = {
LogsScreen() AnimatedVisibility(
} visible = navBarState.showBottom,
composable<Route.Config> { enter = slideInVertically(initialOffsetY = { it }),
val args = it.toRoute<Route.Config>() exit = slideOutVertically(targetOffsetY = { it }),
val config = ) {
appUiState.tunnels.firstOrNull { it.id == args.id } BottomNavbar(appUiState = appUiState)
ConfigScreen(config, viewModel) }
} },
composable<Route.TunnelOptions> { ) { padding ->
val args = it.toRoute<Route.TunnelOptions>() Box(
val config = appUiState.tunnels.first { it.id == args.id } modifier =
OptionsScreen(config) Modifier.fillMaxSize()
} .background(MaterialTheme.colorScheme.surface)
composable<Route.Lock> { .padding(padding)
PinLockScreen(viewModel) .consumeWindowInsets(padding)
} .imePadding()
composable<Route.Scanner> { ) {
ScannerScreen() NavHost(
} navController,
composable<Route.KillSwitch> { startDestination =
KillSwitchScreen(appUiState, viewModel) (if (appUiState.appState.isPinLockEnabled) Route.Lock
} else Route.Main),
composable<Route.SplitTunnel> { ) {
val args = it.toRoute<Route.SplitTunnel>() composable<Route.Main> {
val config = appUiState.tunnels.first { it.id == args.id } MainScreen(appUiState, appViewState, viewModel)
SplitTunnelScreen(config, viewModel) }
} composable<Route.Settings> {
composable<Route.TunnelAutoTunnel> { SettingsScreen(appUiState, viewModel)
val args = it.toRoute<Route.TunnelOptions>() }
val config = appUiState.tunnels.first { it.id == args.id } composable<Route.SettingsAdvanced> {
TunnelAutoTunnelScreen(config, appUiState.appSettings) SettingsAdvancedScreen(appUiState, viewModel)
} }
} composable<Route.LocationDisclosure> {
BackHandler { LocationDisclosureScreen(appUiState, viewModel)
if (navController.previousBackStackEntry == null || !navController.popBackStack()) { }
this@MainActivity.finish() composable<Route.AutoTunnel> {
} AutoTunnelScreen(appUiState, viewModel)
} }
} composable<Route.Appearance> { AppearanceScreen() }
} composable<Route.Language> {
} LanguageScreen(appUiState, viewModel)
} }
} composable<Route.Display> {
} DisplayScreen(appUiState, viewModel)
} }
composable<Route.Support> {
SupportScreen(appViewModel = viewModel)
}
composable<Route.License> { LicenseScreen() }
composable<Route.AutoTunnelAdvanced> {
AutoTunnelAdvancedScreen(appUiState, viewModel)
}
composable<Route.Logs> { LogsScreen(appViewState, viewModel) }
composable<Route.Config> { backStack ->
val args = backStack.toRoute<Route.Config>()
val config =
appUiState.tunnels.firstOrNull { it.id == args.id }
ConfigScreen(config, viewModel)
}
composable<Route.TunnelOptions> { backStack ->
val args = backStack.toRoute<Route.TunnelOptions>()
appUiState.tunnels
.firstOrNull { it.id == args.id }
?.let { config ->
TunnelOptionsScreen(config, viewModel)
}
}
composable<Route.Lock> { PinLockScreen(viewModel) }
composable<Route.Scanner> { ScannerScreen(viewModel) }
composable<Route.KillSwitch> {
KillSwitchScreen(appUiState, viewModel)
}
composable<Route.SplitTunnel> { SplitTunnelScreen(viewModel) }
composable<Route.TunnelAutoTunnel> { backStack ->
val args = backStack.toRoute<Route.TunnelOptions>()
appUiState.tunnels
.firstOrNull { it.id == args.id }
?.let {
TunnelAutoTunnelScreen(
it,
appUiState.appSettings,
viewModel,
)
}
}
}
}
}
}
}
}
}
}
override fun onResume() {
super.onResume()
checkPermissionAndNotify()
}
private fun checkPermissionAndNotify() {
val hasLocation =
ContextCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) ==
PackageManager.PERMISSION_GRANTED
if (lastLocationPermissionState != hasLocation) {
Timber.d("Location permission changed to: $hasLocation")
if (hasLocation) {
networkMonitor.sendLocationPermissionsGrantedBroadcast()
}
lastLocationPermissionState = hasLocation
}
}
} }
@@ -11,6 +11,7 @@ import androidx.work.Configuration
import com.wireguard.android.backend.GoBackend import com.wireguard.android.backend.GoBackend
import com.zaneschepke.logcatter.LogReader import com.zaneschepke.logcatter.LogReader
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.core.worker.ServiceWorker
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.di.MainDispatcher import com.zaneschepke.wireguardautotunnel.di.MainDispatcher
@@ -18,121 +19,116 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.util.LocaleUtil import com.zaneschepke.wireguardautotunnel.util.LocaleUtil
import com.zaneschepke.wireguardautotunnel.util.ReleaseTree import com.zaneschepke.wireguardautotunnel.util.ReleaseTree
import com.zaneschepke.wireguardautotunnel.util.extensions.isRunningOnTv
import dagger.hilt.android.HiltAndroidApp import dagger.hilt.android.HiltAndroidApp
import javax.inject.Inject
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject
@HiltAndroidApp @HiltAndroidApp
class WireGuardAutoTunnel : Application(), Configuration.Provider { class WireGuardAutoTunnel : Application(), Configuration.Provider {
@Inject @Inject lateinit var workerFactory: HiltWorkerFactory
lateinit var workerFactory: HiltWorkerFactory
override val workManagerConfiguration: Configuration override val workManagerConfiguration: Configuration
get() = Configuration.Builder() get() = Configuration.Builder().setWorkerFactory(workerFactory).build()
.setWorkerFactory(workerFactory)
.build()
@Inject @Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
@ApplicationScope
lateinit var applicationScope: CoroutineScope
@Inject @Inject lateinit var logReader: LogReader
lateinit var logReader: LogReader
@Inject @Inject lateinit var appDataRepository: AppDataRepository
lateinit var appDataRepository: AppDataRepository
@Inject @Inject @IoDispatcher lateinit var ioDispatcher: CoroutineDispatcher
@IoDispatcher
lateinit var ioDispatcher: CoroutineDispatcher
@Inject @Inject @MainDispatcher lateinit var mainDispatcher: CoroutineDispatcher
@MainDispatcher
lateinit var mainDispatcher: CoroutineDispatcher
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var tunnelManager: TunnelManager
override fun onCreate() { override fun onCreate() {
super.onCreate() super.onCreate()
instance = this instance = this
ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifecycleObserver()) ProcessLifecycleOwner.get().lifecycle.addObserver(AppLifecycleObserver())
if (BuildConfig.DEBUG) { if (BuildConfig.DEBUG) {
Timber.plant(Timber.DebugTree()) Timber.plant(Timber.DebugTree())
StrictMode.setThreadPolicy( StrictMode.setThreadPolicy(
ThreadPolicy.Builder() ThreadPolicy.Builder()
.detectDiskReads() .detectDiskReads()
.detectDiskWrites() .detectDiskWrites()
.detectNetwork() .detectNetwork()
.penaltyLog() .penaltyLog()
.build(), .build()
) )
} else { } else {
Timber.plant(ReleaseTree()) Timber.plant(ReleaseTree())
} }
GoBackend.setAlwaysOnCallback { GoBackend.setAlwaysOnCallback {
applicationScope.launch { applicationScope.launch {
val settings = appDataRepository.settings.get() val settings = appDataRepository.settings.get()
if (settings.isAlwaysOnVpnEnabled) { if (settings.isAlwaysOnVpnEnabled) {
val tunnel = appDataRepository.getPrimaryOrFirstTunnel() val tunnel = appDataRepository.getPrimaryOrFirstTunnel()
tunnel?.let { tunnel?.let { tunnelManager.startTunnel(it) }
tunnelManager.startTunnel(it) } else {
} Timber.w("Always-on VPN is not enabled in app settings")
} else { }
Timber.Forest.w("Always-on VPN is not enabled in app settings") }
} }
}
}
applicationScope.launch { ServiceWorker.start(this)
withContext(mainDispatcher) {
if (appDataRepository.appState.isLocalLogsEnabled() && !isRunningOnTv()) logReader.initialize()
}
if (!appDataRepository.settings.get().isKernelEnabled) {
tunnelManager.setBackendState(BackendState.SERVICE_ACTIVE, emptyList())
}
appDataRepository.appState.getLocale()?.let {
withContext(mainDispatcher) {
LocaleUtil.changeLocale(it)
}
}
}
}
override fun onTerminate() { applicationScope.launch {
applicationScope.launch { appDataRepository.appState.getLocale()?.let {
tunnelManager.setBackendState(BackendState.SERVICE_ACTIVE, emptyList()) withContext(mainDispatcher) { LocaleUtil.changeLocale(it) }
} }
super.onTerminate() appDataRepository.appState.isLocalLogsEnabled().let { enabled ->
} if (enabled) logReader.start()
}
}
}
class AppLifecycleObserver : DefaultLifecycleObserver { override fun onTerminate() {
applicationScope.launch {
tunnelManager.setBackendState(BackendState.INACTIVE, emptyList())
}
super.onTerminate()
}
override fun onStart(owner: LifecycleOwner) { class AppLifecycleObserver : DefaultLifecycleObserver {
Timber.d("Application entered foreground")
foreground = true
}
override fun onPause(owner: LifecycleOwner) {
Timber.d("Application entered background")
foreground = false
}
}
companion object { override fun onStart(owner: LifecycleOwner) {
private var foreground = false Timber.d("Application entered foreground")
foreground = true
}
fun isForeground(): Boolean { override fun onPause(owner: LifecycleOwner) {
return foreground Timber.d("Application entered background")
} foreground = false
}
}
lateinit var instance: WireGuardAutoTunnel companion object {
private set private var foreground = false
}
fun isForeground(): Boolean {
return foreground
}
@Volatile private var lastActiveTunnels: List<Int> = emptyList()
@Synchronized
fun getLastActiveTunnels(): List<Int> {
return lastActiveTunnels
}
@Synchronized
fun setLastActiveTunnels(newTunnels: List<Int>) {
lastActiveTunnels = newTunnels
}
lateinit var instance: WireGuardAutoTunnel
private set
}
} }
@@ -1,45 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import javax.inject.Inject
@AndroidEntryPoint
class AppUpdateReceiver : BroadcastReceiver() {
@Inject
@ApplicationScope
lateinit var applicationScope: CoroutineScope
@Inject
lateinit var appDataRepository: AppDataRepository
@Inject
lateinit var serviceManager: ServiceManager
@Inject
lateinit var tunnelManager: TunnelManager
override fun onReceive(context: Context, intent: Intent) {
if (intent.action != Intent.ACTION_MY_PACKAGE_REPLACED) return
serviceManager.updateTunnelTile()
serviceManager.updateAutoTunnelTile()
applicationScope.launch {
with(appDataRepository.settings.get()) {
if (isRestoreOnBootEnabled) {
// If auto tunnel is enabled, just start it and let auto tunnel start appropriate tun
if (isAutoTunnelEnabled && !serviceManager.autoTunnelActive.value) return@launch serviceManager.startAutoTunnel(true)
tunnelManager.restorePreviousState()
}
}
}
}
}
@@ -1,44 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import javax.inject.Inject
@AndroidEntryPoint
class BootReceiver : BroadcastReceiver() {
@Inject
lateinit var appDataRepository: AppDataRepository
@Inject
@ApplicationScope
lateinit var applicationScope: CoroutineScope
@Inject
lateinit var serviceManager: ServiceManager
@Inject
lateinit var tunnelManager: TunnelManager
override fun onReceive(context: Context, intent: Intent) {
if (Intent.ACTION_BOOT_COMPLETED != intent.action) return
serviceManager.updateTunnelTile()
serviceManager.updateAutoTunnelTile()
applicationScope.launch {
with(appDataRepository.settings.get()) {
if (isRestoreOnBootEnabled) {
// If auto tunnel is enabled, just start it and let auto tunnel start appropriate tun
if (isAutoTunnelEnabled && !serviceManager.autoTunnelActive.value) return@launch serviceManager.startAutoTunnel(true)
tunnelManager.restorePreviousState()
}
}
}
}
}
@@ -3,47 +3,40 @@ package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver import android.content.BroadcastReceiver
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import javax.inject.Inject
@AndroidEntryPoint @AndroidEntryPoint
class KernelReceiver : BroadcastReceiver() { class KernelReceiver : BroadcastReceiver() {
@Inject @Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
@ApplicationScope
lateinit var applicationScope: CoroutineScope
@Inject @Inject lateinit var tunnelRepository: TunnelRepository
lateinit var tunnelRepository: TunnelRepository
@Inject @Inject lateinit var serviceManager: ServiceManager
lateinit var serviceManager: ServiceManager
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var tunnelManager: TunnelManager
override fun onReceive(context: Context, intent: Intent) { override fun onReceive(context: Context, intent: Intent) {
val action = intent.action ?: return val action = intent.action ?: return
applicationScope.launch { applicationScope.launch {
if (action == REFRESH_TUNNELS_ACTION) { if (action == REFRESH_TUNNELS_ACTION) {
tunnelManager.runningTunnelNames().forEach { tunnelManager.runningTunnelNames().forEach { name ->
val tunnel = tunnelRepository.findByTunnelName(it) val tunnel = tunnelRepository.findByTunnelName(name)
tunnel?.let { tunnel?.let { tunnelRepository.save(it.copy(isActive = true)) }
tunnelRepository.save(it.copy(isActive = true)) }
} serviceManager.updateTunnelTile()
} }
serviceManager.updateTunnelTile() }
} }
}
}
companion object { companion object {
const val REFRESH_TUNNELS_ACTION = "com.wireguard.android.action.REFRESH_TUNNEL_STATES" const val REFRESH_TUNNELS_ACTION = "com.wireguard.android.action.REFRESH_TUNNEL_STATES"
} }
} }
@@ -4,43 +4,42 @@ import android.content.BroadcastReceiver
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import javax.inject.Inject
@AndroidEntryPoint @AndroidEntryPoint
class NotificationActionReceiver : BroadcastReceiver() { class NotificationActionReceiver : BroadcastReceiver() {
@Inject @Inject lateinit var serviceManager: ServiceManager
lateinit var serviceManager: ServiceManager
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var tunnelManager: TunnelManager
@Inject @Inject lateinit var tunnelRepository: TunnelRepository
lateinit var tunnelRepository: TunnelRepository
@Inject @Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
@ApplicationScope
lateinit var applicationScope: CoroutineScope
override fun onReceive(context: Context, intent: Intent) { override fun onReceive(context: Context, intent: Intent) {
applicationScope.launch { applicationScope.launch {
when (intent.action) { when (intent.action) {
NotificationAction.AUTO_TUNNEL_OFF.name -> serviceManager.stopAutoTunnel() NotificationAction.AUTO_TUNNEL_OFF.name -> serviceManager.stopAutoTunnel()
NotificationAction.TUNNEL_OFF.name -> { NotificationAction.TUNNEL_OFF.name -> {
val tunnelId = intent.getIntExtra(NotificationManager.EXTRA_ID, 0) val tunnelId = intent.getIntExtra(NotificationManager.EXTRA_ID, 0)
if (tunnelId == 0) return@launch tunnelManager.stopTunnel() if (tunnelId == STOP_ALL_TUNNELS_ID) return@launch tunnelManager.stopTunnel()
val tunnel = tunnelRepository.getById(tunnelId) val tunnel = tunnelRepository.getById(tunnelId)
tunnelManager.stopTunnel(tunnel) tunnelManager.stopTunnel(tunnel)
} }
} }
} }
} }
companion object {
const val STOP_ALL_TUNNELS_ID = 0
}
} }
@@ -0,0 +1,96 @@
package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.util.Constants
import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import timber.log.Timber
@AndroidEntryPoint
class RemoteControlReceiver : BroadcastReceiver() {
@Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
@Inject lateinit var appDataRepository: AppDataRepository
@Inject lateinit var serviceManager: ServiceManager
@Inject lateinit var tunnelManager: TunnelManager
enum class Action(private val suffix: String) {
START_TUNNEL("START_TUNNEL"),
STOP_TUNNEL("STOP_TUNNEL"),
START_AUTO_TUNNEL("START_AUTO_TUNNEL"),
STOP_AUTO_TUNNEL("STOP_AUTO_TUNNEL");
fun getFullAction(): String {
return "${Constants.BASE_PACKAGE}.$suffix"
}
companion object {
fun fromAction(action: String): Action? {
for (a in entries) {
if (a.getFullAction() == action) {
return a
}
}
return null
}
}
}
override fun onReceive(context: Context, intent: Intent) {
Timber.i("onReceive")
val action = intent.action ?: return
val appAction = Action.fromAction(action) ?: return Timber.w("Unknown action $action")
applicationScope.launch {
if (!appDataRepository.appState.isRemoteControlEnabled())
return@launch Timber.w("Remote control disabled")
val key =
appDataRepository.appState.getRemoteKey()
?: return@launch Timber.w("Remote control key missing")
if (key != intent.getStringExtra(EXTRA_KEY)?.trim())
return@launch Timber.w("Invalid remote control key")
when (appAction) {
Action.START_TUNNEL -> {
val tunnelName =
intent.getStringExtra(EXTRA_TUN_NAME) ?: return@launch startDefaultTunnel()
val tunnel =
appDataRepository.tunnels.findByTunnelName(tunnelName)
?: return@launch startDefaultTunnel()
tunnelManager.startTunnel(tunnel)
}
Action.STOP_TUNNEL -> {
val tunnelName =
intent.getStringExtra(EXTRA_TUN_NAME)
?: return@launch tunnelManager.stopTunnel()
val tunnel =
appDataRepository.tunnels.findByTunnelName(tunnelName)
?: return@launch tunnelManager.stopTunnel()
tunnelManager.stopTunnel(tunnel)
}
Action.START_AUTO_TUNNEL -> serviceManager.startAutoTunnel()
Action.STOP_AUTO_TUNNEL -> serviceManager.stopAutoTunnel()
}
}
}
private suspend fun startDefaultTunnel() {
appDataRepository.getPrimaryOrFirstTunnel()?.let { tunnel ->
tunnelManager.startTunnel(tunnel)
}
}
companion object {
const val EXTRA_TUN_NAME = "tunnelName"
const val EXTRA_KEY = "key"
}
}
@@ -0,0 +1,55 @@
package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import com.zaneschepke.logcatter.LogReader
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import timber.log.Timber
@AndroidEntryPoint
class RestartReceiver : BroadcastReceiver() {
@Inject lateinit var appDataRepository: AppDataRepository
@Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
@Inject lateinit var serviceManager: ServiceManager
@Inject lateinit var tunnelManager: TunnelManager
@Inject lateinit var logReader: LogReader
@Inject @IoDispatcher lateinit var ioDispatcher: CoroutineDispatcher
override fun onReceive(context: Context, intent: Intent) {
Timber.d("RestartReceiver triggered with action: ${intent.action}")
serviceManager.updateTunnelTile()
serviceManager.updateAutoTunnelTile()
applicationScope.launch(ioDispatcher) {
val settings = appDataRepository.settings.get()
if (settings.isRestoreOnBootEnabled) {
if (
settings.isAutoTunnelEnabled && serviceManager.autoTunnelService.value == null
) {
Timber.d("Starting auto-tunnel on boot/update")
serviceManager.startAutoTunnel()
} else {
Timber.d("Restoring previous tunnel state")
tunnelManager.restorePreviousState()
}
} else {
Timber.d("Restore on boot disabled, skipping")
}
if (intent.action == Intent.ACTION_MY_PACKAGE_REPLACED) logReader.deleteAndClearLogs()
}
}
}
@@ -1,161 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.network
import android.content.Context
import android.net.ConnectivityManager
import android.net.Network
import android.net.NetworkCapabilities
import android.net.NetworkRequest
import android.net.wifi.SupplicantState
import android.net.wifi.WifiManager
import android.os.Build
import com.zaneschepke.wireguardautotunnel.domain.state.ConnectivityState
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import dagger.hilt.android.qualifiers.ApplicationContext
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.channels.awaitClose
import kotlinx.coroutines.flow.callbackFlow
import kotlinx.coroutines.flow.flowOn
import javax.inject.Inject
class InternetConnectivityMonitor
@Inject
constructor(
@ApplicationContext private val context: Context,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) : NetworkMonitor {
private val connectivityManager =
context.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager
@get:Synchronized @set:Synchronized
private var wifiCapabilities: NetworkCapabilities? = null
@get:Synchronized @set:Synchronized
private var wifiNetworkChanged: Boolean = false
override val didWifiChangeSinceLastCapabilitiesQuery: Boolean
get() = wifiNetworkChanged
override val status = callbackFlow {
var wifiState: Boolean = false
var ethernetState: Boolean = false
var cellularState: Boolean = false
fun emitState() {
trySend(ConnectivityState(wifiState, ethernetState, cellularState))
}
val currentNetwork = connectivityManager.activeNetwork
if (currentNetwork == null) {
emitState()
}
fun updateCapabilityState(up: Boolean, network: Network) {
with(connectivityManager.getNetworkCapabilities(network)) {
when {
this == null -> return
hasTransport(NetworkCapabilities.TRANSPORT_WIFI) -> wifiState = up
hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) ->
cellularState = up
hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET) ->
ethernetState = up
}
}
}
fun onWifiChange(network: Network, callback: () -> Unit) {
if (connectivityManager.getNetworkCapabilities(network)?.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) == true) {
callback()
}
}
fun onAvailable(network: Network) {
onWifiChange(network) {
wifiNetworkChanged = true
}
}
fun onCapabilitiesChanged(network: Network, networkCapabilities: NetworkCapabilities) {
onWifiChange(network) {
wifiCapabilities = networkCapabilities
}
updateCapabilityState(true, network)
emitState()
}
val networkStatusCallback =
when (Build.VERSION.SDK_INT) {
in Build.VERSION_CODES.S..Int.MAX_VALUE -> {
object :
ConnectivityManager.NetworkCallback(
FLAG_INCLUDE_LOCATION_INFO,
) {
override fun onAvailable(network: Network) {
onAvailable(network)
}
override fun onLost(network: Network) {
updateCapabilityState(false, network)
emitState()
}
override fun onCapabilitiesChanged(network: Network, networkCapabilities: NetworkCapabilities) {
onCapabilitiesChanged(network, networkCapabilities)
}
}
}
else -> {
object : ConnectivityManager.NetworkCallback() {
override fun onAvailable(network: Network) {
onAvailable(network)
}
override fun onLost(network: Network) {
updateCapabilityState(false, network)
emitState()
}
override fun onCapabilitiesChanged(network: Network, networkCapabilities: NetworkCapabilities) {
onCapabilitiesChanged(network, networkCapabilities)
}
}
}
}
val request =
NetworkRequest.Builder()
.addTransportType(NetworkCapabilities.TRANSPORT_WIFI)
.addTransportType(NetworkCapabilities.TRANSPORT_ETHERNET)
.addTransportType(NetworkCapabilities.TRANSPORT_CELLULAR)
.addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET)
.addCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED)
.build()
connectivityManager.registerNetworkCallback(request, networkStatusCallback)
awaitClose { connectivityManager.unregisterNetworkCallback(networkStatusCallback) }
}.flowOn(ioDispatcher)
override fun getWifiCapabilities(): NetworkCapabilities? {
wifiNetworkChanged = false
return wifiCapabilities
}
companion object {
fun getNetworkName(networkCapabilities: NetworkCapabilities, context: Context): String? {
var ssid = networkCapabilities.getWifiName()
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.S) {
val wifiManager =
context.applicationContext.getSystemService(Context.WIFI_SERVICE) as WifiManager
@Suppress("DEPRECATION")
val info = wifiManager.connectionInfo
if (info.supplicantState === SupplicantState.COMPLETED) {
ssid = info.ssid
}
}
return ssid?.trim('"')
}
}
}
@@ -1,16 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.network
import android.net.NetworkCapabilities
import android.net.wifi.WifiInfo
import android.os.Build
fun NetworkCapabilities.getWifiName(): String? {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
val info: WifiInfo
if (transportInfo is WifiInfo) {
info = transportInfo as WifiInfo
return info.ssid
}
}
return null
}
@@ -1,13 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.network
import android.net.NetworkCapabilities
import com.zaneschepke.wireguardautotunnel.domain.state.ConnectivityState
import kotlinx.coroutines.flow.Flow
interface NetworkMonitor {
val status: Flow<ConnectivityState>
// util to help limit location queries
val didWifiChangeSinceLastCapabilitiesQuery: Boolean
fun getWifiCapabilities(): NetworkCapabilities?
}
@@ -4,44 +4,47 @@ import android.app.Notification
import android.app.NotificationManager import android.app.NotificationManager
import android.content.Context import android.content.Context
import androidx.core.app.NotificationCompat import androidx.core.app.NotificationCompat
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification.NotificationChannels import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification.NotificationChannels
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
interface NotificationManager { interface NotificationManager {
val context: Context val context: Context
fun createNotification(
channel: NotificationChannels,
title: String = "",
actions: Collection<NotificationCompat.Action> = emptyList(),
description: String = "",
showTimestamp: Boolean = false,
importance: Int = NotificationManager.IMPORTANCE_HIGH,
onGoing: Boolean = true,
onlyAlertOnce: Boolean = true,
): Notification
fun createNotification( fun createNotification(
channel: NotificationChannels, channel: NotificationChannels,
title: StringValue, title: String = "",
actions: Collection<NotificationCompat.Action> = emptyList(), actions: Collection<NotificationCompat.Action> = emptyList(),
description: StringValue, description: String = "",
showTimestamp: Boolean = false, showTimestamp: Boolean = false,
importance: Int = NotificationManager.IMPORTANCE_HIGH, importance: Int = NotificationManager.IMPORTANCE_HIGH,
onGoing: Boolean = true, onGoing: Boolean = true,
onlyAlertOnce: Boolean = true, onlyAlertOnce: Boolean = true,
): Notification ): Notification
fun createNotificationAction(notificationAction: NotificationAction, extraId: Int? = null): NotificationCompat.Action fun createNotification(
channel: NotificationChannels,
title: StringValue,
actions: Collection<NotificationCompat.Action> = emptyList(),
description: StringValue,
showTimestamp: Boolean = false,
importance: Int = NotificationManager.IMPORTANCE_HIGH,
onGoing: Boolean = true,
onlyAlertOnce: Boolean = true,
): Notification
fun remove(notificationId: Int) fun createNotificationAction(
notificationAction: NotificationAction,
extraId: Int? = null,
): NotificationCompat.Action
fun show(notificationId: Int, notification: Notification) fun remove(notificationId: Int)
companion object { fun show(notificationId: Int, notification: Notification)
const val KERNEL_SERVICE_NOTIFICATION_ID = 123
const val AUTO_TUNNEL_NOTIFICATION_ID = 122 companion object {
const val VPN_NOTIFICATION_ID = 100 const val AUTO_TUNNEL_NOTIFICATION_ID = 122
const val EXTRA_ID = "id" const val VPN_NOTIFICATION_ID = 100
} const val EXTRA_ID = "id"
}
} }
@@ -12,158 +12,165 @@ import android.graphics.Color
import androidx.core.app.ActivityCompat import androidx.core.app.ActivityCompat
import androidx.core.app.NotificationCompat import androidx.core.app.NotificationCompat
import androidx.core.app.NotificationManagerCompat import androidx.core.app.NotificationManagerCompat
import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.broadcast.NotificationActionReceiver import com.zaneschepke.wireguardautotunnel.core.broadcast.NotificationActionReceiver
import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager.Companion.EXTRA_ID import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager.Companion.EXTRA_ID
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
import dagger.hilt.android.qualifiers.ApplicationContext import dagger.hilt.android.qualifiers.ApplicationContext
import javax.inject.Inject import javax.inject.Inject
class WireGuardNotification class WireGuardNotification @Inject constructor(@ApplicationContext override val context: Context) :
@Inject com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager {
constructor(
@ApplicationContext override val context: Context,
) : com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager {
enum class NotificationChannels { enum class NotificationChannels {
VPN, VPN,
AUTO_TUNNEL, AUTO_TUNNEL,
} }
private val notificationManager = NotificationManagerCompat.from(context) private val notificationManager = NotificationManagerCompat.from(context)
override fun createNotification( override fun createNotification(
channel: NotificationChannels, channel: NotificationChannels,
title: String, title: String,
actions: Collection<NotificationCompat.Action>, actions: Collection<NotificationCompat.Action>,
description: String, description: String,
showTimestamp: Boolean, showTimestamp: Boolean,
importance: Int, importance: Int,
onGoing: Boolean, onGoing: Boolean,
onlyAlertOnce: Boolean, onlyAlertOnce: Boolean,
): Notification { ): Notification {
notificationManager.createNotificationChannel(channel.asChannel()) notificationManager.createNotificationChannel(channel.asChannel())
return channel.asBuilder().apply { return channel
actions.forEach { .asBuilder()
addAction(it) .apply {
} actions.forEach { addAction(it) }
setContentTitle(title) setContentTitle(title)
setContentIntent( setContentIntent(
PendingIntent.getActivity( PendingIntent.getActivity(
context, context,
0, 0,
Intent(context, MainActivity::class.java), Intent(context, MainActivity::class.java),
PendingIntent.FLAG_IMMUTABLE, PendingIntent.FLAG_IMMUTABLE,
), )
) )
setContentText(description) setContentText(description)
setOnlyAlertOnce(onlyAlertOnce) setOnlyAlertOnce(onlyAlertOnce)
setOngoing(onGoing) setOngoing(onGoing)
setPriority(NotificationCompat.PRIORITY_HIGH) setPriority(NotificationCompat.PRIORITY_HIGH)
setShowWhen(showTimestamp) setShowWhen(showTimestamp)
setSmallIcon(R.drawable.ic_launcher) setSmallIcon(R.drawable.ic_notification)
}.build() }
} .build()
}
override fun createNotification( override fun createNotification(
channel: NotificationChannels, channel: NotificationChannels,
title: StringValue, title: StringValue,
actions: Collection<NotificationCompat.Action>, actions: Collection<NotificationCompat.Action>,
description: StringValue, description: StringValue,
showTimestamp: Boolean, showTimestamp: Boolean,
importance: Int, importance: Int,
onGoing: Boolean, onGoing: Boolean,
onlyAlertOnce: Boolean, onlyAlertOnce: Boolean,
): Notification { ): Notification {
return createNotification( return createNotification(
channel, channel,
title.asString(context), title.asString(context),
actions, actions,
description.asString(context), description.asString(context),
showTimestamp, showTimestamp,
importance, importance,
onGoing, onGoing,
onlyAlertOnce, onlyAlertOnce,
) )
} }
override fun createNotificationAction(notificationAction: NotificationAction, extraId: Int?): NotificationCompat.Action { override fun createNotificationAction(
val pendingIntent = PendingIntent.getBroadcast( notificationAction: NotificationAction,
context, extraId: Int?,
0, ): NotificationCompat.Action {
Intent(context, NotificationActionReceiver::class.java).apply { val pendingIntent =
action = notificationAction.name PendingIntent.getBroadcast(
if (extraId != null) putExtra(EXTRA_ID, extraId) context,
}, 0,
PendingIntent.FLAG_IMMUTABLE, Intent(context, NotificationActionReceiver::class.java).apply {
) action = notificationAction.name
return NotificationCompat.Action.Builder( if (extraId != null) putExtra(EXTRA_ID, extraId)
R.drawable.ic_launcher, },
notificationAction.title(context).uppercase(), PendingIntent.FLAG_IMMUTABLE,
pendingIntent, )
).build() return NotificationCompat.Action.Builder(
} R.drawable.ic_notification,
notificationAction.title(context).uppercase(),
pendingIntent,
)
.build()
}
override fun remove(notificationId: Int) { override fun remove(notificationId: Int) {
notificationManager.cancel(notificationId) notificationManager.cancel(notificationId)
} }
override fun show(notificationId: Int, notification: Notification) { override fun show(notificationId: Int, notification: Notification) {
with(notificationManager) { with(notificationManager) {
if (ActivityCompat.checkSelfPermission(context, Manifest.permission.POST_NOTIFICATIONS) != PackageManager.PERMISSION_GRANTED) { if (
return ActivityCompat.checkSelfPermission(
} context,
notify(notificationId, notification) Manifest.permission.POST_NOTIFICATIONS,
} ) != PackageManager.PERMISSION_GRANTED
} ) {
return
}
notify(notificationId, notification)
}
}
private fun NotificationChannels.asBuilder(): NotificationCompat.Builder { private fun NotificationChannels.asBuilder(): NotificationCompat.Builder {
return when (this) { return when (this) {
NotificationChannels.AUTO_TUNNEL -> { NotificationChannels.AUTO_TUNNEL -> {
NotificationCompat.Builder( NotificationCompat.Builder(
context, context,
context.getString(R.string.auto_tunnel_channel_id), context.getString(R.string.auto_tunnel_channel_id),
) )
} }
NotificationChannels.VPN -> { NotificationChannels.VPN -> {
NotificationCompat.Builder( NotificationCompat.Builder(context, context.getString(R.string.vpn_channel_id))
context, }
context.getString(R.string.vpn_channel_id), }
) }
}
}
}
private fun NotificationChannels.asChannel(): NotificationChannel { private fun NotificationChannels.asChannel(): NotificationChannel {
return when (this) { return when (this) {
NotificationChannels.VPN -> { NotificationChannels.VPN -> {
NotificationChannel( NotificationChannel(
context.getString(R.string.vpn_channel_id), context.getString(R.string.vpn_channel_id),
context.getString(R.string.vpn_channel_name), context.getString(R.string.vpn_channel_name),
NotificationManager.IMPORTANCE_HIGH, NotificationManager.IMPORTANCE_HIGH,
).apply { )
description = context.getString(R.string.vpn_channel_description) .apply {
enableLights(true) description = context.getString(R.string.vpn_channel_description)
lightColor = Color.WHITE enableLights(true)
enableVibration(false) lightColor = Color.WHITE
vibrationPattern = longArrayOf(100, 200, 300) enableVibration(false)
} vibrationPattern = longArrayOf(100, 200, 300)
} }
NotificationChannels.AUTO_TUNNEL -> { }
NotificationChannel( NotificationChannels.AUTO_TUNNEL -> {
context.getString(R.string.auto_tunnel_channel_id), NotificationChannel(
context.getString(R.string.auto_tunnel_channel_name), context.getString(R.string.auto_tunnel_channel_id),
NotificationManager.IMPORTANCE_HIGH, context.getString(R.string.auto_tunnel_channel_name),
).apply { NotificationManager.IMPORTANCE_HIGH,
description = context.getString(R.string.auto_tunnel_channel_description) )
enableLights(true) .apply {
lightColor = Color.WHITE description = context.getString(R.string.auto_tunnel_channel_description)
enableVibration(false) enableLights(true)
vibrationPattern = longArrayOf(100, 200, 300) lightColor = Color.WHITE
} enableVibration(false)
} vibrationPattern = longArrayOf(100, 200, 300)
} }
} }
}
}
} }
@@ -1,119 +1,150 @@
package com.zaneschepke.wireguardautotunnel.core.service package com.zaneschepke.wireguardautotunnel.core.service
import android.app.Service import android.content.ComponentName
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import android.content.ServiceConnection
import android.net.VpnService
import android.os.IBinder
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService
import com.zaneschepke.wireguardautotunnel.core.service.tile.AutoTunnelControlTile import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.tile.TunnelControlTile import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.di.MainDispatcher
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.util.extensions.requestAutoTunnelTileServiceUpdate import com.zaneschepke.wireguardautotunnel.util.extensions.requestAutoTunnelTileServiceUpdate
import com.zaneschepke.wireguardautotunnel.util.extensions.requestTunnelTileServiceStateUpdate import com.zaneschepke.wireguardautotunnel.util.extensions.requestTunnelTileServiceStateUpdate
import jakarta.inject.Inject import jakarta.inject.Inject
import kotlinx.coroutines.CompletableDeferred
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.asStateFlow
import kotlinx.coroutines.flow.update import kotlinx.coroutines.flow.update
import kotlinx.coroutines.launch
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import timber.log.Timber import timber.log.Timber
@OptIn(ExperimentalCoroutinesApi::class)
class ServiceManager class ServiceManager
@Inject constructor(private val context: Context, private val ioDispatcher: CoroutineDispatcher, private val appDataRepository: AppDataRepository) { @Inject
constructor(
private val context: Context,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher,
@ApplicationScope private val applicationScope: CoroutineScope,
@MainDispatcher private val mainDispatcher: CoroutineDispatcher,
private val appDataRepository: AppDataRepository,
) {
private val _autoTunnelActive = MutableStateFlow(false) private val autoTunnelMutex = Mutex()
val autoTunnelActive = _autoTunnelActive.asStateFlow() private val _tunnelService = MutableStateFlow<TunnelForegroundService?>(null)
private val _autoTunnelService = MutableStateFlow<AutoTunnelService?>(null)
val autoTunnelService = _autoTunnelService.asStateFlow()
var autoTunnelService = CompletableDeferred<AutoTunnelService>() private val tunnelServiceConnection =
var backgroundService = CompletableDeferred<TunnelForegroundService>() object : ServiceConnection {
var autoTunnelTile = CompletableDeferred<AutoTunnelControlTile>() override fun onServiceConnected(name: ComponentName, service: IBinder) {
var tunnelControlTile = CompletableDeferred<TunnelControlTile>() val binder = service as? TunnelForegroundService.LocalBinder
_tunnelService.value = binder?.service
Timber.d("TunnelForegroundService connected")
}
private fun <T : Service> startService(cls: Class<T>, background: Boolean) { override fun onServiceDisconnected(name: ComponentName) {
runCatching { _tunnelService.value = null
val intent = Intent(context, cls) Timber.d("TunnelForegroundService disconnected")
if (background) { }
context.startForegroundService(intent) }
} else {
context.startService(intent)
}
}.onFailure { Timber.e(it) }
}
suspend fun startAutoTunnel(background: Boolean) { private val autoTunnelServiceConnection =
val settings = appDataRepository.settings.get() object : ServiceConnection {
appDataRepository.settings.save(settings.copy(isAutoTunnelEnabled = true)) override fun onServiceConnected(name: ComponentName, service: IBinder) {
if (autoTunnelService.isCompleted) return _autoTunnelActive.update { true } val binder = service as? AutoTunnelService.LocalBinder
runCatching { _autoTunnelService.value = binder?.service
startService(AutoTunnelService::class.java, background) Timber.d("AutoTunnelService connected")
autoTunnelService.await() }
autoTunnelService.getCompleted().start()
_autoTunnelActive.update { true }
updateAutoTunnelTile()
}.onFailure {
Timber.e(it)
}
}
suspend fun startBackgroundService(tunnelConf: TunnelConf) { override fun onServiceDisconnected(name: ComponentName) {
if (backgroundService.isCompleted) return _autoTunnelService.value = null
runCatching { Timber.d("AutoTunnelService disconnected")
startService(TunnelForegroundService::class.java, true) }
backgroundService.await() }
backgroundService.getCompleted().start(tunnelConf)
}.onFailure {
Timber.e(it)
}
}
fun stopBackgroundService() { fun hasVpnPermission(): Boolean {
if (!backgroundService.isCompleted) return return VpnService.prepare(context) == null
runCatching { }
backgroundService.getCompleted().stop()
}.onFailure {
Timber.e(it)
}
}
suspend fun toggleAutoTunnel(background: Boolean) { suspend fun startAutoTunnel() {
withContext(ioDispatcher) { autoTunnelMutex.withLock {
if (_autoTunnelActive.value) return@withContext stopAutoTunnel() val settings = appDataRepository.settings.get()
startAutoTunnel(background) appDataRepository.settings.save(settings.copy(isAutoTunnelEnabled = true))
} if (_autoTunnelService.value != null) return
} withContext(ioDispatcher) {
val intent = Intent(context, AutoTunnelService::class.java)
context.startForegroundService(intent)
context.bindService(intent, autoTunnelServiceConnection, Context.BIND_AUTO_CREATE)
withContext(mainDispatcher) { updateAutoTunnelTile() }
}
}
}
fun updateAutoTunnelTile() { suspend fun stopAutoTunnel() {
if (autoTunnelTile.isCompleted) { autoTunnelMutex.withLock {
autoTunnelTile.getCompleted().updateTileState() val settings = appDataRepository.settings.get()
} else { appDataRepository.settings.save(settings.copy(isAutoTunnelEnabled = false))
context.requestAutoTunnelTileServiceUpdate() if (_autoTunnelService.value == null) return
} _autoTunnelService.value?.let { service ->
} service.stop()
try {
context.unbindService(autoTunnelServiceConnection)
} finally {
_tunnelService.value = null
}
}
withContext(mainDispatcher) { updateAutoTunnelTile() }
}
}
fun updateTunnelTile() { suspend fun startTunnelForegroundService() {
if (tunnelControlTile.isCompleted) { if (_tunnelService.value != null) return
tunnelControlTile.getCompleted().updateTileState() withContext(ioDispatcher) {
} else { applicationScope.launch(ioDispatcher) {
context.requestTunnelTileServiceStateUpdate() val intent = Intent(context, TunnelForegroundService::class.java)
} context.startForegroundService(intent)
} context.bindService(intent, tunnelServiceConnection, Context.BIND_AUTO_CREATE)
}
}
}
suspend fun stopAutoTunnel() { fun stopTunnelForegroundService() {
withContext(ioDispatcher) { _tunnelService.value?.let { service ->
val settings = appDataRepository.settings.get() service.stop()
appDataRepository.settings.save(settings.copy(isAutoTunnelEnabled = false)) try {
if (!autoTunnelService.isCompleted) return@withContext context.unbindService(tunnelServiceConnection)
runCatching { } finally {
autoTunnelService.getCompleted().stop() _tunnelService.value = null
_autoTunnelActive.update { false } }
updateAutoTunnelTile() }
}.onFailure { }
Timber.e(it)
} fun toggleAutoTunnel() {
} applicationScope.launch(ioDispatcher) {
} if (_autoTunnelService.value != null) stopAutoTunnel() else startAutoTunnel()
}
}
fun updateAutoTunnelTile() {
context.requestAutoTunnelTileServiceUpdate()
}
fun updateTunnelTile() {
context.requestTunnelTileServiceStateUpdate()
}
fun handleTunnelServiceDestroy() {
_tunnelService.update { null }
}
fun handleAutoTunnelServiceDestroy() {
_autoTunnelService.update { null }
}
} }
@@ -2,70 +2,327 @@ package com.zaneschepke.wireguardautotunnel.core.service
import android.app.Notification import android.app.Notification
import android.content.Intent import android.content.Intent
import android.os.Binder
import android.os.IBinder import android.os.IBinder
import androidx.core.app.ServiceCompat import androidx.core.app.ServiceCompat
import androidx.lifecycle.LifecycleService import androidx.lifecycle.LifecycleService
import androidx.lifecycle.lifecycleScope
import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.networkmonitor.NetworkStatus
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import com.zaneschepke.wireguardautotunnel.util.Constants import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.extensions.distinctByKeys
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CompletableDeferred import java.util.concurrent.ConcurrentHashMap
import javax.inject.Inject import javax.inject.Inject
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.Job
import kotlinx.coroutines.NonCancellable
import kotlinx.coroutines.coroutineScope
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.collectLatest
import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.filterNotNull
import kotlinx.coroutines.flow.flowOn
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock
import kotlinx.coroutines.withContext
import timber.log.Timber
@AndroidEntryPoint @AndroidEntryPoint
class TunnelForegroundService : LifecycleService() { class TunnelForegroundService : LifecycleService() {
@Inject @Inject lateinit var notificationManager: NotificationManager
lateinit var notificationManager: NotificationManager
@Inject @Inject lateinit var serviceManager: ServiceManager
lateinit var serviceManager: ServiceManager
override fun onCreate() { @Inject lateinit var networkMonitor: NetworkMonitor
super.onCreate()
serviceManager.backgroundService.complete(this)
}
override fun onBind(intent: Intent): IBinder? { @Inject @IoDispatcher lateinit var ioDispatcher: CoroutineDispatcher
super.onBind(intent)
return null
}
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { @Inject lateinit var tunnelRepo: TunnelRepository
super.onStartCommand(intent, flags, startId)
serviceManager.backgroundService.complete(this)
return START_NOT_STICKY
}
fun start(tunnelConf: TunnelConf) { @Inject lateinit var tunnelManager: TunnelManager
ServiceCompat.startForeground(
this@TunnelForegroundService,
NotificationManager.KERNEL_SERVICE_NOTIFICATION_ID,
createNotification(tunnelConf),
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
}
fun stop() { private val isNetworkConnected = MutableStateFlow(true)
stopForeground(STOP_FOREGROUND_REMOVE)
stopSelf()
}
override fun onDestroy() { private val tunnelJobs = ConcurrentHashMap<TunnelConf, Job>()
serviceManager.backgroundService = CompletableDeferred() private val pingJobs = ConcurrentHashMap<TunnelConf, Job>()
super.onDestroy()
}
private fun createNotification(tunnelConf: TunnelConf): Notification { private val jobsMutex = Mutex()
return notificationManager.createNotification(
WireGuardNotification.NotificationChannels.VPN, class LocalBinder(val service: TunnelForegroundService) : Binder()
title = "${getString(R.string.tunnel_running)} - ${tunnelConf.tunName}",
actions = listOf( private val binder = LocalBinder(this)
notificationManager.createNotificationAction(NotificationAction.TUNNEL_OFF, tunnelConf.id),
), override fun onCreate() {
) super.onCreate()
} ServiceCompat.startForeground(
this@TunnelForegroundService,
NotificationManager.VPN_NOTIFICATION_ID,
onCreateNotification(),
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
}
override fun onBind(intent: Intent): IBinder {
super.onBind(intent)
return binder
}
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
super.onStartCommand(intent, flags, startId)
ServiceCompat.startForeground(
this@TunnelForegroundService,
NotificationManager.VPN_NOTIFICATION_ID,
onCreateNotification(),
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
start()
return START_STICKY
}
fun start() =
lifecycleScope.launch(ioDispatcher) {
tunnelManager.activeTunnels.distinctByKeys().collect { activeTunnels ->
// No active tunnels and no jobs: nothing to do
if (activeTunnels.isEmpty() && tunnelJobs.isEmpty()) return@collect
// Synchronize jobs with active tunnels
synchronizeJobs(activeTunnels)
updateServiceNotification()
}
}
private suspend fun synchronizeJobs(activeTunnels: Map<TunnelConf, TunnelState>) {
jobsMutex.withLock {
// Stop jobs for tunnels that are no longer active
stopInactiveJobs(activeTunnels)
// Start jobs for new tunnels
startNewJobs(activeTunnels)
}
}
private fun stopInactiveJobs(activeTunnels: Map<TunnelConf, TunnelState>) {
// If no active tunnels, clear all jobs
if (activeTunnels.isEmpty()) {
clearAllJobs()
return
}
// Stop jobs for tunnels not in activeTunnels
val tunnelsToStop = tunnelJobs.keys - activeTunnels.keys
tunnelsToStop.forEach { tun -> stopTunnelJobs(tun) }
}
private fun clearAllJobs() {
tunnelJobs.forEach { (tun, job) ->
Timber.d("Stopping tunnel job for ${tun.tunName}")
job.cancel()
}
tunnelJobs.clear()
pingJobs.forEach { (tun, job) ->
if (isPingBounce(tun)) {
Timber.d("Preserving ping job for ${tun.tunName} due to PING bounce")
return@forEach
}
Timber.d("Stopping ping job for ${tun.tunName}")
job.cancel()
}
pingJobs.entries.removeIf { (tun, _) -> !isPingBounce(tun) }
}
private fun stopTunnelJobs(tun: TunnelConf) {
tunnelJobs.remove(tun)?.cancel()
Timber.d("Stopped tunnel job for ${tun.tunName}")
if (isPingBounce(tun))
return Timber.d("Preserving ${tun.tunName} ping job due to ping bounce")
pingJobs.remove(tun)?.cancel()
Timber.d("Stopped ping job for ${tun.tunName}")
}
private fun startNewJobs(activeTunnels: Map<TunnelConf, TunnelState>) {
val tunnelsToStart = activeTunnels.keys - tunnelJobs.keys
tunnelsToStart.forEach { tun ->
tunnelJobs[tun] = startTunnelJobs(tun)
Timber.d("Started tunnel job for ${tun.tunName}")
if (pingJobs[tun]?.isActive == true) {
Timber.d("Reusing active ping job for ${tun.tunName}")
} else {
pingJobs[tun]?.cancel() // Cancel any stale job
if (tun.isPingEnabled) {
if (tun.isStaticallyConfigured()) {
Timber.d("Skipping ping for statically configured tunnel")
} else {
pingJobs[tun] = startPingJob(tun)
Timber.d("Started ping job for ${tun.tunName}")
}
}
}
}
}
private fun isPingBounce(tun: TunnelConf): Boolean =
tunnelManager.bouncingTunnelIds[tun.id] == TunnelStatus.StopReason.PING
// TODO Would be cool to have this include kill switch
// TODO also we need to include errors
private fun updateServiceNotification() {
val notification =
when (tunnelJobs.size) {
0 -> onCreateNotification()
1 -> createTunnelNotification(tunnelJobs.keys.first())
else -> createTunnelsNotification()
}
ServiceCompat.startForeground(
this@TunnelForegroundService,
NotificationManager.VPN_NOTIFICATION_ID,
notification,
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
}
// use same scope so we can cancel all of these
private fun startTunnelJobs(tunnelConf: TunnelConf) =
lifecycleScope.launch(ioDispatcher) {
// monitor if we have internet connectivity
launch { startNetworkMonitorJob() }
// job to trigger stats emit on interval
launch { startTunnelStatsJob(tunnelConf) }
// monitor changes to the tunnel config
launch { startTunnelConfChangesJob(tunnelConf) }
}
private suspend fun startTunnelConfChangesJob(tunnelConf: TunnelConf) {
tunnelRepo.flow
.flowOn(ioDispatcher)
.map { storedTunnels -> storedTunnels.firstOrNull { it.id == tunnelConf.id } }
.filterNotNull()
// only emit when one of these 3 values change
.distinctUntilChanged { old, new -> old == new }
.collect { storedTunnel ->
if (tunnelConf != storedTunnel) {
Timber.d("Config changed for ${storedTunnel.tunName}, bouncing")
// let this complete, even after cancel
withContext(NonCancellable) {
tunnelManager.bounceTunnel(
storedTunnel,
TunnelStatus.StopReason.CONFIG_CHANGED,
)
}
}
}
}
private suspend fun startNetworkMonitorJob() {
networkMonitor.networkStatusFlow.flowOn(ioDispatcher).collectLatest { status ->
val isAvailable = status !is NetworkStatus.Disconnected
isNetworkConnected.value = isAvailable
Timber.d("Network available: $status")
}
}
private suspend fun startTunnelStatsJob(tunnel: TunnelConf) = coroutineScope {
while (isActive) {
tunnelManager.updateTunnelStatistics(tunnel)
delay(STATS_DELAY)
}
}
private fun startPingJob(tunnel: TunnelConf) =
lifecycleScope.launch(ioDispatcher) {
// delay for initial duration
delay(tunnel.pingInterval ?: Constants.PING_INTERVAL)
while (isActive) {
val shouldBounce = shouldBounceTunnel(tunnel)
val delayMs =
if (shouldBounce) {
// let this complete, even after cancel
withContext(NonCancellable) {
tunnelManager.bounceTunnel(tunnel, TunnelStatus.StopReason.PING)
}
tunnel.pingCooldown ?: Constants.PING_COOLDOWN
} else {
tunnel.pingInterval ?: Constants.PING_INTERVAL
}
delay(delayMs)
}
}
private suspend fun shouldBounceTunnel(tunnel: TunnelConf): Boolean {
if (!isNetworkConnected.value) {
Timber.d("Network disconnected, skipping ping for ${tunnel.tunName}")
return false
}
return runCatching { !tunnel.isTunnelPingable(ioDispatcher) }
.onFailure { e -> Timber.e(e, "Ping check failed for ${tunnel.tunName}") }
.getOrDefault(true)
}
fun stop() {
ServiceCompat.stopForeground(this, ServiceCompat.STOP_FOREGROUND_REMOVE)
stopSelf()
}
override fun onDestroy() {
serviceManager.handleTunnelServiceDestroy()
ServiceCompat.stopForeground(this, ServiceCompat.STOP_FOREGROUND_REMOVE)
super.onDestroy()
}
private fun createTunnelNotification(tunnelConf: TunnelConf): Notification {
return notificationManager.createNotification(
WireGuardNotification.NotificationChannels.VPN,
title = "${getString(R.string.tunnel_running)} - ${tunnelConf.tunName}",
actions =
listOf(
notificationManager.createNotificationAction(
NotificationAction.TUNNEL_OFF,
tunnelConf.id,
)
),
)
}
private fun createTunnelsNotification(): Notification {
return notificationManager.createNotification(
WireGuardNotification.NotificationChannels.VPN,
title = "${getString(R.string.tunnel_running)} - ${getString(R.string.multiple)}",
actions =
listOf(
notificationManager.createNotificationAction(NotificationAction.TUNNEL_OFF, 0)
),
)
}
private fun onCreateNotification(): Notification {
return notificationManager.createNotification(
WireGuardNotification.NotificationChannels.VPN,
title = getString(R.string.tunnel_starting),
)
}
// TODO add notification handling and optional log reading for restart on handshake failures
companion object {
const val STATS_DELAY = 1_000L
// ipv6 disabled or block on network
// Failed to send handshake initiation: write udp [::]"
// Failed to send data packets: write udp [::]
// Failed to send data packets: write udp 0.0.0.0:51820
// Handshake did not complete after 5 seconds, retrying
}
} }
@@ -1,255 +1,278 @@
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel package com.zaneschepke.wireguardautotunnel.core.service.autotunnel
import android.content.Intent import android.content.Intent
import android.net.NetworkCapabilities import android.os.Binder
import android.os.IBinder import android.os.IBinder
import android.os.PowerManager import android.os.PowerManager
import androidx.core.app.ServiceCompat import androidx.core.app.ServiceCompat
import androidx.lifecycle.LifecycleService import androidx.lifecycle.LifecycleService
import androidx.lifecycle.lifecycleScope import androidx.lifecycle.lifecycleScope
import com.wireguard.android.util.RootShell import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.networkmonitor.NetworkStatus
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.di.AppShell
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.di.MainImmediateDispatcher
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.domain.state.AutoTunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.NetworkState
import com.zaneschepke.wireguardautotunnel.core.network.InternetConnectivityMonitor
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.domain.state.ConnectivityState
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent
import com.zaneschepke.wireguardautotunnel.domain.events.KillSwitchEvent import com.zaneschepke.wireguardautotunnel.domain.events.KillSwitchEvent
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.state.AutoTunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.NetworkState
import com.zaneschepke.wireguardautotunnel.util.Constants import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.extensions.Tunnels import com.zaneschepke.wireguardautotunnel.util.extensions.Tunnels
import com.zaneschepke.wireguardautotunnel.util.extensions.getCurrentWifiName
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CompletableDeferred import javax.inject.Inject
import javax.inject.Provider
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.FlowPreview import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.Job
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.debounce import kotlinx.coroutines.flow.debounce
import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.flatMapLatest
import kotlinx.coroutines.flow.flowOn
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.update import kotlinx.coroutines.flow.update
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject
import javax.inject.Provider
@AndroidEntryPoint @AndroidEntryPoint
class AutoTunnelService : LifecycleService() { class AutoTunnelService : LifecycleService() {
@Inject @Inject lateinit var networkMonitor: NetworkMonitor
@AppShell
lateinit var rootShell: Provider<RootShell>
@Inject @Inject lateinit var appDataRepository: Provider<AppDataRepository>
lateinit var networkMonitor: NetworkMonitor
@Inject @Inject lateinit var notificationManager: NotificationManager
lateinit var appDataRepository: Provider<AppDataRepository>
@Inject @Inject @IoDispatcher lateinit var ioDispatcher: CoroutineDispatcher
lateinit var notificationManager: NotificationManager
@Inject @Inject lateinit var serviceManager: ServiceManager
@IoDispatcher
lateinit var ioDispatcher: CoroutineDispatcher
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var serviceManager: ServiceManager
@Inject private val defaultState = AutoTunnelState()
@MainImmediateDispatcher
lateinit var mainImmediateDispatcher: CoroutineDispatcher
@Inject private val autoTunnelStateFlow = MutableStateFlow(defaultState)
lateinit var tunnelManager: TunnelManager
private val defaultState = AutoTunnelState() private var wakeLock: PowerManager.WakeLock? = null
private val autoTunnelStateFlow = MutableStateFlow(defaultState) private var killSwitchJob: Job? = null
private var wakeLock: PowerManager.WakeLock? = null class LocalBinder(val service: AutoTunnelService) : Binder()
override fun onCreate() { private val binder = LocalBinder(this)
super.onCreate()
serviceManager.autoTunnelService.complete(this)
lifecycleScope.launch(mainImmediateDispatcher) {
runCatching {
launchWatcherNotification()
}.onFailure {
Timber.e(it)
}
}
}
override fun onBind(intent: Intent): IBinder? { override fun onCreate() {
super.onBind(intent) super.onCreate()
return null launchWatcherNotification()
} }
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { override fun onBind(intent: Intent): IBinder {
super.onStartCommand(intent, flags, startId) super.onBind(intent)
Timber.d("onStartCommand executed with startId: $startId") return binder
serviceManager.autoTunnelService.complete(this) }
return START_NOT_STICKY
}
fun start() { override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
kotlin.runCatching { super.onStartCommand(intent, flags, startId)
lifecycleScope.launch(mainImmediateDispatcher) { Timber.d("onStartCommand executed with startId: $startId")
launchWatcherNotification() start()
initWakeLock() return START_STICKY
} }
startAutoTunnelJob()
startAutoTunnelStateJob()
startKillSwitchJob()
}.onFailure {
Timber.e(it)
}
}
fun stop() { fun start() {
wakeLock?.let { if (it.isHeld) it.release() } kotlin
stopSelf() .runCatching {
} launchWatcherNotification()
initWakeLock()
startAutoTunnelJob()
startAutoTunnelStateJob()
killSwitchJob = startKillSwitchJob()
}
.onFailure { Timber.e(it) }
}
override fun onDestroy() { fun stop() {
serviceManager.autoTunnelService = CompletableDeferred() wakeLock?.let { if (it.isHeld) it.release() }
super.onDestroy() stopSelf()
} }
private fun launchWatcherNotification(description: String = getString(R.string.monitoring_state_changes)) { override fun onDestroy() {
val notification = serviceManager.handleAutoTunnelServiceDestroy()
notificationManager.createNotification( restoreVpnKillSwitch()
WireGuardNotification.NotificationChannels.AUTO_TUNNEL, super.onDestroy()
title = getString(R.string.auto_tunnel_title), }
description = description,
actions = listOf(
notificationManager.createNotificationAction(NotificationAction.AUTO_TUNNEL_OFF),
),
)
ServiceCompat.startForeground(
this,
NotificationManager.AUTO_TUNNEL_NOTIFICATION_ID,
notification,
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
}
private fun initWakeLock() { private fun restoreVpnKillSwitch() {
wakeLock = (getSystemService(POWER_SERVICE) as PowerManager).run { with(autoTunnelStateFlow.value) {
val tag = this.javaClass.name if (
newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "$tag::lock").apply { settings.isVpnKillSwitchEnabled &&
try { tunnelManager.getBackendState() != BackendState.KILL_SWITCH_ACTIVE
Timber.i("Initiating wakelock with 10 min timeout") ) {
acquire(Constants.BATTERY_SAVER_WATCHER_WAKE_LOCK_TIMEOUT) killSwitchJob?.cancel()
} finally { val allowedIps =
release() if (settings.isLanOnKillSwitchEnabled) TunnelConf.LAN_BYPASS_ALLOWED_IPS
} else emptyList()
} tunnelManager.setBackendState(BackendState.KILL_SWITCH_ACTIVE, allowedIps)
} }
} }
}
private suspend fun buildNetworkState(connectivityState: ConnectivityState): NetworkState { private fun launchWatcherNotification(
return with(autoTunnelStateFlow.value.networkState) { description: String = getString(R.string.monitoring_state_changes)
val wifiName = when { ) {
connectivityState.wifiAvailable && val notification =
(wifiName == null || wifiName == Constants.UNREADABLE_SSID || networkMonitor.didWifiChangeSinceLastCapabilitiesQuery) -> { notificationManager.createNotification(
networkMonitor.getWifiCapabilities()?.let { getWifiName(it) } ?: wifiName WireGuardNotification.NotificationChannels.AUTO_TUNNEL,
} title = getString(R.string.auto_tunnel_title),
!connectivityState.wifiAvailable -> null description = description,
else -> wifiName actions =
} listOf(
copy( notificationManager.createNotificationAction(
isWifiConnected = connectivityState.wifiAvailable, NotificationAction.AUTO_TUNNEL_OFF
isMobileDataConnected = connectivityState.cellularAvailable, )
isEthernetConnected = isEthernetConnected, ),
wifiName = wifiName, )
) ServiceCompat.startForeground(
} this,
} NotificationManager.AUTO_TUNNEL_NOTIFICATION_ID,
notification,
Constants.SYSTEM_EXEMPT_SERVICE_TYPE_ID,
)
}
private fun startAutoTunnelStateJob() = lifecycleScope.launch(ioDispatcher) { private fun initWakeLock() {
combine( wakeLock =
combineSettings(), (getSystemService(POWER_SERVICE) as PowerManager).run {
networkMonitor.status.map { val tag = this.javaClass.name
buildNetworkState(it) newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "$tag::lock").apply {
}.distinctUntilChanged(), try {
) { double, networkState -> Timber.i("Initiating wakelock with 10 min timeout")
AutoTunnelState(tunnelManager.activeTunnels().value, networkState, double.first, double.second) acquire(Constants.BATTERY_SAVER_WATCHER_WAKE_LOCK_TIMEOUT)
}.collect { state -> } finally {
autoTunnelStateFlow.update { release()
it.copy(activeTunnels = state.activeTunnels, networkState = state.networkState, settings = state.settings, tunnels = state.tunnels) }
} }
} }
} }
private suspend fun getWifiName(wifiCapabilities: NetworkCapabilities): String? { private fun buildNetworkState(networkStatus: NetworkStatus): NetworkState {
val setting = appDataRepository.get().settings.get() return with(autoTunnelStateFlow.value.networkState) {
return if (setting.isWifiNameByShellEnabled) { val wifiName =
rootShell.get().getCurrentWifiName() when (networkStatus) {
} else { is NetworkStatus.Connected -> {
InternetConnectivityMonitor.getNetworkName(wifiCapabilities, this@AutoTunnelService) networkStatus.wifiSsid
} }
} else -> null
}
copy(
isWifiConnected = networkStatus.wifiConnected,
isMobileDataConnected = networkStatus.cellularConnected,
isEthernetConnected = networkStatus.ethernetConnected,
wifiName = wifiName,
)
}
}
private fun combineSettings(): Flow<Pair<AppSettings, Tunnels>> { @OptIn(ExperimentalCoroutinesApi::class)
return combine( private fun startAutoTunnelStateJob() =
appDataRepository.get().settings.flow, lifecycleScope.launch(ioDispatcher) {
appDataRepository.get().tunnels.flow.map { tunnels -> combine(
// isActive is ignored for equality checks so user can manually toggle off tunnel with auto-tunnel combineSettings(),
tunnels.map { it.copy(isActive = false) } appDataRepository
}, .get()
) { settings, tunnels -> .settings
Pair(settings, tunnels) .flow
}.distinctUntilChanged() .distinctUntilChanged { old, new ->
} old.isKernelEnabled == new.isKernelEnabled
} // Only emit when isKernelEnabled changes
.flatMapLatest {
networkMonitor.networkStatusFlow.flowOn(ioDispatcher).map {
buildNetworkState(it)
}
}
.distinctUntilChanged(),
) { double, networkState ->
AutoTunnelState(
tunnelManager.activeTunnels.value,
networkState,
double.first,
double.second,
)
}
.collect { state ->
autoTunnelStateFlow.update {
it.copy(
activeTunnels = state.activeTunnels,
networkState = state.networkState,
settings = state.settings,
tunnels = state.tunnels,
)
}
}
}
private fun startKillSwitchJob() = lifecycleScope.launch(ioDispatcher) { private fun combineSettings(): Flow<Pair<AppSettings, Tunnels>> {
autoTunnelStateFlow.collect { return combine(
if (it == defaultState) return@collect appDataRepository.get().settings.flow,
when (val event = it.asKillSwitchEvent()) { appDataRepository.get().tunnels.flow.map { tunnels ->
KillSwitchEvent.DoNothing -> Unit // isActive is ignored for equality checks so user can manually toggle off
is KillSwitchEvent.Start -> { // tunnel with auto-tunnel
Timber.d("Starting kill switch") tunnels.map { it.copy(isActive = false) }
tunnelManager.setBackendState(BackendState.KILL_SWITCH_ACTIVE, event.allowedIps) },
} ) { settings, tunnels ->
KillSwitchEvent.Stop -> { Pair(settings, tunnels)
Timber.d("Stopping kill switch") }
tunnelManager.setBackendState(BackendState.SERVICE_ACTIVE, emptySet()) .distinctUntilChanged()
} }
}
}
}
@OptIn(FlowPreview::class) private fun startKillSwitchJob() =
private fun startAutoTunnelJob() = lifecycleScope.launch(ioDispatcher) { lifecycleScope.launch(ioDispatcher) {
Timber.i("Starting auto-tunnel network event watcher") autoTunnelStateFlow.collect {
val settings = appDataRepository.get().settings.get() if (it == defaultState) return@collect
Timber.d("Starting with debounce delay of: ${settings.debounceDelaySeconds} seconds") when (val event = it.asKillSwitchEvent()) {
autoTunnelStateFlow.debounce(settings.debounceDelayMillis()).collect { watcherState -> KillSwitchEvent.DoNothing -> Unit
if (watcherState == defaultState) return@collect is KillSwitchEvent.Start -> {
Timber.d("New auto tunnel state emitted") Timber.d("Starting kill switch")
when (val event = watcherState.asAutoTunnelEvent()) { tunnelManager.setBackendState(
is AutoTunnelEvent.Start -> (event.tunnelConf ?: appDataRepository.get().getPrimaryOrFirstTunnel())?.let { BackendState.KILL_SWITCH_ACTIVE,
tunnelManager.startTunnel(it) event.allowedIps,
} )
// TODO improve this to target specific tunnels to better support multi-tunnel }
is AutoTunnelEvent.Stop -> tunnelManager.stopTunnel() KillSwitchEvent.Stop -> {
AutoTunnelEvent.DoNothing -> Timber.i("Auto-tunneling: no condition met") Timber.d("Stopping kill switch")
} tunnelManager.setBackendState(BackendState.SERVICE_ACTIVE, emptySet())
} }
} }
}
}
@OptIn(FlowPreview::class)
private fun startAutoTunnelJob() =
lifecycleScope.launch(ioDispatcher) {
Timber.i("Starting auto-tunnel network event watcher")
val settings = appDataRepository.get().settings.get()
Timber.d("Starting with debounce delay of: ${settings.debounceDelaySeconds} seconds")
autoTunnelStateFlow.debounce(settings.debounceDelayMillis()).collect { watcherState ->
if (watcherState == defaultState) return@collect
Timber.d("New auto tunnel state emitted ${watcherState.networkState}")
when (val event = watcherState.asAutoTunnelEvent()) {
is AutoTunnelEvent.Start ->
(event.tunnelConf ?: appDataRepository.get().getPrimaryOrFirstTunnel())
?.let { tunnelManager.startTunnel(it) }
// TODO improve this to target specific tunnels to better support multi-tunnel
is AutoTunnelEvent.Stop -> tunnelManager.stopTunnel()
AutoTunnelEvent.DoNothing -> Timber.i("Auto-tunneling: no condition met")
}
}
}
} }
@@ -4,97 +4,101 @@ import android.content.Intent
import android.os.IBinder import android.os.IBinder
import android.service.quicksettings.Tile import android.service.quicksettings.Tile
import android.service.quicksettings.TileService import android.service.quicksettings.TileService
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import androidx.lifecycle.Lifecycle
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.LifecycleRegistry
import androidx.lifecycle.lifecycleScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CompletableDeferred import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject
@AndroidEntryPoint @AndroidEntryPoint
class AutoTunnelControlTile : TileService() { class AutoTunnelControlTile : TileService(), LifecycleOwner {
@Inject @Inject lateinit var appDataRepository: AppDataRepository
lateinit var appDataRepository: AppDataRepository
@Inject @Inject lateinit var serviceManager: ServiceManager
lateinit var serviceManager: ServiceManager
@Inject private val lifecycleRegistry: LifecycleRegistry = LifecycleRegistry(this)
@ApplicationScope
lateinit var applicationScope: CoroutineScope
override fun onCreate() { override fun onCreate() {
super.onCreate() super.onCreate()
serviceManager.autoTunnelTile.complete(this) lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_CREATE)
} }
override fun onDestroy() { override fun onDestroy() {
super.onDestroy() super.onDestroy()
serviceManager.autoTunnelTile = CompletableDeferred() lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_DESTROY)
} }
override fun onStartListening() { override fun onStartListening() {
super.onStartListening() super.onStartListening()
serviceManager.autoTunnelTile.complete(this) lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_START)
applicationScope.launch { Timber.d("Start listening called for auto tunnel tile")
if (appDataRepository.tunnels.getAll().isEmpty()) return@launch setUnavailable() lifecycleScope.launch {
updateTileState() serviceManager.autoTunnelService.collect {
} if (it != null) return@collect setActive()
} setInactive()
}
}
lifecycleScope.launch {
appDataRepository.tunnels.flow.collect {
if (it.isEmpty()) {
setUnavailable()
}
}
}
}
fun updateTileState() { override fun onClick() {
serviceManager.autoTunnelActive.value.let { super.onClick()
if (it) setActive() else setInactive() unlockAndRun {
} lifecycleScope.launch {
} if (serviceManager.autoTunnelService.value != null) {
serviceManager.stopAutoTunnel()
setInactive()
} else {
serviceManager.startAutoTunnel()
setActive()
}
}
}
}
override fun onClick() { private fun setActive() {
super.onClick() runCatching {
unlockAndRun { qsTile.state = Tile.STATE_ACTIVE
applicationScope.launch { qsTile.updateTile()
if (serviceManager.autoTunnelActive.value) { }
serviceManager.stopAutoTunnel() }
setInactive()
} else {
serviceManager.startAutoTunnel(true)
setActive()
}
}
}
}
private fun setActive() { private fun setInactive() {
runCatching { runCatching {
qsTile.state = Tile.STATE_ACTIVE qsTile.state = Tile.STATE_INACTIVE
qsTile.updateTile() qsTile.updateTile()
} }
} }
private fun setInactive() { /* This works around an annoying unsolved frameworks bug some people are hitting. */
runCatching { override fun onBind(intent: Intent): IBinder? {
qsTile.state = Tile.STATE_INACTIVE var ret: IBinder? = null
qsTile.updateTile() try {
} ret = super.onBind(intent)
} } catch (_: Throwable) {
Timber.e("Failed to bind to TunnelControlTile")
}
return ret
}
/* This works around an annoying unsolved frameworks bug some people are hitting. */ private fun setUnavailable() {
override fun onBind(intent: Intent): IBinder? { runCatching {
var ret: IBinder? = null qsTile.state = Tile.STATE_UNAVAILABLE
try { qsTile.updateTile()
ret = super.onBind(intent) }
} catch (_: Throwable) { }
Timber.e("Failed to bind to TunnelControlTile")
}
return ret
}
private fun setUnavailable() { override val lifecycle: Lifecycle
runCatching { get() = lifecycleRegistry
qsTile.state = Tile.STATE_UNAVAILABLE
qsTile.updateTile()
}
}
} }
@@ -5,126 +5,180 @@ import android.os.Build
import android.os.IBinder import android.os.IBinder
import android.service.quicksettings.Tile import android.service.quicksettings.Tile
import android.service.quicksettings.TileService import android.service.quicksettings.TileService
import androidx.lifecycle.Lifecycle
import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.LifecycleRegistry
import androidx.lifecycle.lifecycleScope
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.WireGuardAutoTunnel
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import kotlinx.coroutines.CompletableDeferred import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject
@AndroidEntryPoint @AndroidEntryPoint
class TunnelControlTile : TileService() { class TunnelControlTile : TileService(), LifecycleOwner {
@Inject @Inject lateinit var appDataRepository: AppDataRepository
lateinit var appDataRepository: AppDataRepository
@Inject @Inject lateinit var serviceManager: ServiceManager
@ApplicationScope
lateinit var applicationScope: CoroutineScope
@Inject @Inject lateinit var tunnelManager: TunnelManager
lateinit var serviceManager: ServiceManager
@Inject private val lifecycleRegistry: LifecycleRegistry = LifecycleRegistry(this)
lateinit var tunnelManager: TunnelManager
override fun onCreate() { private var isCollecting = false
super.onCreate()
serviceManager.tunnelControlTile.complete(this)
}
override fun onDestroy() { override fun onCreate() {
super.onDestroy() super.onCreate()
serviceManager.tunnelControlTile = CompletableDeferred() lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_CREATE)
} }
override fun onStartListening() { override fun onDestroy() {
super.onStartListening() super.onDestroy()
Timber.d("Start listening called") lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_DESTROY)
serviceManager.tunnelControlTile.complete(this) }
applicationScope.launch {
updateTileState()
}
}
fun updateTileState() = applicationScope.launch { override fun onStartListening() {
if (appDataRepository.tunnels.getAll().isEmpty()) return@launch setUnavailable() super.onStartListening()
with(tunnelManager.activeTunnels().value) { lifecycleRegistry.handleLifecycleEvent(Lifecycle.Event.ON_START)
if (isNotEmpty()) return@launch updateTile(if (size == 1) first().tunName else getString(R.string.multiple), true) Timber.d("Start listening called for tunnel tile")
} if (isCollecting) return
appDataRepository.getStartTunnelConfig()?.let { isCollecting = true
updateTile(it.tunName, false) lifecycleScope.launch { tunnelManager.activeTunnels.collect { updateTileState() } }
} }
}
override fun onClick() { private suspend fun updateTileState() {
super.onClick() try {
unlockAndRun { val tunnels = appDataRepository.tunnels.getAll()
applicationScope.launch { if (tunnels.isEmpty()) {
if (tunnelManager.activeTunnels().value.isNotEmpty()) return@launch tunnelManager.stopTunnel() setUnavailable()
appDataRepository.getStartTunnelConfig()?.let { return
tunnelManager.startTunnel(it) }
}
}
}
}
private fun setActive() { val activeTunnels =
runCatching { tunnelManager.activeTunnels.value.filter { it.value.status.isUpOrStarting() }
qsTile.state = Tile.STATE_ACTIVE
qsTile.updateTile()
}
}
private fun setInactive() { when {
runCatching { activeTunnels.isNotEmpty() -> {
qsTile.state = Tile.STATE_INACTIVE val activeIds = activeTunnels.map { it.key.id }
qsTile.updateTile() // TODO improvements would be needed to make this work well with toggling
} // multiple tunnels
} // this would be better managed elsewhere
WireGuardAutoTunnel.setLastActiveTunnels(activeIds)
updateTileForActiveTunnels(activeTunnels)
}
else -> updateTileForLastActiveTunnels()
}
} catch (e: Exception) {
setUnavailable()
}
}
private fun setUnavailable() { private fun updateTileForActiveTunnels(activeTunnels: Map<TunnelConf, TunnelState>) {
runCatching { val tileName =
qsTile.state = Tile.STATE_UNAVAILABLE when (activeTunnels.size) {
setTileDescription("") 1 -> activeTunnels.keys.first().tunName
qsTile.updateTile() else -> getString(R.string.multiple)
} }
} updateTile(tileName, true)
}
private fun setTileDescription(description: String) { private suspend fun updateTileForLastActiveTunnels() {
runCatching { val lastActiveIds = WireGuardAutoTunnel.getLastActiveTunnels()
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { when {
qsTile.subtitle = description lastActiveIds.isEmpty() -> {
} appDataRepository.getStartTunnelConfig()?.let { config ->
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { updateTile(config.tunName, false)
qsTile.stateDescription = description } ?: setUnavailable()
} }
qsTile.updateTile() lastActiveIds.size > 1 -> updateTile(getString(R.string.multiple), false)
} else -> {
} val tunnelId = lastActiveIds.first()
appDataRepository.tunnels.getById(tunnelId)?.let { tunnel ->
updateTile(tunnel.tunName, false)
} ?: setUnavailable()
}
}
}
/* This works around an annoying unsolved frameworks bug some people are hitting. */ override fun onClick() {
override fun onBind(intent: Intent): IBinder? { super.onClick()
var ret: IBinder? = null unlockAndRun {
try { lifecycleScope.launch {
ret = super.onBind(intent) if (tunnelManager.activeTunnels.value.isNotEmpty())
} catch (_: Throwable) { return@launch tunnelManager.stopTunnel()
Timber.e("Failed to bind to TunnelControlTile") val lastActive = WireGuardAutoTunnel.getLastActiveTunnels()
} if (lastActive.isEmpty()) {
return ret appDataRepository.getStartTunnelConfig()?.let { tunnelManager.startTunnel(it) }
} } else {
lastActive.forEach { id ->
appDataRepository.tunnels.getById(id)?.let { tunnelManager.startTunnel(it) }
}
}
}
}
}
private fun updateTile(name: String, active: Boolean) { private fun setActive() {
runCatching { runCatching {
setTileDescription(name) qsTile.state = Tile.STATE_ACTIVE
if (active) return setActive() qsTile.updateTile()
setInactive() }
}.onFailure { }
Timber.e(it)
} private fun setInactive() {
} runCatching {
qsTile.state = Tile.STATE_INACTIVE
qsTile.updateTile()
}
}
private fun setUnavailable() {
runCatching {
qsTile.state = Tile.STATE_UNAVAILABLE
setTileDescription("")
qsTile.updateTile()
}
}
private fun setTileDescription(description: String) {
runCatching {
if (qsTile == null) return@runCatching
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
qsTile.subtitle = description
qsTile.stateDescription = description
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
qsTile.subtitle = description
}
qsTile.updateTile()
}
}
/* This works around an annoying unsolved frameworks bug some people are hitting. */
override fun onBind(intent: Intent): IBinder? {
var ret: IBinder? = null
try {
ret = super.onBind(intent)
} catch (_: Throwable) {
Timber.e("Failed to bind to TunnelControlTile")
}
return ret
}
private fun updateTile(name: String, active: Boolean) {
runCatching {
setTileDescription(name)
if (active) return setActive()
setInactive()
}
.onFailure { Timber.e(it) }
}
override val lifecycle: Lifecycle
get() = lifecycleRegistry
} }
@@ -10,70 +10,83 @@ import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
class DynamicShortcutManager(private val context: Context, @IoDispatcher private val ioDispatcher: CoroutineDispatcher) : ShortcutManager { class DynamicShortcutManager(
override suspend fun addShortcuts() { private val context: Context,
withContext(ioDispatcher) { @IoDispatcher private val ioDispatcher: CoroutineDispatcher,
ShortcutManagerCompat.setDynamicShortcuts(context, createShortcuts()) ) : ShortcutManager {
} override suspend fun addShortcuts() {
} withContext(ioDispatcher) {
ShortcutManagerCompat.setDynamicShortcuts(context, createShortcuts())
}
}
override suspend fun removeShortcuts() { override suspend fun removeShortcuts() {
withContext(ioDispatcher) { withContext(ioDispatcher) {
ShortcutManagerCompat.removeDynamicShortcuts(context, createShortcuts().map { it.id }) ShortcutManagerCompat.removeDynamicShortcuts(context, createShortcuts().map { it.id })
} }
} }
private fun createShortcuts(): List<ShortcutInfoCompat> { private fun createShortcuts(): List<ShortcutInfoCompat> {
return listOf( return listOf(
buildShortcut( buildShortcut(
context.getString(R.string.vpn_off), context.getString(R.string.vpn_off),
context.getString(R.string.vpn_off), context.getString(R.string.vpn_off),
context.getString(R.string.vpn_off), context.getString(R.string.vpn_off),
intent = Intent(context, ShortcutsActivity::class.java).apply { intent =
putExtra("className", "WireGuardTunnelService") Intent(context, ShortcutsActivity::class.java).apply {
action = ShortcutsActivity.Action.STOP.name putExtra("className", "WireGuardTunnelService")
}, action = ShortcutsActivity.Action.STOP.name
shortcutIcon = R.drawable.vpn_off, },
), shortcutIcon = R.drawable.vpn_off,
buildShortcut( ),
context.getString(R.string.vpn_on), buildShortcut(
context.getString(R.string.vpn_on), context.getString(R.string.vpn_on),
context.getString(R.string.vpn_on), context.getString(R.string.vpn_on),
intent = Intent(context, ShortcutsActivity::class.java).apply { context.getString(R.string.vpn_on),
putExtra("className", "WireGuardTunnelService") intent =
action = ShortcutsActivity.Action.START.name Intent(context, ShortcutsActivity::class.java).apply {
}, putExtra("className", "WireGuardTunnelService")
shortcutIcon = R.drawable.vpn_on, action = ShortcutsActivity.Action.START.name
), },
buildShortcut( shortcutIcon = R.drawable.vpn_on,
context.getString(R.string.start_auto), ),
context.getString(R.string.start_auto), buildShortcut(
context.getString(R.string.start_auto), context.getString(R.string.start_auto),
intent = Intent(context, ShortcutsActivity::class.java).apply { context.getString(R.string.start_auto),
putExtra("className", "WireGuardConnectivityWatcherService") context.getString(R.string.start_auto),
action = ShortcutsActivity.Action.START.name intent =
}, Intent(context, ShortcutsActivity::class.java).apply {
shortcutIcon = R.drawable.auto_play, putExtra("className", "WireGuardConnectivityWatcherService")
), action = ShortcutsActivity.Action.START.name
buildShortcut( },
context.getString(R.string.stop_auto), shortcutIcon = R.drawable.auto_play,
context.getString(R.string.stop_auto), ),
context.getString(R.string.stop_auto), buildShortcut(
intent = Intent(context, ShortcutsActivity::class.java).apply { context.getString(R.string.stop_auto),
putExtra("className", "WireGuardConnectivityWatcherService") context.getString(R.string.stop_auto),
action = ShortcutsActivity.Action.STOP.name context.getString(R.string.stop_auto),
}, intent =
shortcutIcon = R.drawable.auto_pause, Intent(context, ShortcutsActivity::class.java).apply {
), putExtra("className", "WireGuardConnectivityWatcherService")
) action = ShortcutsActivity.Action.STOP.name
} },
shortcutIcon = R.drawable.auto_pause,
),
)
}
private fun buildShortcut(id: String, shortLabel: String, longLabel: String, intent: Intent, shortcutIcon: Int): ShortcutInfoCompat { private fun buildShortcut(
return ShortcutInfoCompat.Builder(context, id) id: String,
.setShortLabel(shortLabel) shortLabel: String,
.setLongLabel(longLabel) longLabel: String,
.setIntent(intent) intent: Intent,
.setIcon(IconCompat.createWithResource(context, shortcutIcon)) shortcutIcon: Int,
.build() ): ShortcutInfoCompat {
} return ShortcutInfoCompat.Builder(context, id)
.setShortLabel(shortLabel)
.setLongLabel(longLabel)
.setIntent(intent)
.setIcon(IconCompat.createWithResource(context, shortcutIcon))
.build()
}
} }
@@ -1,6 +1,7 @@
package com.zaneschepke.wireguardautotunnel.core.shortcut package com.zaneschepke.wireguardautotunnel.core.shortcut
interface ShortcutManager { interface ShortcutManager {
suspend fun addShortcuts() suspend fun addShortcuts()
suspend fun removeShortcuts()
suspend fun removeShortcuts()
} }
@@ -2,72 +2,75 @@ package com.zaneschepke.wireguardautotunnel.core.shortcut
import android.os.Bundle import android.os.Bundle
import androidx.activity.ComponentActivity import androidx.activity.ComponentActivity
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import dagger.hilt.android.AndroidEntryPoint import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject
@AndroidEntryPoint @AndroidEntryPoint
class ShortcutsActivity : ComponentActivity() { class ShortcutsActivity : ComponentActivity() {
@Inject @Inject lateinit var appDataRepository: AppDataRepository
lateinit var appDataRepository: AppDataRepository
@Inject @Inject lateinit var serviceManager: ServiceManager
lateinit var serviceManager: ServiceManager
@Inject @Inject lateinit var tunnelManager: TunnelManager
@ApplicationScope
lateinit var applicationScope: CoroutineScope
override fun onCreate(savedInstanceState: Bundle?) { @Inject @ApplicationScope lateinit var applicationScope: CoroutineScope
super.onCreate(savedInstanceState)
applicationScope.launch {
val settings = appDataRepository.settings.get()
if (settings.isShortcutsEnabled) {
when (intent.getStringExtra(CLASS_NAME_EXTRA_KEY)) {
LEGACY_TUNNEL_SERVICE_NAME, TunnelProvider::class.java.simpleName -> {
val tunnelName = intent.getStringExtra(TUNNEL_NAME_EXTRA_KEY)
Timber.d("Tunnel name extra: $tunnelName")
val tunnelConfig = tunnelName?.let {
appDataRepository.tunnels.getAll()
.firstOrNull { it.tunName == tunnelName }
} ?: appDataRepository.getStartTunnelConfig()
Timber.d("Shortcut action on name: ${tunnelConfig?.tunName}")
// tunnelConfig?.let {
// when (intent.action) {
// Action.START.name -> tunnelService.get().startTunnel(it)
// Action.STOP.name -> tunnelService.get().stopTunnel()
// else -> Unit
// }
// }
}
AutoTunnelService::class.java.simpleName, LEGACY_AUTO_TUNNEL_SERVICE_NAME -> {
when (intent.action) {
Action.START.name -> serviceManager.startAutoTunnel(true)
Action.STOP.name -> serviceManager.stopAutoTunnel()
}
}
}
}
}
finish()
}
enum class Action { override fun onCreate(savedInstanceState: Bundle?) {
START, super.onCreate(savedInstanceState)
STOP, applicationScope.launch {
} val settings = appDataRepository.settings.get()
if (settings.isShortcutsEnabled) {
when (intent.getStringExtra(CLASS_NAME_EXTRA_KEY)) {
LEGACY_TUNNEL_SERVICE_NAME,
TunnelProvider::class.java.simpleName -> {
val tunnelName = intent.getStringExtra(TUNNEL_NAME_EXTRA_KEY)
Timber.d("Tunnel name extra: $tunnelName")
val tunnelConfig =
tunnelName?.let {
appDataRepository.tunnels.getAll().firstOrNull {
it.tunName == tunnelName
}
} ?: appDataRepository.getStartTunnelConfig()
Timber.d("Shortcut action on name: ${tunnelConfig?.tunName}")
tunnelConfig?.let {
when (intent.action) {
Action.START.name -> tunnelManager.startTunnel(it)
Action.STOP.name -> tunnelManager.stopTunnel()
else -> Unit
}
}
}
AutoTunnelService::class.java.simpleName,
LEGACY_AUTO_TUNNEL_SERVICE_NAME -> {
when (intent.action) {
Action.START.name -> serviceManager.startAutoTunnel()
Action.STOP.name -> serviceManager.stopAutoTunnel()
}
}
}
}
}
finish()
}
companion object { enum class Action {
const val LEGACY_TUNNEL_SERVICE_NAME = "WireGuardTunnelService" START,
const val LEGACY_AUTO_TUNNEL_SERVICE_NAME = "WireGuardConnectivityWatcherService" STOP,
const val TUNNEL_NAME_EXTRA_KEY = "tunnelName" }
const val CLASS_NAME_EXTRA_KEY = "className"
} companion object {
const val LEGACY_TUNNEL_SERVICE_NAME = "WireGuardTunnelService"
const val LEGACY_AUTO_TUNNEL_SERVICE_NAME = "WireGuardConnectivityWatcherService"
const val TUNNEL_NAME_EXTRA_KEY = "tunnelName"
const val CLASS_NAME_EXTRA_KEY = "className"
}
} }
@@ -1,219 +1,233 @@
package com.zaneschepke.wireguardautotunnel.core.tunnel package com.zaneschepke.wireguardautotunnel.core.tunnel
import com.zaneschepke.wireguardautotunnel.R import com.wireguard.android.backend.Tunnel
import com.zaneschepke.wireguardautotunnel.WireGuardAutoTunnel
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.core.notification.WireGuardNotification
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider.Companion.CHECK_INTERVAL
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendError import com.zaneschepke.wireguardautotunnel.domain.enums.BackendError
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarController import com.zaneschepke.wireguardautotunnel.util.extensions.asTunnelState
import com.zaneschepke.wireguardautotunnel.util.Constants import java.util.concurrent.ConcurrentHashMap
import com.zaneschepke.wireguardautotunnel.util.StringValue import kotlin.concurrent.thread
import com.zaneschepke.wireguardautotunnel.util.extensions.cancelWithMessage
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Job
import kotlinx.coroutines.coroutineScope
import kotlinx.coroutines.delay import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.*
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.asStateFlow
import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.update
import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock
import timber.log.Timber import timber.log.Timber
import java.util.concurrent.atomic.AtomicBoolean
open class BaseTunnel( abstract class BaseTunnel(
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @ApplicationScope private val applicationScope: CoroutineScope,
@ApplicationScope private val applicationScope: CoroutineScope, private val appDataRepository: AppDataRepository,
private val networkMonitor: NetworkMonitor, private val serviceManager: ServiceManager,
private val appDataRepository: AppDataRepository,
private val serviceManager: ServiceManager,
private val notificationManager: NotificationManager,
) : TunnelProvider { ) : TunnelProvider {
internal val tunnels = MutableStateFlow<List<TunnelConf>>(emptyList()) private val _errorEvents =
MutableSharedFlow<Pair<TunnelConf, BackendError>>(replay = 0, extraBufferCapacity = 1)
override val errorEvents = _errorEvents.asSharedFlow()
private val tunnelJobs = mutableMapOf<TunnelConf, Job>() private val activeTuns = MutableStateFlow<Map<TunnelConf, TunnelState>>(emptyMap())
private val tunThreads = ConcurrentHashMap<Int, Thread>()
override val activeTunnels = activeTuns.asStateFlow()
private val isNetworkAvailable = AtomicBoolean(false) private val tunMutex = Mutex()
private val tunStatusMutex = Mutex()
private val bounceTunnelMutex = Mutex()
init { override val bouncingTunnelIds = ConcurrentHashMap<Int, TunnelStatus.StopReason>()
applicationScope.launch(ioDispatcher) {
launch {
startNetworkJob()
}
tunnels.collect { tuns ->
val previousTuns = tunnelJobs.keys.toSet()
val newTuns = tuns - previousTuns
val removedItems = previousTuns - tuns.toSet()
newTuns.forEach { tun -> abstract suspend fun startBackend(tunnel: TunnelConf)
Timber.d("Starting tunnel jobs for tun ${tun.name}")
tunnelJobs[tun] = startTunnelJobs(tun)
}
removedItems.forEach { tun -> abstract fun stopBackend(tunnel: TunnelConf)
tunnelJobs[tun]?.cancelWithMessage("Canceling tunnel jobs for tunnel: ${tun.name}")
tunnelJobs.remove(tun)
}
serviceManager.updateTunnelTile()
}
}
}
private fun startTunnelJobs(tunnel: TunnelConf) = applicationScope.launch(ioDispatcher) { override fun hasVpnPermission(): Boolean {
launch { return serviceManager.hasVpnPermission()
startTunnelStatisticsJob(tunnel) }
}
launch {
startPingJob(tunnel)
}
launch {
startTunnelConfigChangeJob(tunnel)
}
}
override suspend fun bounceTunnel(tunnelConf: TunnelConf) { protected suspend fun updateTunnelStatus(
if (tunnels.value.any { it.id == tunnelConf.id }) { tunnelConf: TunnelConf,
toggleTunnel(tunnelConf, TunnelStatus.DOWN) status: TunnelStatus? = null,
toggleTunnel(tunnelConf, TunnelStatus.UP) stats: TunnelStatistics? = null,
} ) {
} tunStatusMutex.withLock {
activeTuns.update { currentTuns ->
val originalConf = currentTuns.getKeyById(tunnelConf.id) ?: tunnelConf
val existingState = currentTuns.getValueById(tunnelConf.id) ?: TunnelState()
val newState = status ?: existingState.status
if (newState == TunnelStatus.Down) {
Timber.d("Removing tunnel ${tunnelConf.id} from activeTunnels as state is DOWN")
cleanUpTunThread(tunnelConf)
currentTuns - originalConf
} else if (existingState.status == newState && stats == null) {
Timber.d("Skipping redundant state update for ${tunnelConf.id}: $newState")
currentTuns
} else {
val updated =
existingState.copy(
status = newState,
statistics = stats ?: existingState.statistics,
)
currentTuns + (originalConf to updated)
}
}
}
}
override suspend fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) { private suspend fun stopActiveTunnels() {
} activeTunnels.value.forEach { (config, state) ->
if (state.status.isUpOrStarting()) {
stopTunnel(config)
}
}
}
override suspend fun runningTunnelNames(): Set<String> { private fun configureTunnelCallbacks(tunnelConf: TunnelConf) {
return emptySet() Timber.d("Configuring TunnelConf instance: ${tunnelConf.hashCode()}")
} tunnelConf.setStateChangeCallback { state ->
applicationScope.launch {
Timber.d(
"State change callback triggered for tunnel ${tunnelConf.id}: ${tunnelConf.tunName} with state $state at ${System.currentTimeMillis()}"
)
when (state) {
is Tunnel.State -> updateTunnelStatus(tunnelConf, state.asTunnelState())
is org.amnezia.awg.backend.Tunnel.State ->
updateTunnelStatus(tunnelConf, state.asTunnelState())
}
handleServiceStateOnChange()
}
serviceManager.updateTunnelTile()
}
}
override suspend fun activeTunnels(): StateFlow<List<TunnelConf>> { override suspend fun updateTunnelStatistics(tunnel: TunnelConf) {
return tunnels.asStateFlow() val stats = getStatistics(tunnel)
} updateTunnelStatus(tunnel, null, stats)
}
override suspend fun startTunnel(tunnelConf: TunnelConf) { override suspend fun startTunnel(tunnelConf: TunnelConf) {
if (tunnels.value.any { it.id == tunnelConf.id }) return Timber.w("Tunnel already running") if (activeTuns.exists(tunnelConf.id) || tunThreads.containsKey(tunnelConf.id)) return
serviceManager.startBackgroundService(tunnelConf) if (this@BaseTunnel is UserspaceTunnel) stopActiveTunnels()
appDataRepository.tunnels.save(tunnelConf.copy(isActive = true)) tunMutex.withLock {
} tunThreads[tunnelConf.id] = thread {
runBlocking {
try {
Timber.d("Starting tunnel ${tunnelConf.id}...")
startTunnelInner(tunnelConf)
Timber.d("Started complete for tunnel ${tunnelConf.name}...")
} catch (e: InterruptedException) {
Timber.w(
"Tunnel start has been interrupted as ${tunnelConf.name} failed to start"
)
}
}
}
}
}
override suspend fun stopTunnel(tunnelConf: TunnelConf?) { private suspend fun startTunnelInner(tunnelConf: TunnelConf) {
} configureTunnelCallbacks(tunnelConf)
Timber.d("Starting backend for tunnel ${tunnelConf.id}...")
try {
startBackend(tunnelConf)
updateTunnelStatus(tunnelConf, TunnelStatus.Up)
Timber.d("Started for tun ${tunnelConf.id}...")
saveTunnelActiveState(tunnelConf, true)
serviceManager.startTunnelForegroundService()
} catch (e: BackendError) {
Timber.e(e, "Failed to start backend for ${tunnelConf.name}")
_errorEvents.emit(tunnelConf to e)
updateTunnelStatus(tunnelConf, TunnelStatus.Down)
}
}
open suspend fun toggleTunnel(tunnelConf: TunnelConf, state: TunnelStatus) { private suspend fun saveTunnelActiveState(tunnelConf: TunnelConf, active: Boolean) {
} val tunnelCopy = tunnelConf.copyWithCallback(isActive = active)
appDataRepository.tunnels.save(tunnelCopy)
}
open suspend fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics { override suspend fun stopTunnel(tunnelConf: TunnelConf?, reason: TunnelStatus.StopReason) {
throw NotImplementedError("Get statistics not implemented in base class") if (tunnelConf == null) return stopActiveTunnels()
} tunMutex.withLock {
if (activeTuns.isStarting(tunnelConf.id))
return handleStuckStartingTunnelShutdown(tunnelConf)
updateTunnelStatus(tunnelConf, TunnelStatus.Stopping(reason))
stopTunnelInner(tunnelConf)
}
}
internal suspend fun onTunnelStop(tunnelConf: TunnelConf) { private suspend fun stopTunnelInner(tunnelConf: TunnelConf) {
appDataRepository.tunnels.save(tunnelConf.copy(isActive = false)) try {
removeFromActiveTunnels(tunnelConf) val tunnel = activeTuns.findTunnel(tunnelConf.id) ?: return
if (tunnels.value.isEmpty()) serviceManager.stopBackgroundService() stopBackend(tunnel)
} saveTunnelActiveState(tunnelConf, false)
removeActiveTunnel(tunnel)
} catch (e: BackendError) {
Timber.e(e, "Failed to stop tunnel ${tunnelConf.id}")
_errorEvents.emit(tunnelConf to e)
updateTunnelStatus(tunnelConf, TunnelStatus.Down)
}
}
internal suspend fun stopAllTunnels() { private fun handleServiceStateOnChange() {
tunnels.value.forEach { if (activeTuns.value.isEmpty() && bouncingTunnelIds.isEmpty())
stopTunnel(it) serviceManager.stopTunnelForegroundService()
} }
}
internal fun addToActiveTunnels(conf: TunnelConf) { private suspend fun handleStuckStartingTunnelShutdown(tunnel: TunnelConf) {
tunnels.update { Timber.d("Stuck in starting state so shutting down tunnel thread for tunnel ${tunnel.name}")
it.toMutableList().apply { try {
add(conf) tunThreads[tunnel.id]?.let {
} if (it.state != Thread.State.TERMINATED) {
} it.interrupt()
} } else {
Timber.d("Thread already terminated")
}
}
} catch (e: Exception) {
Timber.e(e, "Failed to stop tunnel thread for ${tunnel.name}")
} finally {
updateTunnelStatus(tunnel, TunnelStatus.Down)
}
}
private fun removeFromActiveTunnels(conf: TunnelConf) { private fun cleanUpTunThread(tunnel: TunnelConf) {
tunnels.update { Timber.d("Removing thread for ${tunnel.name}")
it.toMutableList().apply { tunThreads -= tunnel.id
remove(conf) }
}
}
}
private suspend fun startNetworkJob() = coroutineScope { private fun removeActiveTunnel(tunnelConf: TunnelConf) {
networkMonitor.status.distinctUntilChanged().collect { activeTuns.update { current -> current.toMutableMap().apply { remove(tunnelConf) } }
isNetworkAvailable.set(!it.allOffline) }
}
}
private suspend fun startPingJob(tunnel: TunnelConf) = coroutineScope { override suspend fun bounceTunnel(tunnelConf: TunnelConf, reason: TunnelStatus.StopReason) {
while (isActive) { bounceTunnelMutex.withLock {
if (isNetworkAvailable.get() && tunnel.isActive) { Timber.i(
val pingResult = tunnel.pingTunnel(ioDispatcher) "Bounce tunnel ${tunnelConf.name} for reason: $reason, current bouncing: ${bouncingTunnelIds.size}"
handlePingResult(tunnel, pingResult) )
} bouncingTunnelIds[tunnelConf.id] = reason
delay(CHECK_INTERVAL) try {
} stopTunnel(tunnelConf, reason)
} delay(BOUNCE_DELAY)
startTunnel(tunnelConf)
} finally {
bouncingTunnelIds.remove(tunnelConf.id)
handleServiceStateOnChange()
Timber.d(
"Cleared bounce state for ${tunnelConf.name}, remaining: ${bouncingTunnelIds.size}"
)
}
}
}
private suspend fun handlePingResult(tunnel: TunnelConf, pingResult: List<Boolean>) { override suspend fun runningTunnelNames(): Set<String> =
if (pingResult.contains(false)) { activeTuns.value.keys.map { it.tunName }.toSet()
if (isNetworkAvailable.get()) {
Timber.i("Ping result: target was not reachable, bouncing the tunnel")
bounceTunnel(tunnel)
delay(tunnel.pingCooldown ?: Constants.PING_COOLDOWN)
} else {
Timber.i("Ping result: target was not reachable, but no network available")
}
} else {
Timber.i("Ping result: all ping targets were reached successfully")
}
}
internal fun handleBackendThrowable(backendError: BackendError) { companion object {
val message = when (backendError) { const val BOUNCE_DELAY = 300L
BackendError.Config -> StringValue.StringResource(R.string.start_failed_config) }
BackendError.DNS -> StringValue.StringResource(R.string.dns_error)
BackendError.Unauthorized -> StringValue.StringResource(R.string.unauthorized)
}
if (WireGuardAutoTunnel.isForeground()) {
SnackbarController.showMessage(message)
} else {
notificationManager.show(
NotificationManager.VPN_NOTIFICATION_ID,
notificationManager.createNotification(
WireGuardNotification.NotificationChannels.VPN,
title = StringValue.StringResource(R.string.tunne_start_failed_title),
description = message,
),
)
}
}
private suspend fun startTunnelConfigChangeJob(tunnel: TunnelConf) = coroutineScope {
appDataRepository.tunnels.flow.collect { storageTuns ->
storageTuns.firstOrNull { it.id == tunnel.id }?.let { storageTun ->
if (tunnel.isQuickConfigChanged(storageTun) || tunnel.isPingConfigMatching(storageTun)) {
bounceTunnel(tunnel)
}
}
}
}
private suspend fun startTunnelStatisticsJob(tunnel: TunnelConf) = coroutineScope {
while (isActive) {
val stats = getStatistics(tunnel)
tunnel.state.update {
it.copy(statistics = stats)
}
delay(CHECK_INTERVAL)
}
}
} }
@@ -0,0 +1,50 @@
package com.zaneschepke.wireguardautotunnel.core.tunnel
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import kotlinx.coroutines.flow.MutableStateFlow
fun Map<TunnelConf, TunnelState>.allDown(): Boolean {
return this.all { it.value.status.isDown() }
}
fun Map<TunnelConf, TunnelState>.hasActive(): Boolean {
return this.any { it.value.status.isUp() }
}
fun Map<TunnelConf, TunnelState>.getValueById(id: Int): TunnelState? {
val key = this.keys.find { it.id == id }
return key?.let { this@getValueById[it] }
}
fun Map<TunnelConf, TunnelState>.getKeyById(id: Int): TunnelConf? {
return this.keys.find { it.id == id }
}
fun Map<TunnelConf, TunnelState>.isUp(tunnelConf: TunnelConf): Boolean {
return this.getValueById(tunnelConf.id)?.status?.isUp() ?: false
}
fun MutableStateFlow<Map<TunnelConf, TunnelState>>.exists(id: Int): Boolean {
return this.value.any { it.key.id == id }
}
fun MutableStateFlow<Map<TunnelConf, TunnelState>>.isUp(id: Int): Boolean {
return this.value.any { it.key.id == id && it.value.status == TunnelStatus.Up }
}
fun MutableStateFlow<Map<TunnelConf, TunnelState>>.isStarting(id: Int): Boolean {
return this.value.any { it.key.id == id && it.value.status == TunnelStatus.Starting }
}
fun MutableStateFlow<Map<TunnelConf, TunnelState>>.findTunnel(id: Int): TunnelConf? {
return this.value.keys.find { it.id == id }
}
private val URL_PATTERN =
Regex("""^([a-zA-Z0-9]([a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}:[0-9]{1,5}$""")
fun String.isUrl(): Boolean {
return URL_PATTERN.matches(this)
}
@@ -3,11 +3,8 @@ package com.zaneschepke.wireguardautotunnel.core.tunnel
import com.wireguard.android.backend.Backend import com.wireguard.android.backend.Backend
import com.wireguard.android.backend.BackendException import com.wireguard.android.backend.BackendException
import com.wireguard.android.backend.Tunnel import com.wireguard.android.backend.Tunnel
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
@@ -15,69 +12,55 @@ import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics
import com.zaneschepke.wireguardautotunnel.domain.state.WireGuardStatistics import com.zaneschepke.wireguardautotunnel.domain.state.WireGuardStatistics
import com.zaneschepke.wireguardautotunnel.util.extensions.toBackendError import com.zaneschepke.wireguardautotunnel.util.extensions.toBackendError
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.withContext
import timber.log.Timber
import javax.inject.Inject import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope
import timber.log.Timber
class KernelTunnel @Inject constructor( class KernelTunnel
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @Inject
@ApplicationScope private val applicationScope: CoroutineScope, constructor(
serviceManager: ServiceManager, @ApplicationScope private val applicationScope: CoroutineScope,
appDataRepository: AppDataRepository, serviceManager: ServiceManager,
notificationManager: NotificationManager, appDataRepository: AppDataRepository,
private val backend: Backend, private val backend: Backend,
networkMonitor: NetworkMonitor, ) : BaseTunnel(applicationScope, appDataRepository, serviceManager) {
) : BaseTunnel(ioDispatcher, applicationScope, networkMonitor, appDataRepository, serviceManager, notificationManager) {
override suspend fun startTunnel(tunnelConf: TunnelConf) { override fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics? {
withContext(ioDispatcher) { return try {
super.startTunnel(tunnelConf) WireGuardStatistics(backend.getStatistics(tunnelConf))
runCatching { } catch (e: Exception) {
backend.setState(tunnelConf, Tunnel.State.UP, tunnelConf.toWgConfig()) Timber.e(e)
addToActiveTunnels(tunnelConf) null
}.onFailure { }
onTunnelStop(tunnelConf) }
if (it is BackendException) {
handleBackendThrowable(it.toBackendError())
} else {
Timber.e(it)
}
}
}
}
override suspend fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics { override suspend fun startBackend(tunnel: TunnelConf) {
return WireGuardStatistics(backend.getStatistics(tunnelConf)) try {
} updateTunnelStatus(tunnel, TunnelStatus.Starting)
backend.setState(tunnel, Tunnel.State.UP, tunnel.toWgConfig())
} catch (e: BackendException) {
throw e.toBackendError()
}
}
override suspend fun stopTunnel(tunnelConf: TunnelConf?) { override fun stopBackend(tunnel: TunnelConf) {
withContext(ioDispatcher) { Timber.i("Stopping tunnel ${tunnel.id} kernel")
val tunnel = tunnels.value.firstOrNull { it.id == tunnelConf?.id } try {
runCatching { backend.setState(tunnel, Tunnel.State.DOWN, tunnel.toWgConfig())
tunnel?.let { } catch (e: BackendException) {
backend.setState(it, Tunnel.State.DOWN, it.toWgConfig()) throw e.toBackendError()
onTunnelStop(it) }
} ?: stopAllTunnels() }
}.onFailure {
Timber.e(it)
}
}
}
override suspend fun toggleTunnel(tunnelConf: TunnelConf, status: TunnelStatus) { override fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) {
when (status) { Timber.w("Not yet implemented for kernel")
TunnelStatus.UP -> backend.setState(tunnelConf, Tunnel.State.UP, tunnelConf.toWgConfig()) }
TunnelStatus.DOWN -> backend.setState(tunnelConf, Tunnel.State.DOWN, tunnelConf.toWgConfig())
}
}
override suspend fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) { override fun getBackendState(): BackendState {
Timber.w("Not yet implemented for kernel") return BackendState.INACTIVE
} }
override suspend fun runningTunnelNames(): Set<String> { override suspend fun runningTunnelNames(): Set<String> {
return backend.runningTunnelNames return backend.runningTunnelNames
} }
} }
@@ -4,96 +4,122 @@ import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.di.Kernel import com.zaneschepke.wireguardautotunnel.di.Kernel
import com.zaneschepke.wireguardautotunnel.di.Userspace import com.zaneschepke.wireguardautotunnel.di.Userspace
import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendError
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.util.extensions.withData import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics
import java.util.concurrent.ConcurrentHashMap
import javax.inject.Inject
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.SharedFlow
import kotlinx.coroutines.flow.SharingStarted import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.filterNotNull import kotlinx.coroutines.flow.filterNotNull
import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.flatMapLatest
import kotlinx.coroutines.flow.stateIn import kotlinx.coroutines.flow.stateIn
import kotlinx.coroutines.launch
import kotlinx.coroutines.plus import kotlinx.coroutines.plus
import kotlinx.coroutines.withContext
import javax.inject.Inject
class TunnelManager @Inject constructor( class TunnelManager
@Kernel private val kernelTunnel: TunnelProvider, @Inject
@Userspace private val userspaceTunnel: TunnelProvider, constructor(
private val appDataRepository: AppDataRepository, @Kernel private val kernelTunnel: TunnelProvider,
@ApplicationScope private val applicationScope: CoroutineScope, @Userspace private val userspaceTunnel: TunnelProvider,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, private val appDataRepository: AppDataRepository,
@ApplicationScope private val applicationScope: CoroutineScope,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) : TunnelProvider { ) : TunnelProvider {
val appSettings: StateFlow<AppSettings?> = appDataRepository.settings.flow.stateIn( @OptIn(ExperimentalCoroutinesApi::class)
scope = applicationScope.plus(ioDispatcher), private val tunnelProviderFlow =
started = SharingStarted.Eagerly, appDataRepository.settings.flow
initialValue = null, .filterNotNull()
) .flatMapLatest { settings ->
MutableStateFlow(if (settings.isKernelEnabled) kernelTunnel else userspaceTunnel)
}
.stateIn(
scope = applicationScope.plus(ioDispatcher),
started = SharingStarted.Eagerly,
initialValue = userspaceTunnel,
)
override suspend fun activeTunnels(): StateFlow<List<TunnelConf>> { @OptIn(ExperimentalCoroutinesApi::class)
return withContext(ioDispatcher) { override val activeTunnels =
appSettings.filterNotNull().first().let { appDataRepository.settings.flow
if (it.isKernelEnabled) return@withContext kernelTunnel.activeTunnels() .filterNotNull()
userspaceTunnel.activeTunnels() .flatMapLatest { settings ->
} if (settings.isKernelEnabled) {
} kernelTunnel.activeTunnels
} } else {
userspaceTunnel.activeTunnels
}
}
.stateIn(
scope = applicationScope.plus(ioDispatcher),
started = SharingStarted.Eagerly,
initialValue = emptyMap(),
)
override suspend fun startTunnel(tunnelConf: TunnelConf) { override val errorEvents: SharedFlow<Pair<TunnelConf, BackendError>>
appSettings.withData { get() = tunnelProviderFlow.value.errorEvents
if (it.isKernelEnabled) return@withData kernelTunnel.startTunnel(tunnelConf)
userspaceTunnel.startTunnel(tunnelConf)
}
}
override suspend fun stopTunnel(tunnelConf: TunnelConf?) { override val bouncingTunnelIds: ConcurrentHashMap<Int, TunnelStatus.StopReason> =
appSettings.withData { tunnelProviderFlow.value.bouncingTunnelIds
if (it.isKernelEnabled) return@withData kernelTunnel.stopTunnel(tunnelConf)
userspaceTunnel.stopTunnel(tunnelConf)
}
}
override suspend fun bounceTunnel(tunnelConf: TunnelConf) { override fun hasVpnPermission(): Boolean {
appSettings.withData { return userspaceTunnel.hasVpnPermission()
if (it.isKernelEnabled) return@withData kernelTunnel.stopTunnel(tunnelConf) }
userspaceTunnel.stopTunnel(tunnelConf)
}
}
override suspend fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) { override suspend fun updateTunnelStatistics(tunnel: TunnelConf) {
appSettings.withData { tunnelProviderFlow.value.updateTunnelStatistics(tunnel)
if (it.isKernelEnabled) return@withData kernelTunnel.setBackendState(backendState, allowedIps) }
userspaceTunnel.setBackendState(backendState, allowedIps)
}
}
override suspend fun runningTunnelNames(): Set<String> { override suspend fun startTunnel(tunnelConf: TunnelConf) {
appSettings.filterNotNull().first().let { tunnelProviderFlow.value.startTunnel(tunnelConf)
if (it.isKernelEnabled) return kernelTunnel.runningTunnelNames() }
return userspaceTunnel.runningTunnelNames()
}
}
suspend fun restorePreviousState() { override suspend fun stopTunnel(tunnelConf: TunnelConf?, reason: TunnelStatus.StopReason) {
withContext(ioDispatcher) { tunnelProviderFlow.value.stopTunnel(tunnelConf, reason)
with(appDataRepository.settings.get()) { }
if (isRestoreOnBootEnabled) {
val previouslyActiveTuns = appDataRepository.tunnels.getActive() override suspend fun bounceTunnel(tunnelConf: TunnelConf, reason: TunnelStatus.StopReason) {
// handle kernel mode tunnelProviderFlow.value.bounceTunnel(tunnelConf, reason)
val tunsToStart = previouslyActiveTuns.filterNot { tun -> activeTunnels().value.any { tun.id == it.id } } }
if (isKernelEnabled) {
return@withContext tunsToStart.forEach { override fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) {
startTunnel(it) tunnelProviderFlow.value.setBackendState(backendState, allowedIps)
} }
}
// handle userspace override fun getBackendState(): BackendState {
if (activeTunnels().value.isEmpty()) tunsToStart.firstOrNull()?.let { startTunnel(it) } return tunnelProviderFlow.value.getBackendState()
} }
}
} override suspend fun runningTunnelNames(): Set<String> {
} return tunnelProviderFlow.value.runningTunnelNames()
}
override fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics? {
return tunnelProviderFlow.value.getStatistics(tunnelConf)
}
fun restorePreviousState() =
applicationScope.launch(ioDispatcher) {
val settings = appDataRepository.settings.get()
if (settings.isRestoreOnBootEnabled) {
val previouslyActiveTuns = appDataRepository.tunnels.getActive()
val tunsToStart =
previouslyActiveTuns.filterNot { tun ->
activeTunnels.value.any { tun.id == it.key.id }
}
if (settings.isKernelEnabled) {
return@launch tunsToStart.forEach { startTunnel(it) }
} else {
tunsToStart.firstOrNull()?.let { startTunnel(it) }
}
}
}
} }
@@ -1,24 +1,58 @@
package com.zaneschepke.wireguardautotunnel.core.tunnel package com.zaneschepke.wireguardautotunnel.core.tunnel
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendError
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics
import java.util.concurrent.ConcurrentHashMap
import kotlinx.coroutines.flow.SharedFlow
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
interface TunnelProvider { interface TunnelProvider {
/** Starts the specified tunnel configuration. */
suspend fun startTunnel(tunnelConf: TunnelConf)
suspend fun activeTunnels(): StateFlow<List<TunnelConf>> /**
* Stops the specified tunnel, or all tunnels if none is provided.
*
* @param tunnelConf The tunnel to stop, or null to stop all active tunnels.
* @param reason The reason for stopping, defaults to USER for manual stops. Callers should
* override with specific reasons (e.g., PING, CONFIG_CHANGED) when applicable.
*/
suspend fun stopTunnel(
tunnelConf: TunnelConf? = null,
reason: TunnelStatus.StopReason = TunnelStatus.StopReason.USER,
)
suspend fun startTunnel(tunnelConf: TunnelConf) /**
* Bounces (stops and restarts) the specified tunnel.
*
* @param tunnelConf The tunnel to bounce.
* @param reason The reason for bouncing, defaults to USER for manual actions. Callers should
* override with specific reasons (e.g., PING, CONFIG_CHANGED) when applicable.
*/
suspend fun bounceTunnel(
tunnelConf: TunnelConf,
reason: TunnelStatus.StopReason = TunnelStatus.StopReason.USER,
)
suspend fun stopTunnel(tunnelConf: TunnelConf? = null) fun setBackendState(backendState: BackendState, allowedIps: Collection<String>)
suspend fun bounceTunnel(tunnelConf: TunnelConf) fun getBackendState(): BackendState
suspend fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) suspend fun runningTunnelNames(): Set<String>
suspend fun runningTunnelNames(): Set<String> fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics?
companion object { val activeTunnels: StateFlow<Map<TunnelConf, TunnelState>>
const val CHECK_INTERVAL = 1_000L
} val errorEvents: SharedFlow<Pair<TunnelConf, BackendError>>
val bouncingTunnelIds: ConcurrentHashMap<Int, TunnelStatus.StopReason>
fun hasVpnPermission(): Boolean
suspend fun updateTunnelStatistics(tunnel: TunnelConf)
} }
@@ -1,11 +1,7 @@
package com.zaneschepke.wireguardautotunnel.core.tunnel package com.zaneschepke.wireguardautotunnel.core.tunnel
import com.wireguard.android.backend.BackendException
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.di.ApplicationScope import com.zaneschepke.wireguardautotunnel.di.ApplicationScope
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
@@ -13,71 +9,100 @@ import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.state.AmneziaStatistics import com.zaneschepke.wireguardautotunnel.domain.state.AmneziaStatistics
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics import com.zaneschepke.wireguardautotunnel.domain.state.TunnelStatistics
import com.zaneschepke.wireguardautotunnel.util.extensions.asAmBackendState import com.zaneschepke.wireguardautotunnel.util.extensions.asAmBackendState
import com.zaneschepke.wireguardautotunnel.util.extensions.asBackendState
import com.zaneschepke.wireguardautotunnel.util.extensions.toBackendError import com.zaneschepke.wireguardautotunnel.util.extensions.toBackendError
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.withContext
import org.amnezia.awg.backend.Backend
import org.amnezia.awg.backend.Tunnel
import timber.log.Timber
import javax.inject.Inject import javax.inject.Inject
import kotlin.jvm.optionals.getOrNull
import kotlinx.coroutines.CoroutineScope
import org.amnezia.awg.backend.Backend
import org.amnezia.awg.backend.BackendException
import org.amnezia.awg.backend.Tunnel
import org.amnezia.awg.config.Config
import timber.log.Timber
class UserspaceTunnel @Inject constructor( class UserspaceTunnel
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @Inject
@ApplicationScope private val applicationScope: CoroutineScope, constructor(
serviceManager: ServiceManager, @ApplicationScope private val applicationScope: CoroutineScope,
appDataRepository: AppDataRepository, val serviceManager: ServiceManager,
notificationManager: NotificationManager, val appDataRepository: AppDataRepository,
private val backend: Backend, private val backend: Backend,
networkMonitor: NetworkMonitor, ) : BaseTunnel(applicationScope, appDataRepository, serviceManager) {
) : TunnelProvider, BaseTunnel(ioDispatcher, applicationScope, networkMonitor, appDataRepository, serviceManager, notificationManager) {
override suspend fun startTunnel(tunnelConf: TunnelConf) { private var previousBackendState: Pair<BackendState, Boolean>? = null
withContext(ioDispatcher) {
super.startTunnel(tunnelConf)
runCatching {
backend.setState(tunnelConf, Tunnel.State.UP, tunnelConf.toAmConfig())
addToActiveTunnels(tunnelConf)
}.onFailure {
onTunnelStop(tunnelConf)
if (it is BackendException) {
handleBackendThrowable(it.toBackendError())
} else {
Timber.e(it)
}
}
}
}
override suspend fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics { override suspend fun startBackend(tunnel: TunnelConf) {
return AmneziaStatistics(backend.getStatistics(tunnelConf)) try {
} updateTunnelStatus(tunnel, TunnelStatus.Starting)
val amConfig = tunnel.toAmConfig()
handleVpnKillSwitchWithDomainEndpoints(amConfig)
backend.setState(tunnel, Tunnel.State.UP, amConfig)
} catch (e: BackendException) {
Timber.e(e, "Failed to start up backend for tunnel ${tunnel.name}")
throw e.toBackendError()
}
}
override suspend fun toggleTunnel(tunnelConf: TunnelConf, status: TunnelStatus) { override fun stopBackend(tunnel: TunnelConf) {
when (status) { Timber.i("Stopping tunnel ${tunnel.name} userspace")
TunnelStatus.UP -> backend.setState(tunnelConf, Tunnel.State.UP, tunnelConf.toAmConfig()) try {
TunnelStatus.DOWN -> backend.setState(tunnelConf, Tunnel.State.DOWN, tunnelConf.toAmConfig()) backend.setState(tunnel, Tunnel.State.DOWN, tunnel.toAmConfig())
} } catch (e: BackendException) {
} Timber.e(e, "Failed to stop tunnel ${tunnel.id}")
throw e.toBackendError()
} finally {
handlePreviouslyEnabledVpnKillSwitch()
}
}
override suspend fun stopTunnel(tunnelConf: TunnelConf?) { // stop vpn kill switch if we need to resolve DNS for peer endpoints
withContext(ioDispatcher) { private suspend fun handleVpnKillSwitchWithDomainEndpoints(config: Config) {
runCatching { if (
tunnels.value.firstOrNull { it.id == tunnelConf?.id }?.let { config.peers.any { it.endpoint.getOrNull()?.toString()?.isUrl() == true } &&
backend.setState(it, Tunnel.State.DOWN, it.toAmConfig()) backend.backendState.asBackendState() == BackendState.KILL_SWITCH_ACTIVE
onTunnelStop(it) ) {
} ?: stopAllTunnels() val bypassLan = appDataRepository.settings.get().isLanOnKillSwitchEnabled
}.onFailure { previousBackendState = Pair(BackendState.KILL_SWITCH_ACTIVE, bypassLan)
Timber.e(it) setBackendState(BackendState.SERVICE_ACTIVE, emptyList())
} }
} }
}
override suspend fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) { // restore vpn kill switch if needed
backend.setBackendState(backendState.asAmBackendState(), allowedIps) private fun handlePreviouslyEnabledVpnKillSwitch() {
} // let auto tunnel handle this if it is active
if (serviceManager.autoTunnelService.value == null) {
previousBackendState?.let { (state, lanEnabled) ->
Timber.d("Restoring kill switch configuration")
val lan = if (lanEnabled) TunnelConf.LAN_BYPASS_ALLOWED_IPS else emptyList()
backend.setBackendState(state.asAmBackendState(), lan)
}
}
previousBackendState = null
}
override suspend fun runningTunnelNames(): Set<String> { override fun setBackendState(backendState: BackendState, allowedIps: Collection<String>) {
return backend.runningTunnelNames Timber.d("Setting backend state: $backendState with allowedIps: $allowedIps")
} try {
backend.setBackendState(backendState.asAmBackendState(), allowedIps)
} catch (e: BackendException) {
throw e.toBackendError()
}
}
override fun getBackendState(): BackendState {
return backend.backendState.asBackendState()
}
override suspend fun runningTunnelNames(): Set<String> {
return backend.runningTunnelNames
}
override fun getStatistics(tunnelConf: TunnelConf): TunnelStatistics? {
return try {
AmneziaStatistics(backend.getStatistics(tunnelConf))
} catch (e: Exception) {
Timber.e(e, "Failed to get stats for ${tunnelConf.tunName}")
null
}
}
} }
@@ -13,48 +13,55 @@ import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import dagger.assisted.Assisted import dagger.assisted.Assisted
import dagger.assisted.AssistedInject import dagger.assisted.AssistedInject
import java.util.concurrent.TimeUnit
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import timber.log.Timber import timber.log.Timber
import java.util.concurrent.TimeUnit
@HiltWorker @HiltWorker
class ServiceWorker @AssistedInject constructor( class ServiceWorker
@Assisted private val context: Context, @AssistedInject
@Assisted private val params: WorkerParameters, constructor(
private val serviceManager: ServiceManager, @Assisted private val context: Context,
private val appDataRepository: AppDataRepository, @Assisted private val params: WorkerParameters,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, private val serviceManager: ServiceManager,
private val tunnelManager: TunnelManager, private val appDataRepository: AppDataRepository,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher,
private val tunnelManager: TunnelManager,
) : CoroutineWorker(context, params) { ) : CoroutineWorker(context, params) {
companion object { companion object {
private const val TAG = "service_worker" private const val TAG = "service_worker"
fun stop(context: Context) { fun stop(context: Context) {
WorkManager.getInstance(context).cancelAllWorkByTag(TAG) WorkManager.getInstance(context).cancelAllWorkByTag(TAG)
} }
fun start(context: Context) { fun start(context: Context) {
val periodicWorkRequest = PeriodicWorkRequestBuilder<ServiceWorker>( val periodicWorkRequest =
repeatInterval = 15, PeriodicWorkRequestBuilder<ServiceWorker>(
repeatIntervalTimeUnit = TimeUnit.MINUTES, repeatInterval = 15,
).build() repeatIntervalTimeUnit = TimeUnit.MINUTES,
WorkManager.getInstance(context) )
.enqueueUniquePeriodicWork( .build()
TAG, WorkManager.getInstance(context)
ExistingPeriodicWorkPolicy.KEEP, .enqueueUniquePeriodicWork(
periodicWorkRequest, TAG,
) ExistingPeriodicWorkPolicy.KEEP,
} periodicWorkRequest,
} )
}
}
override suspend fun doWork(): Result = withContext(ioDispatcher) { override suspend fun doWork(): Result =
Timber.i("Service worker started") withContext(ioDispatcher) {
with(appDataRepository.settings.get()) { Timber.i("Service worker started")
if (isAutoTunnelEnabled && !serviceManager.autoTunnelActive.value) return@with serviceManager.startAutoTunnel(true) with(appDataRepository.settings.get()) {
if (tunnelManager.activeTunnels().value.isEmpty()) tunnelManager.restorePreviousState() if (isAutoTunnelEnabled && serviceManager.autoTunnelService.value == null)
} return@with serviceManager.startAutoTunnel()
Result.success() if (tunnelManager.activeTunnels.value.isEmpty())
} tunnelManager.restorePreviousState()
}
Result.success()
}
} }
@@ -12,79 +12,38 @@ import com.zaneschepke.wireguardautotunnel.data.model.Settings
import com.zaneschepke.wireguardautotunnel.data.model.TunnelConfig import com.zaneschepke.wireguardautotunnel.data.model.TunnelConfig
@Database( @Database(
entities = [Settings::class, TunnelConfig::class], entities = [Settings::class, TunnelConfig::class],
version = 16, version = 16,
autoMigrations = autoMigrations =
[ [
AutoMigration(from = 1, to = 2), AutoMigration(from = 1, to = 2),
AutoMigration(from = 2, to = 3), AutoMigration(from = 2, to = 3),
AutoMigration( AutoMigration(from = 3, to = 4),
from = 3, AutoMigration(from = 4, to = 5),
to = 4, AutoMigration(from = 5, to = 6),
), AutoMigration(from = 6, to = 7, spec = RemoveLegacySettingColumnsMigration::class),
AutoMigration( AutoMigration(7, 8),
from = 4, AutoMigration(8, 9),
to = 5, AutoMigration(9, 10),
), AutoMigration(from = 10, to = 11, spec = RemoveTunnelPauseMigration::class),
AutoMigration( AutoMigration(from = 11, to = 12),
from = 5, AutoMigration(from = 12, to = 13),
to = 6, AutoMigration(from = 13, to = 14),
), AutoMigration(from = 14, to = 15),
AutoMigration( AutoMigration(from = 15, to = 16),
from = 6, ],
to = 7, exportSchema = true,
spec = RemoveLegacySettingColumnsMigration::class,
),
AutoMigration(7, 8),
AutoMigration(8, 9),
AutoMigration(9, 10),
AutoMigration(
from = 10,
to = 11,
spec = RemoveTunnelPauseMigration::class,
),
AutoMigration(
from = 11,
to = 12,
),
AutoMigration(
from = 12,
to = 13,
),
AutoMigration(
from = 13,
to = 14,
),
AutoMigration(
from = 14,
to = 15,
),
AutoMigration(
from = 15,
to = 16,
),
],
exportSchema = true,
) )
@TypeConverters(DatabaseListConverters::class) @TypeConverters(DatabaseListConverters::class)
abstract class AppDatabase : RoomDatabase() { abstract class AppDatabase : RoomDatabase() {
abstract fun settingDao(): SettingsDao abstract fun settingDao(): SettingsDao
abstract fun tunnelConfigDoa(): TunnelConfigDao abstract fun tunnelConfigDoa(): TunnelConfigDao
} }
@DeleteColumn( @DeleteColumn(tableName = "Settings", columnName = "default_tunnel")
tableName = "Settings", @DeleteColumn(tableName = "Settings", columnName = "is_battery_saver_enabled")
columnName = "default_tunnel",
)
@DeleteColumn(
tableName = "Settings",
columnName = "is_battery_saver_enabled",
)
class RemoveLegacySettingColumnsMigration : AutoMigrationSpec class RemoveLegacySettingColumnsMigration : AutoMigrationSpec
@DeleteColumn( @DeleteColumn(tableName = "Settings", columnName = "is_auto_tunnel_paused")
tableName = "Settings",
columnName = "is_auto_tunnel_paused",
)
class RemoveTunnelPauseMigration : AutoMigrationSpec class RemoveTunnelPauseMigration : AutoMigrationSpec
@@ -7,6 +7,7 @@ import androidx.datastore.preferences.core.edit
import androidx.datastore.preferences.core.stringPreferencesKey import androidx.datastore.preferences.core.stringPreferencesKey
import androidx.datastore.preferences.preferencesDataStore import androidx.datastore.preferences.preferencesDataStore
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import java.io.IOException
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.first
@@ -15,80 +16,77 @@ import kotlinx.coroutines.flow.map
import kotlinx.coroutines.runBlocking import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import timber.log.Timber import timber.log.Timber
import java.io.IOException
class DataStoreManager( class DataStoreManager(
private val context: Context, private val context: Context,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) { ) {
companion object { companion object {
val locationDisclosureShown = booleanPreferencesKey("LOCATION_DISCLOSURE_SHOWN") val locationDisclosureShown = booleanPreferencesKey("LOCATION_DISCLOSURE_SHOWN")
val batteryDisableShown = booleanPreferencesKey("BATTERY_OPTIMIZE_DISABLE_SHOWN") val batteryDisableShown = booleanPreferencesKey("BATTERY_OPTIMIZE_DISABLE_SHOWN")
val currentSSID = stringPreferencesKey("CURRENT_SSID") val pinLockEnabled = booleanPreferencesKey("PIN_LOCK_ENABLED")
val pinLockEnabled = booleanPreferencesKey("PIN_LOCK_ENABLED") val expandedTunnelIds = stringPreferencesKey("EXPANDED_TUNNEL_IDS")
val tunnelStatsExpanded = booleanPreferencesKey("TUNNEL_STATS_EXPANDED") val isLocalLogsEnabled = booleanPreferencesKey("LOCAL_LOGS_ENABLED")
val isLocalLogsEnabled = booleanPreferencesKey("LOCAL_LOGS_ENABLED") val locale = stringPreferencesKey("LOCALE")
val locale = stringPreferencesKey("LOCALE") val theme = stringPreferencesKey("THEME")
val theme = stringPreferencesKey("THEME") val isRemoteControlEnabled = booleanPreferencesKey("IS_REMOTE_CONTROL_ENABLED")
} val remoteKey = stringPreferencesKey("REMOTE_KEY")
}
// preferences // preferences
private val preferencesKey = "preferences" private val preferencesKey = "preferences"
private val Context.dataStore by private val Context.dataStore by preferencesDataStore(name = preferencesKey)
preferencesDataStore(
name = preferencesKey,
)
suspend fun init() { suspend fun init() {
withContext(ioDispatcher) { withContext(ioDispatcher) {
try { try {
context.dataStore.data.first() context.dataStore.data.first()
} catch (e: IOException) { } catch (e: IOException) {
Timber.Forest.e(e) Timber.Forest.e(e)
} }
} }
} }
suspend fun <T> saveToDataStore(key: Preferences.Key<T>, value: T) { suspend fun <T> saveToDataStore(key: Preferences.Key<T>, value: T) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
try { try {
context.dataStore.edit { it[key] = value } context.dataStore.edit { it[key] = value }
} catch (e: IOException) { } catch (e: IOException) {
Timber.Forest.e(e) Timber.Forest.e(e)
} catch (e: Exception) { } catch (e: Exception) {
Timber.Forest.e(e) Timber.Forest.e(e)
} }
} }
} }
suspend fun <T> removeFromDataStore(key: Preferences.Key<T>) { suspend fun <T> removeFromDataStore(key: Preferences.Key<T>) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
try { try {
context.dataStore.edit { it.remove(key) } context.dataStore.edit { it.remove(key) }
} catch (e: IOException) { } catch (e: IOException) {
Timber.Forest.e(e) Timber.Forest.e(e)
} catch (e: Exception) { } catch (e: Exception) {
Timber.Forest.e(e) Timber.Forest.e(e)
} }
} }
} }
fun <T> getFromStoreFlow(key: Preferences.Key<T>) = context.dataStore.data.map { it[key] } fun <T> getFromStoreFlow(key: Preferences.Key<T>) = context.dataStore.data.map { it[key] }
suspend fun <T> getFromStore(key: Preferences.Key<T>): T? { suspend fun <T> getFromStore(key: Preferences.Key<T>): T? {
return withContext(ioDispatcher) { return withContext(ioDispatcher) {
try { try {
context.dataStore.data.map { it[key] }.first() context.dataStore.data.map { it[key] }.first()
} catch (e: IOException) { } catch (e: IOException) {
Timber.Forest.e(e) Timber.Forest.e(e)
null null
} }
} }
} }
fun <T> getFromStoreBlocking(key: Preferences.Key<T>) = runBlocking { fun <T> getFromStoreBlocking(key: Preferences.Key<T>) = runBlocking {
context.dataStore.data.map { it[key] }.first() context.dataStore.data.map { it[key] }.first()
} }
val preferencesFlow: Flow<Preferences?> = context.dataStore.data.flowOn(ioDispatcher) val preferencesFlow: Flow<Preferences?> = context.dataStore.data.flowOn(ioDispatcher)
} }
@@ -5,17 +5,17 @@ import androidx.sqlite.db.SupportSQLiteDatabase
import timber.log.Timber import timber.log.Timber
class DatabaseCallback : RoomDatabase.Callback() { class DatabaseCallback : RoomDatabase.Callback() {
override fun onCreate(db: SupportSQLiteDatabase) = db.run { override fun onCreate(db: SupportSQLiteDatabase) =
// Notice non-ui thread is here db.run {
beginTransaction() beginTransaction()
try { try {
execSQL(Queries.createDefaultSettings()) execSQL(Queries.createDefaultSettings())
Timber.i("Bootstrapping settings data") Timber.i("Bootstrapping settings data")
setTransactionSuccessful() setTransactionSuccessful()
} catch (e: Exception) { } catch (e: Exception) {
Timber.e(e) Timber.e(e)
} finally { } finally {
endTransaction() endTransaction()
} }
} }
} }
@@ -1,24 +1,23 @@
package com.zaneschepke.wireguardautotunnel.data package com.zaneschepke.wireguardautotunnel.data
import androidx.room.TypeConverter import androidx.room.TypeConverter
import kotlinx.serialization.encodeToString
import kotlinx.serialization.json.Json import kotlinx.serialization.json.Json
class DatabaseListConverters { class DatabaseListConverters {
@TypeConverter @TypeConverter
fun listToString(value: MutableList<String>): String { fun listToString(value: MutableList<String>): String {
return Json.encodeToString(value) return Json.encodeToString(value)
} }
@TypeConverter @TypeConverter
fun stringToList(value: String): MutableList<String> { fun stringToList(value: String): MutableList<String> {
if (value.isBlank() || value.isEmpty()) return mutableListOf() if (value.isBlank() || value.isEmpty()) return mutableListOf()
return try { return try {
Json.decodeFromString<MutableList<String>>(value) Json.decodeFromString<MutableList<String>>(value)
} catch (e: Exception) { } catch (e: Exception) {
val list = value.split(",").toMutableList() val list = value.split(",").toMutableList()
val json = listToString(list) val json = listToString(list)
Json.decodeFromString<MutableList<String>>(json) Json.decodeFromString<MutableList<String>>(json)
} }
} }
} }
@@ -1,8 +1,8 @@
package com.zaneschepke.wireguardautotunnel.data package com.zaneschepke.wireguardautotunnel.data
object Queries { object Queries {
fun createDefaultSettings(): String { fun createDefaultSettings(): String {
return """ return """
INSERT INTO Settings (is_tunnel_enabled, INSERT INTO Settings (is_tunnel_enabled,
is_tunnel_on_mobile_data_enabled, is_tunnel_on_mobile_data_enabled,
trusted_network_ssids, trusted_network_ssids,
@@ -24,12 +24,14 @@ object Queries {
'false', 'false',
'false', 'false',
'false') 'false')
""".trimIndent() """
} .trimIndent()
}
fun createTunnelConfig(): String { fun createTunnelConfig(): String {
return """ return """
INSERT INTO TunnelConfig (name, wg_quick) VALUES ('test', 'test') INSERT INTO TunnelConfig (name, wg_quick) VALUES ('test', 'test')
""".trimIndent() """
} .trimIndent()
}
} }
@@ -10,27 +10,19 @@ import kotlinx.coroutines.flow.Flow
@Dao @Dao
interface SettingsDao { interface SettingsDao {
@Insert(onConflict = OnConflictStrategy.REPLACE) @Insert(onConflict = OnConflictStrategy.REPLACE) suspend fun save(t: Settings)
suspend fun save(t: Settings)
@Insert(onConflict = OnConflictStrategy.REPLACE) @Insert(onConflict = OnConflictStrategy.REPLACE) suspend fun saveAll(t: List<Settings>)
suspend fun saveAll(t: List<Settings>)
@Query("SELECT * FROM settings WHERE id=:id") @Query("SELECT * FROM settings WHERE id=:id") suspend fun getById(id: Long): Settings?
suspend fun getById(id: Long): Settings?
@Query("SELECT * FROM settings") @Query("SELECT * FROM settings") suspend fun getAll(): List<Settings>
suspend fun getAll(): List<Settings>
@Query("SELECT * FROM settings LIMIT 1") @Query("SELECT * FROM settings LIMIT 1") fun getSettingsFlow(): Flow<Settings>
fun getSettingsFlow(): Flow<Settings>
@Query("SELECT * FROM settings") @Query("SELECT * FROM settings") fun getAllFlow(): Flow<MutableList<Settings>>
fun getAllFlow(): Flow<MutableList<Settings>>
@Delete @Delete suspend fun delete(t: Settings)
suspend fun delete(t: Settings)
@Query("SELECT COUNT('id') FROM settings") @Query("SELECT COUNT('id') FROM settings") suspend fun count(): Long
suspend fun count(): Long
} }
@@ -11,48 +11,40 @@ import kotlinx.coroutines.flow.Flow
@Dao @Dao
interface TunnelConfigDao { interface TunnelConfigDao {
@Insert(onConflict = OnConflictStrategy.REPLACE) @Insert(onConflict = OnConflictStrategy.REPLACE) suspend fun save(t: TunnelConfig)
suspend fun save(t: TunnelConfig)
@Insert(onConflict = OnConflictStrategy.REPLACE) @Insert(onConflict = OnConflictStrategy.REPLACE) suspend fun saveAll(t: TunnelConfigs)
suspend fun saveAll(t: TunnelConfigs)
@Query("SELECT * FROM TunnelConfig WHERE id=:id") @Query("SELECT * FROM TunnelConfig WHERE id=:id") suspend fun getById(id: Long): TunnelConfig?
suspend fun getById(id: Long): TunnelConfig?
@Query("SELECT * FROM TunnelConfig WHERE name=:name") @Query("SELECT * FROM TunnelConfig WHERE name=:name")
suspend fun getByName(name: String): TunnelConfig? suspend fun getByName(name: String): TunnelConfig?
@Query("SELECT * FROM TunnelConfig WHERE is_Active=1") @Query("SELECT * FROM TunnelConfig WHERE is_Active=1") suspend fun getActive(): TunnelConfigs
suspend fun getActive(): TunnelConfigs
@Query("SELECT * FROM TunnelConfig") @Query("SELECT * FROM TunnelConfig") suspend fun getAll(): TunnelConfigs
suspend fun getAll(): TunnelConfigs
@Delete @Delete suspend fun delete(t: TunnelConfig)
suspend fun delete(t: TunnelConfig)
@Query("SELECT COUNT('id') FROM TunnelConfig") @Query("SELECT COUNT('id') FROM TunnelConfig") suspend fun count(): Long
suspend fun count(): Long
@Query("SELECT * FROM TunnelConfig WHERE tunnel_networks LIKE '%' || :name || '%'") @Query("SELECT * FROM TunnelConfig WHERE tunnel_networks LIKE '%' || :name || '%'")
suspend fun findByTunnelNetworkName(name: String): TunnelConfigs suspend fun findByTunnelNetworkName(name: String): TunnelConfigs
@Query("UPDATE TunnelConfig SET is_primary_tunnel = 0 WHERE is_primary_tunnel =1") @Query("UPDATE TunnelConfig SET is_primary_tunnel = 0 WHERE is_primary_tunnel =1")
suspend fun resetPrimaryTunnel() suspend fun resetPrimaryTunnel()
@Query("UPDATE TunnelConfig SET is_mobile_data_tunnel = 0 WHERE is_mobile_data_tunnel =1") @Query("UPDATE TunnelConfig SET is_mobile_data_tunnel = 0 WHERE is_mobile_data_tunnel =1")
suspend fun resetMobileDataTunnel() suspend fun resetMobileDataTunnel()
@Query("UPDATE TunnelConfig SET is_ethernet_tunnel = 0 WHERE is_ethernet_tunnel =1") @Query("UPDATE TunnelConfig SET is_ethernet_tunnel = 0 WHERE is_ethernet_tunnel =1")
suspend fun resetEthernetTunnel() suspend fun resetEthernetTunnel()
@Query("SELECT * FROM TUNNELCONFIG WHERE is_primary_tunnel=1") @Query("SELECT * FROM TUNNELCONFIG WHERE is_primary_tunnel=1")
suspend fun findByPrimary(): TunnelConfigs suspend fun findByPrimary(): TunnelConfigs
@Query("SELECT * FROM TUNNELCONFIG WHERE is_mobile_data_tunnel=1") @Query("SELECT * FROM TUNNELCONFIG WHERE is_mobile_data_tunnel=1")
suspend fun findByMobileDataTunnel(): TunnelConfigs suspend fun findByMobileDataTunnel(): TunnelConfigs
@Query("SELECT * FROM tunnelconfig") @Query("SELECT * FROM tunnelconfig") fun getAllFlow(): Flow<MutableList<TunnelConfig>>
fun getAllFlow(): Flow<MutableList<TunnelConfig>>
} }
@@ -0,0 +1,10 @@
package com.zaneschepke.wireguardautotunnel.data.model
import kotlinx.serialization.SerialName
import kotlinx.serialization.Serializable
@Serializable
data class Asset(
val name: String,
@SerialName("browser_download_url") val browserDownloadUrl: String,
)
@@ -4,43 +4,51 @@ import com.zaneschepke.wireguardautotunnel.domain.entity.AppState
import com.zaneschepke.wireguardautotunnel.ui.theme.Theme import com.zaneschepke.wireguardautotunnel.ui.theme.Theme
data class GeneralState( data class GeneralState(
val isLocationDisclosureShown: Boolean = LOCATION_DISCLOSURE_SHOWN_DEFAULT, val isLocationDisclosureShown: Boolean = LOCATION_DISCLOSURE_SHOWN_DEFAULT,
val isBatteryOptimizationDisableShown: Boolean = BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT, val isBatteryOptimizationDisableShown: Boolean = BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT,
val isPinLockEnabled: Boolean = PIN_LOCK_ENABLED_DEFAULT, val isPinLockEnabled: Boolean = PIN_LOCK_ENABLED_DEFAULT,
val isTunnelStatsExpanded: Boolean = IS_TUNNEL_STATS_EXPANDED, val expandedTunnelIds: List<Int> = emptyList(),
val isLocalLogsEnabled: Boolean = IS_LOGS_ENABLED_DEFAULT, val isLocalLogsEnabled: Boolean = IS_LOGS_ENABLED_DEFAULT,
val locale: String? = null, val isRemoteControlEnabled: Boolean = IS_REMOTE_CONTROL_ENABLED,
val theme: Theme = Theme.AUTOMATIC, val remoteKey: String? = null,
val locale: String? = null,
val theme: Theme = Theme.AUTOMATIC,
) { ) {
fun toAppState(): AppState = AppState( fun toAppState(): AppState =
isLocationDisclosureShown, AppState(
isBatteryOptimizationDisableShown, isLocationDisclosureShown,
isPinLockEnabled, isBatteryOptimizationDisableShown,
isTunnelStatsExpanded, isPinLockEnabled,
isLocationDisclosureShown, expandedTunnelIds,
locale, isLocalLogsEnabled,
theme, isRemoteControlEnabled,
) remoteKey,
locale,
theme,
)
companion object { companion object {
fun from(appState: AppState): GeneralState { fun from(appState: AppState): GeneralState {
return with(appState) { return with(appState) {
GeneralState( GeneralState(
isLocationDisclosureShown, isLocationDisclosureShown,
isBatteryOptimizationDisableShown, isBatteryOptimizationDisableShown,
isPinLockEnabled, isPinLockEnabled,
isTunnelStatsExpanded, expandedTunnelIds,
isLocationDisclosureShown, isLocalLogsEnabled,
locale, isRemoteControlEnabled,
theme, remoteKey,
) locale,
} theme,
} )
const val LOCATION_DISCLOSURE_SHOWN_DEFAULT = false }
const val BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT = false }
const val PIN_LOCK_ENABLED_DEFAULT = false
const val IS_TUNNEL_STATS_EXPANDED = false const val LOCATION_DISCLOSURE_SHOWN_DEFAULT = false
const val IS_LOGS_ENABLED_DEFAULT = false const val BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT = false
} const val PIN_LOCK_ENABLED_DEFAULT = false
const val IS_LOGS_ENABLED_DEFAULT = false
const val IS_REMOTE_CONTROL_ENABLED = false
}
} }
@@ -0,0 +1,24 @@
package com.zaneschepke.wireguardautotunnel.data.model
import com.zaneschepke.wireguardautotunnel.domain.entity.AppUpdate
import kotlinx.serialization.SerialName
import kotlinx.serialization.Serializable
@Serializable
data class GitHubRelease(
@SerialName("tag_name") val tagName: String,
val name: String?,
val body: String?,
val assets: List<Asset>,
) {
fun toAppUpdate(): AppUpdate {
val apkAsset = assets.firstOrNull { it.name.endsWith(".apk") }
return AppUpdate(
version = tagName.removePrefix("v"),
title = name ?: "Update $tagName",
releaseNotes = body ?: "No release notes provided",
apkUrl = apkAsset?.browserDownloadUrl,
apkFileName = apkAsset?.name,
)
}
}
@@ -7,112 +7,100 @@ import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
@Entity @Entity
data class Settings( data class Settings(
@PrimaryKey(autoGenerate = true) val id: Int = 0, @PrimaryKey(autoGenerate = true) val id: Int = 0,
@ColumnInfo(name = "is_tunnel_enabled") val isAutoTunnelEnabled: Boolean = false, @ColumnInfo(name = "is_tunnel_enabled") val isAutoTunnelEnabled: Boolean = false,
@ColumnInfo(name = "is_tunnel_on_mobile_data_enabled") @ColumnInfo(name = "is_tunnel_on_mobile_data_enabled")
val isTunnelOnMobileDataEnabled: Boolean = false, val isTunnelOnMobileDataEnabled: Boolean = false,
@ColumnInfo(name = "trusted_network_ssids") @ColumnInfo(name = "trusted_network_ssids")
val trustedNetworkSSIDs: MutableList<String> = mutableListOf(), val trustedNetworkSSIDs: MutableList<String> = mutableListOf(),
@ColumnInfo(name = "is_always_on_vpn_enabled") val isAlwaysOnVpnEnabled: Boolean = false, @ColumnInfo(name = "is_always_on_vpn_enabled") val isAlwaysOnVpnEnabled: Boolean = false,
@ColumnInfo(name = "is_tunnel_on_ethernet_enabled") @ColumnInfo(name = "is_tunnel_on_ethernet_enabled")
val isTunnelOnEthernetEnabled: Boolean = false, val isTunnelOnEthernetEnabled: Boolean = false,
@ColumnInfo( @ColumnInfo(name = "is_shortcuts_enabled", defaultValue = "false")
name = "is_shortcuts_enabled", val isShortcutsEnabled: Boolean = false,
defaultValue = "false", @ColumnInfo(name = "is_tunnel_on_wifi_enabled", defaultValue = "false")
) val isTunnelOnWifiEnabled: Boolean = false,
val isShortcutsEnabled: Boolean = false, @ColumnInfo(name = "is_kernel_enabled", defaultValue = "false")
@ColumnInfo( val isKernelEnabled: Boolean = false,
name = "is_tunnel_on_wifi_enabled", @ColumnInfo(name = "is_restore_on_boot_enabled", defaultValue = "false")
defaultValue = "false", val isRestoreOnBootEnabled: Boolean = false,
) @ColumnInfo(name = "is_multi_tunnel_enabled", defaultValue = "false")
val isTunnelOnWifiEnabled: Boolean = false, val isMultiTunnelEnabled: Boolean = false,
@ColumnInfo( @ColumnInfo(name = "is_ping_enabled", defaultValue = "false")
name = "is_kernel_enabled", val isPingEnabled: Boolean = false,
defaultValue = "false", @ColumnInfo(name = "is_amnezia_enabled", defaultValue = "false")
) val isAmneziaEnabled: Boolean = false,
val isKernelEnabled: Boolean = false, @ColumnInfo(name = "is_wildcards_enabled", defaultValue = "false")
@ColumnInfo( val isWildcardsEnabled: Boolean = false,
name = "is_restore_on_boot_enabled", @ColumnInfo(name = "is_wifi_by_shell_enabled", defaultValue = "false")
defaultValue = "false", val isWifiNameByShellEnabled: Boolean = false,
) @ColumnInfo(name = "is_stop_on_no_internet_enabled", defaultValue = "false")
val isRestoreOnBootEnabled: Boolean = false, val isStopOnNoInternetEnabled: Boolean = false,
@ColumnInfo( @ColumnInfo(name = "is_vpn_kill_switch_enabled", defaultValue = "false")
name = "is_multi_tunnel_enabled", val isVpnKillSwitchEnabled: Boolean = false,
defaultValue = "false", @ColumnInfo(name = "is_kernel_kill_switch_enabled", defaultValue = "false")
) val isKernelKillSwitchEnabled: Boolean = false,
val isMultiTunnelEnabled: Boolean = false, @ColumnInfo(name = "is_lan_on_kill_switch_enabled", defaultValue = "false")
@ColumnInfo( val isLanOnKillSwitchEnabled: Boolean = false,
name = "is_ping_enabled", @ColumnInfo(name = "debounce_delay_seconds", defaultValue = "3")
defaultValue = "false", val debounceDelaySeconds: Int = 3,
) @ColumnInfo(name = "is_disable_kill_switch_on_trusted_enabled", defaultValue = "false")
val isPingEnabled: Boolean = false, val isDisableKillSwitchOnTrustedEnabled: Boolean = false,
@ColumnInfo(
name = "is_amnezia_enabled",
defaultValue = "false",
)
val isAmneziaEnabled: Boolean = false,
@ColumnInfo(
name = "is_wildcards_enabled",
defaultValue = "false",
)
val isWildcardsEnabled: Boolean = false,
@ColumnInfo(
name = "is_wifi_by_shell_enabled",
defaultValue = "false",
)
val isWifiNameByShellEnabled: Boolean = false,
@ColumnInfo(
name = "is_stop_on_no_internet_enabled",
defaultValue = "false",
)
val isStopOnNoInternetEnabled: Boolean = false,
@ColumnInfo(
name = "is_vpn_kill_switch_enabled",
defaultValue = "false",
)
val isVpnKillSwitchEnabled: Boolean = false,
@ColumnInfo(
name = "is_kernel_kill_switch_enabled",
defaultValue = "false",
)
val isKernelKillSwitchEnabled: Boolean = false,
@ColumnInfo(
name = "is_lan_on_kill_switch_enabled",
defaultValue = "false",
)
val isLanOnKillSwitchEnabled: Boolean = false,
@ColumnInfo(
name = "debounce_delay_seconds",
defaultValue = "3",
)
val debounceDelaySeconds: Int = 3,
@ColumnInfo(
name = "is_disable_kill_switch_on_trusted_enabled",
defaultValue = "false",
)
val isDisableKillSwitchOnTrustedEnabled: Boolean = false,
) { ) {
fun toAppSettings(): AppSettings { fun toAppSettings(): AppSettings {
return AppSettings( return AppSettings(
id, isAutoTunnelEnabled, isTunnelOnMobileDataEnabled, trustedNetworkSSIDs, isAlwaysOnVpnEnabled, isTunnelOnEthernetEnabled, id,
isShortcutsEnabled, isTunnelOnWifiEnabled, isKernelEnabled, isRestoreOnBootEnabled, isMultiTunnelEnabled, isPingEnabled, isAutoTunnelEnabled,
isAmneziaEnabled, isWildcardsEnabled, isWifiNameByShellEnabled, isStopOnNoInternetEnabled, isVpnKillSwitchEnabled, isTunnelOnMobileDataEnabled,
isKernelKillSwitchEnabled, isLanOnKillSwitchEnabled, debounceDelaySeconds, isDisableKillSwitchOnTrustedEnabled, trustedNetworkSSIDs,
) isAlwaysOnVpnEnabled,
} isTunnelOnEthernetEnabled,
isShortcutsEnabled,
isTunnelOnWifiEnabled,
isKernelEnabled,
isRestoreOnBootEnabled,
isMultiTunnelEnabled,
isPingEnabled,
isAmneziaEnabled,
isWildcardsEnabled,
isWifiNameByShellEnabled,
isStopOnNoInternetEnabled,
isVpnKillSwitchEnabled,
isKernelKillSwitchEnabled,
isLanOnKillSwitchEnabled,
debounceDelaySeconds,
isDisableKillSwitchOnTrustedEnabled,
)
}
companion object { companion object {
fun from(appSettings: AppSettings): Settings { fun from(appSettings: AppSettings): Settings {
return with(appSettings) { return with(appSettings) {
Settings( Settings(
id, isAutoTunnelEnabled, isTunnelOnMobileDataEnabled, trustedNetworkSSIDs.toMutableList(), isAlwaysOnVpnEnabled, id,
isTunnelOnEthernetEnabled, isShortcutsEnabled, isTunnelOnWifiEnabled, isKernelEnabled, isRestoreOnBootEnabled, isAutoTunnelEnabled,
isMultiTunnelEnabled, isPingEnabled, isAmneziaEnabled, isWildcardsEnabled, isWifiNameByShellEnabled, isTunnelOnMobileDataEnabled,
isStopOnNoInternetEnabled, isVpnKillSwitchEnabled, isKernelKillSwitchEnabled, isLanOnKillSwitchEnabled, trustedNetworkSSIDs.toMutableList(),
debounceDelaySeconds, isDisableKillSwitchOnTrustedEnabled, isAlwaysOnVpnEnabled,
) isTunnelOnEthernetEnabled,
} isShortcutsEnabled,
} isTunnelOnWifiEnabled,
} isKernelEnabled,
isRestoreOnBootEnabled,
isMultiTunnelEnabled,
isPingEnabled,
isAmneziaEnabled,
isWildcardsEnabled,
isWifiNameByShellEnabled,
isStopOnNoInternetEnabled,
isVpnKillSwitchEnabled,
isKernelKillSwitchEnabled,
isLanOnKillSwitchEnabled,
debounceDelaySeconds,
isDisableKillSwitchOnTrustedEnabled,
)
}
}
}
} }
@@ -8,86 +8,70 @@ import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
@Entity(indices = [Index(value = ["name"], unique = true)]) @Entity(indices = [Index(value = ["name"], unique = true)])
data class TunnelConfig( data class TunnelConfig(
@PrimaryKey(autoGenerate = true) val id: Int = 0, @PrimaryKey(autoGenerate = true) val id: Int = 0,
@ColumnInfo(name = "name") val name: String, @ColumnInfo(name = "name") val name: String,
@ColumnInfo(name = "wg_quick") val wgQuick: String, @ColumnInfo(name = "wg_quick") val wgQuick: String,
@ColumnInfo( @ColumnInfo(name = "tunnel_networks", defaultValue = "")
name = "tunnel_networks", val tunnelNetworks: MutableList<String> = mutableListOf(),
defaultValue = "", @ColumnInfo(name = "is_mobile_data_tunnel", defaultValue = "false")
) val isMobileDataTunnel: Boolean = false,
val tunnelNetworks: MutableList<String> = mutableListOf(), @ColumnInfo(name = "is_primary_tunnel", defaultValue = "false")
@ColumnInfo( val isPrimaryTunnel: Boolean = false,
name = "is_mobile_data_tunnel", @ColumnInfo(name = "am_quick", defaultValue = "") val amQuick: String = AM_QUICK_DEFAULT,
defaultValue = "false", @ColumnInfo(name = "is_Active", defaultValue = "false") val isActive: Boolean = false,
) @ColumnInfo(name = "is_ping_enabled", defaultValue = "false")
val isMobileDataTunnel: Boolean = false, val isPingEnabled: Boolean = false,
@ColumnInfo( @ColumnInfo(name = "ping_interval", defaultValue = "null") val pingInterval: Long? = null,
name = "is_primary_tunnel", @ColumnInfo(name = "ping_cooldown", defaultValue = "null") val pingCooldown: Long? = null,
defaultValue = "false", @ColumnInfo(name = "ping_ip", defaultValue = "null") var pingIp: String? = null,
) @ColumnInfo(name = "is_ethernet_tunnel", defaultValue = "false")
val isPrimaryTunnel: Boolean = false, var isEthernetTunnel: Boolean = false,
@ColumnInfo( @ColumnInfo(name = "is_ipv4_preferred", defaultValue = "true")
name = "am_quick", var isIpv4Preferred: Boolean = true,
defaultValue = "",
)
val amQuick: String = AM_QUICK_DEFAULT,
@ColumnInfo(
name = "is_Active",
defaultValue = "false",
)
val isActive: Boolean = false,
@ColumnInfo(
name = "is_ping_enabled",
defaultValue = "false",
)
val isPingEnabled: Boolean = false,
@ColumnInfo(
name = "ping_interval",
defaultValue = "null",
)
val pingInterval: Long? = null,
@ColumnInfo(
name = "ping_cooldown",
defaultValue = "null",
)
val pingCooldown: Long? = null,
@ColumnInfo(
name = "ping_ip",
defaultValue = "null",
)
var pingIp: String? = null,
@ColumnInfo(
name = "is_ethernet_tunnel",
defaultValue = "false",
)
var isEthernetTunnel: Boolean = false,
@ColumnInfo(
name = "is_ipv4_preferred",
defaultValue = "true",
)
var isIpv4Preferred: Boolean = true,
) { ) {
fun toTunnel(): TunnelConf { fun toTunnel(): TunnelConf {
return TunnelConf( return TunnelConf(
id, name, wgQuick, tunnelNetworks, isMobileDataTunnel, id,
isPrimaryTunnel, amQuick, isActive, isPingEnabled, pingInterval, name,
pingCooldown, pingIp, isEthernetTunnel, isIpv4Preferred, wgQuick,
) tunnelNetworks,
} isMobileDataTunnel,
isPrimaryTunnel,
amQuick,
isActive,
isPingEnabled,
pingInterval,
pingCooldown,
pingIp,
isEthernetTunnel,
isIpv4Preferred,
)
}
companion object { companion object {
const val AM_QUICK_DEFAULT = "" const val AM_QUICK_DEFAULT = ""
fun from(tunnelConf: TunnelConf): TunnelConfig { fun from(tunnelConf: TunnelConf): TunnelConfig {
return with(tunnelConf) { return with(tunnelConf) {
return TunnelConfig( return TunnelConfig(
id, tunName, wgQuick, tunnelNetworks.toMutableList(), isMobileDataTunnel, id,
isPrimaryTunnel, amQuick, isActive, isPingEnabled, pingInterval, tunName,
pingCooldown, pingIp, isEthernetTunnel, isIpv4Preferred, wgQuick,
) tunnelNetworks.toMutableList(),
} isMobileDataTunnel,
} isPrimaryTunnel,
} amQuick,
isActive,
isPingEnabled,
pingInterval,
pingCooldown,
pingIp,
isEthernetTunnel,
isIpv4Preferred,
)
}
}
}
} }
@@ -0,0 +1,9 @@
package com.zaneschepke.wireguardautotunnel.data.network
import com.zaneschepke.wireguardautotunnel.data.model.GitHubRelease
interface GitHubApi {
suspend fun getLatestRelease(owner: String, repo: String): Result<GitHubRelease>
suspend fun getNightlyRelease(owner: String, repo: String): Result<GitHubRelease>
}
@@ -0,0 +1,28 @@
package com.zaneschepke.wireguardautotunnel.data.network
import io.ktor.client.HttpClient
import io.ktor.client.engine.okhttp.OkHttp
import io.ktor.client.plugins.HttpTimeout
import io.ktor.client.plugins.contentnegotiation.ContentNegotiation
import io.ktor.serialization.kotlinx.json.json
import kotlinx.serialization.json.Json
object KtorClient {
fun create(): HttpClient {
return HttpClient(OkHttp) {
install(ContentNegotiation) {
json(
Json {
ignoreUnknownKeys = true
isLenient = true
}
)
}
install(HttpTimeout) {
requestTimeoutMillis = 15000
connectTimeoutMillis = 15000
socketTimeoutMillis = 15000
}
}
}
}
@@ -0,0 +1,56 @@
package com.zaneschepke.wireguardautotunnel.data.network
import com.zaneschepke.wireguardautotunnel.data.model.GitHubRelease
import io.ktor.client.HttpClient
import io.ktor.client.call.body
import io.ktor.client.plugins.ClientRequestException
import io.ktor.client.request.get
import io.ktor.http.HttpStatusCode
class KtorGitHubApi(private val client: HttpClient) : GitHubApi {
override suspend fun getLatestRelease(owner: String, repo: String): Result<GitHubRelease> {
return try {
val response: GitHubRelease =
client.get("https://api.github.com/repos/$owner/$repo/releases/latest").body()
Result.success(response)
} catch (e: ClientRequestException) {
when (e.response.status) {
HttpStatusCode.Forbidden -> Result.failure(Exception("Rate limit exceeded"))
HttpStatusCode.NotFound ->
Result.failure(Exception("Repository or release not found"))
else -> Result.failure(e)
}
} catch (e: Exception) {
Result.failure(e)
}
}
override suspend fun getNightlyRelease(owner: String, repo: String): Result<GitHubRelease> {
return try {
// Fetch all releases
val releases: List<GitHubRelease> =
client.get("https://api.github.com/repos/$owner/$repo/releases").body()
// Find the first release with "nightly" in the tag_name (case-insensitive)
val nightlyRelease =
releases.firstOrNull { release ->
release.tagName.contains("nightly", ignoreCase = true)
}
if (nightlyRelease != null) {
Result.success(nightlyRelease)
} else {
Result.failure(Exception("No release with 'nightly' tag found"))
}
} catch (e: ClientRequestException) {
when (e.response.status) {
HttpStatusCode.Forbidden -> Result.failure(Exception("Rate limit exceeded"))
HttpStatusCode.NotFound ->
Result.failure(Exception("Repository or release not found"))
else -> Result.failure(e)
}
} catch (e: Exception) {
Result.failure(e)
}
}
}
@@ -1,28 +1,28 @@
package com.zaneschepke.wireguardautotunnel.data.repository package com.zaneschepke.wireguardautotunnel.data.repository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import javax.inject.Inject import javax.inject.Inject
class AppDataRoomRepository class AppDataRoomRepository
@Inject @Inject
constructor( constructor(
override val settings: AppSettingRepository, override val settings: AppSettingRepository,
override val tunnels: TunnelRepository, override val tunnels: TunnelRepository,
override val appState: AppStateRepository, override val appState: AppStateRepository,
) : AppDataRepository { ) : AppDataRepository {
override suspend fun getPrimaryOrFirstTunnel(): TunnelConf? { override suspend fun getPrimaryOrFirstTunnel(): TunnelConf? {
return tunnels.findPrimary().firstOrNull() ?: tunnels.getAll().firstOrNull() return tunnels.findPrimary().firstOrNull() ?: tunnels.getAll().firstOrNull()
} }
override suspend fun getStartTunnelConfig(): TunnelConf? { override suspend fun getStartTunnelConfig(): TunnelConf? {
tunnels.getActive().let { tunnels.getActive().let {
if (it.isNotEmpty()) return it.first() if (it.isNotEmpty()) return it.first()
return getPrimaryOrFirstTunnel() return getPrimaryOrFirstTunnel()
} }
} }
} }
@@ -1,106 +1,157 @@
package com.zaneschepke.wireguardautotunnel.data.repository package com.zaneschepke.wireguardautotunnel.data.repository
import com.zaneschepke.wireguardautotunnel.data.DataStoreManager import com.zaneschepke.wireguardautotunnel.data.DataStoreManager
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.data.model.GeneralState import com.zaneschepke.wireguardautotunnel.data.model.GeneralState
import com.zaneschepke.wireguardautotunnel.domain.entity.AppState
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.ui.theme.Theme import com.zaneschepke.wireguardautotunnel.ui.theme.Theme
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import timber.log.Timber import timber.log.Timber
class DataStoreAppStateRepository( class DataStoreAppStateRepository(private val dataStoreManager: DataStoreManager) :
private val dataStoreManager: DataStoreManager, AppStateRepository {
) : override suspend fun isLocationDisclosureShown(): Boolean {
AppStateRepository { return dataStoreManager.getFromStore(DataStoreManager.locationDisclosureShown)
override suspend fun isLocationDisclosureShown(): Boolean { ?: GeneralState.LOCATION_DISCLOSURE_SHOWN_DEFAULT
return dataStoreManager.getFromStore(DataStoreManager.locationDisclosureShown) }
?: GeneralState.LOCATION_DISCLOSURE_SHOWN_DEFAULT
}
override suspend fun setLocationDisclosureShown(shown: Boolean) { override suspend fun setLocationDisclosureShown(shown: Boolean) {
dataStoreManager.saveToDataStore(DataStoreManager.locationDisclosureShown, shown) dataStoreManager.saveToDataStore(DataStoreManager.locationDisclosureShown, shown)
} }
override suspend fun isPinLockEnabled(): Boolean { override suspend fun isPinLockEnabled(): Boolean {
return dataStoreManager.getFromStore(DataStoreManager.pinLockEnabled) return dataStoreManager.getFromStore(DataStoreManager.pinLockEnabled)
?: GeneralState.PIN_LOCK_ENABLED_DEFAULT ?: GeneralState.PIN_LOCK_ENABLED_DEFAULT
} }
override suspend fun setPinLockEnabled(enabled: Boolean) { override suspend fun setPinLockEnabled(enabled: Boolean) {
dataStoreManager.saveToDataStore(DataStoreManager.pinLockEnabled, enabled) dataStoreManager.saveToDataStore(DataStoreManager.pinLockEnabled, enabled)
} }
override suspend fun isBatteryOptimizationDisableShown(): Boolean { override suspend fun isBatteryOptimizationDisableShown(): Boolean {
return dataStoreManager.getFromStore(DataStoreManager.batteryDisableShown) return dataStoreManager.getFromStore(DataStoreManager.batteryDisableShown)
?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT ?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT
} }
override suspend fun setBatteryOptimizationDisableShown(shown: Boolean) { override suspend fun setBatteryOptimizationDisableShown(shown: Boolean) {
dataStoreManager.saveToDataStore(DataStoreManager.batteryDisableShown, shown) dataStoreManager.saveToDataStore(DataStoreManager.batteryDisableShown, shown)
} }
override suspend fun isTunnelStatsExpanded(): Boolean { override suspend fun setTunnelExpanded(id: Int) {
return dataStoreManager.getFromStore(DataStoreManager.tunnelStatsExpanded) val ids =
?: GeneralState.IS_TUNNEL_STATS_EXPANDED dataStoreManager
} .getFromStore(DataStoreManager.expandedTunnelIds)
?.split(",")
?.mapNotNull { it.toIntOrNull() } ?: emptyList()
override suspend fun setTunnelStatsExpanded(expanded: Boolean) { if (ids.contains(id)) return
dataStoreManager.saveToDataStore(DataStoreManager.tunnelStatsExpanded, expanded)
}
override suspend fun setTheme(theme: Theme) { val updatedList = ids.toMutableList().apply { add(id) }
dataStoreManager.saveToDataStore(DataStoreManager.theme, theme.name) dataStoreManager.saveToDataStore(
} DataStoreManager.expandedTunnelIds,
updatedList.joinToString(","),
)
}
override suspend fun getTheme(): Theme { override suspend fun removeTunnelExpanded(id: Int) {
return dataStoreManager.getFromStore(DataStoreManager.theme)?.let { val ids =
try { dataStoreManager
Theme.valueOf(it) .getFromStore(DataStoreManager.expandedTunnelIds)
} catch (_: IllegalArgumentException) { ?.split(",")
Theme.AUTOMATIC ?.mapNotNull { it.toIntOrNull() } ?: emptyList()
}
} ?: Theme.AUTOMATIC
}
override suspend fun isLocalLogsEnabled(): Boolean { if (ids.isEmpty() || !ids.contains(id)) return
return dataStoreManager.getFromStore(DataStoreManager.isLocalLogsEnabled) ?: GeneralState.IS_LOGS_ENABLED_DEFAULT
}
override suspend fun setLocalLogsEnabled(enabled: Boolean) { val updatedList = ids.toMutableList().apply { remove(id) }
dataStoreManager.saveToDataStore(DataStoreManager.isLocalLogsEnabled, enabled) dataStoreManager.saveToDataStore(
} DataStoreManager.expandedTunnelIds,
updatedList.joinToString(","),
)
}
override suspend fun setLocale(localeTag: String) { override suspend fun setTheme(theme: Theme) {
dataStoreManager.saveToDataStore(DataStoreManager.locale, localeTag) dataStoreManager.saveToDataStore(DataStoreManager.theme, theme.name)
} }
override suspend fun getLocale(): String? { override suspend fun getTheme(): Theme {
return dataStoreManager.getFromStore(DataStoreManager.locale) return dataStoreManager.getFromStore(DataStoreManager.theme)?.let {
} try {
Theme.valueOf(it)
} catch (_: IllegalArgumentException) {
Theme.AUTOMATIC
}
} ?: Theme.AUTOMATIC
}
override val flow: Flow<GeneralState> = override suspend fun isLocalLogsEnabled(): Boolean {
dataStoreManager.preferencesFlow.map { prefs -> return dataStoreManager.getFromStore(DataStoreManager.isLocalLogsEnabled)
prefs?.let { pref -> ?: GeneralState.IS_LOGS_ENABLED_DEFAULT
try { }
GeneralState(
isLocationDisclosureShown = override suspend fun setLocalLogsEnabled(enabled: Boolean) {
pref[DataStoreManager.locationDisclosureShown] dataStoreManager.saveToDataStore(DataStoreManager.isLocalLogsEnabled, enabled)
?: GeneralState.LOCATION_DISCLOSURE_SHOWN_DEFAULT, }
isBatteryOptimizationDisableShown =
pref[DataStoreManager.batteryDisableShown] override suspend fun setLocale(localeTag: String) {
?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT, dataStoreManager.saveToDataStore(DataStoreManager.locale, localeTag)
isPinLockEnabled = }
pref[DataStoreManager.pinLockEnabled]
?: GeneralState.PIN_LOCK_ENABLED_DEFAULT, override suspend fun getLocale(): String? {
isTunnelStatsExpanded = pref[DataStoreManager.tunnelStatsExpanded] ?: GeneralState.IS_TUNNEL_STATS_EXPANDED, return dataStoreManager.getFromStore(DataStoreManager.locale)
isLocalLogsEnabled = pref[DataStoreManager.isLocalLogsEnabled] ?: GeneralState.IS_LOGS_ENABLED_DEFAULT, }
locale = pref[DataStoreManager.locale],
theme = getTheme(), override suspend fun setIsRemoteControlEnabled(enabled: Boolean) {
) dataStoreManager.saveToDataStore(DataStoreManager.isRemoteControlEnabled, enabled)
} catch (e: IllegalArgumentException) { }
Timber.e(e)
GeneralState() override suspend fun isRemoteControlEnabled(): Boolean {
} return dataStoreManager.getFromStore(DataStoreManager.isRemoteControlEnabled)
} ?: GeneralState() ?: GeneralState.IS_REMOTE_CONTROL_ENABLED
} }
override suspend fun setRemoteKey(key: String) {
dataStoreManager.saveToDataStore(DataStoreManager.remoteKey, key)
}
override suspend fun getRemoteKey(): String? {
return dataStoreManager.getFromStore(DataStoreManager.remoteKey)
}
override val flow: Flow<AppState> =
dataStoreManager.preferencesFlow
.map { prefs ->
prefs?.let { pref ->
try {
GeneralState(
isLocationDisclosureShown =
pref[DataStoreManager.locationDisclosureShown]
?: GeneralState.LOCATION_DISCLOSURE_SHOWN_DEFAULT,
isBatteryOptimizationDisableShown =
pref[DataStoreManager.batteryDisableShown]
?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT,
isPinLockEnabled =
pref[DataStoreManager.pinLockEnabled]
?: GeneralState.PIN_LOCK_ENABLED_DEFAULT,
expandedTunnelIds =
pref[DataStoreManager.expandedTunnelIds]?.split(",")?.mapNotNull {
it.toIntOrNull()
} ?: emptyList(),
isLocalLogsEnabled =
pref[DataStoreManager.isLocalLogsEnabled]
?: GeneralState.IS_LOGS_ENABLED_DEFAULT,
isRemoteControlEnabled =
pref[DataStoreManager.isRemoteControlEnabled]
?: GeneralState.IS_REMOTE_CONTROL_ENABLED,
remoteKey = pref[DataStoreManager.remoteKey],
locale = pref[DataStoreManager.locale],
theme = getTheme(),
)
} catch (e: IllegalArgumentException) {
Timber.e(e)
GeneralState()
}
} ?: GeneralState()
}
.map { it.toAppState() }
} }
@@ -0,0 +1,98 @@
package com.zaneschepke.wireguardautotunnel.data.repository
import android.content.Context
import com.zaneschepke.wireguardautotunnel.BuildConfig
import com.zaneschepke.wireguardautotunnel.data.network.GitHubApi
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.AppUpdate
import com.zaneschepke.wireguardautotunnel.domain.repository.UpdateRepository
import com.zaneschepke.wireguardautotunnel.util.NumberUtils
import io.ktor.client.HttpClient
import io.ktor.client.request.get
import io.ktor.client.statement.HttpResponse
import io.ktor.client.statement.bodyAsChannel
import io.ktor.http.contentLength
import io.ktor.utils.io.ByteReadChannel
import io.ktor.utils.io.readAvailable
import java.io.File
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.withContext
import timber.log.Timber
class GitHubUpdateRepository(
private val gitHubApi: GitHubApi,
private val httpClient: HttpClient,
private val githubOwner: String,
private val githubRepo: String,
private val context: Context,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) : UpdateRepository {
override suspend fun checkForUpdate(currentVersion: String): Result<AppUpdate?> =
withContext(ioDispatcher) {
Timber.i("Checking for update")
val release =
if (BuildConfig.VERSION_NAME.contains("nightly")) {
gitHubApi.getNightlyRelease(githubOwner, githubRepo)
} else {
gitHubApi.getLatestRelease(githubOwner, githubRepo)
}
release.map { release ->
val apkAsset =
release.assets.find { asset ->
asset.name.startsWith("wgtunnel-full-v") && asset.name.endsWith(".apk")
}
val newVersion =
apkAsset?.name?.removePrefix("wgtunnel-full-v")?.removeSuffix(".apk")
?: return@map null
Timber.i("Latest version: $newVersion, current version: $currentVersion")
if (NumberUtils.compareVersions(newVersion, currentVersion) > 0) {
release.toAppUpdate()
} else {
null
}
}
}
override suspend fun downloadApk(
apkUrl: String,
fileName: String,
onProgress: (Float) -> Unit,
): Result<File> =
withContext(ioDispatcher) {
try {
// clean up old files
context.getExternalFilesDir(null)?.listFiles()?.forEach { file ->
if (file.extension == "apk") file.delete()
}
val response: HttpResponse = httpClient.get(apkUrl)
val apkFile = File(context.getExternalFilesDir(null), fileName)
val channel: ByteReadChannel = response.bodyAsChannel()
val totalBytes: Long = response.contentLength() ?: -1L
var bytesCopied = 0L
apkFile.outputStream().use { output ->
val buffer = ByteArray(8 * 1024)
while (!channel.isClosedForRead) {
val bytesRead = channel.readAvailable(buffer)
if (bytesRead <= 0) break
output.write(buffer, 0, bytesRead)
bytesCopied += bytesRead
if (totalBytes > 0) {
val progress = bytesCopied.toFloat() / totalBytes
onProgress(progress.coerceIn(0f, 1f))
}
}
}
Result.success(apkFile)
} catch (e: Exception) {
Result.failure(e)
}
}
}
@@ -1,31 +1,30 @@
package com.zaneschepke.wireguardautotunnel.data.repository package com.zaneschepke.wireguardautotunnel.data.repository
import com.zaneschepke.wireguardautotunnel.data.dao.SettingsDao import com.zaneschepke.wireguardautotunnel.data.dao.SettingsDao
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import com.zaneschepke.wireguardautotunnel.data.model.Settings import com.zaneschepke.wireguardautotunnel.data.model.Settings
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.flow.flowOn import kotlinx.coroutines.flow.flowOn
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
class RoomSettingsRepository( class RoomSettingsRepository(
private val settingsDoa: SettingsDao, private val settingsDoa: SettingsDao,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) : AppSettingRepository { ) : AppSettingRepository {
override suspend fun save(appSettings: AppSettings) { override suspend fun save(appSettings: AppSettings) {
withContext(ioDispatcher) { withContext(ioDispatcher) { settingsDoa.save(Settings.from(appSettings)) }
settingsDoa.save(Settings.from(appSettings)) }
}
}
override val flow = settingsDoa.getSettingsFlow().flowOn(ioDispatcher).map { it.toAppSettings() } override val flow =
settingsDoa.getSettingsFlow().flowOn(ioDispatcher).map { it.toAppSettings() }
override suspend fun get(): AppSettings { override suspend fun get(): AppSettings {
return withContext(ioDispatcher) { return withContext(ioDispatcher) {
(settingsDoa.getAll().firstOrNull() ?: Settings()).toAppSettings() (settingsDoa.getAll().firstOrNull() ?: Settings()).toAppSettings()
} }
} }
} }
@@ -1,10 +1,10 @@
package com.zaneschepke.wireguardautotunnel.data.repository package com.zaneschepke.wireguardautotunnel.data.repository
import com.zaneschepke.wireguardautotunnel.data.dao.TunnelConfigDao import com.zaneschepke.wireguardautotunnel.data.dao.TunnelConfigDao
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.data.model.TunnelConfig import com.zaneschepke.wireguardautotunnel.data.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.di.IoDispatcher import com.zaneschepke.wireguardautotunnel.di.IoDispatcher
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.util.extensions.Tunnels import com.zaneschepke.wireguardautotunnel.util.extensions.Tunnels
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.flow.flowOn import kotlinx.coroutines.flow.flowOn
@@ -12,96 +12,81 @@ import kotlinx.coroutines.flow.map
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
class RoomTunnelRepository( class RoomTunnelRepository(
private val tunnelConfigDao: TunnelConfigDao, private val tunnelConfigDao: TunnelConfigDao,
@IoDispatcher private val ioDispatcher: CoroutineDispatcher, @IoDispatcher private val ioDispatcher: CoroutineDispatcher,
) : TunnelRepository { ) : TunnelRepository {
override val flow = tunnelConfigDao.getAllFlow().flowOn(ioDispatcher).map { it.map { it.toTunnel() } } override val flow =
tunnelConfigDao.getAllFlow().flowOn(ioDispatcher).map { it.map { it.toTunnel() } }
override suspend fun getAll(): Tunnels { override suspend fun getAll(): Tunnels {
return withContext(ioDispatcher) { return withContext(ioDispatcher) { tunnelConfigDao.getAll().map { it.toTunnel() } }
tunnelConfigDao.getAll().map { it.toTunnel() } }
}
}
override suspend fun save(tunnelConf: TunnelConf) { override suspend fun save(tunnelConf: TunnelConf) {
withContext(ioDispatcher) { withContext(ioDispatcher) { tunnelConfigDao.save(TunnelConfig.from(tunnelConf)) }
tunnelConfigDao.save(TunnelConfig.from(tunnelConf)) }
}
}
override suspend fun updatePrimaryTunnel(tunnelConf: TunnelConf?) { override suspend fun saveAll(tunnelConfList: List<TunnelConf>) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
tunnelConfigDao.resetPrimaryTunnel() tunnelConfigDao.saveAll(tunnelConfList.map(TunnelConfig::from))
tunnelConf?.let { }
save( }
it.copy(
isPrimaryTunnel = true,
),
)
}
}
}
override suspend fun updateMobileDataTunnel(tunnelConf: TunnelConf?) { override suspend fun updatePrimaryTunnel(tunnelConf: TunnelConf?) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
tunnelConfigDao.resetMobileDataTunnel() tunnelConfigDao.resetPrimaryTunnel()
tunnelConf?.let { tunnelConf?.let { save(it.copy(isPrimaryTunnel = true)) }
save( }
it.copy( }
isMobileDataTunnel = true,
),
)
}
}
}
override suspend fun updateEthernetTunnel(tunnelConf: TunnelConf?) { override suspend fun updateMobileDataTunnel(tunnelConf: TunnelConf?) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
tunnelConfigDao.resetEthernetTunnel() tunnelConfigDao.resetMobileDataTunnel()
tunnelConf?.let { tunnelConf?.let { save(it.copy(isMobileDataTunnel = true)) }
save( }
it.copy( }
isEthernetTunnel = true,
),
)
}
}
}
override suspend fun delete(tunnelConf: TunnelConf) { override suspend fun updateEthernetTunnel(tunnelConf: TunnelConf?) {
withContext(ioDispatcher) { withContext(ioDispatcher) {
tunnelConfigDao.delete(TunnelConfig.from(tunnelConf)) tunnelConfigDao.resetEthernetTunnel()
} tunnelConf?.let { save(it.copy(isEthernetTunnel = true)) }
} }
}
override suspend fun getById(id: Int): TunnelConf? { override suspend fun delete(tunnelConf: TunnelConf) {
return withContext(ioDispatcher) { tunnelConfigDao.getById(id.toLong())?.toTunnel() } withContext(ioDispatcher) { tunnelConfigDao.delete(TunnelConfig.from(tunnelConf)) }
} }
override suspend fun getActive(): Tunnels { override suspend fun getById(id: Int): TunnelConf? {
return withContext(ioDispatcher) { return withContext(ioDispatcher) { tunnelConfigDao.getById(id.toLong())?.toTunnel() }
tunnelConfigDao.getActive().map { it.toTunnel() } }
}
}
override suspend fun count(): Int { override suspend fun getActive(): Tunnels {
return withContext(ioDispatcher) { tunnelConfigDao.count().toInt() } return withContext(ioDispatcher) { tunnelConfigDao.getActive().map { it.toTunnel() } }
} }
override suspend fun findByTunnelName(name: String): TunnelConf? { override suspend fun count(): Int {
return withContext(ioDispatcher) { tunnelConfigDao.getByName(name)?.toTunnel() } return withContext(ioDispatcher) { tunnelConfigDao.count().toInt() }
} }
override suspend fun findByTunnelNetworksName(name: String): Tunnels { override suspend fun findByTunnelName(name: String): TunnelConf? {
return withContext(ioDispatcher) { tunnelConfigDao.findByTunnelNetworkName(name).map { it.toTunnel() } } return withContext(ioDispatcher) { tunnelConfigDao.getByName(name)?.toTunnel() }
} }
override suspend fun findByMobileDataTunnel(): Tunnels { override suspend fun findByTunnelNetworksName(name: String): Tunnels {
return withContext(ioDispatcher) { tunnelConfigDao.findByMobileDataTunnel().map { it.toTunnel() } } return withContext(ioDispatcher) {
} tunnelConfigDao.findByTunnelNetworkName(name).map { it.toTunnel() }
}
}
override suspend fun findPrimary(): Tunnels { override suspend fun findByMobileDataTunnel(): Tunnels {
return withContext(ioDispatcher) { tunnelConfigDao.findByPrimary().map { it.toTunnel() } } return withContext(ioDispatcher) {
} tunnelConfigDao.findByMobileDataTunnel().map { it.toTunnel() }
}
}
override suspend fun findPrimary(): Tunnels {
return withContext(ioDispatcher) { tunnelConfigDao.findByPrimary().map { it.toTunnel() } }
}
} }
@@ -12,35 +12,39 @@ import dagger.Provides
import dagger.hilt.InstallIn import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent import dagger.hilt.components.SingletonComponent
import javax.inject.Singleton
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.SupervisorJob import kotlinx.coroutines.SupervisorJob
import javax.inject.Singleton
@Module @Module
@InstallIn(SingletonComponent::class) @InstallIn(SingletonComponent::class)
class AppModule { class AppModule {
@Singleton @Singleton
@ApplicationScope @ApplicationScope
@Provides @Provides
fun providesApplicationScope(@DefaultDispatcher defaultDispatcher: CoroutineDispatcher): CoroutineScope = fun providesApplicationScope(
CoroutineScope(SupervisorJob() + defaultDispatcher) @DefaultDispatcher defaultDispatcher: CoroutineDispatcher
): CoroutineScope = CoroutineScope(SupervisorJob() + defaultDispatcher)
@Singleton @Singleton
@Provides @Provides
fun provideLogCollect(@ApplicationContext context: Context): LogReader { fun provideLogCollect(@ApplicationContext context: Context): LogReader {
return LogcatReader.init(storageDir = context.filesDir.absolutePath) return LogcatReader.init(storageDir = context.filesDir.absolutePath)
} }
@Singleton @Singleton
@Provides @Provides
fun provideNotificationService(@ApplicationContext context: Context): NotificationManager { fun provideNotificationService(@ApplicationContext context: Context): NotificationManager {
return WireGuardNotification(context) return WireGuardNotification(context)
} }
@Singleton @Singleton
@Provides @Provides
fun provideShortcutManager(@ApplicationContext context: Context, @IoDispatcher ioDispatcher: CoroutineDispatcher): ShortcutManager { fun provideShortcutManager(
return DynamicShortcutManager(context, ioDispatcher) @ApplicationContext context: Context,
} @IoDispatcher ioDispatcher: CoroutineDispatcher,
): ShortcutManager {
return DynamicShortcutManager(context, ioDispatcher)
}
} }
@@ -2,18 +2,10 @@ package com.zaneschepke.wireguardautotunnel.di
import javax.inject.Qualifier import javax.inject.Qualifier
@Qualifier @Qualifier @Retention(AnnotationRetention.BINARY) annotation class TunnelShell
@Retention(AnnotationRetention.BINARY)
annotation class TunnelShell
@Qualifier @Qualifier @Retention(AnnotationRetention.BINARY) annotation class AppShell
@Retention(AnnotationRetention.BINARY)
annotation class AppShell
@Qualifier @Qualifier @Retention(AnnotationRetention.BINARY) annotation class Kernel
@Retention(AnnotationRetention.BINARY)
annotation class Kernel
@Qualifier @Qualifier @Retention(AnnotationRetention.BINARY) annotation class Userspace
@Retention(AnnotationRetention.BINARY)
annotation class Userspace
@@ -2,26 +2,14 @@ package com.zaneschepke.wireguardautotunnel.di
import javax.inject.Qualifier import javax.inject.Qualifier
@Retention(AnnotationRetention.RUNTIME) @Retention(AnnotationRetention.RUNTIME) @Qualifier annotation class DefaultDispatcher
@Qualifier
annotation class DefaultDispatcher
@Retention(AnnotationRetention.RUNTIME) @Retention(AnnotationRetention.RUNTIME) @Qualifier annotation class IoDispatcher
@Qualifier
annotation class IoDispatcher
@Retention(AnnotationRetention.RUNTIME) @Retention(AnnotationRetention.RUNTIME) @Qualifier annotation class MainDispatcher
@Qualifier
annotation class MainDispatcher
@Retention(AnnotationRetention.BINARY) @Retention(AnnotationRetention.BINARY) @Qualifier annotation class MainImmediateDispatcher
@Qualifier
annotation class MainImmediateDispatcher
@Retention(AnnotationRetention.RUNTIME) @Retention(AnnotationRetention.RUNTIME) @Qualifier annotation class ApplicationScope
@Qualifier
annotation class ApplicationScope
@Retention(AnnotationRetention.RUNTIME) @Retention(AnnotationRetention.RUNTIME) @Qualifier annotation class ServiceScope
@Qualifier
annotation class ServiceScope
@@ -10,19 +10,15 @@ import kotlinx.coroutines.Dispatchers
@Module @Module
@InstallIn(SingletonComponent::class) @InstallIn(SingletonComponent::class)
object CoroutinesDispatchersModule { object CoroutinesDispatchersModule {
@DefaultDispatcher @DefaultDispatcher
@Provides @Provides
fun providesDefaultDispatcher(): CoroutineDispatcher = Dispatchers.Default fun providesDefaultDispatcher(): CoroutineDispatcher = Dispatchers.Default
@IoDispatcher @IoDispatcher @Provides fun providesIoDispatcher(): CoroutineDispatcher = Dispatchers.IO
@Provides
fun providesIoDispatcher(): CoroutineDispatcher = Dispatchers.IO
@MainDispatcher @MainDispatcher @Provides fun providesMainDispatcher(): CoroutineDispatcher = Dispatchers.Main
@Provides
fun providesMainDispatcher(): CoroutineDispatcher = Dispatchers.Main
@MainImmediateDispatcher @MainImmediateDispatcher
@Provides @Provides
fun providesMainImmediateDispatcher(): CoroutineDispatcher = Dispatchers.Main.immediate fun providesMainImmediateDispatcher(): CoroutineDispatcher = Dispatchers.Main.immediate
} }
@@ -4,85 +4,130 @@ import android.content.Context
import androidx.room.Room import androidx.room.Room
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.data.AppDatabase import com.zaneschepke.wireguardautotunnel.data.AppDatabase
import com.zaneschepke.wireguardautotunnel.data.DataStoreManager
import com.zaneschepke.wireguardautotunnel.data.DatabaseCallback import com.zaneschepke.wireguardautotunnel.data.DatabaseCallback
import com.zaneschepke.wireguardautotunnel.data.dao.SettingsDao import com.zaneschepke.wireguardautotunnel.data.dao.SettingsDao
import com.zaneschepke.wireguardautotunnel.data.dao.TunnelConfigDao import com.zaneschepke.wireguardautotunnel.data.dao.TunnelConfigDao
import com.zaneschepke.wireguardautotunnel.data.DataStoreManager import com.zaneschepke.wireguardautotunnel.data.network.GitHubApi
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.data.network.KtorClient
import com.zaneschepke.wireguardautotunnel.data.network.KtorGitHubApi
import com.zaneschepke.wireguardautotunnel.data.repository.AppDataRoomRepository import com.zaneschepke.wireguardautotunnel.data.repository.AppDataRoomRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.data.repository.DataStoreAppStateRepository import com.zaneschepke.wireguardautotunnel.data.repository.DataStoreAppStateRepository
import com.zaneschepke.wireguardautotunnel.data.repository.GitHubUpdateRepository
import com.zaneschepke.wireguardautotunnel.data.repository.RoomSettingsRepository import com.zaneschepke.wireguardautotunnel.data.repository.RoomSettingsRepository
import com.zaneschepke.wireguardautotunnel.data.repository.RoomTunnelRepository import com.zaneschepke.wireguardautotunnel.data.repository.RoomTunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppStateRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.UpdateRepository
import dagger.Module import dagger.Module
import dagger.Provides import dagger.Provides
import dagger.hilt.InstallIn import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent import dagger.hilt.components.SingletonComponent
import kotlinx.coroutines.CoroutineDispatcher import io.ktor.client.HttpClient
import javax.inject.Singleton import javax.inject.Singleton
import kotlinx.coroutines.CoroutineDispatcher
@Module @Module
@InstallIn(SingletonComponent::class) @InstallIn(SingletonComponent::class)
class RepositoryModule { class RepositoryModule {
@Provides @Provides
@Singleton @Singleton
fun provideDatabase(@ApplicationContext context: Context): AppDatabase { fun provideDatabase(@ApplicationContext context: Context): AppDatabase {
return Room.databaseBuilder( return Room.databaseBuilder(
context, context,
AppDatabase::class.java, AppDatabase::class.java,
context.getString(R.string.db_name), context.getString(R.string.db_name),
) )
.fallbackToDestructiveMigration() .fallbackToDestructiveMigration(true)
.addCallback(DatabaseCallback()) .addCallback(DatabaseCallback())
.build() .build()
} }
@Singleton @Singleton
@Provides @Provides
fun provideSettingsDoa(appDatabase: AppDatabase): SettingsDao { fun provideSettingsDoa(appDatabase: AppDatabase): SettingsDao {
return appDatabase.settingDao() return appDatabase.settingDao()
} }
@Singleton @Singleton
@Provides @Provides
fun provideTunnelConfigDoa(appDatabase: AppDatabase): TunnelConfigDao { fun provideTunnelConfigDoa(appDatabase: AppDatabase): TunnelConfigDao {
return appDatabase.tunnelConfigDoa() return appDatabase.tunnelConfigDoa()
} }
@Singleton @Singleton
@Provides @Provides
fun provideTunnelConfigRepository(tunnelConfigDao: TunnelConfigDao, @IoDispatcher ioDispatcher: CoroutineDispatcher): TunnelRepository { fun provideTunnelConfigRepository(
return RoomTunnelRepository(tunnelConfigDao, ioDispatcher) tunnelConfigDao: TunnelConfigDao,
} @IoDispatcher ioDispatcher: CoroutineDispatcher,
): TunnelRepository {
return RoomTunnelRepository(tunnelConfigDao, ioDispatcher)
}
@Singleton @Singleton
@Provides @Provides
fun provideSettingsRepository(settingsDao: SettingsDao, @IoDispatcher ioDispatcher: CoroutineDispatcher): AppSettingRepository { fun provideSettingsRepository(
return RoomSettingsRepository(settingsDao, ioDispatcher) settingsDao: SettingsDao,
} @IoDispatcher ioDispatcher: CoroutineDispatcher,
): AppSettingRepository {
return RoomSettingsRepository(settingsDao, ioDispatcher)
}
@Singleton @Singleton
@Provides @Provides
fun providePreferencesDataStore(@ApplicationContext context: Context, @IoDispatcher ioDispatcher: CoroutineDispatcher): DataStoreManager { fun providePreferencesDataStore(
return DataStoreManager(context, ioDispatcher) @ApplicationContext context: Context,
} @IoDispatcher ioDispatcher: CoroutineDispatcher,
): DataStoreManager {
return DataStoreManager(context, ioDispatcher)
}
@Provides @Provides
@Singleton @Singleton
fun provideGeneralStateRepository(dataStoreManager: DataStoreManager): AppStateRepository { fun provideGeneralStateRepository(dataStoreManager: DataStoreManager): AppStateRepository {
return DataStoreAppStateRepository(dataStoreManager) return DataStoreAppStateRepository(dataStoreManager)
} }
@Provides @Provides
@Singleton @Singleton
fun provideAppDataRepository( fun provideAppDataRepository(
settingsRepository: AppSettingRepository, settingsRepository: AppSettingRepository,
tunnelRepository: TunnelRepository, tunnelRepository: TunnelRepository,
appStateRepository: AppStateRepository, appStateRepository: AppStateRepository,
): AppDataRepository { ): AppDataRepository {
return AppDataRoomRepository(settingsRepository, tunnelRepository, appStateRepository) return AppDataRoomRepository(settingsRepository, tunnelRepository, appStateRepository)
} }
@Provides
@Singleton
fun provideHttpClient(): HttpClient {
return KtorClient.create()
}
@Provides
@Singleton
fun provideGitHubApi(client: HttpClient): GitHubApi {
return KtorGitHubApi(client)
}
@Provides
@Singleton
fun provideUpdateRepository(
gitHubApi: GitHubApi,
client: HttpClient,
@IoDispatcher ioDispatcher: CoroutineDispatcher,
@ApplicationContext context: Context,
): UpdateRepository {
return GitHubUpdateRepository(
gitHubApi,
client,
"wgtunnel",
"wgtunnel",
context,
ioDispatcher,
)
}
} }
@@ -1,16 +0,0 @@
package com.zaneschepke.wireguardautotunnel.di
import com.zaneschepke.wireguardautotunnel.core.network.InternetConnectivityMonitor
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor
import dagger.Binds
import dagger.Module
import dagger.hilt.InstallIn
import dagger.hilt.components.SingletonComponent
@Module
@InstallIn(SingletonComponent::class)
abstract class ServiceModule {
@Binds
abstract fun provideInternetConnectivityService(wifiService: InternetConnectivityMonitor): NetworkMonitor
}
@@ -4,107 +4,135 @@ import android.content.Context
import com.wireguard.android.backend.WgQuickBackend import com.wireguard.android.backend.WgQuickBackend
import com.wireguard.android.util.RootShell import com.wireguard.android.util.RootShell
import com.wireguard.android.util.ToolsInstaller import com.wireguard.android.util.ToolsInstaller
import com.zaneschepke.wireguardautotunnel.core.network.NetworkMonitor import com.zaneschepke.networkmonitor.AndroidNetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationManager import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.KernelTunnel import com.zaneschepke.wireguardautotunnel.core.tunnel.KernelTunnel
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.core.tunnel.UserspaceTunnel import com.zaneschepke.wireguardautotunnel.core.tunnel.UserspaceTunnel
import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository import com.zaneschepke.wireguardautotunnel.domain.repository.AppDataRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.AppSettingRepository
import dagger.Module import dagger.Module
import dagger.Provides import dagger.Provides
import dagger.hilt.InstallIn import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent import dagger.hilt.components.SingletonComponent
import javax.inject.Singleton
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.runBlocking
import org.amnezia.awg.backend.Backend import org.amnezia.awg.backend.Backend
import org.amnezia.awg.backend.GoBackend import org.amnezia.awg.backend.GoBackend
import org.amnezia.awg.backend.RootTunnelActionHandler import org.amnezia.awg.backend.RootTunnelActionHandler
import javax.inject.Singleton
@Module @Module
@InstallIn(SingletonComponent::class) @InstallIn(SingletonComponent::class)
class TunnelModule { class TunnelModule {
@Provides @Provides
@Singleton @Singleton
@TunnelShell @TunnelShell
fun provideTunnelRootShell(@ApplicationContext context: Context): RootShell { fun provideTunnelRootShell(@ApplicationContext context: Context): RootShell {
return RootShell(context) return RootShell(context)
} }
@Provides @Provides
@Singleton @Singleton
@AppShell @AppShell
fun provideAppRootShell(@ApplicationContext context: Context): RootShell { fun provideAppRootShell(@ApplicationContext context: Context): RootShell {
return RootShell(context) return RootShell(context)
} }
@Provides @Provides
@Singleton @Singleton
fun provideAmneziaBackend(@ApplicationContext context: Context): Backend { fun provideAmneziaBackend(@ApplicationContext context: Context): Backend {
return GoBackend(context, RootTunnelActionHandler(org.amnezia.awg.util.RootShell(context))) return GoBackend(context, RootTunnelActionHandler(org.amnezia.awg.util.RootShell(context)))
} }
@Provides @Provides
@Singleton @Singleton
fun provideKernelBackend(@ApplicationContext context: Context, @TunnelShell shell: RootShell): com.wireguard.android.backend.Backend { fun provideKernelBackend(
return WgQuickBackend(context, shell, ToolsInstaller(context, shell), com.wireguard.android.backend.RootTunnelActionHandler(shell)).also { @ApplicationContext context: Context,
it.setMultipleTunnels(true) @TunnelShell shell: RootShell,
} ): com.wireguard.android.backend.Backend {
} return WgQuickBackend(
context,
shell,
ToolsInstaller(context, shell),
com.wireguard.android.backend.RootTunnelActionHandler(shell),
)
.also { it.setMultipleTunnels(true) }
}
@Provides @Provides
@Singleton @Singleton
@Kernel @Kernel
fun provideKernelProvider( fun provideKernelProvider(
@IoDispatcher ioDispatcher: CoroutineDispatcher, @ApplicationScope applicationScope: CoroutineScope,
@ApplicationScope applicationScope: CoroutineScope, serviceManager: ServiceManager,
serviceManager: ServiceManager, appDataRepository: AppDataRepository,
appDataRepository: AppDataRepository, backend: com.wireguard.android.backend.Backend,
networkMonitor: NetworkMonitor, ): TunnelProvider {
notificationManager: NotificationManager, return KernelTunnel(applicationScope, serviceManager, appDataRepository, backend)
backend: com.wireguard.android.backend.Backend, }
): TunnelProvider {
return KernelTunnel(ioDispatcher, applicationScope, serviceManager, appDataRepository, notificationManager, backend, networkMonitor)
}
@Provides @Provides
@Singleton @Singleton
@Userspace @Userspace
fun provideUserspaceProvider( fun provideUserspaceProvider(
@IoDispatcher ioDispatcher: CoroutineDispatcher, @ApplicationScope applicationScope: CoroutineScope,
@ApplicationScope applicationScope: CoroutineScope, serviceManager: ServiceManager,
serviceManager: ServiceManager, appDataRepository: AppDataRepository,
appDataRepository: AppDataRepository, backend: Backend,
notificationManager: NotificationManager, ): TunnelProvider {
networkMonitor: NetworkMonitor, return UserspaceTunnel(applicationScope, serviceManager, appDataRepository, backend)
backend: Backend, }
): TunnelProvider {
return UserspaceTunnel(ioDispatcher, applicationScope, serviceManager, appDataRepository, notificationManager, backend, networkMonitor)
}
@Provides @Provides
@Singleton @Singleton
fun provideTunnelManager( fun provideTunnelManager(
@Kernel kernelTunnel: TunnelProvider, @Kernel kernelTunnel: TunnelProvider,
@Userspace userspaceTunnel: TunnelProvider, @Userspace userspaceTunnel: TunnelProvider,
appDataRepository: AppDataRepository, appDataRepository: AppDataRepository,
@IoDispatcher ioDispatcher: CoroutineDispatcher, @IoDispatcher ioDispatcher: CoroutineDispatcher,
@ApplicationScope applicationScope: CoroutineScope, @ApplicationScope applicationScope: CoroutineScope,
): TunnelManager { ): TunnelManager {
return TunnelManager(kernelTunnel, userspaceTunnel, appDataRepository, applicationScope, ioDispatcher) return TunnelManager(
} kernelTunnel,
userspaceTunnel,
appDataRepository,
applicationScope,
ioDispatcher,
)
}
@Singleton @Provides
@Provides @Singleton
fun provideServiceManager( fun provideNetworkMonitor(
@ApplicationContext context: Context, @ApplicationContext context: Context,
@IoDispatcher ioDispatcher: CoroutineDispatcher, settingsRepository: AppSettingRepository,
appDataRepository: AppDataRepository, ): NetworkMonitor {
): ServiceManager { return AndroidNetworkMonitor(context) {
return ServiceManager(context, ioDispatcher, appDataRepository) runBlocking { settingsRepository.get().isWifiNameByShellEnabled }
} }
}
@Singleton
@Provides
fun provideServiceManager(
@ApplicationContext context: Context,
@IoDispatcher ioDispatcher: CoroutineDispatcher,
@MainDispatcher mainCoroutineDispatcher: CoroutineDispatcher,
@ApplicationScope applicationScope: CoroutineScope,
appDataRepository: AppDataRepository,
): ServiceManager {
return ServiceManager(
context,
ioDispatcher,
applicationScope,
mainCoroutineDispatcher,
appDataRepository,
)
}
} }
@@ -13,9 +13,12 @@ import kotlinx.coroutines.CoroutineDispatcher
@Module @Module
@InstallIn(ViewModelComponent::class) @InstallIn(ViewModelComponent::class)
class ViewModelModule { class ViewModelModule {
@ViewModelScoped @ViewModelScoped
@Provides @Provides
fun provideFileUtils(@ApplicationContext context: Context, @IoDispatcher ioDispatcher: CoroutineDispatcher): FileUtils { fun provideFileUtils(
return FileUtils(context, ioDispatcher) @ApplicationContext context: Context,
} @IoDispatcher ioDispatcher: CoroutineDispatcher,
): FileUtils {
return FileUtils(context, ioDispatcher)
}
} }
@@ -1,29 +1,29 @@
package com.zaneschepke.wireguardautotunnel.domain.entity package com.zaneschepke.wireguardautotunnel.domain.entity
data class AppSettings( data class AppSettings(
val id: Int = 0, val id: Int = 0,
val isAutoTunnelEnabled: Boolean = false, val isAutoTunnelEnabled: Boolean = false,
val isTunnelOnMobileDataEnabled: Boolean = false, val isTunnelOnMobileDataEnabled: Boolean = false,
val trustedNetworkSSIDs: List<String> = emptyList(), val trustedNetworkSSIDs: List<String> = emptyList(),
val isAlwaysOnVpnEnabled: Boolean = false, val isAlwaysOnVpnEnabled: Boolean = false,
val isTunnelOnEthernetEnabled: Boolean = false, val isTunnelOnEthernetEnabled: Boolean = false,
val isShortcutsEnabled: Boolean = false, val isShortcutsEnabled: Boolean = false,
val isTunnelOnWifiEnabled: Boolean = false, val isTunnelOnWifiEnabled: Boolean = false,
val isKernelEnabled: Boolean = false, val isKernelEnabled: Boolean = false,
val isRestoreOnBootEnabled: Boolean = false, val isRestoreOnBootEnabled: Boolean = false,
val isMultiTunnelEnabled: Boolean = false, val isMultiTunnelEnabled: Boolean = false,
val isPingEnabled: Boolean = false, val isPingEnabled: Boolean = false,
val isAmneziaEnabled: Boolean = false, val isAmneziaEnabled: Boolean = false,
val isWildcardsEnabled: Boolean = false, val isWildcardsEnabled: Boolean = false,
val isWifiNameByShellEnabled: Boolean = false, val isWifiNameByShellEnabled: Boolean = false,
val isStopOnNoInternetEnabled: Boolean = false, val isStopOnNoInternetEnabled: Boolean = false,
val isVpnKillSwitchEnabled: Boolean = false, val isVpnKillSwitchEnabled: Boolean = false,
val isKernelKillSwitchEnabled: Boolean = false, val isKernelKillSwitchEnabled: Boolean = false,
val isLanOnKillSwitchEnabled: Boolean = false, val isLanOnKillSwitchEnabled: Boolean = false,
val debounceDelaySeconds: Int = 3, val debounceDelaySeconds: Int = 3,
val isDisableKillSwitchOnTrustedEnabled: Boolean = false, val isDisableKillSwitchOnTrustedEnabled: Boolean = false,
) { ) {
fun debounceDelayMillis(): Long { fun debounceDelayMillis(): Long {
return debounceDelaySeconds * 1000L return debounceDelaySeconds * 1000L
} }
} }
@@ -3,11 +3,13 @@ package com.zaneschepke.wireguardautotunnel.domain.entity
import com.zaneschepke.wireguardautotunnel.ui.theme.Theme import com.zaneschepke.wireguardautotunnel.ui.theme.Theme
data class AppState( data class AppState(
val isLocationDisclosureShown: Boolean, val isLocationDisclosureShown: Boolean,
val isBatteryOptimizationDisableShown: Boolean, val isBatteryOptimizationDisableShown: Boolean,
val isPinLockEnabled: Boolean, val isPinLockEnabled: Boolean,
val isTunnelStatsExpanded: Boolean, val expandedTunnelIds: List<Int>,
val isLocalLogsEnabled: Boolean, val isLocalLogsEnabled: Boolean,
val locale: String?, val isRemoteControlEnabled: Boolean,
val theme: Theme, val remoteKey: String?,
val locale: String?,
val theme: Theme,
) )
@@ -0,0 +1,9 @@
package com.zaneschepke.wireguardautotunnel.domain.entity
data class AppUpdate(
val version: String,
val title: String,
val releaseNotes: String,
val apkUrl: String?,
val apkFileName: String?,
)
@@ -1,126 +1,219 @@
package com.zaneschepke.wireguardautotunnel.domain.entity package com.zaneschepke.wireguardautotunnel.domain.entity
import com.wireguard.android.backend.Tunnel
import com.wireguard.config.Config import com.wireguard.config.Config
import com.zaneschepke.wireguardautotunnel.domain.state.TunnelState
import com.zaneschepke.wireguardautotunnel.util.Constants import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.extensions.asTunnelState import com.zaneschepke.wireguardautotunnel.util.extensions.*
import com.zaneschepke.wireguardautotunnel.util.extensions.isReachable
import com.zaneschepke.wireguardautotunnel.util.extensions.toWgQuickString
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.update
import kotlinx.coroutines.withContext
import org.amnezia.awg.backend.Tunnel
import timber.log.Timber
import java.io.InputStream import java.io.InputStream
import java.net.InetAddress import java.net.InetAddress
import java.nio.charset.StandardCharsets
import kotlin.coroutines.CoroutineContext import kotlin.coroutines.CoroutineContext
import kotlinx.coroutines.withContext
import timber.log.Timber
data class TunnelConf( data class TunnelConf(
val id: Int = 0, val id: Int = 0,
val tunName: String, val tunName: String,
val wgQuick: String, val wgQuick: String,
val tunnelNetworks: List<String> = emptyList(), val tunnelNetworks: List<String> = emptyList(),
val isMobileDataTunnel: Boolean = false, val isMobileDataTunnel: Boolean = false,
val isPrimaryTunnel: Boolean = false, val isPrimaryTunnel: Boolean = false,
val amQuick: String, val amQuick: String,
val isActive: Boolean = false, val isActive: Boolean = false,
val isPingEnabled: Boolean = false, val isPingEnabled: Boolean = false,
val pingInterval: Long? = null, val pingInterval: Long? = null,
val pingCooldown: Long? = null, val pingCooldown: Long? = null,
val pingIp: String? = null, val pingIp: String? = null,
val isEthernetTunnel: Boolean = false, val isEthernetTunnel: Boolean = false,
val isIpv4Preferred: Boolean = false, val isIpv4Preferred: Boolean = true,
) : Tunnel, com.wireguard.android.backend.Tunnel { val useCache: Boolean = false,
@Transient private var stateChangeCallback: ((Any) -> Unit)? = null,
) : Tunnel, org.amnezia.awg.backend.Tunnel {
val state = MutableStateFlow(TunnelState()) fun setStateChangeCallback(callback: (Any) -> Unit) {
stateChangeCallback = callback
}
fun toAmConfig(): org.amnezia.awg.config.Config { override fun equals(other: Any?): Boolean {
return configFromAmQuick(amQuick.ifBlank { wgQuick }) if (this === other) return true
} if (other !is TunnelConf) return false
return id == other.id &&
tunName == other.tunName &&
wgQuick == other.wgQuick &&
amQuick == other.amQuick &&
isPrimaryTunnel == other.isPrimaryTunnel &&
isMobileDataTunnel == other.isMobileDataTunnel &&
isEthernetTunnel == other.isEthernetTunnel &&
isPingEnabled == other.isPingEnabled &&
pingIp == other.pingIp &&
pingCooldown == other.pingCooldown &&
pingInterval == other.pingInterval &&
tunnelNetworks == other.tunnelNetworks &&
isIpv4Preferred == other.isIpv4Preferred
}
fun toWgConfig(): Config { override fun hashCode(): Int {
return configFromWgQuick(wgQuick) var result = id
} result = 31 * result + tunName.hashCode()
result = 31 * result + wgQuick.hashCode()
result = 31 * result + amQuick.hashCode()
return result
}
override fun getName(): String { fun isStaticallyConfigured(): Boolean {
return tunName return toAmConfig().peers.all { it.endpoint.get().host.isValidIpv4orIpv6Address() }
} }
override fun isIpv4ResolutionPreferred(): Boolean { fun copyWithCallback(
return isIpv4Preferred id: Int = this.id,
} tunName: String = this.tunName,
wgQuick: String = this.wgQuick,
tunnelNetworks: List<String> = this.tunnelNetworks,
isMobileDataTunnel: Boolean = this.isMobileDataTunnel,
isPrimaryTunnel: Boolean = this.isPrimaryTunnel,
amQuick: String = this.amQuick,
isActive: Boolean = this.isActive,
isPingEnabled: Boolean = this.isPingEnabled,
pingInterval: Long? = this.pingInterval,
pingCooldown: Long? = this.pingCooldown,
pingIp: String? = this.pingIp,
isEthernetTunnel: Boolean = this.isEthernetTunnel,
isIpv4Preferred: Boolean = this.isIpv4Preferred,
): TunnelConf {
return TunnelConf(
id,
tunName,
wgQuick,
tunnelNetworks,
isMobileDataTunnel,
isPrimaryTunnel,
amQuick,
isActive,
isPingEnabled,
pingInterval,
pingCooldown,
pingIp,
isEthernetTunnel,
isIpv4Preferred,
)
.apply { stateChangeCallback = this@TunnelConf.stateChangeCallback }
}
override fun onStateChange(newState: Tunnel.State) { fun toAmConfig(): org.amnezia.awg.config.Config {
state.update { return configFromAmQuick(amQuick.ifBlank { wgQuick })
it.copy(state = newState.asTunnelState()) }
}
}
override fun onStateChange(newState: com.wireguard.android.backend.Tunnel.State) { fun toWgConfig(): Config {
state.update { return configFromWgQuick(wgQuick)
it.copy(state = newState.asTunnelState()) }
}
}
fun isQuickConfigChanged(updatedConf: TunnelConf): Boolean { override fun getName(): String = tunName
return updatedConf.wgQuick != wgQuick ||
updatedConf.amQuick != amQuick
}
fun isPingConfigMatching(updatedConf: TunnelConf): Boolean { override fun isIpv4ResolutionPreferred(): Boolean = isIpv4Preferred
return updatedConf.isPingEnabled == isPingEnabled &&
pingIp == updatedConf.pingIp &&
updatedConf.pingCooldown == pingCooldown &&
updatedConf.pingInterval == pingInterval
}
suspend fun pingTunnel(context: CoroutineContext): List<Boolean> { override fun useCache(): Boolean = useCache
return withContext(context) {
val config = toWgConfig()
if (pingIp != null) {
Timber.i("Pinging custom ip")
listOf(InetAddress.getByName(pingIp).isReachable(Constants.PING_TIMEOUT.toInt()))
} else {
Timber.i("Pinging all peers")
config.peers.map { peer ->
peer.isReachable(isIpv4Preferred)
}
}
}
}
companion object { override fun onStateChange(newState: org.amnezia.awg.backend.Tunnel.State) {
fun configFromWgQuick(wgQuick: String): Config { stateChangeCallback?.invoke(newState)
val inputStream: InputStream = wgQuick.byteInputStream() }
return inputStream.bufferedReader(Charsets.UTF_8).use {
Config.parse(it)
}
}
fun configFromAmQuick(amQuick: String): org.amnezia.awg.config.Config { override fun onStateChange(newState: Tunnel.State) {
val inputStream: InputStream = amQuick.byteInputStream() stateChangeCallback?.invoke(newState)
return inputStream.bufferedReader(Charsets.UTF_8).use { }
org.amnezia.awg.config.Config.parse(it)
}
}
fun tunnelConfigFromAmConfig(config: org.amnezia.awg.config.Config, name: String): TunnelConf { fun generateUniqueName(tunnelNames: List<String>): String {
val amQuick = config.toAwgQuickString(true) var tunnelName = this.tunName
val wgQuick = config.toWgQuickString() var num = 1
return TunnelConf(tunName = name, wgQuick = wgQuick, amQuick = amQuick) while (tunnelNames.any { it == tunnelName }) {
} tunnelName =
if (!tunnelName.hasNumberInParentheses()) {
"$name($num)"
} else {
val pair = tunnelName.extractNameAndNumber()
"${pair?.first}($num)"
}
num++
}
return tunnelName
}
private const val IPV6_ALL_NETWORKS = "::/0" suspend fun isTunnelPingable(context: CoroutineContext): Boolean {
private const val IPV4_ALL_NETWORKS = "0.0.0.0/0" return withContext(context) {
val ALL_IPS = listOf(IPV4_ALL_NETWORKS, IPV6_ALL_NETWORKS) val config = toWgConfig()
private val IPV4_PUBLIC_NETWORKS = listOf( if (pingIp != null) {
"0.0.0.0/5", "8.0.0.0/7", "11.0.0.0/8", "12.0.0.0/6", "16.0.0.0/4", "32.0.0.0/3", return@withContext InetAddress.getByName(pingIp)
"64.0.0.0/2", "128.0.0.0/3", "160.0.0.0/5", "168.0.0.0/6", "172.0.0.0/12", .isReachable(Constants.PING_TIMEOUT.toInt())
"172.32.0.0/11", "172.64.0.0/10", "172.128.0.0/9", "173.0.0.0/8", "174.0.0.0/7", .also { Timber.i("Ping reachable $pingIp: $it") }
"176.0.0.0/4", "192.0.0.0/9", "192.128.0.0/11", "192.160.0.0/13", "192.169.0.0/16", }
"192.170.0.0/15", "192.172.0.0/14", "192.176.0.0/12", "192.192.0.0/10", config.peers
"193.0.0.0/8", "194.0.0.0/7", "196.0.0.0/6", "200.0.0.0/5", "208.0.0.0/4", .map { peer -> peer.isReachable() }
) .all { true }
val LAN_BYPASS_ALLOWED_IPS = listOf(IPV6_ALL_NETWORKS) + IPV4_PUBLIC_NETWORKS .also { Timber.i("Ping of all peers reachable: $it") }
} }
}
companion object {
fun configFromWgQuick(wgQuick: String): Config {
val inputStream: InputStream = wgQuick.byteInputStream()
return inputStream.bufferedReader(StandardCharsets.UTF_8).use { Config.parse(it) }
}
fun configFromAmQuick(amQuick: String): org.amnezia.awg.config.Config {
val inputStream: InputStream = amQuick.byteInputStream()
return inputStream.bufferedReader(StandardCharsets.UTF_8).use {
org.amnezia.awg.config.Config.parse(it)
}
}
fun tunnelConfigFromAmConfig(
config: org.amnezia.awg.config.Config,
name: String? = null,
): TunnelConf {
val amQuick = config.toAwgQuickString(true)
val wgQuick = config.toWgQuickString()
return TunnelConf(
tunName = name ?: config.defaultName(),
wgQuick = wgQuick,
amQuick = amQuick,
)
}
private const val IPV6_ALL_NETWORKS = "::/0"
private const val IPV4_ALL_NETWORKS = "0.0.0.0/0"
val ALL_IPS = listOf(IPV4_ALL_NETWORKS, IPV6_ALL_NETWORKS)
private val IPV4_PUBLIC_NETWORKS =
listOf(
"0.0.0.0/5",
"8.0.0.0/7",
"11.0.0.0/8",
"12.0.0.0/6",
"16.0.0.0/4",
"32.0.0.0/3",
"64.0.0.0/2",
"128.0.0.0/3",
"160.0.0.0/5",
"168.0.0.0/6",
"172.0.0.0/12",
"172.32.0.0/11",
"172.64.0.0/10",
"172.128.0.0/9",
"173.0.0.0/8",
"174.0.0.0/7",
"176.0.0.0/4",
"192.0.0.0/9",
"192.128.0.0/11",
"192.160.0.0/13",
"192.169.0.0/16",
"192.170.0.0/15",
"192.172.0.0/14",
"192.176.0.0/12",
"192.192.0.0/10",
"193.0.0.0/8",
"194.0.0.0/7",
"196.0.0.0/6",
"200.0.0.0/5",
"208.0.0.0/4",
)
val LAN_BYPASS_ALLOWED_IPS = listOf(IPV6_ALL_NETWORKS) + IPV4_PUBLIC_NETWORKS
}
} }
@@ -1,7 +1,33 @@
package com.zaneschepke.wireguardautotunnel.domain.enums package com.zaneschepke.wireguardautotunnel.domain.enums
sealed class BackendError() { import com.zaneschepke.wireguardautotunnel.R
data object DNS : BackendError()
data object Unauthorized : BackendError() sealed class BackendError : Exception() {
data object Config : BackendError() data object DNS : BackendError()
data object Unauthorized : BackendError()
data object Config : BackendError()
data object KernelModuleName : BackendError()
data object InvalidConfig : BackendError()
data object NotAuthorized : BackendError()
data object ServiceNotRunning : BackendError()
data object Unknown : BackendError()
fun toStringRes() =
when (this) {
Config -> R.string.config_error
DNS -> R.string.dns_resolve_error
InvalidConfig -> R.string.invalid_config_error
KernelModuleName -> R.string.kernel_name_error
NotAuthorized,
Unauthorized -> R.string.auth_error
ServiceNotRunning -> R.string.service_running_error
Unknown -> R.string.unknown_error
}
} }
@@ -1,7 +1,7 @@
package com.zaneschepke.wireguardautotunnel.domain.enums package com.zaneschepke.wireguardautotunnel.domain.enums
enum class BackendState { enum class BackendState {
KILL_SWITCH_ACTIVE, KILL_SWITCH_ACTIVE,
SERVICE_ACTIVE, SERVICE_ACTIVE,
INACTIVE, INACTIVE,
} }
@@ -1,6 +1,6 @@
package com.zaneschepke.wireguardautotunnel.domain.enums package com.zaneschepke.wireguardautotunnel.domain.enums
enum class ConfigType { enum class ConfigType {
AMNEZIA, AMNEZIA,
WG, WG,
} }
@@ -1,17 +1,16 @@
package com.zaneschepke.wireguardautotunnel.domain.enums package com.zaneschepke.wireguardautotunnel.domain.enums
enum class HandshakeStatus { enum class HandshakeStatus {
HEALTHY, HEALTHY,
STALE, STALE,
UNKNOWN, UNKNOWN,
NOT_STARTED, NOT_STARTED;
;
companion object { companion object {
private const val WG_TYPICAL_HANDSHAKE_INTERVAL_WHEN_HEALTHY_SEC = 180 private const val WG_TYPICAL_HANDSHAKE_INTERVAL_WHEN_HEALTHY_SEC = 180
const val STATUS_CHANGE_TIME_BUFFER = 30 const val STATUS_CHANGE_TIME_BUFFER = 30
const val STALE_TIME_LIMIT_SEC = const val STALE_TIME_LIMIT_SEC =
WG_TYPICAL_HANDSHAKE_INTERVAL_WHEN_HEALTHY_SEC + STATUS_CHANGE_TIME_BUFFER WG_TYPICAL_HANDSHAKE_INTERVAL_WHEN_HEALTHY_SEC + STATUS_CHANGE_TIME_BUFFER
const val NEVER_CONNECTED_TO_UNHEALTHY_TIME_LIMIT_SEC = 30 const val NEVER_CONNECTED_TO_UNHEALTHY_TIME_LIMIT_SEC = 30
} }
} }
@@ -4,14 +4,13 @@ import android.content.Context
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
enum class NotificationAction { enum class NotificationAction {
TUNNEL_OFF, TUNNEL_OFF,
AUTO_TUNNEL_OFF, AUTO_TUNNEL_OFF;
;
fun title(context: Context): String { fun title(context: Context): String {
return when (this) { return when (this) {
TUNNEL_OFF -> context.getString(R.string.stop) TUNNEL_OFF -> context.getString(R.string.stop)
AUTO_TUNNEL_OFF -> context.getString(R.string.stop) AUTO_TUNNEL_OFF -> context.getString(R.string.stop)
} }
} }
} }
@@ -1,15 +1,34 @@
package com.zaneschepke.wireguardautotunnel.domain.enums package com.zaneschepke.wireguardautotunnel.domain.enums
enum class TunnelStatus { sealed class TunnelStatus {
UP,
DOWN,
;
fun isDown(): Boolean { data object Up : TunnelStatus()
return this == DOWN
}
fun isUp(): Boolean { data object Down : TunnelStatus()
return this == UP
} data class Stopping(val reason: StopReason) : TunnelStatus()
data object Starting : TunnelStatus()
enum class StopReason {
USER,
PING,
CONFIG_CHANGED,
}
fun isDown(): Boolean {
return this == Down
}
fun isUp(): Boolean {
return this == Up
}
fun isUpOrStarting(): Boolean {
return this == Up || this == Starting
}
fun isDownOrStopping(): Boolean {
return this == Down || this is Stopping
}
} }
@@ -3,7 +3,9 @@ package com.zaneschepke.wireguardautotunnel.domain.events
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
sealed class AutoTunnelEvent { sealed class AutoTunnelEvent {
data class Start(val tunnelConf: TunnelConf? = null) : AutoTunnelEvent() data class Start(val tunnelConf: TunnelConf? = null) : AutoTunnelEvent()
data object Stop : AutoTunnelEvent()
data object DoNothing : AutoTunnelEvent() data object Stop : AutoTunnelEvent()
data object DoNothing : AutoTunnelEvent()
} }
@@ -1,7 +1,9 @@
package com.zaneschepke.wireguardautotunnel.domain.events package com.zaneschepke.wireguardautotunnel.domain.events
sealed class KillSwitchEvent { sealed class KillSwitchEvent {
data class Start(val allowedIps: List<String>) : KillSwitchEvent() data class Start(val allowedIps: List<String>) : KillSwitchEvent()
data object Stop : KillSwitchEvent()
data object DoNothing : KillSwitchEvent() data object Stop : KillSwitchEvent()
data object DoNothing : KillSwitchEvent()
} }
@@ -3,11 +3,11 @@ package com.zaneschepke.wireguardautotunnel.domain.repository
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
interface AppDataRepository { interface AppDataRepository {
suspend fun getPrimaryOrFirstTunnel(): TunnelConf? suspend fun getPrimaryOrFirstTunnel(): TunnelConf?
suspend fun getStartTunnelConfig(): TunnelConf? suspend fun getStartTunnelConfig(): TunnelConf?
val settings: AppSettingRepository val settings: AppSettingRepository
val tunnels: TunnelRepository val tunnels: TunnelRepository
val appState: AppStateRepository val appState: AppStateRepository
} }
@@ -4,7 +4,9 @@ import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
interface AppSettingRepository { interface AppSettingRepository {
suspend fun save(appSettings: AppSettings) suspend fun save(appSettings: AppSettings)
val flow: Flow<AppSettings>
suspend fun get(): AppSettings val flow: Flow<AppSettings>
suspend fun get(): AppSettings
} }
@@ -1,37 +1,45 @@
package com.zaneschepke.wireguardautotunnel.domain.repository package com.zaneschepke.wireguardautotunnel.domain.repository
import com.zaneschepke.wireguardautotunnel.data.model.GeneralState import com.zaneschepke.wireguardautotunnel.domain.entity.AppState
import com.zaneschepke.wireguardautotunnel.ui.theme.Theme import com.zaneschepke.wireguardautotunnel.ui.theme.Theme
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
interface AppStateRepository { interface AppStateRepository {
suspend fun isLocationDisclosureShown(): Boolean suspend fun isLocationDisclosureShown(): Boolean
suspend fun setLocationDisclosureShown(shown: Boolean) suspend fun setLocationDisclosureShown(shown: Boolean)
suspend fun isPinLockEnabled(): Boolean suspend fun isPinLockEnabled(): Boolean
suspend fun setPinLockEnabled(enabled: Boolean) suspend fun setPinLockEnabled(enabled: Boolean)
suspend fun isBatteryOptimizationDisableShown(): Boolean suspend fun isBatteryOptimizationDisableShown(): Boolean
suspend fun setBatteryOptimizationDisableShown(shown: Boolean) suspend fun setBatteryOptimizationDisableShown(shown: Boolean)
suspend fun isTunnelStatsExpanded(): Boolean suspend fun setTunnelExpanded(id: Int)
suspend fun setTunnelStatsExpanded(expanded: Boolean) suspend fun removeTunnelExpanded(id: Int)
suspend fun setTheme(theme: Theme) suspend fun setTheme(theme: Theme)
suspend fun getTheme(): Theme suspend fun getTheme(): Theme
suspend fun isLocalLogsEnabled(): Boolean suspend fun isLocalLogsEnabled(): Boolean
suspend fun setLocalLogsEnabled(enabled: Boolean) suspend fun setLocalLogsEnabled(enabled: Boolean)
suspend fun setLocale(localeTag: String) suspend fun setLocale(localeTag: String)
suspend fun getLocale(): String? suspend fun getLocale(): String?
val flow: Flow<GeneralState> suspend fun setIsRemoteControlEnabled(enabled: Boolean)
suspend fun isRemoteControlEnabled(): Boolean
suspend fun setRemoteKey(key: String)
suspend fun getRemoteKey(): String?
val flow: Flow<AppState>
} }
@@ -5,31 +5,33 @@ import com.zaneschepke.wireguardautotunnel.util.extensions.Tunnels
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
interface TunnelRepository { interface TunnelRepository {
val flow: Flow<List<TunnelConf>> val flow: Flow<List<TunnelConf>>
suspend fun getAll(): Tunnels suspend fun getAll(): Tunnels
suspend fun save(tunnelConf: TunnelConf) suspend fun save(tunnelConf: TunnelConf)
suspend fun updatePrimaryTunnel(tunnelConf: TunnelConf?) suspend fun saveAll(tunnelConfList: List<TunnelConf>)
suspend fun updateMobileDataTunnel(tunnelConf: TunnelConf?) suspend fun updatePrimaryTunnel(tunnelConf: TunnelConf?)
suspend fun updateEthernetTunnel(tunnelConf: TunnelConf?) suspend fun updateMobileDataTunnel(tunnelConf: TunnelConf?)
suspend fun delete(tunnelConf: TunnelConf) suspend fun updateEthernetTunnel(tunnelConf: TunnelConf?)
suspend fun getById(id: Int): TunnelConf? suspend fun delete(tunnelConf: TunnelConf)
suspend fun getActive(): Tunnels suspend fun getById(id: Int): TunnelConf?
suspend fun count(): Int suspend fun getActive(): Tunnels
suspend fun findByTunnelName(name: String): TunnelConf? suspend fun count(): Int
suspend fun findByTunnelNetworksName(name: String): Tunnels suspend fun findByTunnelName(name: String): TunnelConf?
suspend fun findByMobileDataTunnel(): Tunnels suspend fun findByTunnelNetworksName(name: String): Tunnels
suspend fun findPrimary(): Tunnels suspend fun findByMobileDataTunnel(): Tunnels
suspend fun findPrimary(): Tunnels
} }
@@ -0,0 +1,14 @@
package com.zaneschepke.wireguardautotunnel.domain.repository
import com.zaneschepke.wireguardautotunnel.domain.entity.AppUpdate
import java.io.File
interface UpdateRepository {
suspend fun checkForUpdate(currentVersion: String): Result<AppUpdate?>
suspend fun downloadApk(
apkUrl: String,
fileName: String,
onProgress: (Float) -> Unit,
): Result<File>
}
@@ -4,31 +4,32 @@ import org.amnezia.awg.backend.Statistics
import org.amnezia.awg.crypto.Key import org.amnezia.awg.crypto.Key
class AmneziaStatistics(private val statistics: Statistics) : TunnelStatistics() { class AmneziaStatistics(private val statistics: Statistics) : TunnelStatistics() {
override fun peerStats(peer: Key): PeerStats? { override fun peerStats(peer: Key): PeerStats? {
val key = Key.fromBase64(peer.toBase64()) val key = Key.fromBase64(peer.toBase64())
val stats = statistics.peer(key) val stats = statistics.peer(key)
return stats?.let { return stats?.let {
PeerStats( PeerStats(
rxBytes = stats.rxBytes, rxBytes = stats.rxBytes,
txBytes = stats.txBytes, txBytes = stats.txBytes,
latestHandshakeEpochMillis = stats.latestHandshakeEpochMillis, latestHandshakeEpochMillis = stats.latestHandshakeEpochMillis,
) resolvedEndpoint = stats.resolvedEndpoint,
} )
} }
}
override fun isTunnelStale(): Boolean { override fun isTunnelStale(): Boolean {
return statistics.isStale return statistics.isStale
} }
override fun getPeers(): Array<Key> { override fun getPeers(): Array<Key> {
return statistics.peers() return statistics.peers()
} }
override fun rx(): Long { override fun rx(): Long {
return statistics.totalRx() return statistics.totalRx()
} }
override fun tx(): Long { override fun tx(): Long {
return statistics.totalTx() return statistics.totalTx()
} }
} }
@@ -1,156 +1,198 @@
package com.zaneschepke.wireguardautotunnel.domain.state package com.zaneschepke.wireguardautotunnel.domain.state
import com.zaneschepke.wireguardautotunnel.domain.events.KillSwitchEvent import com.zaneschepke.wireguardautotunnel.core.tunnel.allDown
import com.zaneschepke.wireguardautotunnel.core.tunnel.hasActive
import com.zaneschepke.wireguardautotunnel.core.tunnel.isUp
import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings import com.zaneschepke.wireguardautotunnel.domain.entity.AppSettings
import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf import com.zaneschepke.wireguardautotunnel.domain.entity.TunnelConf
import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent
import com.zaneschepke.wireguardautotunnel.domain.events.KillSwitchEvent
import com.zaneschepke.wireguardautotunnel.util.extensions.isMatchingToWildcardList import com.zaneschepke.wireguardautotunnel.util.extensions.isMatchingToWildcardList
data class AutoTunnelState( data class AutoTunnelState(
val activeTunnels: List<TunnelConf> = emptyList(), val activeTunnels: Map<TunnelConf, TunnelState> = emptyMap(),
val networkState: NetworkState = NetworkState(), val networkState: NetworkState = NetworkState(),
val settings: AppSettings = AppSettings(), val settings: AppSettings = AppSettings(),
val tunnels: List<TunnelConf> = emptyList(), val tunnels: List<TunnelConf> = emptyList(),
) { ) {
private fun isMobileDataActive(): Boolean { private fun isMobileDataActive(): Boolean {
return !networkState.isEthernetConnected && !networkState.isWifiConnected && networkState.isMobileDataConnected return !networkState.isEthernetConnected &&
} !networkState.isWifiConnected &&
networkState.isMobileDataConnected
}
private fun isMobileTunnelDataChangeNeeded(): Boolean { private fun isMobileTunnelDataChangeNeeded(): Boolean {
val preferredTunnel = preferredMobileDataTunnel() val preferredTunnel = preferredMobileDataTunnel()
return preferredTunnel != null && return preferredTunnel != null &&
activeTunnels.isNotEmpty() && !activeTunnels.any { it.id == preferredTunnel.id } activeTunnels.isNotEmpty() &&
} !activeTunnels.isUp(preferredTunnel)
}
private fun isEthernetTunnelChangeNeeded(): Boolean { private fun isEthernetTunnelChangeNeeded(): Boolean {
val preferredTunnel = preferredEthernetTunnel() val preferredTunnel = preferredEthernetTunnel()
return preferredTunnel != null && activeTunnels.isNotEmpty() && !activeTunnels.any { it.id == preferredTunnel.id } return preferredTunnel != null &&
} activeTunnels.isNotEmpty() &&
!activeTunnels.isUp(preferredTunnel)
}
private fun preferredMobileDataTunnel(): TunnelConf? { private fun preferredMobileDataTunnel(): TunnelConf? {
return tunnels.firstOrNull { it.isMobileDataTunnel } ?: tunnels.firstOrNull { it.isPrimaryTunnel } return tunnels.firstOrNull { it.isMobileDataTunnel }
} ?: tunnels.firstOrNull { it.isPrimaryTunnel }
}
private fun preferredEthernetTunnel(): TunnelConf? { private fun preferredEthernetTunnel(): TunnelConf? {
return tunnels.firstOrNull { it.isEthernetTunnel } ?: tunnels.firstOrNull { it.isPrimaryTunnel } return tunnels.firstOrNull { it.isEthernetTunnel }
} ?: tunnels.firstOrNull { it.isPrimaryTunnel }
}
private fun preferredWifiTunnel(): TunnelConf? { private fun preferredWifiTunnel(): TunnelConf? {
return getTunnelWithMatchingTunnelNetwork() ?: tunnels.firstOrNull { it.isPrimaryTunnel } return getTunnelWithMatchingTunnelNetwork() ?: tunnels.firstOrNull { it.isPrimaryTunnel }
} }
private fun isWifiActive(): Boolean { private fun isWifiActive(): Boolean {
return !networkState.isEthernetConnected && networkState.isWifiConnected return !networkState.isEthernetConnected && networkState.isWifiConnected
} }
private fun startOnEthernet(): Boolean { private fun startOnEthernet(): Boolean {
return networkState.isEthernetConnected && settings.isTunnelOnEthernetEnabled && activeTunnels.isEmpty() return networkState.isEthernetConnected &&
} settings.isTunnelOnEthernetEnabled &&
activeTunnels.allDown()
}
private fun stopOnEthernet(): Boolean { private fun stopOnEthernet(): Boolean {
return networkState.isEthernetConnected && !settings.isTunnelOnEthernetEnabled && activeTunnels.isNotEmpty() return networkState.isEthernetConnected &&
} !settings.isTunnelOnEthernetEnabled &&
activeTunnels.hasActive()
}
// TODO test removed kill switch state check // TODO test removed kill switch state check
private fun stopKillSwitchOnTrusted(): Boolean { private fun stopKillSwitchOnTrusted(): Boolean {
return networkState.isWifiConnected && settings.isVpnKillSwitchEnabled && settings.isDisableKillSwitchOnTrustedEnabled && isCurrentSSIDTrusted() return networkState.isWifiConnected &&
} settings.isVpnKillSwitchEnabled &&
settings.isDisableKillSwitchOnTrustedEnabled &&
isCurrentSSIDTrusted()
}
// TODO test, removed kill switch state check // TODO test, removed kill switch state check
private fun startKillSwitch(): Boolean { private fun startKillSwitch(): Boolean {
return settings.isVpnKillSwitchEnabled && (!settings.isDisableKillSwitchOnTrustedEnabled || !isCurrentSSIDTrusted()) return settings.isVpnKillSwitchEnabled &&
} (!settings.isDisableKillSwitchOnTrustedEnabled || !isCurrentSSIDTrusted())
}
fun isNoConnectivity(): Boolean { private fun isNoConnectivity(): Boolean {
return !networkState.isEthernetConnected && !networkState.isWifiConnected && !networkState.isMobileDataConnected return !networkState.isEthernetConnected &&
} !networkState.isWifiConnected &&
!networkState.isMobileDataConnected
}
private fun stopOnMobileData(): Boolean { private fun stopOnMobileData(): Boolean {
return isMobileDataActive() && !settings.isTunnelOnMobileDataEnabled && activeTunnels.isNotEmpty() return isMobileDataActive() &&
} !settings.isTunnelOnMobileDataEnabled &&
activeTunnels.hasActive()
}
private fun startOnMobileData(): Boolean { private fun startOnMobileData(): Boolean {
return isMobileDataActive() && settings.isTunnelOnMobileDataEnabled && activeTunnels.isEmpty() return isMobileDataActive() &&
} settings.isTunnelOnMobileDataEnabled &&
activeTunnels.allDown()
}
private fun changeOnMobileData(): Boolean { private fun changeOnMobileData(): Boolean {
return isMobileDataActive() && settings.isTunnelOnMobileDataEnabled && isMobileTunnelDataChangeNeeded() return isMobileDataActive() &&
} settings.isTunnelOnMobileDataEnabled &&
isMobileTunnelDataChangeNeeded()
}
private fun changeOnEthernet(): Boolean { private fun changeOnEthernet(): Boolean {
return networkState.isEthernetConnected && settings.isTunnelOnEthernetEnabled && isEthernetTunnelChangeNeeded() return networkState.isEthernetConnected &&
} settings.isTunnelOnEthernetEnabled &&
isEthernetTunnelChangeNeeded()
}
private fun stopOnWifi(): Boolean { private fun stopOnWifi(): Boolean {
return isWifiActive() && !settings.isTunnelOnWifiEnabled && activeTunnels.isNotEmpty() return isWifiActive() && !settings.isTunnelOnWifiEnabled && activeTunnels.hasActive()
} }
private fun stopOnTrustedWifi(): Boolean { private fun stopOnTrustedWifi(): Boolean {
return isWifiActive() && settings.isTunnelOnWifiEnabled && activeTunnels.isNotEmpty() && isCurrentSSIDTrusted() return isWifiActive() &&
} settings.isTunnelOnWifiEnabled &&
activeTunnels.hasActive() &&
isCurrentSSIDTrusted()
}
private fun startOnUntrustedWifi(): Boolean { private fun startOnUntrustedWifi(): Boolean {
return isWifiActive() && settings.isTunnelOnWifiEnabled && activeTunnels.isEmpty() && !isCurrentSSIDTrusted() return isWifiActive() &&
} settings.isTunnelOnWifiEnabled &&
activeTunnels.allDown() &&
!isCurrentSSIDTrusted()
}
private fun changeOnUntrustedWifi(): Boolean { private fun changeOnUntrustedWifi(): Boolean {
return isWifiActive() && settings.isTunnelOnWifiEnabled && activeTunnels.isNotEmpty() && !isCurrentSSIDTrusted() && !isWifiTunnelPreferred() return isWifiActive() &&
} settings.isTunnelOnWifiEnabled &&
activeTunnels.hasActive() &&
!isCurrentSSIDTrusted() &&
!isWifiTunnelPreferred()
}
private fun isWifiTunnelPreferred(): Boolean { private fun isWifiTunnelPreferred(): Boolean {
val preferred = preferredWifiTunnel() val preferred = preferredWifiTunnel()
return activeTunnels.any { it.id == preferred?.id } return preferred?.let { activeTunnels.isUp(it) } ?: true
} }
fun asAutoTunnelEvent(): AutoTunnelEvent { fun asAutoTunnelEvent(): AutoTunnelEvent {
return when { return when {
// ethernet scenarios // ethernet scenarios
stopOnEthernet() -> AutoTunnelEvent.Stop stopOnEthernet() -> AutoTunnelEvent.Stop
startOnEthernet() || changeOnEthernet() -> AutoTunnelEvent.Start(preferredEthernetTunnel()) startOnEthernet() || changeOnEthernet() ->
// mobile data scenarios AutoTunnelEvent.Start(preferredEthernetTunnel())
stopOnMobileData() -> AutoTunnelEvent.Stop // mobile data scenarios
startOnMobileData() || changeOnMobileData() -> AutoTunnelEvent.Start(preferredMobileDataTunnel()) stopOnMobileData() -> AutoTunnelEvent.Stop
// wifi scenarios startOnMobileData() || changeOnMobileData() ->
stopOnWifi() -> AutoTunnelEvent.Stop AutoTunnelEvent.Start(preferredMobileDataTunnel())
stopOnTrustedWifi() -> AutoTunnelEvent.Stop // wifi scenarios
startOnUntrustedWifi() || changeOnUntrustedWifi() -> AutoTunnelEvent.Start(preferredWifiTunnel()) stopOnWifi() -> AutoTunnelEvent.Stop
// no connectivity stopOnTrustedWifi() -> AutoTunnelEvent.Stop
isNoConnectivity() && settings.isStopOnNoInternetEnabled -> AutoTunnelEvent.Stop startOnUntrustedWifi() || changeOnUntrustedWifi() ->
else -> AutoTunnelEvent.DoNothing AutoTunnelEvent.Start(preferredWifiTunnel())
} // no connectivity
} isNoConnectivity() && settings.isStopOnNoInternetEnabled -> AutoTunnelEvent.Stop
else -> AutoTunnelEvent.DoNothing
}
}
fun asKillSwitchEvent(): KillSwitchEvent { fun asKillSwitchEvent(): KillSwitchEvent {
return when { return when {
stopKillSwitchOnTrusted() -> KillSwitchEvent.Stop stopKillSwitchOnTrusted() -> KillSwitchEvent.Stop
startKillSwitch() -> { startKillSwitch() -> {
val allowedIps = if (settings.isLanOnKillSwitchEnabled) TunnelConf.LAN_BYPASS_ALLOWED_IPS else emptyList() val allowedIps =
KillSwitchEvent.Start(allowedIps) if (settings.isLanOnKillSwitchEnabled) TunnelConf.LAN_BYPASS_ALLOWED_IPS
} else emptyList()
else -> KillSwitchEvent.DoNothing KillSwitchEvent.Start(allowedIps)
} }
} else -> KillSwitchEvent.DoNothing
}
}
private fun isCurrentSSIDTrusted(): Boolean { private fun isCurrentSSIDTrusted(): Boolean {
return networkState.wifiName?.let { return networkState.wifiName?.let { hasTrustedWifiName(it) } == true
hasTrustedWifiName(it) }
} == true
}
private fun hasTrustedWifiName(wifiName: String, wifiNames: List<String> = settings.trustedNetworkSSIDs): Boolean { private fun hasTrustedWifiName(
return if (settings.isWildcardsEnabled) { wifiName: String,
wifiNames.isMatchingToWildcardList(wifiName) wifiNames: List<String> = settings.trustedNetworkSSIDs,
} else { ): Boolean {
wifiNames.contains(wifiName) return if (settings.isWildcardsEnabled) {
} wifiNames.isMatchingToWildcardList(wifiName)
} } else {
wifiNames.contains(wifiName)
}
}
private fun getTunnelWithMatchingTunnelNetwork(): TunnelConf? { private fun getTunnelWithMatchingTunnelNetwork(): TunnelConf? {
return networkState.wifiName?.let { wifiName -> return networkState.wifiName?.let { wifiName ->
tunnels.firstOrNull { tunnels.firstOrNull { hasTrustedWifiName(wifiName, it.tunnelNetworks) }
hasTrustedWifiName(wifiName, it.tunnelNetworks) }
} }
}
}
} }
@@ -1,9 +1,9 @@
package com.zaneschepke.wireguardautotunnel.domain.state package com.zaneschepke.wireguardautotunnel.domain.state
data class ConnectivityState( data class ConnectivityState(
val wifiAvailable: Boolean, val wifiAvailable: Boolean,
val ethernetAvailable: Boolean, val ethernetAvailable: Boolean,
val cellularAvailable: Boolean, val cellularAvailable: Boolean,
) { ) {
val allOffline = !wifiAvailable && !ethernetAvailable && !cellularAvailable val allOffline = !wifiAvailable && !ethernetAvailable && !cellularAvailable
} }
@@ -1,12 +1,12 @@
package com.zaneschepke.wireguardautotunnel.domain.state package com.zaneschepke.wireguardautotunnel.domain.state
data class NetworkState( data class NetworkState(
val isWifiConnected: Boolean = false, val isWifiConnected: Boolean = false,
val isMobileDataConnected: Boolean = false, val isMobileDataConnected: Boolean = false,
val isEthernetConnected: Boolean = false, val isEthernetConnected: Boolean = false,
val wifiName: String? = null, val wifiName: String? = null,
) { ) {
fun hasNoCapabilities(): Boolean { fun hasNoCapabilities(): Boolean {
return !isWifiConnected && !isMobileDataConnected && !isEthernetConnected return !isWifiConnected && !isMobileDataConnected && !isEthernetConnected
} }
} }
@@ -4,7 +4,7 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.BackendState
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelStatus
data class TunnelState( data class TunnelState(
val state: TunnelStatus = TunnelStatus.DOWN, val status: TunnelStatus = TunnelStatus.Down,
val backendState: BackendState = BackendState.INACTIVE, val backendState: BackendState = BackendState.INACTIVE,
val statistics: TunnelStatistics? = null, val statistics: TunnelStatistics? = null,
) )
@@ -3,16 +3,21 @@ package com.zaneschepke.wireguardautotunnel.domain.state
import org.amnezia.awg.crypto.Key import org.amnezia.awg.crypto.Key
abstract class TunnelStatistics { abstract class TunnelStatistics {
@JvmRecord @JvmRecord
data class PeerStats(val rxBytes: Long, val txBytes: Long, val latestHandshakeEpochMillis: Long) data class PeerStats(
val rxBytes: Long,
val txBytes: Long,
val latestHandshakeEpochMillis: Long,
val resolvedEndpoint: String,
)
abstract fun peerStats(peer: Key): PeerStats? abstract fun peerStats(peer: Key): PeerStats?
abstract fun isTunnelStale(): Boolean abstract fun isTunnelStale(): Boolean
abstract fun getPeers(): Array<Key> abstract fun getPeers(): Array<Key>
abstract fun rx(): Long abstract fun rx(): Long
abstract fun tx(): Long abstract fun tx(): Long
} }
@@ -4,33 +4,32 @@ import com.wireguard.android.backend.Statistics
import org.amnezia.awg.crypto.Key import org.amnezia.awg.crypto.Key
class WireGuardStatistics(private val statistics: Statistics) : TunnelStatistics() { class WireGuardStatistics(private val statistics: Statistics) : TunnelStatistics() {
override fun peerStats(peer: Key): PeerStats? { override fun peerStats(peer: Key): PeerStats? {
val key = com.wireguard.crypto.Key.fromBase64(peer.toBase64()) val key = com.wireguard.crypto.Key.fromBase64(peer.toBase64())
val peerStats = statistics.peer(key) val peerStats = statistics.peer(key)
return peerStats?.let { return peerStats?.let {
PeerStats( PeerStats(
txBytes = peerStats.txBytes, txBytes = peerStats.txBytes,
rxBytes = peerStats.rxBytes, rxBytes = peerStats.rxBytes,
latestHandshakeEpochMillis = peerStats.latestHandshakeEpochMillis, latestHandshakeEpochMillis = peerStats.latestHandshakeEpochMillis,
) resolvedEndpoint = peerStats.resolvedEndpoint,
} )
} }
}
override fun isTunnelStale(): Boolean { override fun isTunnelStale(): Boolean {
return statistics.isStale return statistics.isStale
} }
override fun getPeers(): Array<Key> { override fun getPeers(): Array<Key> {
return statistics.peers().map { return statistics.peers().map { Key.fromBase64(it.toBase64()) }.toTypedArray()
Key.fromBase64(it.toBase64()) }
}.toTypedArray()
}
override fun rx(): Long { override fun rx(): Long {
return statistics.totalRx() return statistics.totalRx()
} }
override fun tx(): Long { override fun tx(): Long {
return statistics.totalTx() return statistics.totalTx()
} }
} }
@@ -3,62 +3,46 @@ package com.zaneschepke.wireguardautotunnel.ui
import kotlinx.serialization.Serializable import kotlinx.serialization.Serializable
sealed class Route { sealed class Route {
@Serializable @Serializable data object Support : Route()
data object Support : Route()
@Serializable @Serializable data object Settings : Route()
data object Settings : Route()
@Serializable @Serializable data object SettingsAdvanced : Route()
data object AutoTunnel : Route()
@Serializable @Serializable data object AutoTunnel : Route()
data object AutoTunnelAdvanced : Route()
@Serializable @Serializable data object AutoTunnelAdvanced : Route()
data object LocationDisclosure : Route()
@Serializable @Serializable data object LocationDisclosure : Route()
data object Appearance : Route()
@Serializable @Serializable data object Appearance : Route()
data object Display : Route()
@Serializable @Serializable data object Display : Route()
data object KillSwitch : Route()
@Serializable @Serializable data object KillSwitch : Route()
data object Language : Route()
@Serializable @Serializable data object Language : Route()
data object Main : Route()
@Serializable @Serializable data object Main : Route()
data class TunnelOptions(
val id: Int,
) : Route()
@Serializable @Serializable data class TunnelOptions(val id: Int) : Route()
data object Lock : Route()
@Serializable @Serializable data object Lock : Route()
data object Scanner : Route()
@Serializable @Serializable data object Scanner : Route()
data class Config(
val id: Int,
) : Route()
@Serializable @Serializable data object License : Route()
data class SplitTunnel(
val id: Int,
) : Route()
@Serializable @Serializable data class Config(val id: Int) : Route()
data class TunnelAutoTunnel(
val id: Int,
) : Route()
@Serializable @Serializable
data object Logs : Route() data class SplitTunnel(val id: Int) : Route() {
companion object {
const val KEY_ID = "id"
}
}
@Serializable data class TunnelAutoTunnel(val id: Int) : Route()
@Serializable data object Logs : Route()
} }

Some files were not shown because too many files have changed in this diff Show More