Compare commits

..

191 commits

Author SHA1 Message Date
egorovnikita d42ea75978 remove unnecessary cloning 2024-03-15 19:37:04 +03:00
egorovnikita ecda670d1e add shared modules to use libcanberra in Flatpak 2024-03-15 19:37:04 +03:00
Maxim Logaev 2587a7e6a9 Merge remote-tracking branch 'kkonsw/resync' into master-windows-changes-test 2024-03-12 18:51:50 +03:00
Maxim Logaev f1aed531e2 Merge remote-tracking branch 'smalishevskiy/test' into master-windows-changes-test 2024-03-12 18:34:26 +03:00
Maxim Logaev 75092caced
Added CI for MSYS2 (MINGW64) (#2)
- Use quotes in windows build script;
- Added missing gstreamer, webrtc-audio-processing and git;
- Added CI for Windows.
---------
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-12 20:01:36 +03:00
Psayker ca1f97e243 Move download yolort headers logic into prepare stage, delete yolort download script 2024-03-12 16:20:13 +03:00
Konstantin Kuznetsov ecb3fb0056 Fetch MAM history for conversations without local mam catchup data 2024-03-12 16:17:39 +03:00
Konstantin Kuznetsov 1a9c55ecc9 Add button to fetch MAM history for conversation 2024-03-12 12:49:07 +03:00
emil 735e6f544a Add gap to ringing and dialing sounds 2024-03-12 11:33:04 +03:00
emil a3958949f0 Phone ringing and dialing 2024-03-12 11:33:04 +03:00
emil 734e1503c7 Remove Gst.deinit (proactive bug fix)
Gst.deinit caused the Dino proces to linger in the background when I
tried using Gst in the phone ringer plugin. My reasoning for leaving it
out even though Gst is not in use anymore in the ringer plugin is that
this is a nasty bug that might crop up again in the future when someone
tries to do anything else at all with Gst.

I did the following things:

- checked this https://gstreamer.freedesktop.org/documentation/gstreamer/gst.html#gst_deinit
- tested that Dino works without the Gst.deinit
- tried looking for a specific reason for the deinit with git log -L 276,+10 -- plugins/rtp/src/plugin.vala

I didn't find anything so this made me conclude that it is better to
leave it out.
2024-03-12 11:33:03 +03:00
Konstantin Kuznetsov bf07ecddb3 Add button to fetch MAM history for conversation 2024-03-12 11:30:39 +03:00
Stanislav Malishevskiy 1a73e9d53d Fix issue #261. 2024-03-12 10:43:40 +03:00
Miquel Lionel fc8fdc166b wip, fixes https://github.com/dino/dino/issues/261
- server refuses to responds in any way to the sent iq stanza
	  and i don't know why, when i send it with the xml console in
	  gajim it works well
2024-03-12 10:43:40 +03:00
Maxim Logaev 9c15a81b40 Fixed build dependencies
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-11 17:10:07 +03:00
Maxim Logaev 3848cf3129 Added build-installer target to build-win64.sh
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-06 23:01:49 +03:00
Maxim Logaev 43502806bc Removed unnecessary installer files
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-06 21:19:13 +03:00
Maxim Logaev b886e2a2f2 Fixed dist-install dir
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-06 21:11:15 +03:00
Maxim Logaev 4674e90d16 Added basic support for NSIS installer
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-06 20:42:19 +03:00
Maxim Logaev a3171e12df Added README-WIN64.md
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-05 23:13:25 +03:00
Maxim Logaev 5f500b4d42 Added build script for windows
Signed-off-by: Maxim Logaev <maxlogaev@proton.me>
2024-03-05 22:58:00 +03:00
Maxim Logaev f9fd66dba5 Merge branch 'origin/master' into master-windows-changes 2024-03-05 15:07:52 +03:00
LAGonauta 1c156e8c9f Fix GPGME 2023-10-09 10:54:55 -04:00
LAGonauta f6fe383ce8 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-10-09 10:54:15 -04:00
LAGonauta 49f331627d Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-07-17 17:28:47 -04:00
LAGonauta 7603990740 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-04-24 19:01:02 -03:00
LAGonauta 7872d21f03 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-04-21 20:43:40 -03:00
LAGonauta 86182a7db9 Check YoloRT checksum before building 2023-02-12 11:17:57 -03:00
LAGonauta 82e6937f21 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-02-12 10:59:16 -03:00
LAGonauta 3d61f175a6 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-01-30 08:53:45 -03:00
LAGonauta 471f8c5f4a Merge remote-tracking branch 'upstream/master' into master-windows-changes 2023-01-08 15:41:07 -03:00
LAGonauta fb05c83c8e Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-12-28 09:13:59 -03:00
LAGonauta ef41c27cb2 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-10-15 13:00:56 -03:00
LAGonauta e638ef0694 Fixed using GTK3 instead of GTK4 2022-08-25 09:29:47 -03:00
LAGonauta a7f7a6d388 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-08-25 08:54:04 -03:00
LAGonauta d507808e40 Automatically set PANGOCAIRO_BACKEND to fontconfig on win32 2022-06-17 18:37:49 -03:00
LAGonauta 2d82a4c195 Removed version from Dino executable
We need a better way to get the version number
2022-06-17 18:37:34 -03:00
LAGonauta c29032bcc6 :Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-06-05 11:12:23 -03:00
LAGonauta 90bd9df891 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-02-25 07:02:20 -03:00
LAGonauta d10badd690 Ignore wasapi devices as they do not work well yet 2022-02-25 07:02:06 -03:00
LAGonauta 11828cc2cf no need to ignore wasapi 2022-02-09 21:09:44 -03:00
LAGonauta a583a44994 fix call notifications buttons not working 2022-02-09 21:01:26 -03:00
LAGonauta dd71e943c3 Merge remote-tracking branch 'upstream/master' into master-windows-changes 2022-02-08 20:32:22 -03:00
LAGonauta b8b74817f7 Merge remote-tracking branch 'upstream/master' into master-windows-changes
# Conflicts:
#	main/src/ui/conversation_content_view/file_widget.vala
#	plugins/CMakeLists.txt
#	plugins/rtp/src/device.vala
#	plugins/rtp/src/plugin.vala
2022-02-08 18:33:09 -03:00
mjk 43fd04ce41 define _POSIX_C_SOURCE=1 on windows
Fixes "undefined reference to `localtime_r`" in, e.g., Vala's GLib.Time.local
when building on mingw-w64.
2022-02-08 18:09:49 -03:00
Felipe 733d612b18
Merge pull request #5 from danielreuther/master-windows-changes-gcc11-fix
Fix compilation on GCC 11
2022-02-08 18:04:58 -03:00
Daniel Reuther 5a335f89d3 Fix compilation on gcc 11 2021-10-23 02:45:52 +02:00
LAGonauta c57fc768aa Do not generate WinRT headers, just download them on build 2021-06-19 11:43:12 -03:00
LAGonauta 88a376c1c2 Build YoloRT on project build 2021-06-19 08:13:39 -03:00
LAGonauta 97753bd5c4 Remove YoloRT from tree 2021-06-19 06:58:40 -03:00
LAGonauta d7118c1b93 Allow devices with properties and use has_classes 2021-06-18 07:46:39 -03:00
LAGonauta 96fbbdd8bb Use correct generic type for ArrayList
Nullable crashes Dino
2021-06-10 20:03:30 -03:00
LAGonauta 4b0c3b8ef0 Add initial call notifications 2021-06-09 07:20:56 -03:00
LAGonauta ee11a2dbcf Do not allow selection of WASAPI devices
Dino would have to resample it own audio, do more book keeping and
somehow find out manually which is the default device.
2021-06-09 07:20:45 -03:00
LAGonauta 064541dd2b Prefer primary DirectSound device on Windows
It automatically selects the default device for use,
there is no book keeping necessary and things just work

The primary DirectSound device has a (NULL) guid, making
it wasy to be found.
2021-06-09 07:20:33 -03:00
LAGonauta 854818133e DelegateToUi must be an owned function 2021-06-09 07:20:14 -03:00
LAGonauta 878bf9ac43 Allow null header, body, applogo, and image on notification builder 2021-06-09 07:20:14 -03:00
LAGonauta e6a521ada9 Add support for inline images to notification 2021-06-09 07:20:14 -03:00
LAGonauta 359dcbf70c Add support for adaptive Windows 10 notifications 2021-06-09 07:20:14 -03:00
mjk c40775b7da comment out unused arguments [-Wunused-parameter] 2021-06-09 07:20:13 -03:00
mjk 934b492357 init Callback completely always
The `token` pointer was left dangerously uninitialized after construction.
2021-06-09 07:20:12 -03:00
mjk 0258980977 replace lists with vectors 2021-06-09 07:20:12 -03:00
mjk 8034a6f344 remove the no-longer-needed -municode compile option 2021-06-09 07:20:10 -03:00
mjk c89671af04 log SetCurrentProcessExplicitAppUserModelID errors 2021-06-09 07:20:10 -03:00
mjk 822b91f40c stop checking for empty AUMIDs
The downstream code handles them just fine.
2021-06-09 07:20:09 -03:00
mjk 332216e63c reflow/respace 2021-06-09 07:20:09 -03:00
mjk 7b6f3ea9f6 replace C-style concurrent initialization of statics
C++ statics are thread-safe as is and are usually implemented more
efficiently. Besides, `volatile` is likely misused here anyway.
2021-06-09 07:20:04 -03:00
mjk 8988b77d57 don't use FALSE as a null pointer constant 2021-06-09 07:20:03 -03:00
mjk 987eb672d5 fix / work around uninitialized fields warnings 2021-06-09 07:20:03 -03:00
mjk 1698d15f7d silence enum stringification warnings by first casting to underlying types 2021-06-09 07:20:02 -03:00
mjk 17c1172d84 work around a (pedantic) format specifier warning 2021-06-09 07:20:02 -03:00
mjk 8f8dfa2e70 move module loading functions out of shortcutcreator.cpp 2021-06-09 07:20:02 -03:00
mjk e8d84d2eda handle empty menu-relative shortcut paths 2021-06-09 07:20:01 -03:00
mjk f1bcb6604f rewrite InitApartment and protect callers from (the rest of the) exceptions
Initializing COM by calling `winrt::init_apartment()` would always cause
stack unwinding *in practice*, which is suboptimal at best, and even using
`apartment_type::single_threaded` still would require exception filtering
*just in case*.
2021-06-09 07:20:01 -03:00
mjk c855d5e7cb protect GetTemplateContent callers from exceptions 2021-06-09 07:20:00 -03:00
mjk ed7c23c8d2 generalize glib::try_invoke to any return-by-value type and void 2021-06-09 07:19:59 -03:00
mjk b1cb64d716 add nullary version of g_try_invoke macro 2021-06-09 07:19:58 -03:00
mjk da424fb73b fiddle with punctuation 2021-06-09 07:19:58 -03:00
mjk 39b51b2d4c make glib::impl::varstring less explosive 2021-06-09 07:19:57 -03:00
mjk 7cb6b3f4b3 fix UB in glib::describe_arguments called with no arguments
Makes failure logging of nullary invokables non-crashy.
2021-06-09 07:19:56 -03:00
mjk 15abe70fc6 handle theoretical future failures of wsview_to_char 2021-06-09 07:19:56 -03:00
mjk ab6ce05aec handle empty hresult_error message specially 2021-06-09 07:19:55 -03:00
mjk c44b1daaf7 make meta-error messages more precise 2021-06-09 07:19:54 -03:00
mjk 5e50d1d2d7 remove an unused include 2021-06-09 07:19:53 -03:00
mjk d3214188d5 use lower-case 0x prefix for hresult code formatting everywhere 2021-06-09 07:19:51 -03:00
mjk 229ff697c4 move some whitespace around 2021-06-09 07:19:50 -03:00
mjk c3624f8490 handle exe paths longer than 259 chars 2021-06-09 07:19:48 -03:00
mjk 597061aed1 work around GetEnvironmentVariable not resetting last error 2021-06-09 07:19:48 -03:00
mjk fd035084de unimplement accidentally implemented wide string overloads of describe_argument 2021-06-09 07:19:47 -03:00
LAGonauta 426f997f9b Fix notifications not hiding 2021-06-09 07:19:47 -03:00
LAGonauta c60e0a828f Add README.md to Windows notification plugin 2021-06-09 07:19:46 -03:00
LAGonauta cc0e206c27 Removed unused library on linker
It is loaded dynamically
2021-06-09 07:19:45 -03:00
LAGonauta 60791cb3f0 Add ginvoke to CMakeLists 2021-06-09 07:19:45 -03:00
mjk f9a3b28624 switch to runtime loading of PropVariantToStringAlloc
Now it really should work.
2021-06-09 07:19:44 -03:00
mjk cb3c6874bc add restoration of shortcut's target path 2021-06-09 07:19:44 -03:00
mjk eeda464ca9 rewrite shortcut management code with RAII, error logging and exceptions
It actually works now.
2021-06-09 07:19:43 -03:00
mjk f193948f4e constify all the things 2021-06-09 07:19:43 -03:00
mjk 401c4a1bb1 change some local functions' signatures 2021-06-09 07:19:43 -03:00
mjk 34519e96bd make GetEnv more robust and not limit length of variables 2021-06-09 07:19:42 -03:00
mjk fb754b0d43 generalize GetShortcutPath into GetEnv 2021-06-09 07:19:42 -03:00
mjk 2eb1eea06e clarify GetCurrentModulePath's name 2021-06-09 07:19:41 -03:00
mjk 5b40d166d2 make GetCurrentModulePath and GetShortcutPath throw win32 errors 2021-06-09 07:19:41 -03:00
mjk 2ad659f777 clarify some entry points' names 2021-06-09 07:19:40 -03:00
mjk b21066c89e mark exception-safe C entry points as such 2021-06-09 07:19:40 -03:00
mjk 1bd1376cea stop exceptions from crossing ABI boundary in a few places 2021-06-09 07:19:40 -03:00
mjk f1b5633ce6 introduce try_invoke -- a logging exception catcher 2021-06-09 07:19:39 -03:00
LAGonauta 3d9dcbcf87 Valac think that getters are always owned by the struct 2021-06-09 07:19:38 -03:00
LAGonauta 8856fcbceb Use g_new0 and g_free to generate raw strings 2021-06-09 07:19:37 -03:00
LAGonauta 94944a9e2a Use async 2021-06-09 07:19:37 -03:00
LAGonauta 7de1c01fe8 Add comment to builder 2021-06-09 07:19:36 -03:00
LAGonauta 0f555da7a9 Fixed crash with multiple notifications
Sometimes an invalid function pointer was called with an invalid context
2021-06-09 07:19:35 -03:00
LAGonauta f2c689fa12 Delegate activate_action to UI thread 2021-06-09 07:19:34 -03:00
LAGonauta 839d2a5316 Initial notification provider using WinRT
Crashes when activating actions, might be related to threads.
2021-06-09 07:19:32 -03:00
LAGonauta be0e1841b8 Anitial builder 2021-06-09 07:19:32 -03:00
LAGonauta 2476b5e04b Initial experiments with notification XML building 2021-06-09 07:19:32 -03:00
LAGonauta bc9b9b95e0 Add template getter 2021-06-09 07:19:31 -03:00
LAGonauta 9d9b9e8e42 Rename plugin vala file 2021-06-09 07:19:31 -03:00
LAGonauta 0153953b5d Rename enums to better match what Vala expects 2021-06-09 07:19:30 -03:00
LAGonauta 363b200272 Add template type enum 2021-06-09 07:19:28 -03:00
LAGonauta c6a96fc025 Cleanup dismissed actions on toast notification finalizer 2021-06-09 07:19:27 -03:00
LAGonauta 898470ed23 Add failed and dimissed actions 2021-06-09 07:19:25 -03:00
LAGonauta 6cdaad315a Fields must be created in the private struct
Also change unordered_map to list, we do not need hashing and stuff.
2021-06-09 07:19:24 -03:00
LAGonauta 91d61843f2 No need to enable coroutines 2021-06-09 07:19:22 -03:00
LAGonauta c4f526329d Remove unity compilation unit 2021-06-09 07:19:21 -03:00
LAGonauta 40bf3d2fd4 Actions can stack 2021-06-09 07:19:20 -03:00
LAGonauta 7fd918f32d Fix string conversion 2021-06-09 07:19:19 -03:00
LAGonauta 1d1b00222f Add toastnotifier 2021-06-09 07:19:17 -03:00
LAGonauta 8df226ec29 Increase ref on event token 2021-06-09 07:19:15 -03:00
LAGonauta 13e0a5c0c4 Use string_view 2021-06-09 07:19:14 -03:00
LAGonauta ca0cab0e36 Initial code to allow buttons and text 2021-06-09 07:19:13 -03:00
LAGonauta ae9671716f Initial GObject wrapper for WinRT notifications
Still missing a lot of stuff
2021-06-09 07:19:11 -03:00
LAGonauta ce0deed0dc Initial callback support 2021-06-09 07:19:09 -03:00
LAGonauta 95051d304a Initial code for using winrt headers 2021-06-09 07:19:08 -03:00
LAGonauta 915dd4c738 Add MIT licensed winrt headers 2021-06-09 07:19:06 -03:00
LAGonauta e3356bb3aa Do not use GLib to open links in messages
Use ShellExecute
2021-06-09 07:19:05 -03:00
LAGonauta 03774a3756 Fix muc invite and voide request not working 2021-06-09 07:19:05 -03:00
LAGonauta 2d63c8ae77 Rename field 2021-06-09 07:19:04 -03:00
LAGonauta 96f96ead7e Use list with all notifications 2021-06-09 07:19:04 -03:00
LAGonauta 5f2e636868 Add notification retraction 2021-06-09 07:19:03 -03:00
LAGonauta d8bb9897a4 Add support for custom actions on notification 2021-06-09 07:19:02 -03:00
LAGonauta 08f2391acc Initial plugin using new notification provider 2021-06-09 07:19:02 -03:00
LAGonauta eae628758c Use VAPI and generate template in-app 2021-06-09 07:19:01 -03:00
LAGonauta 931f09504d Added 32-bit wintoast linker library 2021-06-09 07:19:01 -03:00
LAGonauta 5702b323c9 Use dynamic linking instead of runtime loading
Also made me notice that the signature of the function with the callback was wrong. Oops.
2021-06-09 07:19:00 -03:00
LAGonauta 198bce4a84 Allow null image_path 2021-06-09 07:18:59 -03:00
LAGonauta 8553a7cd86 Add callback support 2021-06-09 07:18:59 -03:00
LAGonauta 642ed6ab1b Convert C code to Vala 2021-06-09 07:18:58 -03:00
LAGonauta d36de2b9ea Use code from Dino.Ui.Util 2021-06-09 07:18:56 -03:00
LAGonauta 49978edebf Clean up 2021-06-09 07:18:55 -03:00
LAGonauta 65a26bce53 Refactor windows-notification plugin 2021-06-09 07:18:53 -03:00
LAGonauta 1d4bb774a5 Initial notification support 2021-06-09 07:18:51 -03:00
LAGonauta 3dd19fad71 Use last_index_of instead of index_of 2021-06-09 07:18:47 -03:00
LAGonauta 4a3306b479 Allow 32-bit linking
Win32 apis are __stdcall
2021-06-09 07:18:45 -03:00
LAGonauta 0b9bebd97e Use Dino.Util.get_content_type also on preview 2021-06-09 07:18:45 -03:00
LAGonauta 24890ca38a Export all plugin symbols on Windows 2021-06-09 07:18:44 -03:00
LAGonauta 50b2a0dc0a Do not hardcode GPG path on Windows 2021-06-09 07:18:43 -03:00
LAGonauta a6515ed826 Add implicit link directories to package HINT path on MingW
Instead of blacklisting those libraries
2021-06-09 07:18:43 -03:00
LAGonauta e7fd7f4726 Add --export-all-symbols to Windows compilation 2021-06-09 07:18:42 -03:00
LAGonauta 9022ac596b Every call to CoInitialize() must be balanced
with a call to CoUninitialize()
2021-06-09 07:18:42 -03:00
LAGonauta 823df3ed70 Add WIN32 fonts as a plugin 2021-06-09 07:18:26 -03:00
LAGonauta 6d5ee03e9e Set Windows executable version from PROJECT_VERSION 2021-06-09 07:17:22 -03:00
LAGonauta 1673ec3481 Added information and Dino icon to Windows executable 2021-06-09 07:17:22 -03:00
LAGonauta 16d0cc6fd9 Set alternate file stream for downloaded files. 2021-06-09 07:17:21 -03:00
LAGonauta 03f17b8f39 Fixed some Windows not appearing when opening file 2021-06-09 07:17:21 -03:00
LAGonauta 25673ab145 Use slight larger font on Windows so it matches Linux style
Also fixes some fuzzy fonts.
2021-06-09 07:17:21 -03:00
LAGonauta 8cb15fc325 Use ShellExecute instead of AppInfo to open files on Windows 2021-06-09 07:17:21 -03:00
LAGonauta 281a84a14c Add support for OpenPGP on Windows 2021-06-09 07:17:20 -03:00
LAGonauta e2fd821bb4 Add missing _WIN32 2021-06-09 07:17:20 -03:00
LAGonauta 0d384f83ae Added _WIN32 define to VALAC on Windows 2021-06-09 07:17:20 -03:00
LAGonauta 1841b348f1 Do not search for the built-in libraries when compiling with MINGW 2021-06-09 07:17:19 -03:00
LAGonauta 5f5b8fdfa0 fix build on newest MSYS2 2021-06-09 07:17:19 -03:00
LAGonauta eb7cf3297c Windows compatibility Tweaks 2021-06-09 07:17:18 -03:00
Martin Dosch 5d551360df Fix syntax error for setting AppID. 2020-11-06 14:12:50 +01:00
Martin Dosch a94acccd1d Add AppID (untested). 2020-10-29 13:47:26 +01:00
Martin Dosch d0d676e72d Add compression to achieve smaller installer size. 2020-09-18 16:51:41 +02:00
Martin Dosch 81721307ca Remove installation type "OpenPGP" support
This is no longer needed (see previous commit)
but was forgotten to remove in the previous
commit.
2020-07-10 17:04:13 +02:00
Martin Dosch 2d73f97d34 Removed compenent section
This section was only introduced to be able to
disable the OpenPGP plugin as Dino often crashed
on Windows if OpenPGP was not installed but the
plugin enabled.

This is no more necessary as the OpenPGP plugin
is now disabled by default.
2020-07-10 16:55:00 +02:00
Martin Dosch e24aa9aabc Add option to install without OpenPGP plugin 2020-06-28 09:00:03 +02:00
Martin Dosch 26a85ac109 Add german language. 2020-06-28 07:13:52 +02:00
Martin Dosch 8adf6731e7 Simplify installer script
The current build script already places the files in the right
folder structure so the installer doesn't have to do it itself
2020-05-19 08:38:17 +02:00
Martin Dosch 9154edaee1 Add dino logo again
The dino logo for the startmenu was accidentally
no longer included since the last commit.
2020-05-19 01:05:22 +02:00
Martin Dosch fd916bd6e6 Prevent duplicated DLLs 2020-05-19 00:55:14 +02:00
Martin Dosch 48619ee84d Add startmenu folder with several items
Added a startmenu folder with the following items:
* Dino launcher
* License
* Link to Dino website
* Uninstaller
2020-05-05 22:20:49 +02:00
Martin Dosch 70900cd73e Add license to windows installer 2020-05-05 19:17:45 +02:00
Martin Dosch 64260ee8c0 Merge branch 'master' into master-windows-changes 2020-05-03 11:11:19 +02:00
Martin Dosch 246000ce6a Fix typo in Dino slogan 2020-05-03 01:26:59 +02:00
Martin Dosch 9779abe701 Prepare signing
Collected some infos regarding signing
a windows build.
2020-05-03 01:01:30 +02:00
Martin Dosch 3ffec591a9 Revert "Prepare signing"
I copied the files into the wrong folder…
… it's late, sorry.

This reverts commit 7d6b9e7f4c.
2020-05-03 00:59:42 +02:00
Martin Dosch 7d6b9e7f4c Prepare signing
Collected some infos regarding signing
a windows build.
2020-05-03 00:55:59 +02:00
Martin Dosch 43f118bd4e Add experimental windows installer
This nsis script should create a windows installer.
Although the installer worked for the first tests
you should handle it with care and consider it
highly experimental
2020-05-02 22:03:20 +02:00
LAGonauta 057a72cd9f Windows compatibility Tweaks 2020-05-02 12:48:48 -03:00
129 changed files with 930 additions and 3815 deletions

View file

@ -12,31 +12,15 @@ jobs:
install: git install: git
- run: git config --global core.autocrlf input - run: git config --global core.autocrlf input
- uses: actions/checkout@v4 - uses: actions/checkout@v4
with: - name: Build Dino
fetch-depth: 0
- name: Install build-dependencies
run: | run: |
msys2 -c './build-win64.sh --prepare' msys2 -c './build-win64.sh --prepare'
- name: Build Dino+ (Meson, without saving) msys2 -c './build-win64.sh'
run: | - name: Build Dino Installer
msys2 -c './build-win64.sh -s meson -c -b -t -w'
- name: Build Dino+ (CMake)
run: |
msys2 -c './build-win64.sh -s cmake -c -b -t -i'
- name: Build Dino+ installer
run: | run: |
msys2 -c './build-win64.sh --build-installer' msys2 -c './build-win64.sh --build-installer'
- name: Upload Dino+ installer - name: Upload Dino Installer
uses: actions/upload-artifact@v4 uses: actions/upload-artifact@v4
with: with:
name: dino-plus-installer name: dino-installer
path: windows-installer/dino-installer.exe path: windows-installer/dino-installer.exe
- name: Release Dino+ installer
if: ${{ github.ref_type == 'tag' }}
uses: svenstaro/upload-release-action@2.9.0
with:
repo_token: ${{ secrets.GITHUB_TOKEN }}
file: windows-installer/dino-installer.exe
asset_name: dino-plus-installer.exe
tag: ${{ github.ref }}
release_name: Dino+ ${{ github.ref_name }}

View file

@ -7,64 +7,35 @@ jobs:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
with: with:
fetch-depth: 0 fetch-depth: 0
- run: sudo rm /etc/apt/sources.list.d/microsoft-prod.list
- run: sudo apt-get update - run: sudo apt-get update
- run: sudo apt-get remove libunwind-14-dev - run: sudo apt-get remove libunwind-14-dev
- run: sudo apt-get install -y build-essential gettext cmake valac libgee-0.8-dev libsqlite3-dev libgtk-4-dev libnotify-dev libgpgme-dev libsoup2.4-dev libgcrypt20-dev libqrencode-dev libnice-dev libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev libsrtp2-dev libadwaita-1-dev libsignal-protocol-c-dev libcanberra-dev - run: sudo apt-get install -y build-essential gettext cmake valac libgee-0.8-dev libsqlite3-dev libgtk-4-dev libnotify-dev libgpgme-dev libsoup2.4-dev libgcrypt20-dev libqrencode-dev libnice-dev libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev libsrtp2-dev libwebrtc-audio-processing-dev libadwaita-1-dev libsignal-protocol-c-dev libcanberra-dev
- run: ./configure --release --no-debug --with-tests --enable-plugin=notification-sound --prefix=/usr --without-webrtc - run: ./configure --with-tests --with-libsignal-in-tree
- run: cmake --build build - run: make
- run: cmake --build build --target=test - run: build/xmpp-vala-test
- name: Build DEB-package - run: build/omemo-test
run: cd build && cpack -G DEB
- name: Upload Dino+ DEB-package
uses: actions/upload-artifact@v4
with:
name: dino-plus-deb
path: _packages/dino-plus.deb
- name: Release Dino+ DEB-package
if: ${{ github.ref_type == 'tag' }}
uses: svenstaro/upload-release-action@2.9.0
with:
repo_token: ${{ secrets.GITHUB_TOKEN }}
file: _packages/dino-plus.deb
asset_name: dino-plus.deb
tag: ${{ github.ref }}
release_name: Dino+ ${{ github.ref_name }}
build-meson: build-meson:
runs-on: ubuntu-22.04 runs-on: ubuntu-22.04
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
with: with:
fetch-depth: 0 fetch-depth: 0
- run: sudo rm /etc/apt/sources.list.d/microsoft-prod.list
- run: sudo apt-get update - run: sudo apt-get update
- run: sudo apt-get remove libunwind-14-dev - run: sudo apt-get remove libunwind-14-dev
- run: sudo apt-get install -y build-essential gettext libadwaita-1-dev libcanberra-dev libgcrypt20-dev libgee-0.8-dev libgpgme-dev libgstreamer-plugins-base1.0-dev libgstreamer1.0-dev libgtk-4-dev libnice-dev libnotify-dev libqrencode-dev libsignal-protocol-c-dev libsoup2.4-dev libsqlite3-dev libsrtp2-dev meson valac - run: sudo apt-get install -y build-essential gettext libadwaita-1-dev libcanberra-dev libgcrypt20-dev libgee-0.8-dev libgpgme-dev libgstreamer-plugins-base1.0-dev libgstreamer1.0-dev libgtk-4-dev libnice-dev libnotify-dev libqrencode-dev libsignal-protocol-c-dev libsoup-3.0-dev libsqlite3-dev libsrtp2-dev libwebrtc-audio-processing-dev meson valac
- run: meson setup build -Duse-soup2=true -Dplugin-rtp-webrtc-audio-processing=disabled - run: meson setup build
- run: meson compile -C build - run: meson compile -C build
- run: meson test -C build
build-flatpak: build-flatpak:
runs-on: ubuntu-22.04 runs-on: ubuntu-22.04
container: container:
image: bilelmoussaoui/flatpak-github-actions:gnome-44 image: bilelmoussaoui/flatpak-github-actions:gnome-44
options: --privileged options: --privileged
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v2
with: with:
fetch-depth: 0 fetch-depth: 0
submodules: true submodules: true
- run: flatpak repair --user
- uses: flatpak/flatpak-github-actions/flatpak-builder@v6.1 - uses: flatpak/flatpak-github-actions/flatpak-builder@v6.1
with: with:
manifest-path: im.dino.Dino.json manifest-path: im.dino.Dino.json
build-bundle: true build-bundle: false
- name: Release Dino+ flatpak
if: ${{ github.ref_type == 'tag' }}
uses: svenstaro/upload-release-action@2.9.0
with:
repo_token: ${{ secrets.GITHUB_TOKEN }}
file: app.flatpak
asset_name: app.flatpak
tag: ${{ github.ref }}
release_name: Dino+ ${{ github.ref_name }}

5
.gitignore vendored
View file

@ -6,10 +6,7 @@ Makefile
.idea .idea
.sqlite3 .sqlite3
gschemas.compiled gschemas.compiled
windows-installer/win64-*/ windows-installer/win64-dist/
*.exe *.exe
*.dll *.dll
*.flatpak
flatpak-dist
.flatpak-builder .flatpak-builder
*.zst

View file

@ -1,54 +0,0 @@
Here are the instructions to anyone whom wants to try the build themselves.
It is assumed that you have `brew` installed, know your way around terminal, and generally understand what you're doing.
## Build
1. Start with an already existing build instructions of the original project:
`https://github.com/dino/dino/wiki/macOS`
2. On step 4, replace the path to the upstream repo with this one:
```
git clone https://github.com/mxlgv/dino
cd dino
```
3. On step 5 run `./configure --with-libsoup3`.
4. Continue the build according to the original instructions.
## Install with Brew
To build Dino using this formula, follow these instructions:
```
brew tap mxlgv/homebrew-dino
brew install mxlgv/homebrew-dino/dino
```
You can start the Dino client installed via homebrew with the following command:
```
./opt/homebrew/Cellar/dino/3/bin/dino
```
You can create a shortcut with a symbolic link pointing to this file, so that you can open it in a more convenient way.
If you encounter an error related to rpath, you need to add `DYLD_LIBRARY_PATH` to the environment variable:
```
export DYLD_LIBRARY_PATH=/opt/homebrew/Cellar/dino/3/lib
```
## Notifications
There is a draft PR which can be used to enable notifications: https://github.com/mxlgv/dino/pull/45
## Start Dino
In order to run Dino, just run in the build folder:
```
./dino
```

View file

@ -10,16 +10,6 @@ else ()
set(PROJECT_VERSION ${VERSION_FULL}) set(PROJECT_VERSION ${VERSION_FULL})
endif () endif ()
include(CTest)
option(PLUGIN_RTP_WEBRTC_AUDIO_PROCESSING "Use WebRTC audio processing" ON)
option(WITH_WASAPI "Use wasapi instead of directsound on windows" ON)
# https://gitlab.kitware.com/cmake/cmake/-/issues/19804
if (WIN32)
list(APPEND CMAKE_FIND_LIBRARY_SUFFIXES .dll)
endif()
# Prepare Plugins # Prepare Plugins
set(DEFAULT_PLUGINS omemo;openpgp;http-files;ice;rtp) set(DEFAULT_PLUGINS omemo;openpgp;http-files;ice;rtp)
if (WIN32) if (WIN32)
@ -81,8 +71,6 @@ macro(set_path what val desc)
endif() endif()
endmacro(set_path) endmacro(set_path)
set(LOCALEDIR_NAME "share/locale")
string(REGEX REPLACE "^liblib" "lib" LIBDIR_NAME "lib${LIB_SUFFIX}") string(REGEX REPLACE "^liblib" "lib" LIBDIR_NAME "lib${LIB_SUFFIX}")
set_path(CMAKE_INSTALL_PREFIX "${CMAKE_INSTALL_PREFIX}" "Installation directory for architecture-independent files") set_path(CMAKE_INSTALL_PREFIX "${CMAKE_INSTALL_PREFIX}" "Installation directory for architecture-independent files")
set_path(EXEC_INSTALL_PREFIX "${CMAKE_INSTALL_PREFIX}" "Installation directory for architecture-dependent files") set_path(EXEC_INSTALL_PREFIX "${CMAKE_INSTALL_PREFIX}" "Installation directory for architecture-dependent files")
@ -96,7 +84,7 @@ set_path(SERVICE_FILE_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/dbus-1/services" "Ins
set_path(ICON_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/icons" "Installation directory for icons") set_path(ICON_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/icons" "Installation directory for icons")
set_path(INCLUDE_INSTALL_DIR "${EXEC_INSTALL_PREFIX}/include" "Installation directory for C header files") set_path(INCLUDE_INSTALL_DIR "${EXEC_INSTALL_PREFIX}/include" "Installation directory for C header files")
set_path(LIB_INSTALL_DIR "${EXEC_INSTALL_PREFIX}/${LIBDIR_NAME}" "Installation directory for object code libraries") set_path(LIB_INSTALL_DIR "${EXEC_INSTALL_PREFIX}/${LIBDIR_NAME}" "Installation directory for object code libraries")
set_path(LOCALE_INSTALL_DIR "${EXEC_INSTALL_PREFIX}/${LOCALEDIR_NAME}" "Installation directory for locale files") set_path(LOCALE_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/locale" "Installation directory for locale files")
set_path(PLUGIN_INSTALL_DIR "${LIB_INSTALL_DIR}/dino/plugins" "Installation directory for dino plugin object code files") set_path(PLUGIN_INSTALL_DIR "${LIB_INSTALL_DIR}/dino/plugins" "Installation directory for dino plugin object code files")
set_path(VAPI_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/vala/vapi" "Installation directory for Vala API files") set_path(VAPI_INSTALL_DIR "${SHARE_INSTALL_PREFIX}/vala/vapi" "Installation directory for Vala API files")
@ -233,9 +221,6 @@ add_subdirectory(main)
add_subdirectory(crypto-vala) add_subdirectory(crypto-vala)
add_subdirectory(plugins) add_subdirectory(plugins)
# To generate a DEB package
include(Packing)
# uninstall target # uninstall target
configure_file("${CMAKE_SOURCE_DIR}/cmake/cmake_uninstall.cmake.in" "${CMAKE_BINARY_DIR}/cmake_uninstall.cmake" IMMEDIATE @ONLY) configure_file("${CMAKE_SOURCE_DIR}/cmake/cmake_uninstall.cmake.in" "${CMAKE_BINARY_DIR}/cmake_uninstall.cmake" IMMEDIATE @ONLY)
add_custom_target(uninstall COMMAND ${CMAKE_COMMAND} -P ${CMAKE_BINARY_DIR}/cmake_uninstall.cmake COMMENT "Uninstall the project...") add_custom_target(uninstall COMMAND ${CMAKE_COMMAND} -P ${CMAKE_BINARY_DIR}/cmake_uninstall.cmake COMMENT "Uninstall the project...")

56
README-WIN64.md Normal file
View file

@ -0,0 +1,56 @@
![Dino (WIN64)](https://dino.im/img/readme_header.svg)
=======
![screenshots](https://dino.im/img/screenshot-main.png)
Build on Windows (x86_64)
------------
- Install and configure the [MSYS2](https://www.msys2.org/) package;
- Go to `MINGW64` environment;
- Clone project:
```sh
git clone https://github.com/mxlgv/dino && cd dino
```
- Run the script to install dependencies:
```sh
./build-win64.sh --prepare
```
- Start the build (the builded distribution is available in the `windows-installer/dist-win64` folder):
```sh
./build-win64.sh
```
Note: the build script has some other options, their description can be found using the `--help`.
Build Windows Installer (NSIS)
------------
Before this, you must build the project according to the instructions above. It's worth making sure that `windows-installer/dist-win64` is not empty.
Now you should run:
```sh
./build-win64.sh --build-installer
```
The builded installer will be available in the directory `windows-installer/dino-installer.exe`.
Resources
---------
- Check out the [Dino website](https://dino.im).
- Join our XMPP channel at `chat@dino.im`.
- The [wiki](https://github.com/dino/dino/wiki) provides additional information.
License
-------
Dino - Modern Jabber/XMPP Client using GTK+/Vala
Copyright (C) 2016-2023 Dino contributors
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.

150
README.md
View file

@ -1,163 +1,37 @@
This is Dino+<br /> ![Dino](https://dino.im/img/readme_header.svg)
A modern XMPP/Jabber client software, based on [Dino](https://github.com/dino/dino/)
======= =======
![screenshots](dino_plus.png) ![screenshots](https://dino.im/img/screenshot-main.png)
Project description Installation
------------ ------------
Dino+ is a fork of [Dino](https://github.com/dino/dino), a modern XMPP/Jabber client written in Vala using GTK+, which includes a few relatively minor but important quality-of-life features. Have a look at the [prebuilt packages](https://github.com/dino/dino/wiki/Distribution-Packages).
It was created due to upstream project's maintainers being seemingly unwilling to accept PRs from the community. As of March 2024, there are over 50 PRs pending, most of which are silently ignored. Build
Some of the features that Dino+ implements:
* Windows support
* Proper sync for chats history, including "scroll-to-load"
* Forced MAM sync for chats
* Password change via GUI
* Optional support for Meson build system
* Some additional QoL-features
What's currently in the works:
* Multiple UI fixes
* Fixes for sound notifications
* MacOS support for Mac silicon
* and more
Dino+ is currently to be considered an alpha-quality product. Please open an issue or send a PR if you spot or fix any bugs.
OS support
------------
* Linux (flatpaks are targeted for Ubuntu 22.04+)
* Windows 10\11
* MacOS via [brew](https://brew.sh/) (very experimental, see below)
Officially, we support only x86_64 architecture. Whilst you may be able to compile this code for ARM-based platforms (like Pinephone or MacOS), there's zero guarantees that it'll compile or function properly, as we don't have the hardware and the time to test it out.
Installation (prebuilt packages & AUR)
------------
Have a look at our [releases](https://github.com/mxlgv/dino/releases).
Windows installer is self-explanatory, and will place a shortcut on your desktop. Ignore the possible warning from Windows Defender: it's caused by the fact that installer does not have a digital signature.
Deb package is built and tested only for Ubuntu 22.04, but it may work on its derivatives as well, such as Pop!_OS or Linux Mint.
Flatpak is not present in Flathub yet, but you can install it manually. The commands below assume that you have "flatpak" package of your distro installed in your system. If it is not, please refer to https://flatpak.org/setup/
1) Download the .flatpak file from [releases](https://github.com/mxlgv/dino/releases)
2) Add default Flatpak repo:
```
flatpak remote-add --if-not-exists flathub https://dl.flathub.org/repo/flathub.flatpakrepo
```
3) Change the directory to the one you've downloaded the .flatpak to and run (you may have to change the file name in this command):
```
flatpak install ./im.dino.Dino.flatpak
```
4) To launch the program, run either
```
/usr/bin/dino
```
or
```
flatpak run im.dino.Dino
```
Flatpak distribution is confirmed to be working on Arch Testing, Manjaro Stable, Void Linux and Linux Mint.
Arch Linux & Manjaro users may install `dino-plus-git` package from [AUR](https://aur.archlinux.org/packages/dino-plus-git) via any helper of their choice.
Build on Linux
----- -----
Make sure to install all [dependencies](https://github.com/dino/dino/wiki/Build#dependencies). Make sure to install all [dependencies](https://github.com/dino/dino/wiki/Build#dependencies).
./configure --release --no-debug --with-tests --enable-plugin=notification-sound --prefix=/usr ./configure
make make
build/dino build/dino
Additionally, you might need to install `gst-plugins-bad` and `webrtc-audio-processing` packages (or however they might be called in your distro), if you want to have noise cancelling and gain control. If you don't want that or your distro don't have that package, add "--without-webrtc" (without quotes) to the first command.
If you want to use `meson` build system, follow the next instructions:
meson setup build
meson configure --prefix $PWD/build/install --libdir lib build
meson compile -C build
meson install -C build
LD_LIBRARY_PATH+=:$PWD/build/install/lib build/install/bin/dino
If your `nice` library depends on `libsoup-2.4` (consider `ldd` output for the `libnice.so`), you should additionally specify `-Duse-soup2=true` option.
`LD_LIBRARY_PATH` should point to the directory containing the `libdino.so` library.
Skip `meson configure` step, if you want to install the program globally.
You can specify any convenient directory in the option `--prefix` where the program will be installed.
If there is no `webrtcdsp` plugin in your system (check this by calling `gst-inspect-1.0 webrtcdsp`) you should pass extra argument:
* `--without-webrtcdsp` for `./configure`;
* `-Dplugin-rtp-webrtc-audio-processing=disabled` for `meson`;
* `-DPLUGIN_RTP_WEBRTC_AUDIO_PROCESSING=OFF` for `cmake`.
Build on Windows (x86_64)
------------
- Install and configure the [MSYS2](https://www.msys2.org/) package;
- Go to `MINGW64` environment;
- Clone project:
```sh
git clone https://github.com/mxlgv/dino && cd dino
```
- Run the script to install dependencies:
```sh
./build-win64.sh --prepare
```
- Start the build (the builded distribution is available in the `windows-installer/dist-win64` folder):
```sh
./build-win64.sh
```
If you want to use meson build system, please use `-s meson` key as the first argument, i.e.
```sh
bash build-win64.sh -s meson -c -b
```
will do the same as commands above, but using meson.
Note: the build script has some other options, their description can be found using the `--help`.
Build Windows Installer (NSIS)
------------
Before this, you must build the project according to the instructions above. It's worth making sure that `windows-installer/dist-win64` is not empty.
Now you should run:
```sh
./build-win64.sh --build-installer
```
Build on MacOS
------------
Builds for Mac are currently available only as a [brew](https://brew.sh/) formulae. Please follow [these instructions](https://github.com/mxlgv/dino/blob/master/BUILD_MACOS.md) to produce a build.
Please note that Mac support is very experimental, and currently missing support for event notifications, among other things. You have been warned.
Resources Resources
--------- ---------
- Original project's [website](https://dino.im). - Check out the [Dino website](https://dino.im).
- Dino's chat room `chat@dino.im` (please don't post there any issues related to this fork!) - Join our XMPP channel at `chat@dino.im`.
- The upstream project's [wiki](https://github.com/dino/dino/wiki) provides additional information. - The [wiki](https://github.com/dino/dino/wiki) provides additional information.
Contribute Contribute
---------- ----------
- Pull requests are welcome! - Pull requests are welcome. [These](https://github.com/dino/dino/issues?q=is%3Aopen+is%3Aissue+label%3A%22good+first+issue%22) might be good first issues. Please discuss bigger changes in our channel first.
- Look at [how to debug](https://github.com/dino/dino/wiki/Debugging) Dino before you report a bug. - Look at [how to debug](https://github.com/dino/dino/wiki/Debugging) Dino before you report a bug.
- Help [translating](https://github.com/dino/dino/wiki/Translations) Dino into your language. - Help [translating](https://github.com/dino/dino/wiki/Translations) Dino into your language.
- Make a [donation](https://dino.im/#donate).
Special thanks
----------
We'd like to thank all of the contributors whom provided the PRs used in this project.
We also recognise the previous efforts of [LAGonauta](https://github.com/LAGonauta) without whom Windows build wouldn't be available.
License License
------- -------
Dino+, a modern XMPP/Jabber client software based on Dino Dino - Modern Jabber/XMPP Client using GTK+/Vala
Copyright (C) 2016-2023 Dino contributors Copyright (C) 2016-2023 Dino contributors
Copyright (C) 2024 Dino+ contributors
This program is free software: you can redistribute it and/or modify This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by it under the terms of the GNU General Public License as published by

87
build-flatpack.sh Executable file → Normal file
View file

@ -1,90 +1,25 @@
#!/bin/bash #!/bin/bash
set -e set -e
APP_NAME="im.dino.Dino" getFlatpackDependencies(){
DIST_NAME=${DIST_NAME:-"${APP_NAME}.flatpak"}
DIST_DIR="$PWD/flatpak-dist"
BUILD_TEMP_DIR="$DIST_DIR/buildtemp"
BUILD_EXPORT_DIR="$DIST_DIR/export"
msg()
{
echo -e "\e[32m$1\e[0m"
}
fatal()
{
echo -e "\e[31m$1\e[0m"
exit 1
}
get_flatpak_dependencies()
{
msg "Installing Flatpak dependencies..."
flatpak remote-add --if-not-exists flathub https://dl.flathub.org/repo/flathub.flatpakrepo flatpak remote-add --if-not-exists flathub https://dl.flathub.org/repo/flathub.flatpakrepo
flatpak install flathub org.gnome.Sdk//44 flatpak install flathub org.gnome.Sdk//44
flatpak install flathub org.gnome.Platform//44 flatpak install flathub org.gnome.Platform//44
msg "Flatpak dependencies installed"
} }
pull_shared_modules() prepareModules(){
{
msg "Pulling shared modules..."
git submodule init git submodule init
git submodule update git submodule update
msg "Shared modules successfully pulled"
} }
prepare() build(){
{ FP_TEMP_BUILD_DIR=$(mktemp -d)
get_flatpak_dependencies FP_OUTDIR="builds"
pull_shared_modules flatpak-builder ${FP_TEMP_BUILD_DIR} im.dino.Dino.json
flatpak build-export $FP_OUTDIR $FP_TEMP_BUILD_DIR
flatpak build-bundle $FP_OUTDIR dino.flatpak
} }
build() getFlatpackDependencies
{ prepareModules
msg "Build commencing!" build
rm -rf $BUILD_TEMP_DIR
flatpak-builder --install-deps-from=flathub $BUILD_TEMP_DIR "${APP_NAME}.json"
flatpak build-export $BUILD_EXPORT_DIR $BUILD_TEMP_DIR
flatpak build-bundle $BUILD_EXPORT_DIR $DIST_NAME $APP_NAME
msg "Flatpack bundle ready and saved to ${DIST_NAME}"
}
clean()
{
msg "Wiping intermediate files..."
rm -rf $BUILD_TEMP_DIR $BUILD_EXPORT_DIR
msg "Cleanup complete!"
}
help()
{
cat << EOF
usage: $0 [OPTION]
--prepare install build dependencies
--build build the project
--clean remove build artifacts
--help show this help
Bundle is saved to ${APP_NAME}.flatpak by default.
Set DIST_NAME variable to customize output file name:
'DIST_NAME=customname.flatpak $0'
Running without parameters is equivalent to running:
'--prepare', '--build' and '--clean'
EOF
}
case $1 in
"--prepare" ) prepare ;;
"--build" ) build ;;
"--help" ) help ;;
"--clean" ) clean;;
"" )
prepare
build
clean
;;
*) fatal "Unknown argument!"
esac

View file

@ -1,12 +1,8 @@
#!/bin/bash #!/bin/bash
set -e
set -eu DIST_DIR="$PWD/windows-installer/win64-dist"
PROJ_DIR=$PWD
DIST_DIR=${PROJ_DIR}/windows-installer/win64-dist
BUILD_DIR=$PROJ_DIR/build
JOBS=$NUMBER_OF_PROCESSORS JOBS=$NUMBER_OF_PROCESSORS
build_sys='cmake'
msg() msg()
{ {
@ -21,31 +17,17 @@ fatal()
download_yolort() download_yolort()
{ {
file_name=yolort.zip file_name=cppwinrt-2.0.210122.3+windows-10.0.19041+yolort-835cd4e.zip
yolort_dir="$PROJ_DIR/plugins/windows-notification/yolort" yolort_dir="$PWD/plugins/windows-notification/yolort"
rm -rf "$yolort_dir" rm -rf "$yolort_dir"
mkdir "$yolort_dir" mkdir "$yolort_dir"
curl -L -o "$file_name" "https://github.com/mxlgv/YoloRT/releases/download/dev1/$file_name" curl -L -o "$file_name" "https://github.com/LAGonauta/YoloRT/releases/download/v1.0.0/$file_name"
echo "c2727e390da7e842f66e0a4cf0a9f5d9dfb665115bb554152d98f108d322bbc1 $file_name" | sha256sum --check --status echo "675a6d943c97b4acdbfaa473f68d3241d1798b31a67b5529c8d29fc0176a1707 $file_name" | sha256sum --check --status
unzip -o "$file_name" -d "$yolort_dir" unzip -o "$file_name" -d "$yolort_dir"
rm -f "$file_name" rm -f "$file_name"
} }
download_gtk4_git()
{
# FIXME: The bug fix https://gitlab.gnome.org/GNOME/gtk/-/issues/3749 is currently only available in the main branch,
# so GTK4 was builded from it. Needs to be replaced with a package from the MSYS2 repository when the changes get there.
url="https://github.com/mxlgv/mingw-w64-gtk4-git/releases/download/rel1"
gtk_pkg="mingw-w64-x86_64-gtk4-git-4.14.1.r62.gb1eed1c153-1-any.pkg.tar.zst"
gtk_gstreamer_pkg="mingw-w64-x86_64-gtk4-media-gstreamer-git-4.14.1.r62.gb1eed1c153-1-any.pkg.tar.zst"
curl -L -o "$gtk_pkg" "$url/$gtk_pkg"
curl -L -o "$gtk_gstreamer_pkg" "$url/$gtk_gstreamer_pkg"
pacman -U --needed --noconfirm "$gtk_pkg" "$gtk_gstreamer_pkg"
}
prepare() prepare()
{ {
msg "Installing MINGW64 build dependencies" msg "Installing MINGW64 build dependencies"
@ -54,6 +36,7 @@ prepare()
mingw64/mingw-w64-x86_64-gcc \ mingw64/mingw-w64-x86_64-gcc \
mingw64/mingw-w64-x86_64-cmake \ mingw64/mingw-w64-x86_64-cmake \
mingw64/mingw-w64-x86_64-ninja \ mingw64/mingw-w64-x86_64-ninja \
mingw64/mingw-w64-x86_64-gtk4 \
mingw64/mingw-w64-x86_64-libadwaita \ mingw64/mingw-w64-x86_64-libadwaita \
mingw64/mingw-w64-x86_64-sqlite3 \ mingw64/mingw-w64-x86_64-sqlite3 \
mingw64/mingw-w64-x86_64-openssl \ mingw64/mingw-w64-x86_64-openssl \
@ -74,15 +57,12 @@ prepare()
mingw64/mingw-w64-x86_64-nsis \ mingw64/mingw-w64-x86_64-nsis \
mingw64/mingw-w64-x86_64-libsignal-protocol-c \ mingw64/mingw-w64-x86_64-libsignal-protocol-c \
mingw64/mingw-w64-x86_64-icu \ mingw64/mingw-w64-x86_64-icu \
mingw64/mingw-w64-x86_64-meson \ mingw64/mingw-w64-x86_64-webrtc-audio-processing \
git \ git \
make \ make \
unzip \ unzip \
curl curl
msg "Downloading and install git versions of gtk4 and gtk4-media-gstreamer packages"
download_gtk4_git
msg "Successfully installed!" msg "Successfully installed!"
msg "Download YoloRT headers" msg "Download YoloRT headers"
@ -91,100 +71,60 @@ prepare()
} }
configure_cmake() configure()
{ {
msg "Running configuration for Windows" msg "Running configuration for Windows"
./configure --program-prefix="$DIST_DIR" --no-debug --release --disable-fast-vapi --with-libsoup3 --with-tests ./configure --program-prefix="$DIST_DIR" --no-debug --release --disable-fast-vapi --with-libsoup3
msg "Configured!" msg "Configured!"
} }
build_cmake() build()
{ {
msg "Started building on $JOBS threads" msg "Started building on $JOBS threads"
make -j"$JOBS" make -j"$JOBS"
msg "Successfully builded!" msg "Successfully builded!"
msg "Installing Dino .."
make install
}
test_cmake()
{
msg "Run tests"
make test
}
configure_meson()
{
arg=${1:-"none"}
encr=${2:-"auto"}
local cmd=""
if [ x"${arg}" == x"reconfig" ]; then
cmd=--reconfigure
fi
mkdir -p $BUILD_DIR
meson setup ${cmd} --prefix "$DIST_DIR" \
-D crypto-backend=${encr} \
-D plugin-ice=enabled \
$PROJ_DIR $BUILD_DIR
}
build_meson()
{
meson compile -C $BUILD_DIR
meson install -C $BUILD_DIR
}
test_meson()
{
msg "Run tests"
meson test -C $BUILD_DIR
} }
dist_install() dist_install()
{ {
_dist_arg=${1:-$DIST_DIR} msg "Installing Dino in '$DIST_DIR'!"
make install
msg "Generate dino-with-console.exe"
cp -f "$_dist_arg/bin/dino.exe" "$_dist_arg/bin/dino-with-console.exe"
# IMAGE_SUBSYSTEM_WINDOWS_CUI = 0x0003
# SUBSYSTEM_OFFSET = 0xDC (220)
printf '\x03\x00' | dd of="$_dist_arg/bin/dino-with-console.exe" bs=1 seek=220 count=2 conv=notrunc
msg "Copying MINGW64 dependencies" msg "Copying MINGW64 dependencies"
cp /mingw64/bin/gdbus.exe "$_dist_arg/bin" cp /mingw64/bin/gdbus.exe "$DIST_DIR/bin"
cp /mingw64/bin/gspawn-win64-helper.exe "$_dist_arg/bin" cp /mingw64/bin/gspawn-win64-helper.exe "$DIST_DIR/bin"
cp /mingw64/bin/libcrypto-*-x64.dll "$_dist_arg/bin/" cp /mingw64/bin/libcrypto-*-x64.dll "$DIST_DIR/bin/"
cp -r /mingw64/lib/gstreamer-1.0 "$_dist_arg/lib" cp -r /mingw64/lib/gstreamer-1.0 "$DIST_DIR/lib"
mkdir -p "$_dist_arg/lib/gdk-pixbuf-2.0/" && cp -r /mingw64/lib/gdk-pixbuf-2.0 "$_dist_arg/lib/" mkdir -p "$DIST_DIR/lib/gdk-pixbuf-2.0/" && cp -r /mingw64/lib/gdk-pixbuf-2.0 "$DIST_DIR/lib/"
mkdir -p "$_dist_arg/lib/gio/" && cp -r /mingw64/lib/gio "$_dist_arg/lib/" mkdir -p "$DIST_DIR/lib/gio/" && cp -r /mingw64/lib/gio "$DIST_DIR/lib/"
list=`find "$_dist_arg" -type f \( -name "*.exe" -o -name "*.dll" \) -exec \ list=`find "$DIST_DIR" -type f \( -name "*.exe" -o -name "*.dll" \) -exec \
ntldd -R {} + | \ ntldd -R {} + | \
grep "mingw64" | \ grep "mingw64" | \
cut -f1 -d "=" | sort | uniq` cut -f1 -d "=" | sort | uniq`
for a in $list; do for a in $list; do
cp -fv "/mingw64/bin/$a" "$_dist_arg/bin/" cp -fv "/mingw64/bin/$a" "$DIST_DIR/bin/"
done done
msg "Removing debug information from all EXE and DLL files" msg "Removing debug information from all EXE and DLL files"
find "$_dist_arg" -iname "*.exe" -exec strip -s {} + find "$DIST_DIR" -iname "*.exe" -exec strip -s {} +
find "$_dist_arg" -iname "*.dll" -exec strip -s {} + find "$DIST_DIR" -iname "*.dll" -exec strip -s {} +
find "$_dist_arg" -iname "*.a" -exec rm {} + find "$DIST_DIR" -iname "*.a" -exec rm {} +
msg "Removing redudant header files" msg "Removing redudant header files"
rm -rf "$_dist_arg/include" rm -rf "$DIST_DIR/include"
msg "Copy LICENSE" msg "Copy LICENSE"
cp -f "$PWD/LICENSE" "$_dist_arg/LICENSE" cp -f "$PWD/LICENSE" "$DIST_DIR/LICENSE"
msg "Copy icons, themes, locales and fonts" msg "Copy icons, themes, locales and fonts"
cp -f "$PWD/main/dino.ico" "$_dist_arg/dino.ico" cp -f "$PWD/main/dino.ico" "$DIST_DIR/dino.ico"
cp -rf "/mingw64/share/xml" "$_dist_arg/share" cp -rf "/mingw64/share/xml" "$DIST_DIR/share"
mkdir -p "$_dist_arg/etc/fonts" && cp -r /mingw64/etc/fonts "$_dist_arg/etc/" mkdir -p "$DIST_DIR/etc/fonts" && cp -r /mingw64/etc/fonts "$DIST_DIR/etc/"
mkdir -p "$_dist_arg/share/icons" && cp -r /mingw64/share/icons "$_dist_arg/share/" mkdir -p "$DIST_DIR/share/icons" && cp -r /mingw64/share/icons "$DIST_DIR/share/"
mkdir -p "$_dist_arg/share/glib-2.0/schemas" && cp -rf /mingw64/share/glib-2.0/schemas "$_dist_arg/share/glib-2.0/" mkdir -p "$DIST_DIR/share/glib-2.0/schemas" && cp -rf /mingw64/share/glib-2.0/schemas "$DIST_DIR/share/glib-2.0/"
msg "Successfully installed!" msg "Successfully installed!"
} }
@ -200,71 +140,24 @@ build_installer()
clean() clean()
{ {
rm -rf $BUILD_DIR $DIST_DIR rm -rf build "$DIST_DIR"
msg "Build artifacts removed successfull!" msg "Build artifacts removed successfull!"
} }
help() help()
{ {
cat << EOF cat << EOF
Script to build Dino for windows using cmake or meson build-system. usage: $0 [OPTION]
By default it will be build using build directory --prepare install build dependencies
$BUILD_DIR --configure configure the project
and installed to --build build the project
$DIST_DIR --dist-install install the builded project
--build-installer build installer (using NSIS)
--clean remove build artifacts
--help show this help
Usage: $0 [option] Running without parameters is equivalent to running:
'--configure', '--build' and '--dist-install'
Note: you may set the multiple options, but be sure that they will be
processed sequentially (one-by-one), e.g. command
$0 -s meson -c -b
will run buld config and _after_ that run build using meson, while
$0 -c -b -s meson
will run cmake-based configure & build commands and the -s option
wont have any effect. And the one
$0 -b -s meson -c
is incorrect, as it willtry to run build(for cmake), then configure
with for meson build.
--help, -h
print this help message.
--set-buildsys, -s
set (specify) build system name to be used
possible options are: cmake or meson
--prepare, -p
install build dependencies. may be done once.
--configure, -c
configure build using selected build-system.
--build, -b
invoke build.
--test, -t
run tests.
--reconfig, -r
reconfigure project, if minor changes were
done to build config files but build has been
configured already (only for meson!).
--whipe, -w
remove build artifacts from $BUILD_DIR
--verbose, -v
verbose output enable.
--dist-install, -i
install the builded project along with its'
dependencies.
--build-installer
build installer (using NSIS)
Running without parameters will run configure, build & install
using cmake-based build-system as default one.
EOF EOF
} }
@ -272,62 +165,18 @@ if [[ "$(uname)" != "MINGW64_NT"* ]]; then
fatal "This is not a MINGW64 environment!" fatal "This is not a MINGW64 environment!"
fi fi
# no options provided,simply build with defaults case $1 in
if [[ $# == 0 ]]; then "--prepare" ) prepare ;;
prepare "--configure" ) configure ;;
configure_${build_sys} "--build" ) build ;;
build_${build_sys} "--dist-install" ) dist_install ;;
dist_install "--build-installer") build_installer ;;
"--clean" ) clean ;;
exit 0 "--help" ) help ;;
fi "" )
configure
while [[ $# > 0 ]]; build
do dist_install
case $1 in ;;
--prepare|-p) *) fatal "Unknown argument!"
prepare esac
;;
--configure|-c)
configure_${build_sys}
;;
--build|-b)
build_${build_sys}
;;
--test|-t)
test_${build_sys}
;;
--reconfig|-r)
configure_${build_sys} reconfig
;;
--whipe|-w)
clean
;;
--dist-install|-i)
dist_install
;;
--verbose|-v)
set -xv
;;
--help|-h)
help
exit 0;
;;
--build-installer)
build_installer
;;
--set-buildsys|-s)
if [ x"$2" != x"cmake" -a x"$2" != x"meson" ]; then
fatal "Improper build system selected: ${2}!"
exit 1;
fi
build_sys=$2
shift
;;
-*)
echo "Unknown option $1"
exit 1
;;
esac
shift
done

View file

@ -47,7 +47,7 @@ function(_compute_version_from_git)
return() return()
endif (NOT git_result EQUAL 0) endif (NOT git_result EQUAL 0)
if (git_tag MATCHES "^v?([0-9]+[.]?[0-9]*[.]?[0-9]*[.]?[0-9]*)(-[.0-9A-Za-z-]+)?([+][.0-9A-Za-z-]+)?$") if (git_tag MATCHES "^v?([0-9]+[.]?[0-9]*[.]?[0-9]*)(-[.0-9A-Za-z-]+)?([+][.0-9A-Za-z-]+)?$")
set(VERSION_LAST_RELEASE "${CMAKE_MATCH_1}") set(VERSION_LAST_RELEASE "${CMAKE_MATCH_1}")
else () else ()
return() return()

View file

@ -1,11 +0,0 @@
find_library(GstWebrtcDsp_LIBRARY gstwebrtcdsp PATH_SUFFIXES gstreamer-1.0)
if(GstWebrtcDsp_LIBRARY_FOUND)
find_package(Gst)
set(GstWebrtcDsp_VERSION ${Gst_VERSION})
endif()
include(FindPackageHandleStandardArgs)
find_package_handle_standard_args(GstWebrtcDsp
REQUIRED_VARS GstWebrtcDsp_LIBRARY
VERSION_VAR GstWebrtcDsp_VERSION)

View file

@ -0,0 +1,12 @@
include(PkgConfigWithFallback)
find_pkg_config_with_fallback(WebRTCAudioProcessing
PKG_CONFIG_NAME webrtc-audio-processing
LIB_NAMES webrtc_audio_processing
INCLUDE_NAMES webrtc/modules/audio_processing/include/audio_processing.h
INCLUDE_DIR_SUFFIXES webrtc-audio-processing webrtc_audio_processing
)
include(FindPackageHandleStandardArgs)
find_package_handle_standard_args(WebRTCAudioProcessing
REQUIRED_VARS WebRTCAudioProcessing_LIBRARY
VERSION_VAR WebRTCAudioProcessing_VERSION)

View file

@ -1,34 +0,0 @@
# This is a package creation module using CPack.
# Currently only DEB package generation is supported.
set(CPACK_PACKAGE_NAME dino-plus)
set(CPACK_PACKAGE_DESCRIPTION_SUMMARY
"modern XMPP/Jabber client software, based on Dino.
Dino+ is a fork of Dino, a modern XMPP/Jabber client
written in Vala using GTK+, which includes a few
relatively minor but important quality-of-life features.")
set(CPACK_VERBATIM_VARIABLES YES)
set(CPACK_PACKAGE_FILE_NAME ${CPACK_PACKAGE_NAME})
set(CPACK_PACKAGE_INSTALL_DIRECTORY ${CPACK_PACKAGE_NAME})
set(CPACK_OUTPUT_FILE_PREFIX "${CMAKE_SOURCE_DIR}/_packages")
set(CPACK_PACKAGE_VERSION ${VERSION_FULL})
set(CPACK_PACKAGE_HOMEPAGE_URL "https://github.com/mxlgv/dino")
set(CPACK_PACKAGE_CONTACT "maxlogaev@proton.me")
set(CPACK_COMPONENTS_GROUPING ALL_COMPONENTS_IN_ONE)
set(CPACK_STRIP_FILES TRUE)
# For DEB only
set(CPACK_DEBIAN_PACKAGE_MAINTAINER "Maxim Logaev <${CPACK_PACKAGE_CONTACT}>")
set(CPACK_DEB_COMPONENT_INSTALL YES)
set(CPACK_DEBIAN_PACKAGE_SECTION "net")
set(CPACK_DEBIAN_PACKAGE_CONFLICTS "dino-im, dino-im-common")
set(CPACK_DEBIAN_PACKAGE_RECOMMENDS "ca-certificates, dbus, fonts-noto-color-emoji, network-manager")
set(CPACK_DEBIAN_PACKAGE_SHLIBDEPS ON)
set(CPACK_DEBIAN_PACKAGE_GENERATE_SHLIBS ON)
set(CPACK_DEBIAN_PACKAGE_GENERATE_SHLIBS_POLICY ">=")
install(FILES ${CMAKE_SOURCE_DIR}/LICENSE_SHORT DESTINATION ${SHARE_INSTALL_PREFIX}/doc/${CPACK_PACKAGE_NAME} RENAME copyright)
install(FILES ${CMAKE_SOURCE_DIR}/README.md DESTINATION ${SHARE_INSTALL_PREFIX}/doc/${CPACK_PACKAGE_NAME})
include(CPack)

12
configure vendored
View file

@ -1,7 +1,7 @@
#!/bin/sh #!/bin/sh
OPTS=`getopt -o "h" --long \ OPTS=`getopt -o "h" --long \
help,fetch-only,no-debug,disable-fast-vapi,with-tests,release,with-libsoup3,without-webrtcdsp,\ help,fetch-only,no-debug,disable-fast-vapi,with-tests,release,with-libsoup3,\
enable-plugin:,disable-plugin:,\ enable-plugin:,disable-plugin:,\
prefix:,program-prefix:,exec-prefix:,lib-suffix:,\ prefix:,program-prefix:,exec-prefix:,lib-suffix:,\
bindir:,libdir:,includedir:,datadir:,\ bindir:,libdir:,includedir:,datadir:,\
@ -15,13 +15,12 @@ eval set -- "$OPTS"
PREFIX=${PREFIX:-/usr/local} PREFIX=${PREFIX:-/usr/local}
ENABLED_PLUGINS= ENABLED_PLUGINS=
DISABLED_PLUGINS= DISABLED_PLUGINS=
BUILD_TESTS=no BUILD_TESTS=
BUILD_TYPE=Debug BUILD_TYPE=Debug
DISABLE_FAST_VAPI= DISABLE_FAST_VAPI=
LIB_SUFFIX= LIB_SUFFIX=
NO_DEBUG= NO_DEBUG=
USE_SOUP3= USE_SOUP3=
PLUGIN_RTP_WEBRTC_AUDIO_PROCESSING=yes
EXEC_PREFIX= EXEC_PREFIX=
BINDIR= BINDIR=
@ -55,7 +54,6 @@ Configuration:
--no-debug Build without debug symbols --no-debug Build without debug symbols
--release Configure to build an optimized release version --release Configure to build an optimized release version
--with-libsoup3 Build with libsoup-3.0 --with-libsoup3 Build with libsoup-3.0
--without-webrtcdsp Build without WebRTC audio processing
--with-tests Also build tests. --with-tests Also build tests.
Plugin configuration: Plugin configuration:
@ -111,7 +109,6 @@ while true; do
--valac-flags ) VALACFLAGS="$2"; shift; shift ;; --valac-flags ) VALACFLAGS="$2"; shift; shift ;;
--lib-suffix ) LIB_SUFFIX="$2"; shift; shift ;; --lib-suffix ) LIB_SUFFIX="$2"; shift; shift ;;
--with-libsoup3 ) USE_SOUP3=yes; shift ;; --with-libsoup3 ) USE_SOUP3=yes; shift ;;
--without-webrtcdsp ) PLUGIN_RTP_WEBRTC_AUDIO_PROCESSING=no; shift ;;
--disable-fast-vapi ) DISABLE_FAST_VAPI=yes; shift ;; --disable-fast-vapi ) DISABLE_FAST_VAPI=yes; shift ;;
--no-debug ) NO_DEBUG=yes; shift ;; --no-debug ) NO_DEBUG=yes; shift ;;
--release ) BUILD_TYPE=RelWithDebInfo; shift ;; --release ) BUILD_TYPE=RelWithDebInfo; shift ;;
@ -210,7 +207,7 @@ cmake -G "$cmake_type" \
-DCMAKE_BUILD_TYPE="$BUILD_TYPE" \ -DCMAKE_BUILD_TYPE="$BUILD_TYPE" \
-DENABLED_PLUGINS="$ENABLED_PLUGINS" \ -DENABLED_PLUGINS="$ENABLED_PLUGINS" \
-DDISABLED_PLUGINS="$DISABLED_PLUGINS" \ -DDISABLED_PLUGINS="$DISABLED_PLUGINS" \
-DBUILD_TESTING="$BUILD_TESTS" \ -DBUILD_TESTS="$BUILD_TESTS" \
-DUSE_SOUP3="$USE_SOUP3" \ -DUSE_SOUP3="$USE_SOUP3" \
-DVALA_EXECUTABLE="$VALAC" \ -DVALA_EXECUTABLE="$VALAC" \
-DCMAKE_VALA_FLAGS="$VALACFLAGS" \ -DCMAKE_VALA_FLAGS="$VALACFLAGS" \
@ -222,7 +219,6 @@ cmake -G "$cmake_type" \
-DBIN_INSTALL_DIR="$BINDIR" \ -DBIN_INSTALL_DIR="$BINDIR" \
-DINCLUDE_INSTALL_DIR="$INCLUDEDIR" \ -DINCLUDE_INSTALL_DIR="$INCLUDEDIR" \
-DLIB_INSTALL_DIR="$LIBDIR" \ -DLIB_INSTALL_DIR="$LIBDIR" \
-DPLUGIN_RTP_WEBRTC_AUDIO_PROCESSING="$PLUGIN_RTP_WEBRTC_AUDIO_PROCESSING" \
-Wno-dev \ -Wno-dev \
.. || exit 9 .. || exit 9
@ -242,6 +238,8 @@ default:
@sh -c "cd build; $exec_command" @sh -c "cd build; $exec_command"
distclean: clean uninstall distclean: clean uninstall
test: default
echo "make test not yet supported"
%: %:
@sh -c "cd build; $exec_command \"\$@\"" @sh -c "cd build; $exec_command \"\$@\""
EOF EOF

View file

@ -22,8 +22,6 @@ GENERATE_VAPI
crypto-vala crypto-vala
GENERATE_HEADER GENERATE_HEADER
crypto-vala crypto-vala
DEFINITIONS
GCRYPT
) )
add_custom_target(crypto-vala-vapi add_custom_target(crypto-vala-vapi

View file

@ -1,7 +1,7 @@
dependencies = [ dependencies = [
dep_gio, dep_gio,
dep_glib, dep_glib,
dep_libgcrypt_or_openssl, dep_libgcrypt,
dep_libsrtp2, dep_libsrtp2,
] ]
sources = files( sources = files(
@ -17,11 +17,6 @@ c_args = [
vala_args = [ vala_args = [
'--vapidir', meson.current_source_dir() / 'vapi', '--vapidir', meson.current_source_dir() / 'vapi',
] ]
if crypto_backend == 'openssl'
vala_args += ['--pkg', 'openssl'] # Work around https://github.com/mesonbuild/meson/issues/2103.
elif crypto_backend == 'gnutls'
vala_args += ['-D', 'GCRYPT']
endif
lib_crypto_vala = library('crypto-vala', sources, c_args: c_args, vala_args: vala_args, dependencies: dependencies, version: '0.0', install: true, install_dir: [true, true, true]) lib_crypto_vala = library('crypto-vala', sources, c_args: c_args, vala_args: vala_args, dependencies: dependencies, version: '0.0', install: true, install_dir: [true, true, true])
dep_crypto_vala = declare_dependency(link_with: lib_crypto_vala, include_directories: include_directories('.')) dep_crypto_vala = declare_dependency(link_with: lib_crypto_vala, include_directories: include_directories('.'))

View file

@ -1,24 +1,14 @@
namespace Crypto { namespace Crypto {
public class SymmetricCipher { public class SymmetricCipher {
#if GCRYPT
private GCrypt.Cipher.Cipher cipher; private GCrypt.Cipher.Cipher cipher;
#else
bool is_encryption;
private OpenSSL.EVP.CipherContext? cipher;
#endif
public static bool supports(string algo_name) { public static bool supports(string algo_name) {
#if GCRYPT
GCrypt.Cipher.Algorithm algo; GCrypt.Cipher.Algorithm algo;
GCrypt.Cipher.Mode mode; GCrypt.Cipher.Mode mode;
GCrypt.Cipher.Flag flags; GCrypt.Cipher.Flag flags;
return parse(algo_name, out algo, out mode, out flags); return parse(algo_name, out algo, out mode, out flags);
#else
return algo_name == "AES-GCM";
#endif
} }
#if GCRYPT
private static unowned string mode_to_string(GCrypt.Cipher.Mode mode) { private static unowned string mode_to_string(GCrypt.Cipher.Mode mode) {
switch (mode) { switch (mode) {
case GCrypt.Cipher.Mode.ECB: return "ECB"; case GCrypt.Cipher.Mode.ECB: return "ECB";
@ -105,18 +95,8 @@ public class SymmetricCipher {
return algo.to_string(); return algo.to_string();
} }
} }
#endif
public SymmetricCipher.encryption(string algo_name) throws Error { public SymmetricCipher(string algo_name) throws Error {
this.initialize(algo_name, true);
}
public SymmetricCipher.decryption(string algo_name) throws Error {
this.initialize(algo_name, false);
}
private SymmetricCipher.initialize(string algo_name, bool is_encryption) throws Error {
#if GCRYPT
GCrypt.Cipher.Algorithm algo; GCrypt.Cipher.Algorithm algo;
GCrypt.Cipher.Mode mode; GCrypt.Cipher.Mode mode;
GCrypt.Cipher.Flag flags; GCrypt.Cipher.Flag flags;
@ -125,157 +105,48 @@ public class SymmetricCipher {
} else { } else {
throw new Error.ILLEGAL_ARGUMENTS(@"The algorithm $algo_name is not supported"); throw new Error.ILLEGAL_ARGUMENTS(@"The algorithm $algo_name is not supported");
} }
#else
if (algo_name == "AES-GCM") {
this.openssl(is_encryption);
} else {
throw new Error.ILLEGAL_ARGUMENTS(@"The algorithm $algo_name is not supported");
}
#endif
} }
#if GCRYPT
private SymmetricCipher.gcrypt(GCrypt.Cipher.Algorithm algo, GCrypt.Cipher.Mode mode, GCrypt.Cipher.Flag flags) throws Error { private SymmetricCipher.gcrypt(GCrypt.Cipher.Algorithm algo, GCrypt.Cipher.Mode mode, GCrypt.Cipher.Flag flags) throws Error {
may_throw_gcrypt_error(GCrypt.Cipher.Cipher.open(out this.cipher, algo, mode, flags)); may_throw_gcrypt_error(GCrypt.Cipher.Cipher.open(out this.cipher, algo, mode, flags));
} }
#else
private SymmetricCipher.openssl(bool is_encryption) throws Error {
this.is_encryption = is_encryption;
cipher = new OpenSSL.EVP.CipherContext();
if (is_encryption) {
if (cipher.encrypt_init(OpenSSL.EVP.aes_128_gcm(), null, null, null) != 1) {
openssl_error();
}
} else {
if (cipher.decrypt_init(OpenSSL.EVP.aes_128_gcm(), null, null, null) != 1) {
openssl_error();
}
}
}
#endif
public void set_key(uint8[] key) throws Error { public void set_key(uint8[] key) throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.set_key(key)); may_throw_gcrypt_error(cipher.set_key(key));
#else
if (key.length != 16) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("key length must be 16 for AES-GCM");
}
if (is_encryption) {
if (cipher.encrypt_init(null, null, key, null) != 1) {
openssl_error();
}
} else {
if (cipher.decrypt_init(null, null, key, null) != 1) {
openssl_error();
}
}
#endif
} }
public void set_iv(uint8[] iv) throws Error { public void set_iv(uint8[] iv) throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.set_iv(iv)); may_throw_gcrypt_error(cipher.set_iv(iv));
#else }
if (iv.length != 12) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("intialization vector must be of length 16 for AES-GCM"); public void set_counter_vector(uint8[] ctr) throws Error {
} may_throw_gcrypt_error(cipher.set_counter_vector(ctr));
if (is_encryption) {
if (cipher.encrypt_init(null, null, null, iv) != 1) {
openssl_error();
}
} else {
if (cipher.decrypt_init(null, null, null, iv) != 1) {
openssl_error();
}
}
#endif
} }
public void reset() throws Error { public void reset() throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.reset()); may_throw_gcrypt_error(cipher.reset());
#else
throw new Crypto.Error.ILLEGAL_ARGUMENTS("can't reset OpenSSL cipher context");
#endif
} }
public uint8[] get_tag(size_t taglen) throws Error { public uint8[] get_tag(size_t taglen) throws Error {
uint8[] tag = new uint8[taglen]; uint8[] tag = new uint8[taglen];
#if GCRYPT
may_throw_gcrypt_error(cipher.get_tag(tag)); may_throw_gcrypt_error(cipher.get_tag(tag));
#else
if (!is_encryption) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("can't call get_tag on decryption context");
}
uint8[] empty = new uint8[0];
int empty_len = 0;
if (cipher.encrypt_final(empty, out empty_len) != 1) {
openssl_error();
}
if (empty_len != 0) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("get_tag called on a stream with remaining data");
}
if (cipher.ctrl(OpenSSL.EVP.CTRL_GCM_GET_TAG, (int)taglen, tag) != 1) {
openssl_error();
}
#endif
return tag; return tag;
} }
public void check_tag(uint8[] tag) throws Error { public void check_tag(uint8[] tag) throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.check_tag(tag)); may_throw_gcrypt_error(cipher.check_tag(tag));
#else
if (is_encryption) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("can't call check_tag on encryption context");
}
if (cipher.ctrl(OpenSSL.EVP.CTRL_GCM_SET_TAG, tag.length, tag) != 1) {
openssl_error();
}
uint8[] empty = new uint8[0];
int empty_len = 0;
if (cipher.decrypt_final(empty, out empty_len) != 1) {
openssl_error();
}
if (empty_len != 0) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("check_tag called on a stream with remaining data");
}
#endif
} }
public void encrypt(uint8[] output, uint8[] input) throws Error { public void encrypt(uint8[] output, uint8[] input) throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.encrypt(output, input)); may_throw_gcrypt_error(cipher.encrypt(output, input));
#else
if (!is_encryption) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("can't call encrypt on decryption context");
}
int output_length = output.length;
if (cipher.encrypt_update(output, out output_length, input) != 1) {
openssl_error();
}
if (output_length != output.length) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("invalid output array length");
}
#endif
} }
public void decrypt(uint8[] output, uint8[] input) throws Error { public void decrypt(uint8[] output, uint8[] input) throws Error {
#if GCRYPT
may_throw_gcrypt_error(cipher.decrypt(output, input)); may_throw_gcrypt_error(cipher.decrypt(output, input));
#else }
if (is_encryption) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("can't call decrypt on encryption context"); public void sync() throws Error {
} may_throw_gcrypt_error(cipher.sync());
int output_length = output.length;
if (cipher.decrypt_update(output, out output_length, input) != 1) {
openssl_error();
}
if (output_length != output.length) {
throw new Crypto.Error.ILLEGAL_ARGUMENTS("invalid output array length");
}
#endif
} }
} }
} }

View file

@ -3,20 +3,13 @@ namespace Crypto {
public errordomain Error { public errordomain Error {
ILLEGAL_ARGUMENTS, ILLEGAL_ARGUMENTS,
GCRYPT, GCRYPT,
OPENSSL,
AUTHENTICATION_FAILED, AUTHENTICATION_FAILED,
UNKNOWN UNKNOWN
} }
#if GCRYPT
internal void may_throw_gcrypt_error(GCrypt.Error e) throws Error { internal void may_throw_gcrypt_error(GCrypt.Error e) throws Error {
if (((int)e) != 0) { if (((int)e) != 0) {
throw new Crypto.Error.GCRYPT(e.to_string()); throw new Crypto.Error.GCRYPT(e.to_string());
} }
} }
#else
internal void openssl_error() throws Error {
throw new Crypto.Error.OPENSSL(OpenSSL.ERR.reason_error_string(OpenSSL.ERR.get_error()));
}
#endif
} }

View file

@ -1,9 +1,5 @@
namespace Crypto { namespace Crypto {
public static void randomize(uint8[] buffer) { public static void randomize(uint8[] buffer) {
#if GCRYPT
GCrypt.Random.randomize(buffer); GCrypt.Random.randomize(buffer);
#else
OpenSSL.RAND.bytes(buffer);
#endif
} }
} }

View file

@ -1,822 +0,0 @@
/* OpenSSL Vala Bindings
* Copyright 2020 Zuhong Tao <taozuhong@gmail>
* Copyright 2016 Guillaume Poirier-Morency <guillaumepoiriermorency@gmail>
* Copyright 1995-2016 The OpenSSL Project Authors. All Rights Reserved.
*
* Licensed under the OpenSSL license (the "License"). You may not use
* this file except in compliance with the License. You can obtain a copy
* in the file LICENSE in the source distribution or at
* https://www.openssl.org/source/license.html
*/
[CCode (cprefix = "")]
namespace OpenSSL
{
[CCode (cprefix = "AES_", lower_case_cprefix = "AES_", cheader_filename = "openssl/aes.h")]
namespace AES
{
public const int BLOCK_SIZE;
}
[Compact]
[CCode (cname = "ENGINE", lower_case_cprefix = "ENGINE_", cprefix = "ENGINE_", cheader_filename = "openssl/engine.h", free_function = "ENGINE_free")]
public class Engine {
[CCode (cname = "ENGINE_new")]
public Engine ();
[CCode (cname = "ENGINE_by_id")]
public Engine.by_id (string id);
[CCode (cname = "ENGINE_get_default_RSA")]
public Engine.get_default_RSA ();
[CCode (cname = "ENGINE_get_default_DSA")]
public Engine.get_default_DSA ();
[CCode (cname = "ENGINE_get_default_DH")]
public Engine.get_default_DH ();
[CCode (cname = "ENGINE_get_default_RAND")]
public Engine.get_default_RAND ();
[CCode (cname = "ENGINE_get_cipher_engine")]
public Engine.get_cipher_engine (int nid);
[CCode (cname = "ENGINE_get_digest_engine")]
public Engine.get_digest_engine (int nid);
public int init ();
public int finish ();
public int set_default (uint flags);
public int set_default_RSA ();
public int set_default_DSA ();
public int set_default_DH ();
public int set_default_RAND ();
public int set_default_ciphers ();
public int set_default_digests ();
public int set_default_string (string list);
}
[CCode (cprefix = "NID_", cheader_filename = "openssl/objects.h")]
public enum NID
{
sha256
}
[Compact]
[CCode (cname = "BIO_METHOD", cheader_filename = "openssl/bio.h", free_function = "BIO_meth_free")]
public class BIOMethod
{
public const int BIO_TYPE_DESCRIPTOR;
public const int BIO_TYPE_FILTER;
public const int BIO_TYPE_SOURCE_SINK;
public const int BIO_TYPE_NONE;
public const int BIO_TYPE_MEM;
public const int BIO_TYPE_FILE;
public const int BIO_TYPE_FD;
public const int BIO_TYPE_SOCKET;
public const int BIO_TYPE_NULL;
public const int BIO_TYPE_SSL;
public const int BIO_TYPE_MD;
public const int BIO_TYPE_BUFFER;
public const int BIO_TYPE_CIPHER;
public const int BIO_TYPE_BASE64;
public const int BIO_TYPE_CONNECT;
public const int BIO_TYPE_ACCEPT;
public const int BIO_TYPE_NBIO_TEST;
public const int BIO_TYPE_NULL_FILTER;
public const int BIO_TYPE_BIO;
public const int BIO_TYPE_LINEBUFFER;
public const int BIO_TYPE_DGRAM;
public const int BIO_TYPE_ASN1;
public const int BIO_TYPE_COMP;
public const int BIO_TYPE_DGRAM_SCTP;
[CCode (cname = "BIO_meth_new")]
public BIOMethod (int type, string name);
[CCode (cname = "BIO_get_new_index")]
public int get_new_index ();
}
[Compact]
[CCode (lower_case_cprefix = "BUF_MEM_", cheader_filename = "openssl/buffer.h", free_function = "BUF_MEM_free")]
public class Buffer {
[CCode (cname = "BUF_MEM_new")]
public Buffer ();
[CCode (cname = "BUF_MEM_new_ex")]
public Buffer.with_flags ();
public int grow (int len);
public size_t grow_clean (size_t len);
}
[Compact]
[CCode (lower_case_cprefix = "BIO_", cheader_filename = "openssl/bio.h", free_function = "BIO_free")]
public class BIO
{
public const int NOCLOSE;
public static unowned BIOMethod s_mem ();
public static unowned BIOMethod s_secmem ();
public BIO (BIOMethod type);
[CCode (cname = "BIO_new_file")]
public BIO.with_file (string filename, string mode);
[CCode (cname = "BIO_new_fp")]
public BIO.with_stream (GLib.FileStream stream, int flags);
[CCode (cname = "BIO_new_mem_buf")]
public BIO.with_buffer (uint8[] buf);
public int read_filename (string name);
public int write_filename (string name);
public int append_filename (string name);
public int rw_filename (string name);
public int set_mem_eof_return (int v);
public long get_mem_data ([CCode (array_length = false)] out uint8[] pp);
public int set_mem_buf (Buffer bm, int c);
public int get_mem_ptr (out Buffer pp);
public int set (BIOMethod type);
public int read (uint8[] data);
public int write (uint8[] data);
[PrintfFormat]
public int printf (string format, ...);
[PrintfFormat]
public int vprintf (string format, va_list args);
[PrintfFormat]
public static int snprintf (uint8[] buf, string format, ...);
[PrintfFormat]
public static int vsnprintf (uint8[] buf, string format, va_list args);
public int reset ();
public int seek (int ofs);
public int pending ();
public int wpending ();
public int flush ();
public int eof ();
public int tell ();
public int set_close (long flag);
public int get_close ();
public long ctrl (int cmd, long larg, [CCode (array_length = false)] uint8[] parg);
public int read_ex (uint8[] data, out size_t readbytes);
public int write_ex (uint8[] data, out size_t written);
}
[CCode (lower_case_cprefix = "CRYPTO_", cheader_filename = "openssl/crypto.h")]
namespace Crypto
{
public int memcmp (void* v1, void* v2, size_t n);
}
[Compact]
[CCode (cname = "ASN1_PCTX", lower_case_cprefix = "ASN1_PCTX_", free_function = "ASN1_PCTX_free")]
public class ASN1_PCTX {
public ASN1_PCTX ();
public ulong get_flags ();
public void set_flags (ulong flags);
public ulong get_nm_flags ();
public void set_nm_flags (ulong flags);
public ulong get_cert_flags ();
public void set_cert_flags (ulong flags);
public ulong get_oid_flags ();
public void set_oid_flags (ulong flags);
public ulong get_str_flags ();
public void set_str_flags (ulong flags);
}
[Compact]
[CCode (cname = "ASN1_SCTX", lower_case_cprefix = "ASN1_SCTX_", free_function = "ASN1_SCTX_free")]
public class ASN1_SCTX {
public ASN1_SCTX ();
public ulong get_flags ();
}
[CCode (cprefix = "EVP_", lower_case_cprefix = "EVP_", cheader_filename = "openssl/evp.h")]
namespace EVP
{
public const int CIPH_STREAM_CIPHER;
public const int CIPH_ECB_MODE;
public const int CIPH_CBC_MODE;
public const int CIPH_CFB_MODE;
public const int CIPH_OFB_MODE;
public const int CIPH_CTR_MODE;
public const int CIPH_GCM_MODE;
public const int CIPH_CCM_MODE;
public const int CIPH_XTS_MODE;
public const int CIPH_WRAP_MODE;
public const int CIPH_OCB_MODE;
public const int CIPH_MODE;
public const int CIPH_VARIABLE_LENGTH;
public const int CIPH_CUSTOM_IV;
public const int CIPH_ALWAYS_CALL_INIT;
public const int CIPH_CTRL_INIT;
public const int CIPH_CUSTOM_KEY_LENGTH;
public const int CIPH_NO_PADDING;
public const int CIPH_RAND_KEY;
public const int CIPH_CUSTOM_COPY;
public const int CIPH_CUSTOM_IV_LENGTH;
public const int CIPH_FLAG_DEFAULT_ASN1;
public const int CIPH_FLAG_LENGTH_BITS;
public const int CIPH_FLAG_FIPS;
public const int CIPH_FLAG_NON_FIPS_ALLOW;
public const int CIPH_FLAG_CUSTOM_CIPHER;
public const int CIPH_FLAG_AEAD_CIPHER;
public const int CIPH_FLAG_TLS1_1_MULTIBLOCK;
public const int CIPH_FLAG_PIPELINE;
public const int CTRL_INIT;
public const int CTRL_SET_KEY_LENGTH;
public const int CTRL_GET_RC2_KEY_BITS;
public const int CTRL_SET_RC2_KEY_BITS;
public const int CTRL_GET_RC5_ROUNDS;
public const int CTRL_SET_RC5_ROUNDS;
public const int CTRL_RAND_KEY;
public const int CTRL_PBE_PRF_NID;
public const int CTRL_COPY;
public const int CTRL_AEAD_SET_IVLEN;
public const int CTRL_AEAD_GET_TAG;
public const int CTRL_AEAD_SET_TAG;
public const int CTRL_AEAD_SET_IV_FIXED;
public const int CTRL_GCM_SET_IVLEN;
public const int CTRL_GCM_GET_TAG;
public const int CTRL_GCM_SET_TAG;
public const int CTRL_GCM_SET_IV_FIXED;
public const int CTRL_GCM_IV_GEN;
public const int CTRL_CCM_SET_IVLEN;
public const int CTRL_CCM_GET_TAG;
public const int CTRL_CCM_SET_TAG;
public const int CTRL_CCM_SET_IV_FIXED;
public const int CTRL_CCM_SET_L;
public const int CTRL_CCM_SET_MSGLEN;
public const int CTRL_AEAD_TLS1_AAD;
public const int CTRL_AEAD_SET_MAC_KEY;
public const int CTRL_GCM_SET_IV_INV;
public const int CTRL_TLS1_1_MULTIBLOCK_AAD;
public const int CTRL_TLS1_1_MULTIBLOCK_ENCRYPT;
public const int CTRL_TLS1_1_MULTIBLOCK_DECRYPT;
public const int CTRL_TLS1_1_MULTIBLOCK_MAX_BUFSIZE;
public const int CTRL_SSL3_MASTER_SECRET;
public const int CTRL_SET_SBOX;
public const int CTRL_SBOX_USED;
public const int CTRL_KEY_MESH;
public const int CTRL_BLOCK_PADDING_MODE;
public const int CTRL_SET_PIPELINE_OUTPUT_BUFS;
public const int CTRL_SET_PIPELINE_INPUT_BUFS;
public const int CTRL_SET_PIPELINE_INPUT_LENS;
public const int CTRL_GET_IVLEN;
[CCode (cprefix = "EVP_PADDING_")]
public enum Padding {
PKCS7,
ISO7816_4,
ANSI923,
ISO10126,
ZERO,
}
[CCode (cprefix = "EVP_PKEY_OP_")]
public enum PublicKeyOperation {
UNDEFINED,
PARAMGEN,
KEYGEN,
SIGN,
VERIFY,
VERIFYRECOVER,
SIGNCTX,
VERIFYCTX,
ENCRYPT,
DECRYPT,
DERIVE,
}
public const int PKEY_OP_TYPE_SIG;
public const int PKEY_OP_TYPE_CRYPT;
public const int PKEY_OP_TYPE_NOGEN;
public const int PKEY_OP_TYPE_GEN;
[CCode (cprefix = "EVP_PKEY_CTRL_")]
public enum PublicKeyControl {
MD,
PEER_KEY,
PKCS7_ENCRYPT,
PKCS7_DECRYPT,
PKCS7_SIGN,
SET_MAC_KEY,
DIGESTINIT,
SET_IV,
CMS_ENCRYPT,
CMS_DECRYPT,
CMS_SIGN,
CIPHER,
GET_MD,
SET_DIGEST_SIZE,
}
[CCode (cprefix = "EVP_PKEY_CTRL_", cheader_filename = "openssl/rsa.h")]
public enum PublicKeyRsaControl {
RSA_PADDING,
RSA_PSS_SALTLEN,
RSA_KEYGEN_BITS,
RSA_KEYGEN_PUBEXP,
RSA_MGF1_MD,
GET_RSA_PADDING,
GET_RSA_PSS_SALTLEN,
GET_RSA_MGF1_MD,
RSA_OAEP_MD,
RSA_OAEP_LABEL,
GET_RSA_OAEP_MD,
GET_RSA_OAEP_LABEL,
RSA_KEYGEN_PRIMES,
}
[Compact]
[CCode (cname = "EVP_PKEY", lower_case_cprefix = "EVP_PKEY_", cprefix = "EVP_PKEY_", free_function = "EVP_PKEY_free")]
public class PublicKey {
public const int NONE;
public const int RSA;
public const int RSA2;
public const int RSA_PSS;
public const int DSA;
public const int DSA1;
public const int DSA2;
public const int DSA3;
public const int DSA4;
public const int DH;
public const int DHX;
public const int EC;
public const int SM2;
public const int HMAC;
public const int CMAC;
public const int SCRYPT;
public const int TLS1_PRF;
public const int HKDF;
public const int POLY1305;
public const int SIPHASH;
public const int X25519;
public const int ED25519;
public const int X448;
public const int ED448;
public PublicKey ();
[CCode (cname = "EVP_PKEY_new_raw_private_key")]
public PublicKey.raw_private_key (int type, Engine? e, uint8[] key);
[CCode (cname = "EVP_PKEY_new_raw_public_key")]
public PublicKey.raw_public_key (int type, Engine? e, uint8[] key);
[CCode (cname = "EVP_PKEY_new_CMAC_key")]
public PublicKey.CMAC_key (Engine? e, uint8[] priv, Cipher? cipher);
[CCode (cname = "EVP_PKEY_new_mac_key")]
public PublicKey.mac_key (int type, Engine? e, uint8[] key);
public int id ();
public int size ();
public int base_id ();
public static int type (int type);
public int set_alias_type (int type);
public int up_ref ();
public RSA? get1_RSA ();
public RSA? get0_RSA ();
public int set1_RSA (RSA? key);
public int assign_RSA (RSA? key);
public int security_bits ();
[CCode (instance_pos = 1.1)]
public int print_public (BIO out, int indent, ASN1_PCTX? pctx);
[CCode (instance_pos = 1.1)]
public int print_private (BIO out, int indent, ASN1_PCTX? pctx);
public Engine? get0_engine ();
public int set1_engine (Engine? engine);
public int get_raw_private_key ([CCode (array_length = false)] uint8[] priv, out size_t len);
public int get_raw_public_key ([CCode (array_length = false)] uint8[] pub, out size_t len);
}
[Compact]
[CCode (cname = "EVP_PKEY_CTX", lower_case_cprefix = "EVP_PKEY_CTX_", cprefix = "EVP_PKEY_CTX_", free_function = "EVP_PKEY_CTX_free")]
public class PublicKeyContext {
public PublicKeyContext (PublicKey pkey, Engine? e);
public PublicKeyContext.id (int id, Engine? e);
public PublicKeyContext dup ();
public int ctrl_str (string type, string value);
public int ctrl_uint64(int keytype, int optype, int cmd, uint64 value);
public int ctrl (int keytype, int optype, int cmd, int p1, [CCode (array_length = false)] uint8[] p2);
[CCode (cname = "EVP_PKEY_CTX_set_rsa_padding", cheader_filename = "openssl/rsa.h")]
public int set_rsa_padding (int pad);
[CCode (cname = "EVP_PKEY_CTX_get_rsa_padding", cheader_filename = "openssl/rsa.h")]
public int get_rsa_padding (out int pad);
[CCode (cname = "EVP_PKEY_CTX_set_rsa_pss_saltlen", cheader_filename = "openssl/rsa.h")]
public int set_rsa_pss_saltlen (int len);
[CCode (cname = "EVP_PKEY_CTX_get_rsa_pss_saltlen", cheader_filename = "openssl/rsa.h")]
public int get_rsa_pss_saltlen (out int len);
[CCode (cname = "EVP_PKEY_CTX_set_rsa_keygen_bits", cheader_filename = "openssl/rsa.h")]
public int set_rsa_keygen_bits (int mbits);
[CCode (cname = "EVP_PKEY_CTX_set_rsa_keygen_pubexp", cheader_filename = "openssl/rsa.h")]
public int set_rsa_keygen_pubexp (BIGNUM pubexp);
[CCode (cname = "EVP_PKEY_CTX_set_rsa_keygen_primes", cheader_filename = "openssl/rsa.h")]
public int set_rsa_keygen_primes (int primes);
public int md (int optype, int cmd, string md);
public int set_signature_md (MessageDigest md);
public int get_signature_md (out MessageDigest pmd);
public int set_mac_key (uint8[] key);
[CCode (cname = "EVP_PKEY_keygen_init")]
public int keygen_init ();
[CCode (cname = "EVP_PKEY_keygen")]
public int keygen (out PublicKey ppkey);
[CCode (cname = "EVP_PKEY_paramgen_init")]
public int paramgen_init ();
[CCode (cname = "EVP_PKEY_paramgen")]
public int paramgen (out PublicKey ppkey);
[CCode (cname = "EVP_PKEY_encrypt_init")]
public int encrypt_init ();
[CCode (cname = "EVP_PKEY_encrypt")]
public int encrypt ([CCode (array_length = false)] uint8[] out, out size_t outlen, uint8[] in);
[CCode (cname = "EVP_PKEY_decrypt_init")]
public int decrypt_init ();
[CCode (cname = "EVP_PKEY_decrypt")]
public int decrypt ([CCode (array_length = false)] uint8[] out, out size_t outlen, uint8[] in);
[CCode (cname = "EVP_PKEY_derive_init")]
public int derive_init ();
[CCode (cname = "EVP_PKEY_derive_set_peer")]
public int derive_set_peer (PublicKey peer);
[CCode (cname = "EVP_PKEY_derive")]
public int derive ([CCode (array_length = false)] uint8[] key, out size_t keylen);
[CCode (cname = "EVP_PKEY_sign_init")]
public int sign_init ();
[CCode (cname = "EVP_PKEY_sign")]
public int sign ([CCode (array_length = false)] uint8[] sig, out size_t siglen, uint8[] tbs);
[CCode (cname = "EVP_PKEY_verify_init")]
public int verify_init ();
[CCode (cname = "EVP_PKEY_verify")]
public int verify (uint8[] sig, uint8[] tbs);
[CCode (cname = "EVP_PKEY_verify_recover_init")]
public int verify_recover_init ();
[CCode (cname = "EVP_PKEY_verify_recover")]
public int verify_recover ([CCode (array_length = false)] uint8[] rout, out size_t routlen, uint8[] sig);
}
[Compact]
[CCode (cname = "EVP_MD")]
public class MessageDigest
{
}
public unowned MessageDigest? md_null ();
public unowned MessageDigest? md2 ();
public unowned MessageDigest? md4 ();
public unowned MessageDigest? md5 ();
public unowned MessageDigest? md5_sha1 ();
public unowned MessageDigest? blake2b512 ();
public unowned MessageDigest? blake2s256 ();
public unowned MessageDigest? sha1 ();
public unowned MessageDigest? sha224 ();
public unowned MessageDigest? sha256 ();
public unowned MessageDigest? sha384 ();
public unowned MessageDigest? sha512 ();
public unowned MessageDigest? mdc2 ();
public unowned MessageDigest? ripmed160 ();
public unowned MessageDigest? whirlpool ();
[CCode (cname = "get_digestbyname")]
public unowned MessageDigest? get_digest_by_name (string name);
[Compact]
[CCode (cname = "EVP_MD_CTX", lower_case_cprefix = "EVP_MD_CTX_")]
public class MessageDigestContext
{
public MessageDigestContext ();
[CCode (cname = "EVP_DigestInit_ex")]
public int init (MessageDigest type, Engine? engine);
[CCode (cname = "EVP_DigestUpdate")]
public int update (uint8[] d);
[CCode (cname = "EVP_DigestFinal_ex")]
public int final ([CCode (array_length = false)] uchar[] md, out int s);
}
[Compact]
[CCode (cname = "EVP_CIPHER", lower_case_cprefix = "EVP_CIPHER_")]
public class Cipher
{
[CCode (cname = "EVP_CIPHER_meth_new")]
public Cipher (int cipher_type, int block_size, int key_len);
public int key_length ();
public int iv_length ();
}
public unowned Cipher? enc_null ();
public unowned Cipher? des_ecb ();
public unowned Cipher? des_ede ();
public unowned Cipher? des_ede3 ();
public unowned Cipher? des_ede_ecb ();
public unowned Cipher? des_ede3_ecb ();
public unowned Cipher? des_cfb64 ();
public unowned Cipher? des_cfb1 ();
public unowned Cipher? des_cfb8 ();
public unowned Cipher? des_ede_cfb64 ();
public unowned Cipher? des_ede3_cfb64 ();
public unowned Cipher? des_ede3_cfb1 ();
public unowned Cipher? des_ede3_cfb8 ();
public unowned Cipher? des_ofb ();
public unowned Cipher? des_ede_ofb ();
public unowned Cipher? des_ede3_ofb ();
public unowned Cipher? des_cbc ();
public unowned Cipher? des_ede_cbc ();
public unowned Cipher? des_ede3_cbc ();
public unowned Cipher? desx_cbc ();
public unowned Cipher? des_ede3_wrap ();
public unowned Cipher? rc4 ();
public unowned Cipher? rc4_40 ();
public unowned Cipher? rc4_hmac_md5 ();
public unowned Cipher? idea_ecb ();
public unowned Cipher? idea_cfb64 ();
public unowned Cipher? idea_ofb ();
public unowned Cipher? idea_cbc ();
public unowned Cipher? rc2_ecb ();
public unowned Cipher? rc2_cbc ();
public unowned Cipher? rc2_40_cbc ();
public unowned Cipher? rc2_64_cbc ();
public unowned Cipher? rc2_cfb64 ();
public unowned Cipher? rc2_ofb ();
public unowned Cipher? bf_ecb ();
public unowned Cipher? bf_cbc ();
public unowned Cipher? bf_cfb64 ();
public unowned Cipher? bf_ofb ();
public unowned Cipher? cast5_ecb ();
public unowned Cipher? cast5_cbc ();
public unowned Cipher? cast5_cfb64 ();
public unowned Cipher? cast5_ofb ();
public unowned Cipher? rc5_32_12_16_cbc ();
public unowned Cipher? rc5_32_12_16_ecb ();
public unowned Cipher? rc5_32_12_16_cfb64 ();
public unowned Cipher? rc5_32_12_16_ofb ();
public unowned Cipher? aes_128_ecb ();
public unowned Cipher? aes_128_cbc ();
public unowned Cipher? aes_128_cfb1 ();
public unowned Cipher? aes_128_cfb8 ();
public unowned Cipher? aes_128_cfb128 ();
public unowned Cipher? aes_128_ofb ();
public unowned Cipher? aes_128_ctr ();
public unowned Cipher? aes_128_ccm ();
public unowned Cipher? aes_128_gcm ();
public unowned Cipher? aes_128_xts ();
public unowned Cipher? aes_128_wrap ();
public unowned Cipher? aes_128_wrap_pad ();
public unowned Cipher? aes_128_ocb ();
public unowned Cipher? aes_192_ecb ();
public unowned Cipher? aes_192_cbc ();
public unowned Cipher? aes_192_cfb1 ();
public unowned Cipher? aes_192_cfb8 ();
public unowned Cipher? aes_192_cfb128 ();
public unowned Cipher? aes_192_ofb ();
public unowned Cipher? aes_192_ctr ();
public unowned Cipher? aes_192_ccm ();
public unowned Cipher? aes_192_gcm ();
public unowned Cipher? aes_192_wrap ();
public unowned Cipher? aes_192_wrap_pad ();
public unowned Cipher? aes_192_ocb ();
public unowned Cipher? aes_256_ecb ();
public unowned Cipher? aes_256_cbc ();
public unowned Cipher? aes_256_cfb1 ();
public unowned Cipher? aes_256_cfb8 ();
public unowned Cipher? aes_256_cfb128 ();
public unowned Cipher? aes_256_ofb ();
public unowned Cipher? aes_256_ctr ();
public unowned Cipher? aes_256_ccm ();
public unowned Cipher? aes_256_gcm ();
public unowned Cipher? aes_256_xts ();
public unowned Cipher? aes_256_wrap ();
public unowned Cipher? aes_256_wrap_pad ();
public unowned Cipher? aes_256_ocb ();
public unowned Cipher? aes_128_cbc_hmac_sha1 ();
public unowned Cipher? aes_256_cbc_hmac_sha1 ();
public unowned Cipher? aes_128_cbc_hmac_sha256 ();
public unowned Cipher? aes_256_cbc_hmac_sha256 ();
public unowned Cipher? camellia_128_ecb ();
public unowned Cipher? camellia_128_cbc ();
public unowned Cipher? camellia_128_cfb1 ();
public unowned Cipher? camellia_128_cfb8 ();
public unowned Cipher? camellia_128_cfb128 ();
public unowned Cipher? camellia_128_ofb ();
public unowned Cipher? camellia_128_ctr ();
public unowned Cipher? camellia_192_ecb ();
public unowned Cipher? camellia_192_cbc ();
public unowned Cipher? camellia_192_cfb1 ();
public unowned Cipher? camellia_192_cfb8 ();
public unowned Cipher? camellia_192_cfb128 ();
public unowned Cipher? camellia_192_ofb ();
public unowned Cipher? camellia_192_ctr ();
public unowned Cipher? camellia_256_ecb ();
public unowned Cipher? camellia_256_cbc ();
public unowned Cipher? camellia_256_cfb1 ();
public unowned Cipher? camellia_256_cfb8 ();
public unowned Cipher? camellia_256_cfb128 ();
public unowned Cipher? camellia_256_ofb ();
public unowned Cipher? camellia_256_ctr ();
public unowned Cipher? chacha20 ();
public unowned Cipher? chacha20_poly1305 ();
public unowned Cipher? seed_ecb ();
public unowned Cipher? seed_cbc ();
public unowned Cipher? seed_cfb128 ();
public unowned Cipher? seed_ofb ();
[CCode (cname = "EVP_get_cipherbyname")]
public unowned Cipher? get_cipher_by_name (string name);
[CCode (cname = "EVP_BytesToKey")]
public int bytes_to_key (Cipher cipher, MessageDigest md, [CCode (array_length = false)] uchar[] salt, uchar[] key_data, int nrounds, [CCode (array_length = false)] uchar[] key, [CCode (array_length = false)] uchar[] iv);
[Compact]
[CCode (cname = "EVP_CIPHER_CTX", cprefix = "EVP_CIPHER_CTX_", lower_case_cprefix = "EVP_CIPHER_CTX_")]
public class CipherContext
{
public CipherContext ();
public int reset ();
public int set_key_length (int keylen);
public int set_padding (int pad);
[CCode (cname = "EVP_EncryptInit_ex")]
public int encrypt_init (Cipher? cipher, Engine? engine, [CCode (array_length = false)] uchar[]? key, [CCode (array_length = false)] uchar[]? iv);
[CCode (cname = "EVP_EncryptUpdate")]
public int encrypt_update ([CCode (array_length = false)] uchar[] ciphertext, out int ciphertext_len, uchar[] plaintext);
[CCode (cname = "EVP_EncryptFinal_ex")]
public int encrypt_final ([CCode (array_length = false)] uchar[] ciphertext, out int ciphertext_len);
[CCode (cname = "EVP_DecryptInit_ex")]
public int decrypt_init (Cipher? cipher, Engine? engine, [CCode (array_length = false)] uchar[]? key, [CCode (array_length = false)] uchar[]? iv);
[CCode (cname = "EVP_DecryptUpdate")]
public int decrypt_update ([CCode (array_length = false)] uchar[] plaintext, out int plaintext_len, uchar[] ciphertext);
[CCode (cname = "EVP_DecryptFinal_ex")]
public int decrypt_final ([CCode (array_length = false)] uchar[] plaintext, out int plaintext_len);
[CCode (simple_generics = true)]
public int ctrl<T>(int type, int arg, T? ptr);
}
}
[Compact]
[CCode (cname = "BIGNUM", cheader_filename = "openssl/rsa.h", free_function = "BN_free")]
public class BIGNUM {
[CCode (cname = "BN_new", cheader_filename = "openssl/bn.h")]
public BIGNUM ();
[CCode (cname = "BN_secure_new", cheader_filename = "openssl/bn.h")]
public BIGNUM.secure ();
[CCode (cname = "BN_clear", cheader_filename = "openssl/bn.h")]
public void clear ();
[CCode (cname = "BN_set_word", cheader_filename = "openssl/bn.h")]
public int set_word (ulong w);
}
[Compact]
[CCode (cname = "BN_GENCB", cheader_filename = "openssl/rsa.h", free_function = "BN_GENCB_free")]
public class BN_GENCB {
public delegate int BigNumGenCallback (int a, int b, BN_GENCB gcb);
[CCode (cname = "BN_GENCB_new")]
public BN_GENCB ();
[CCode (cname = "BN_GENCB_set")]
public void set (BigNumGenCallback cb, [CCode (array_length = false)] uint8[] cb_arg);
[CCode (cname = "BN_GENCB_call")]
public int call (int a, int b);
}
[Compact]
[CCode (lower_case_cprefix = "RSA_", cprefix = "RSA_", cheader_filename = "openssl/rsa.h", free_function = "RSA_free")]
public class RSA
{
public const int PKCS1_PADDING;
public const int SSLV23_PADDING;
public const int NO_PADDING;
public const int PKCS1_OAEP_PADDING;
public const int X931_PADDING;
public const int PKCS1_PSS_PADDING;
public const int F4;
[CCode (cname = "RSA_3")]
public const int RSA_3;
public RSA ();
public int size ();
public int set0_key (BIGNUM n, BIGNUM e, BIGNUM d);
public void get0_key (out BIGNUM n, out BIGNUM e, out BIGNUM d);
public void clear_flags (int flags);
public int test_flags (int flags);
public void set_flags (int flags);
public Engine get0_engine ();
[CCode (instance_pos = 1.1)]
public int print_fp (GLib.FileStream fp, int offset);
[CCode (instance_pos = 1.1)]
public int print (BIO bp, int offset);
public int generate_key_ex (int bits, BIGNUM e, BN_GENCB? cb = null);
[CCode (instance_pos = 4)]
public bool sign (int type, uint8[] m, [CCode (array_length = false)] uint8[] sigret, out int siglen);
[CCode (instance_pos = 3)]
public int verify (int type, uint8[] m, uint8[] sigbuf);
[CCode (instance_pos = 2.1)]
public int public_encrypt ([CCode (array_length_pos = 0)] uint8[] from, [CCode (array_length = false)] uint8[] to, int padding);
[CCode (instance_pos = 2.1)]
public int private_encrypt ([CCode (array_length_pos = 0)] uint8[] from, [CCode (array_length = false)] uint8[] to, int padding);
[CCode (instance_pos = 2.1)]
public int public_decrypt ([CCode (array_length_pos = 0)] uint8[] from, [CCode (array_length = false)] uint8[] to,int padding);
[CCode (instance_pos = 2.1)]
public int private_decrypt ([CCode (array_length_pos = 0)] uint8[] from, [CCode (array_length = false)] uint8[] to, int padding);
}
[CCode (lower_case_cprefix = "PEM_", cheader_filename = "openssl/pem.h")]
namespace PEM
{
[CCode (cname = "pem_password_cb")]
public delegate int PasswordCallback (uint8[] buf, int flag);
public void read_RSAPrivateKey (GLib.FileStream f, out RSA x, PasswordCallback? cb = null);
public void read_RSAPublicKey (GLib.FileStream f, out RSA x, PasswordCallback? cb = null);
public int write_RSAPrivateKey (GLib.FileStream f, RSA x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_RSAPublicKey (GLib.FileStream f, RSA x);
public void read_bio_RSAPublicKey (BIO bp, out RSA x, PasswordCallback? cb = null);
public void read_bio_RSAPrivateKey (BIO bp, out RSA x, PasswordCallback? cb = null);
public bool write_bio_RSAPublicKey (BIO bp, RSA x);
public bool write_bio_RSAPrivateKey (BIO bp, RSA x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public void read_bio_PUBKEY (BIO bp, out EVP.PublicKey x, PasswordCallback? cb = null);
public int write_bio_PUBKEY (BIO bp, EVP.PublicKey x);
public void read_PUBKEY (GLib.FileStream fp, out EVP.PublicKey x, PasswordCallback? cb = null);
public int write_PUBKEY (GLib.FileStream fp, EVP.PublicKey x);
public void read_bio_PrivateKey (BIO bp, out EVP.PublicKey x, PasswordCallback? cb = null);
public void read_PrivateKey (GLib.FileStream fp, out EVP.PublicKey x, PasswordCallback? cb = null);
public int write_bio_PrivateKey (BIO bp, EVP.PublicKey x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_bio_PrivateKey_traditional (BIO bp, EVP.PublicKey x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_PrivateKey (GLib.FileStream fp, EVP.PublicKey x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_bio_PKCS8PrivateKey (BIO bp, EVP.PublicKey x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_bio_PKCS8PrivateKey_nid (BIO bp, EVP.PublicKey x, int nid, uint8[] kstr, PasswordCallback? cb = null);
public int write_PKCS8PrivateKey (GLib.FileStream fp, EVP.PublicKey x, EVP.Cipher? enc, uint8[] kstr, PasswordCallback? cb = null);
public int write_PKCS8PrivateKey_nid (GLib.FileStream fp, EVP.PublicKey x, int nid, uint8[] kstr, PasswordCallback? cb = null);
}
public RSA? d2i_RSA_PUBKEY (out RSA a, uint8[] ppin);
public RSA? d2i_RSA_PUBKEY_bio (BIO bp, out RSA a);
public RSA? d2i_RSA_PUBKEY_fp (GLib.FileStream fp, out RSA a);
public int i2d_RSA_PUBKEY (RSA rsa, [CCode (array_length = false)] out uint8[] ppout);
public int i2d_RSA_PUBKEY_fp (GLib.FileStream fp, RSA a);
public int i2d_RSA_PUBKEY_bio (BIO bp, RSA a);
[CCode (cprefix = "ERR_", lower_case_cprefix = "ERR_", cheader_filename = "openssl/err.h")]
namespace ERR
{
public ulong get_error();
public unowned string? reason_error_string(ulong e);
}
[CCode (cprefix = "RAND_", lower_case_cprefix = "RAND_", cheader_filename = "openssl/rand.h")]
namespace RAND
{
public int bytes(uint8[] buf);
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 213 KiB

View file

@ -1,7 +1,7 @@
{ {
"id": "im.dino.Dino", "id": "im.dino.Dino",
"runtime": "org.gnome.Platform", "runtime": "org.gnome.Platform",
"runtime-version": "46", "runtime-version": "44",
"sdk": "org.gnome.Sdk", "sdk": "org.gnome.Sdk",
"command": "dino", "command": "dino",
"finish-args": [ "finish-args": [
@ -60,11 +60,7 @@
"buildsystem": "cmake-ninja", "buildsystem": "cmake-ninja",
"builddir": true, "builddir": true,
"config-opts": [ "config-opts": [
"-DSOUP_VERSION=3", "-DSOUP_VERSION=3"
"-DNO_DEBUG=yes",
"-DCMAKE_BUILD_TYPE=Release",
"-DENABLED_PLUGINS=notification-sound",
"-DPLUGIN_RTP_WEBRTC_AUDIO_PROCESSING=OFF"
], ],
"cleanup": [ "cleanup": [
"/include", "/include",

View file

@ -102,8 +102,7 @@ DEPENDS
${CMAKE_BINARY_DIR}/exports/dino_i18n.h ${CMAKE_BINARY_DIR}/exports/dino_i18n.h
) )
add_definitions(${VALA_CFLAGS} -DDINO_SYSTEM_PLUGIN_DIR="${PLUGIN_INSTALL_DIR}" -DDINO_SYSTEM_LIBDIR_NAME="${LIBDIR_NAME}" -DG_LOG_DOMAIN="libdino" add_definitions(${VALA_CFLAGS} -DDINO_SYSTEM_PLUGIN_DIR="${PLUGIN_INSTALL_DIR}" -DDINO_SYSTEM_LIBDIR_NAME="${LIBDIR_NAME}" -DG_LOG_DOMAIN="libdino" -DDINO_VERSION=\"${PROJECT_VERSION}\")
-DDINO_VERSION=\"${PROJECT_VERSION}\" -DDINO_SYSTEM_LOCALEDIR_NAME="${LOCALEDIR_NAME}")
add_library(libdino SHARED ${LIBDINO_VALA_C} ${CMAKE_BINARY_DIR}/exports/dino_i18n.h) add_library(libdino SHARED ${LIBDINO_VALA_C} ${CMAKE_BINARY_DIR}/exports/dino_i18n.h)
add_dependencies(libdino dino-vapi) add_dependencies(libdino dino-vapi)
target_link_libraries(libdino xmpp-vala qlite ${LIBDINO_PACKAGES} m) target_link_libraries(libdino xmpp-vala qlite ${LIBDINO_PACKAGES} m)
@ -113,13 +112,12 @@ install(TARGETS libdino ${TARGET_INSTALL})
install(FILES ${CMAKE_BINARY_DIR}/exports/dino.vapi ${CMAKE_BINARY_DIR}/exports/dino.deps DESTINATION ${VAPI_INSTALL_DIR}) install(FILES ${CMAKE_BINARY_DIR}/exports/dino.vapi ${CMAKE_BINARY_DIR}/exports/dino.deps DESTINATION ${VAPI_INSTALL_DIR})
install(FILES ${CMAKE_BINARY_DIR}/exports/dino.h ${CMAKE_BINARY_DIR}/exports/dino_i18n.h DESTINATION ${INCLUDE_INSTALL_DIR}) install(FILES ${CMAKE_BINARY_DIR}/exports/dino.h ${CMAKE_BINARY_DIR}/exports/dino_i18n.h DESTINATION ${INCLUDE_INSTALL_DIR})
if(BUILD_TESTING) if(BUILD_TESTS)
vala_precompile(LIBDINO_TEST_VALA_C vala_precompile(LIBDINO_TEST_VALA_C
SOURCES SOURCES
"tests/weak_map.vala" "tests/weak_map.vala"
"tests/testcase.vala" "tests/testcase.vala"
"tests/common.vala" "tests/common.vala"
"tests/jid.vala"
CUSTOM_VAPIS CUSTOM_VAPIS
${CMAKE_BINARY_DIR}/exports/dino_internal.vapi ${CMAKE_BINARY_DIR}/exports/dino_internal.vapi
${CMAKE_BINARY_DIR}/exports/xmpp-vala.vapi ${CMAKE_BINARY_DIR}/exports/xmpp-vala.vapi
@ -133,5 +131,4 @@ if(BUILD_TESTING)
add_definitions(${VALA_CFLAGS}) add_definitions(${VALA_CFLAGS})
add_executable(libdino-test ${LIBDINO_TEST_VALA_C}) add_executable(libdino-test ${LIBDINO_TEST_VALA_C})
target_link_libraries(libdino-test libdino) target_link_libraries(libdino-test libdino)
add_test(NAME libdino COMMAND libdino-test) endif(BUILD_TESTS)
endif(BUILD_TESTING)

View file

@ -79,9 +79,8 @@ sources = files(
) )
sources += [version_vala] sources += [version_vala]
c_args = [ c_args = [
'-DDINO_SYSTEM_LIBDIR_NAME="@0@"'.format(get_option('libdir')), '-DDINO_SYSTEM_LIBDIR_NAME="@0@"'.format(get_option('prefix') / get_option('libdir')),
'-DDINO_SYSTEM_LOCALEDIR_NAME="@0@"'.format(get_option('localedir')), '-DDINO_SYSTEM_PLUGIN_DIR="@0@"'.format(get_option('prefix') / get_option('plugindir')),
'-DDINO_SYSTEM_PLUGIN_DIR="@0@"'.format(get_option('prefix') / get_option('libdir') / get_option('plugindir')),
'-DG_LOG_DOMAIN="libdino"', '-DG_LOG_DOMAIN="libdino"',
] ]
lib_dino = library('dino', sources, c_args: c_args, include_directories: include_directories('src'), dependencies: dependencies, version: '0.0', install: true, install_dir: [true, true, true]) lib_dino = library('dino', sources, c_args: c_args, include_directories: include_directories('src'), dependencies: dependencies, version: '0.0', install: true, install_dir: [true, true, true])
@ -89,12 +88,3 @@ dep_dino = declare_dependency(link_with: lib_dino, include_directories: include_
install_data('dino.deps', install_dir: get_option('datadir') / 'vala/vapi') # TODO: workaround for https://github.com/mesonbuild/meson/issues/9756 install_data('dino.deps', install_dir: get_option('datadir') / 'vala/vapi') # TODO: workaround for https://github.com/mesonbuild/meson/issues/9756
install_headers('src/dino_i18n.h') install_headers('src/dino_i18n.h')
sources = files(
'tests/weak_map.vala',
'tests/testcase.vala',
'tests/common.vala',
'tests/jid.vala',
)
test_libdino = executable('test_libdino', sources, dependencies: dependencies + [dep_dino])
test('libdino', test_libdino)

View file

@ -25,13 +25,13 @@ public interface Application : GLib.Application {
public abstract void handle_uri(string jid, string query, Gee.Map<string, string> options); public abstract void handle_uri(string jid, string query, Gee.Map<string, string> options);
public void init(bool default_dark_theme) throws Error { public void init() throws Error {
if (DirUtils.create_with_parents(get_storage_dir(), 0700) == -1) { if (DirUtils.create_with_parents(get_storage_dir(), 0700) == -1) {
throw new Error(-1, 0, "Could not create storage dir \"%s\": %s", get_storage_dir(), FileUtils.error_from_errno(errno).to_string()); throw new Error(-1, 0, "Could not create storage dir \"%s\": %s", get_storage_dir(), FileUtils.error_from_errno(errno).to_string());
} }
this.db = new Database(Path.build_filename(get_storage_dir(), "dino.db")); this.db = new Database(Path.build_filename(get_storage_dir(), "dino.db"));
this.settings = new Dino.Entities.Settings.from_db(db, default_dark_theme); this.settings = new Dino.Entities.Settings.from_db(db);
this.stream_interactor = new StreamInteractor(db); this.stream_interactor = new StreamInteractor(db);
MessageProcessor.start(stream_interactor, db); MessageProcessor.start(stream_interactor, db);

View file

@ -2,8 +2,6 @@ using Xmpp;
namespace Dino.Entities { namespace Dino.Entities {
const int HISTORY_SYNC_MAM_MESSAGES = 20;
public class Conversation : Object { public class Conversation : Object {
public signal void object_updated(Conversation conversation); public signal void object_updated(Conversation conversation);
@ -35,7 +33,7 @@ public class Conversation : Object {
} }
} }
} }
public Encryption encryption { get; set; default = Encryption.UNKNOWN; } public Encryption encryption { get; set; default = Encryption.NONE; }
public Message? read_up_to { get; set; } public Message? read_up_to { get; set; }
public int read_up_to_item { get; set; default=-1; } public int read_up_to_item { get; set; default=-1; }

View file

@ -11,27 +11,6 @@ namespace Dino.Entities {
public bool is_some() { public bool is_some() {
return this != NONE; return this != NONE;
} }
public static Encryption parse(string str) {
switch (str) {
case "DINO_ENTITIES_ENCRYPTION_NONE":
return NONE;
case "DINO_ENTITIES_ENCRYPTION_PGP":
return PGP;
case "DINO_ENTITIES_ENCRYPTION_OMEMO":
return OMEMO;
case "DINO_ENTITIES_ENCRYPTION_DTLS_SRTP":
return DTLS_SRTP;
case "DINO_ENTITIES_ENCRYPTION_SRTP":
return SRTP;
case "DINO_ENTITIES_ENCRYPTION_UNKNOWN":
// Fall through.
default:
break;
}
return UNKNOWN;
}
} }
} }

View file

@ -71,7 +71,6 @@ public class FileTransfer : Object {
public int provider { get; set; } public int provider { get; set; }
public string info { get; set; } public string info { get; set; }
public Cancellable cancellable { get; default=new Cancellable(); } public Cancellable cancellable { get; default=new Cancellable(); }
public uint64 transferred_bytes { get; set; }
private Database? db; private Database? db;
private string storage_dir; private string storage_dir;

View file

@ -4,7 +4,7 @@ public class Settings : Object {
private Database db; private Database db;
public Settings.from_db(Database db, bool default_dark_theme) { public Settings.from_db(Database db) {
this.db = db; this.db = db;
send_typing_ = col_to_bool_or_default("send_typing", true); send_typing_ = col_to_bool_or_default("send_typing", true);
@ -12,10 +12,6 @@ public class Settings : Object {
notifications_ = col_to_bool_or_default("notifications", true); notifications_ = col_to_bool_or_default("notifications", true);
convert_utf8_smileys_ = col_to_bool_or_default("convert_utf8_smileys", true); convert_utf8_smileys_ = col_to_bool_or_default("convert_utf8_smileys", true);
check_spelling = col_to_bool_or_default("check_spelling", true); check_spelling = col_to_bool_or_default("check_spelling", true);
default_encryption = col_to_encryption_or_default("default_encryption", Encryption.UNKNOWN);
send_button = col_to_bool_or_default("send_button", false);
enter_newline = col_to_bool_or_default("enter_newline", false);
dark_theme = col_to_bool_or_default("dark_theme", default_dark_theme);
} }
private bool col_to_bool_or_default(string key, bool def) { private bool col_to_bool_or_default(string key, bool def) {
@ -23,12 +19,6 @@ public class Settings : Object {
return val != null ? bool.parse(val) : def; return val != null ? bool.parse(val) : def;
} }
private Encryption col_to_encryption_or_default(string key, Encryption def) {
var sval = db.settings.value;
string? val = db.settings.select({sval}).with(db.settings.key, "=", key)[sval];
return val != null ? Encryption.parse(val) : def;
}
private bool send_typing_; private bool send_typing_;
public bool send_typing { public bool send_typing {
get { return send_typing_; } get { return send_typing_; }
@ -89,60 +79,6 @@ public class Settings : Object {
check_spelling_ = value; check_spelling_ = value;
} }
} }
private Encryption default_encryption_;
public Encryption default_encryption {
get { return default_encryption_; }
set {
string valstr = value.to_string();
db.settings.upsert()
.value(db.settings.key, "default_encryption", true)
.value(db.settings.value, valstr)
.perform();
default_encryption_ = value;
}
}
public signal void send_button_update(bool visible);
private bool send_button_;
public bool send_button {
get { return send_button_; }
set {
db.settings.upsert()
.value(db.settings.key, "send_button", true)
.value(db.settings.value, value.to_string())
.perform();
send_button_ = value;
send_button_update(value);
}
}
private bool enter_newline_;
public bool enter_newline {
get { return enter_newline_; }
set {
db.settings.upsert()
.value(db.settings.key, "enter_newline", true)
.value(db.settings.value, value.to_string())
.perform();
enter_newline_ = value;
}
}
public signal void dark_theme_update(bool is_dark);
private bool dark_theme_;
public bool dark_theme {
get { return dark_theme_; }
set {
db.settings.upsert()
.value(db.settings.key, "dark_theme", true)
.value(db.settings.value, value.to_string())
.perform();
dark_theme_ = value;
dark_theme_update(value);
}
}
} }
} }

View file

@ -60,7 +60,7 @@ public class Dino.CallState : Object {
XmppStream stream = stream_interactor.get_stream(call.account); XmppStream stream = stream_interactor.get_stream(call.account);
if (stream == null) return; if (stream == null) return;
Gee.List<Jid> occupants = stream_interactor.get_module(MucManager.IDENTITY).get_other_members(muc, call.account); Gee.List<Jid> occupants = stream_interactor.get_module(MucManager.IDENTITY).get_other_occupants(muc, call.account);
foreach (Jid occupant in occupants) { foreach (Jid occupant in occupants) {
Jid? real_jid = stream_interactor.get_module(MucManager.IDENTITY).get_real_jid(occupant, call.account); Jid? real_jid = stream_interactor.get_module(MucManager.IDENTITY).get_real_jid(occupant, call.account);
if (real_jid == null) continue; if (real_jid == null) continue;

View file

@ -11,7 +11,6 @@ public class ContentItemStore : StreamInteractionModule, Object {
public string id { get { return IDENTITY.id; } } public string id { get { return IDENTITY.id; } }
public signal void new_item(ContentItem item, Conversation conversation); public signal void new_item(ContentItem item, Conversation conversation);
public signal void history_loaded(Conversation conversation, ContentItem item, int count);
private StreamInteractor stream_interactor; private StreamInteractor stream_interactor;
private Database db; private Database db;
@ -242,10 +241,8 @@ public class ContentItemStore : StreamInteractionModule, Object {
// return ret; // return ret;
// } // }
public Gee.List<ContentItem> get_before(Conversation conversation, ContentItem item, int count, bool request_from_server = true) { public Gee.List<ContentItem> get_before(Conversation conversation, ContentItem item, int count) {
debug("Fetching earlier messages from the db");
long time = (long) item.time.to_unix(); long time = (long) item.time.to_unix();
QueryBuilder select = db.content_item.select() QueryBuilder select = db.content_item.select()
.where(@"time < ? OR (time = ? AND id < ?)", { time.to_string(), time.to_string(), item.id.to_string() }) .where(@"time < ? OR (time = ? AND id < ?)", { time.to_string(), time.to_string(), item.id.to_string() })
.with(db.content_item.conversation_id, "=", conversation.id) .with(db.content_item.conversation_id, "=", conversation.id)
@ -254,18 +251,7 @@ public class ContentItemStore : StreamInteractionModule, Object {
.order_by(db.content_item.id, "DESC") .order_by(db.content_item.id, "DESC")
.limit(count); .limit(count);
var items = get_items_from_query(select, conversation); return get_items_from_query(select, conversation);
if (items.size == 0 && request_from_server) {
// Async request to get earlier messages from the server
var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync;
history_sync.fetch_conversation_data.begin(conversation, item.time, (_, res) => {
history_sync.fetch_conversation_data.end(res);
debug("History loaded");
history_loaded(conversation, item, count);
});
}
return items;
} }
public Gee.List<ContentItem> get_after(Conversation conversation, ContentItem item, int count) { public Gee.List<ContentItem> get_after(Conversation conversation, ContentItem item, int count) {

View file

@ -46,15 +46,6 @@ public class FileManager : StreamInteractionModule, Object {
return ret; return ret;
} }
private string sanitize_filename(string filename) {
#if _WIN32
GLib.Regex regex = new GLib.Regex("[<>:\"/\\|?*]");
#else
GLib.Regex regex = new GLib.Regex("[/]");
#endif
return regex.replace(filename, -1, 0, "_");
}
public async void send_file(File file, Conversation conversation) { public async void send_file(File file, Conversation conversation) {
FileTransfer file_transfer = new FileTransfer(); FileTransfer file_transfer = new FileTransfer();
file_transfer.account = conversation.account; file_transfer.account = conversation.account;
@ -129,7 +120,6 @@ public class FileManager : StreamInteractionModule, Object {
} }
yield file_sender.send_file(conversation, file_transfer, file_send_data, file_meta); yield file_sender.send_file(conversation, file_transfer, file_send_data, file_meta);
file_transfer.state = FileTransfer.State.COMPLETE;
} catch (Error e) { } catch (Error e) {
warning("Send file error: %s", e.message); warning("Send file error: %s", e.message);
@ -252,7 +242,7 @@ public class FileManager : StreamInteractionModule, Object {
} }
// Save file // Save file
string filename = Random.next_int().to_string("%x") + "_" + sanitize_filename(file_transfer.file_name); string filename = Random.next_int().to_string("%x") + "_" + file_transfer.file_name;
File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename)); File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename));
OutputStream os = file.create(FileCreateFlags.REPLACE_DESTINATION); OutputStream os = file.create(FileCreateFlags.REPLACE_DESTINATION);
@ -260,7 +250,6 @@ public class FileManager : StreamInteractionModule, Object {
ssize_t read; ssize_t read;
while ((read = yield input_stream.read_async(buffer, Priority.LOW, file_transfer.cancellable)) > 0) { while ((read = yield input_stream.read_async(buffer, Priority.LOW, file_transfer.cancellable)) > 0) {
buffer.length = (int) read; buffer.length = (int) read;
file_transfer.transferred_bytes += (uint64)read;
yield os.write_async(buffer, Priority.LOW, file_transfer.cancellable); yield os.write_async(buffer, Priority.LOW, file_transfer.cancellable);
buffer.length = 1024; buffer.length = 1024;
} }
@ -308,7 +297,6 @@ public class FileManager : StreamInteractionModule, Object {
file_transfer.file_name = file_meta.file_name; file_transfer.file_name = file_meta.file_name;
file_transfer.size = (int)file_meta.size; file_transfer.size = (int)file_meta.size;
file_transfer.info = info; file_transfer.info = info;
file_transfer.transferred_bytes = 0;
var encryption = file_provider.get_encryption(file_transfer, receive_data, file_meta); var encryption = file_provider.get_encryption(file_transfer, receive_data, file_meta);
if (encryption != Encryption.NONE) file_transfer.encryption = encryption; if (encryption != Encryption.NONE) file_transfer.encryption = encryption;
@ -341,11 +329,11 @@ public class FileManager : StreamInteractionModule, Object {
private async void save_file(FileTransfer file_transfer) throws FileSendError { private async void save_file(FileTransfer file_transfer) throws FileSendError {
try { try {
string filename = Random.next_int().to_string("%x") + "_" + sanitize_filename(file_transfer.file_name); string filename = Random.next_int().to_string("%x") + "_" + file_transfer.file_name;
File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename)); File file = File.new_for_path(Path.build_filename(get_storage_dir(), filename));
OutputStream os = file.create(FileCreateFlags.REPLACE_DESTINATION); OutputStream os = file.create(FileCreateFlags.REPLACE_DESTINATION);
yield os.splice_async(file_transfer.input_stream, OutputStreamSpliceFlags.CLOSE_SOURCE|OutputStreamSpliceFlags.CLOSE_TARGET); yield os.splice_async(file_transfer.input_stream, OutputStreamSpliceFlags.CLOSE_SOURCE|OutputStreamSpliceFlags.CLOSE_TARGET);
file_transfer.state = FileTransfer.State.IN_PROGRESS; file_transfer.state = FileTransfer.State.COMPLETE;
file_transfer.path = filename; file_transfer.path = filename;
file_transfer.input_stream = yield file.read_async(); file_transfer.input_stream = yield file.read_async();
} catch (Error e) { } catch (Error e) {

View file

@ -22,7 +22,6 @@ public class Dino.HistorySync {
public HashMap<Account, DateTime> catchup_until_time = new HashMap<Account, DateTime>(Account.hash_func, Account.equals_func); public HashMap<Account, DateTime> catchup_until_time = new HashMap<Account, DateTime>(Account.hash_func, Account.equals_func);
private HashMap<string, Gee.List<Xmpp.MessageStanza>> stanzas = new HashMap<string, Gee.List<Xmpp.MessageStanza>>(); private HashMap<string, Gee.List<Xmpp.MessageStanza>> stanzas = new HashMap<string, Gee.List<Xmpp.MessageStanza>>();
private HashMap<string, int> messages_processed = new HashMap<string, int>();
public class HistorySync(Database db, StreamInteractor stream_interactor) { public class HistorySync(Database db, StreamInteractor stream_interactor) {
this.stream_interactor = stream_interactor; this.stream_interactor = stream_interactor;
@ -121,63 +120,6 @@ public class Dino.HistorySync {
} }
} }
private async PageRequestResult fetch_messages(Account account, Xmpp.MessageArchiveManagement.V2.MamQueryParams query_params) {
debug("[%s | %s] Fetch query %s - %s", account.bare_jid.to_string(), query_params.mam_server.to_string(), query_params.start != null ? query_params.start.to_string() : "", query_params.end != null ? query_params.end.to_string() : "");
PageRequestResult? page_result = null;
int processed_pages = 0;
do {
page_result = yield get_mam_page(account, query_params, page_result, null);
processed_pages++;
debug("%d messages left to process, current page is %d", messages_processed[query_params.query_id], processed_pages);
if (messages_processed[query_params.query_id] <= 1) {
debug("Done processing new messages");
break;
}
debug("[%s | %s] Page result %s (got stanzas: %s)", account.bare_jid.to_string(), query_params.mam_server.to_string(), page_result.page_result.to_string(), (page_result.stanzas != null).to_string());
if (page_result.page_result == PageResult.Error || page_result.page_result == PageResult.Cancelled || page_result.query_result.first == null) {
return page_result;
}
} while (page_result.page_result == PageResult.MorePagesAvailable);
return page_result;
}
public async void fetch_data(Account account, Jid target, DateTime latest) {
debug("Fetch history for %s", target.to_string());
var query_params = new Xmpp.MessageArchiveManagement.V2.MamQueryParams.query_before(target, latest, null);
string query_id = query_params.query_id;
messages_processed[query_id] = HISTORY_SYNC_MAM_MESSAGES;
yield fetch_messages(account, query_params);
}
public async void fetch_conversation_data(Conversation? conversation, DateTime latest) {
if (conversation == null) {
warning("Failed to fetch history, conversation is null");
return;
}
var target = conversation.counterpart.bare_jid;
var account = conversation.account;
debug("Fetch history for %s", target.to_string());
var query_params = new Xmpp.MessageArchiveManagement.V2.MamQueryParams.query_before(target, latest, null);
if (conversation.type_ == Conversation.Type.CHAT) {
query_params.mam_server = account.bare_jid;
query_params.with = target;
}
string query_id = query_params.query_id;
messages_processed[query_id] = HISTORY_SYNC_MAM_MESSAGES;
yield fetch_messages(account, query_params);
}
public async void fetch_history(Account account, Jid target, Cancellable? cancellable = null) { public async void fetch_history(Account account, Jid target, Cancellable? cancellable = null) {
debug("Fetch history for %s", target.to_string()); debug("Fetch history for %s", target.to_string());
@ -506,7 +448,6 @@ public class Dino.HistorySync {
} }
} }
} }
if (hitted_range.has_key(query_id) && hitted_range[query_id] == -2) { if (hitted_range.has_key(query_id) && hitted_range[query_id] == -2) {
// Message got filtered out by xmpp-vala, but succesful range fetch nevertheless // Message got filtered out by xmpp-vala, but succesful range fetch nevertheless
yield send_messages_back_into_pipeline(account, query_id); yield send_messages_back_into_pipeline(account, query_id);
@ -521,24 +462,15 @@ public class Dino.HistorySync {
if (cancellable != null && cancellable.is_cancelled()) { if (cancellable != null && cancellable.is_cancelled()) {
page_result = PageResult.Cancelled; page_result = PageResult.Cancelled;
} }
return new PageRequestResult(page_result, query_result, stanzas_for_query); return new PageRequestResult(page_result, query_result, stanzas_for_query);
} }
private async void send_messages_back_into_pipeline(Account account, string query_id, Cancellable? cancellable = null) { private async void send_messages_back_into_pipeline(Account account, string query_id, Cancellable? cancellable = null) {
if (!stanzas.has_key(query_id)) { if (!stanzas.has_key(query_id)) return;
return;
}
foreach (Xmpp.MessageStanza message in stanzas[query_id]) { foreach (Xmpp.MessageStanza message in stanzas[query_id]) {
if (cancellable != null && cancellable.is_cancelled()) break; if (cancellable != null && cancellable.is_cancelled()) break;
bool result = yield stream_interactor.get_module(MessageProcessor.IDENTITY).run_pipeline_announce(account, message); yield stream_interactor.get_module(MessageProcessor.IDENTITY).run_pipeline_announce(account, message);
if (result && messages_processed.has_key(query_id)) {
int count = messages_processed[query_id];
count = count - 1;
messages_processed[query_id] = count;
}
} }
stanzas.unset(query_id); stanzas.unset(query_id);
} }

View file

@ -220,19 +220,7 @@ public class JingleFileSender : FileSender, Object {
} }
} }
try { try {
var? module = stream.get_module(Xep.JingleFileTransfer.Module.IDENTITY); yield stream.get_module(Xep.JingleFileTransfer.Module.IDENTITY).offer_file_stream(stream, full_jid, file_transfer.input_stream, file_transfer.server_file_name, file_meta.size, precondition_name, precondition_options);
if (module == null)
throw new FileSendError.UPLOAD_FAILED("unexpected null module");
module.transferred_bytes.connect((bytes) => {
file_transfer.transferred_bytes += bytes;
});
yield module.offer_file_stream(stream, full_jid,
file_transfer.cancellable, file_transfer.input_stream,
file_transfer.server_file_name, file_meta.size,
precondition_name, precondition_options);
} catch (Error e) { } catch (Error e) {
throw new FileSendError.UPLOAD_FAILED(@"offer_file_stream failed: $(e.message)"); throw new FileSendError.UPLOAD_FAILED(@"offer_file_stream failed: $(e.message)");
} }

View file

@ -130,18 +130,14 @@ public class MessageProcessor : StreamInteractionModule, Object {
run_pipeline_announce.begin(account, message_stanza); run_pipeline_announce.begin(account, message_stanza);
} }
public async bool run_pipeline_announce(Account account, Xmpp.MessageStanza message_stanza) { public async void run_pipeline_announce(Account account, Xmpp.MessageStanza message_stanza) {
Entities.Message message = yield parse_message_stanza(account, message_stanza); Entities.Message message = yield parse_message_stanza(account, message_stanza);
Conversation? conversation = stream_interactor.get_module(ConversationManager.IDENTITY).get_conversation_for_message(message); Conversation? conversation = stream_interactor.get_module(ConversationManager.IDENTITY).get_conversation_for_message(message);
if (conversation == null) { if (conversation == null) return;
return false;
}
bool abort = yield received_pipeline.run(message, message_stanza, conversation); bool abort = yield received_pipeline.run(message, message_stanza, conversation);
if (abort) { if (abort) return;
return false;
}
if (message.direction == Entities.Message.DIRECTION_RECEIVED) { if (message.direction == Entities.Message.DIRECTION_RECEIVED) {
message_received(message, conversation); message_received(message, conversation);
@ -150,7 +146,6 @@ public class MessageProcessor : StreamInteractionModule, Object {
} }
message_sent_or_received(message, conversation); message_sent_or_received(message, conversation);
return true;
} }
public async Entities.Message parse_message_stanza(Account account, Xmpp.MessageStanza message) { public async Entities.Message parse_message_stanza(Account account, Xmpp.MessageStanza message) {
@ -250,7 +245,6 @@ public class MessageProcessor : StreamInteractionModule, Object {
// If the message is a duplicate // If the message is a duplicate
if (builder.count() > 0) { if (builder.count() > 0) {
warning("deduplicate by server id");
history_sync.on_server_id_duplicate(account, stanza, message); history_sync.on_server_id_duplicate(account, stanza, message);
return true; return true;
} }
@ -277,11 +271,6 @@ public class MessageProcessor : StreamInteractionModule, Object {
} }
} }
bool duplicate = builder.single().row().is_present(); bool duplicate = builder.single().row().is_present();
if (duplicate) {
warning("deduplicate by uuid");
}
return duplicate; return duplicate;
} }
@ -302,13 +291,7 @@ public class MessageProcessor : StreamInteractionModule, Object {
} else { } else {
builder.with_null(db.message.counterpart_resource); builder.with_null(db.message.counterpart_resource);
} }
return builder.count() > 0;
bool duplicate = builder.count() > 0;
if (duplicate) {
warning("deduplicate by content and metadata");
}
return duplicate;
} }
private class DeduplicateMessageListener : MessageListener { private class DeduplicateMessageListener : MessageListener {
@ -374,10 +357,7 @@ public class MessageProcessor : StreamInteractionModule, Object {
} }
public override async bool run(Entities.Message message, Xmpp.MessageStanza stanza, Conversation conversation) { public override async bool run(Entities.Message message, Xmpp.MessageStanza stanza, Conversation conversation) {
if (message.body == null) { if (message.body == null) return true;
return true;
}
stream_interactor.get_module(ContentItemStore.IDENTITY).insert_message(message, conversation); stream_interactor.get_module(ContentItemStore.IDENTITY).insert_message(message, conversation);
return false; return false;
} }

View file

@ -106,8 +106,8 @@ public class MucManager : StreamInteractionModule, Object {
if (can_do_mam) { if (can_do_mam) {
var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync; var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync;
if (conversation == null) { if (conversation == null) {
// We never joined the conversation before, fetch latest MAM pages // We never joined the conversation before, just fetch the latest MAM page
yield history_sync.fetch_data(account, jid.bare_jid, new DateTime.now()); yield history_sync.fetch_latest_page(account, jid.bare_jid, null, new DateTime.from_unix_utc(0), cancellable);
} else { } else {
// Fetch everything up to the last time the user actively joined // Fetch everything up to the last time the user actively joined
if (!mucs_sync_cancellables.has_key(account)) { if (!mucs_sync_cancellables.has_key(account)) {
@ -259,23 +259,7 @@ public class MucManager : StreamInteractionModule, Object {
return is_groupchat(jid, account) && !is_private_room(account, jid); return is_groupchat(jid, account) && !is_private_room(account, jid);
} }
public Gee.List<Jid>? get_all_members(Jid jid, Account account) { public Gee.List<Jid>? get_occupants(Jid jid, Account account) {
if (is_groupchat(jid, account)) {
Gee.List<Jid> ret = new ArrayList<Jid>(Jid.equals_func);
// This should return all members of the chat
Gee.List<Jid>? members = get_offline_members(jid, account);
if (members != null) {
ret.add_all(members);
}
return ret;
}
return null;
}
public Gee.List<Jid>? get_members(Jid jid, Account account) {
if (is_groupchat(jid, account)) { if (is_groupchat(jid, account)) {
Gee.List<Jid> ret = new ArrayList<Jid>(Jid.equals_func); Gee.List<Jid> ret = new ArrayList<Jid>(Jid.equals_func);
Gee.List<Jid>? full_jids = stream_interactor.get_module(PresenceManager.IDENTITY).get_full_jids(jid, account); Gee.List<Jid>? full_jids = stream_interactor.get_module(PresenceManager.IDENTITY).get_full_jids(jid, account);
@ -284,21 +268,18 @@ public class MucManager : StreamInteractionModule, Object {
// Remove eventual presence from bare jid // Remove eventual presence from bare jid
ret.remove(jid); ret.remove(jid);
} }
return ret; return ret;
} }
return null; return null;
} }
public Gee.List<Jid>? get_other_members(Jid jid, Account account) { public Gee.List<Jid>? get_other_occupants(Jid jid, Account account) {
Gee.List<Jid>? members = get_members(jid, account); Gee.List<Jid>? occupants = get_occupants(jid, account);
Jid? own_jid = get_own_jid(jid, account); Jid? own_jid = get_own_jid(jid, account);
if (members != null && own_jid != null) { if (occupants != null && own_jid != null) {
members.remove(own_jid); occupants.remove(own_jid);
} }
return occupants;
return members;
} }
public bool is_groupchat(Jid jid, Account account) { public bool is_groupchat(Jid jid, Account account) {
@ -554,25 +535,12 @@ public class MucManager : StreamInteractionModule, Object {
} }
private void on_invite_received(Account account, Jid room_jid, Jid from_jid, string? password, string? reason) { private void on_invite_received(Account account, Jid room_jid, Jid from_jid, string? password, string? reason) {
info("Invite received for room %s", room_jid.bare_jid.to_string());
Gee.List<Conversation> conversations = stream_interactor.get_module(ConversationManager.IDENTITY).get_active_conversations(account);
foreach (Conversation conversation in conversations) {
if (conversation.counterpart.bare_jid.to_string() == room_jid.bare_jid.to_string()) {
// HACK: try to skip duplicate invites for active conversations
warning("Skipping duplicate invite for conversation %s", room_jid.bare_jid.to_string());
return;
}
}
if (!invites.has_key(account)) { if (!invites.has_key(account)) {
invites[account] = new LinkedList<Jid>(Jid.equals_func); invites[account] = new LinkedList<Jid>(Jid.equals_func);
} }
if (invites[account].contains(room_jid)) return;
if (invites[account].contains(room_jid)) {
return;
}
invites[account].add(room_jid); invites[account].add(room_jid);
invite_received(account, room_jid, from_jid, password, reason); invite_received(account, room_jid, from_jid, password, reason);
Timeout.add_seconds(5, () => { Timeout.add_seconds(5, () => {
@ -628,19 +596,17 @@ public class MucManager : StreamInteractionModule, Object {
private void set_autojoin(Account account, XmppStream stream, Jid jid, string? nick, string? password) { private void set_autojoin(Account account, XmppStream stream, Jid jid, string? nick, string? password) {
bookmarks_provider[account].get_conferences.begin(stream, (_, res) => { bookmarks_provider[account].get_conferences.begin(stream, (_, res) => {
Set<Conference>? conferences = bookmarks_provider[account].get_conferences.end(res); Set<Conference>? conferences = bookmarks_provider[account].get_conferences.end(res);
if (conferences == null) return;
if (conferences != null) { foreach (Conference conference in conferences) {
foreach (Conference conference in conferences) { if (conference.jid.equals(jid)) {
if (conference.jid.equals(jid)) { if (!conference.autojoin) {
if (!conference.autojoin) { Conference new_conference = new Conference() { jid=jid, nick=nick ?? conference.nick, name=conference.name, password=password ?? conference.password, autojoin=true };
Conference new_conference = new Conference() { jid=jid, nick=nick ?? conference.nick, name=conference.name, password=password ?? conference.password, autojoin=true }; bookmarks_provider[account].replace_conference.begin(stream, jid, new_conference);
bookmarks_provider[account].replace_conference.begin(stream, jid, new_conference);
}
return;
} }
return;
} }
} }
Conference changed = new Xep.Bookmarks.Bookmarks1Conference(jid) { nick=nick, password=password, autojoin=true }; Conference changed = new Xep.Bookmarks.Bookmarks1Conference(jid) { nick=nick, password=password, autojoin=true };
bookmarks_provider[account].add_conference.begin(stream, changed); bookmarks_provider[account].add_conference.begin(stream, changed);
}); });

View file

@ -6,7 +6,7 @@ namespace Dino {
public class Util { public class Util {
#if _WIN32 #if _WIN32
[CCode (cname = "ShellExecuteA", cheader_filename = "windows.h")] [CCode (cname = "ShellExecuteA", cheader_filename = "windows.h")]
private static extern int* ShellExecuteA(int* hwnd, string operation, string file, string parameters, string directory, int showCmd); private static extern int ShellExecuteA(int* hwnd, string operation, string file, string parameters, string directory, int showCmd);
[CCode (cname = "CoInitialize", cheader_filename = "windows.h")] [CCode (cname = "CoInitialize", cheader_filename = "windows.h")]
private static extern int CoInitialize(void* reserved); private static extern int CoInitialize(void* reserved);
@ -14,7 +14,7 @@ public class Util {
[CCode (cname = "CoUninitialize", cheader_filename = "windows.h")] [CCode (cname = "CoUninitialize", cheader_filename = "windows.h")]
private static extern void CoUninitialize(); private static extern void CoUninitialize();
private static int* ShellExecute(string operation, string file) { private static int ShellExecute(string operation, string file) {
CoInitialize(null); CoInitialize(null);
var result = ShellExecuteA(null, operation, file, null, null, 1); var result = ShellExecuteA(null, operation, file, null, null, 1);
CoUninitialize(); CoUninitialize();

View file

@ -2,7 +2,6 @@ namespace Dino {
private extern const string SYSTEM_LIBDIR_NAME; private extern const string SYSTEM_LIBDIR_NAME;
private extern const string SYSTEM_PLUGIN_DIR; private extern const string SYSTEM_PLUGIN_DIR;
private extern const string SYSTEM_LOCALEDIR_NAME;
public class SearchPathGenerator { public class SearchPathGenerator {
@ -13,21 +12,14 @@ public class SearchPathGenerator {
} }
public string get_locale_path(string gettext_package, string locale_install_dir) { public string get_locale_path(string gettext_package, string locale_install_dir) {
if (exec_path != null) { string? locale_dir = null;
var exec_dir = Path.get_dirname(exec_path); if (Path.get_dirname(exec_path).contains("dino") || Path.get_dirname(exec_path) == "." || Path.get_dirname(exec_path).contains("build")) {
string[] search_paths = new string[] { string exec_locale = Path.build_filename(Path.get_dirname(exec_path), "locale");
Path.build_filename(exec_dir, "locale"), if (FileUtils.test(Path.build_filename(exec_locale, "en", "LC_MESSAGES", gettext_package + ".mo"), FileTest.IS_REGULAR)) {
Path.build_filename(Path.get_dirname(exec_dir), SYSTEM_LOCALEDIR_NAME) locale_dir = exec_locale;
};
foreach (var path in search_paths) {
if (FileUtils.test(Path.build_filename(path, "en", "LC_MESSAGES", gettext_package + ".mo"), FileTest.IS_REGULAR)) {
debug(@"Found locale $(gettext_package).mo in $(path)");
return path;
}
} }
} }
return locale_dir ?? locale_install_dir;
return locale_install_dir;
} }
public string[] get_plugin_paths() { public string[] get_plugin_paths() {

View file

@ -4,7 +4,6 @@ int main(string[] args) {
GLib.Test.init(ref args); GLib.Test.init(ref args);
GLib.Test.set_nonfatal_assertions(); GLib.Test.set_nonfatal_assertions();
TestSuite.get_root().add_suite(new WeakMapTest().get_suite()); TestSuite.get_root().add_suite(new WeakMapTest().get_suite());
TestSuite.get_root().add_suite(new JidTest().get_suite());
return GLib.Test.run(); return GLib.Test.run();
} }

View file

@ -1,5 +1,4 @@
using Dino.Entities; using Dino.Entities;
using Xmpp;
namespace Dino.Test { namespace Dino.Test {
@ -13,39 +12,27 @@ class JidTest : Gee.TestCase {
} }
private void test_parse() { private void test_parse() {
try { Jid jid = new Jid("user@example.com/res");
Jid jid = new Jid("user@example.com/res"); fail_if(jid.localpart != "user");
fail_if(jid.localpart != "user"); fail_if(jid.domainpart != "example.com");
fail_if(jid.domainpart != "example.com"); fail_if(jid.resourcepart != "res");
fail_if(jid.resourcepart != "res"); fail_if(jid.to_string() != "user@example.com/res");
fail_if(jid.to_string() != "user@example.com/res");
} catch (Error e) {
fail_if_reached(@"Throws $(e.message)");
}
} }
private void test_components() { private void test_components() {
try { Jid jid = new Jid.components("user", "example.com", "res");
Jid jid = new Jid.components("user", "example.com", "res"); fail_if(jid.localpart != "user");
fail_if(jid.localpart != "user"); fail_if(jid.domainpart != "example.com");
fail_if(jid.domainpart != "example.com"); fail_if(jid.resourcepart != "res");
fail_if(jid.resourcepart != "res"); fail_if(jid.to_string() != "user@example.com/res");
fail_if(jid.to_string() != "user@example.com/res");
} catch (Error e) {
fail_if_reached(@"Throws $(e.message)");
}
} }
private void test_with_res() { private void test_with_res() {
try { Jid jid = new Jid.with_resource("user@example.com", "res");
Jid jid = new Jid("user@example.com").with_resource("res"); fail_if(jid.localpart != "user");
fail_if(jid.localpart != "user"); fail_if(jid.domainpart != "example.com");
fail_if(jid.domainpart != "example.com"); fail_if(jid.resourcepart != "res");
fail_if(jid.resourcepart != "res"); fail_if(jid.to_string() != "user@example.com/res");
fail_if(jid.to_string() != "user@example.com/res");
} catch (Error e) {
fail_if_reached(@"Throws $(e.message)");
}
} }
} }

View file

@ -49,8 +49,6 @@ set(RESOURCE_LIST
icons/scalable/status/dino-status-chat.svg icons/scalable/status/dino-status-chat.svg
icons/scalable/status/dino-status-dnd.svg icons/scalable/status/dino-status-dnd.svg
icons/scalable/status/dino-status-online.svg icons/scalable/status/dino-status-online.svg
icons/scalable/status/dino-status-online-bright.svg
icons/scalable/status/dino-status-offline.svg
icons/scalable/status/dino-tick-symbolic.svg icons/scalable/status/dino-tick-symbolic.svg
icons/scalable/status/dino-video-off-symbolic.svg icons/scalable/status/dino-video-off-symbolic.svg
icons/scalable/status/dino-video-symbolic.svg icons/scalable/status/dino-video-symbolic.svg
@ -69,7 +67,6 @@ set(RESOURCE_LIST
conversation_list_titlebar_csd.ui conversation_list_titlebar_csd.ui
conversation_row.ui conversation_row.ui
conversation_view.ui conversation_view.ui
default_encryption_dialog.ui
file_default_widget.ui file_default_widget.ui
file_send_overlay.ui file_send_overlay.ui
global_search.ui global_search.ui

View file

@ -17,7 +17,7 @@
<object class="GtkButton" id="file_button"> <object class="GtkButton" id="file_button">
<property name="icon-name">mail-attachment-symbolic</property> <property name="icon-name">mail-attachment-symbolic</property>
<property name="margin-top">2</property> <property name="margin-top">2</property>
<property name="valign">center</property> <property name="valign">start</property>
<style> <style>
<class name="flat"/> <class name="flat"/>
<class name="dino-chatinput-button"/> <class name="dino-chatinput-button"/>
@ -54,7 +54,7 @@
<property name="icon-name">emoji-people-symbolic</property> <property name="icon-name">emoji-people-symbolic</property>
<property name="has-frame">False</property> <property name="has-frame">False</property>
<property name="margin-top">2</property> <property name="margin-top">2</property>
<property name="valign">center</property> <property name="valign">start</property>
<style> <style>
<class name="flat"/> <class name="flat"/>
<class name="dino-chatinput-button"/> <class name="dino-chatinput-button"/>
@ -67,21 +67,7 @@
<property name="icon-name">changes-allow-symbolic</property> <property name="icon-name">changes-allow-symbolic</property>
<property name="has-frame">False</property> <property name="has-frame">False</property>
<property name="margin-top">2</property> <property name="margin-top">2</property>
<property name="valign">center</property> <property name="valign">start</property>
<style>
<class name="flat"/>
<class name="dino-chatinput-button"/>
<class name="image-button"/>
</style>
</object>
</child>
<child>
<object class="GtkButton" id="send_button">
<property name="icon-name">mail-send-symbolic</property>
<property name="has-frame">False</property>
<property name="margin-top">2</property>
<property name="valign">center</property>
<property name="sensitive">false</property>
<style> <style>
<class name="flat"/> <class name="flat"/>
<class name="dino-chatinput-button"/> <class name="dino-chatinput-button"/>

View file

@ -38,12 +38,19 @@
</object> </object>
</child> </child>
<child> <child>
<object class="GtkLabel" id="time_label"> <object class="GtkRevealer" id="time_revealer">
<property name="hexpand">False</property> <property name="transition-type">slide-right</property>
<property name="xalign">1</property> <property name="transition-duration">50</property>
<attributes> <property name="reveal-child">True</property>
<attribute name="scale" value="0.7"/> <child>
</attributes> <object class="GtkLabel" id="time_label">
<property name="hexpand">False</property>
<property name="xalign">1</property>
<attributes>
<attribute name="scale" value="0.7"/>
</attributes>
</object>
</child>
</object> </object>
</child> </child>
</object> </object>
@ -81,26 +88,33 @@
</object> </object>
</child> </child>
<child> <child>
<object class="GtkBox"> <object class="GtkRevealer" id="top_row_revealer">
<property name="orientation">horizontal</property> <property name="transition-type">slide-right</property>
<property name="transition-duration">50</property>
<property name="reveal-child">True</property>
<property name="margin-start">15</property> <property name="margin-start">15</property>
<property name="spacing">6</property>
<child> <child>
<object class="GtkLabel" id="unread_count_label"> <object class="GtkBox">
<property name="vexpand">False</property> <property name="orientation">horizontal</property>
<property name="visible">False</property> <property name="spacing">6</property>
<property name="xalign">0.5</property> <child>
<attributes> <object class="GtkLabel" id="unread_count_label">
<attribute name="scale" value="0.6"/> <property name="vexpand">False</property>
<attribute name="weight" value="PANGO_WEIGHT_BOLD"/> <property name="visible">False</property>
</attributes> <property name="xalign">0.5</property>
</object> <attributes>
</child> <attribute name="scale" value="0.6"/>
<child> <attribute name="weight" value="PANGO_WEIGHT_BOLD"/>
<object class="GtkImage" id="pinned_image"> </attributes>
<property name="icon-name">view-pin-symbolic</property> </object>
<property name="pixel-size">12</property> </child>
<property name="visible">False</property> <child>
<object class="GtkImage" id="pinned_image">
<property name="icon-name">view-pin-symbolic</property>
<property name="pixel-size">12</property>
<property name="visible">False</property>
</object>
</child>
</object> </object>
</child> </child>
</object> </object>
@ -109,6 +123,47 @@
</child> </child>
</object> </object>
</child> </child>
<child>
<object class="GtkBox">
<property name="orientation">vertical</property>
<child>
<object class="GtkBox">
<property name="vexpand">True</property>
</object>
</child>
<child>
<object class="GtkRevealer" id="xbutton_revealer">
<property name="transition-type">slide-left</property>
<property name="transition-duration">100</property>
<property name="reveal-child">False</property>
<child>
<object class="GtkButton" id="x_button">
<property name="width-request">27</property>
<property name="height-request">27</property>
<property name="vexpand">False</property>
<property name="margin-start">5</property>
<style>
<class name="conversation_list_row_xbutton"/>
<class name="circular"/>
<class name="flat"/>
</style>
<child>
<object class="GtkImage">
<property name="icon-name">window-close-symbolic</property>
<property name="icon-size">1</property>
</object>
</child>
</object>
</child>
</object>
</child>
<child>
<object class="GtkBox">
<property name="vexpand">True</property>
</object>
</child>
</object>
</child>
</object> </object>
</child> </child>
</object> </object>

View file

@ -1,83 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<interface>
<requires lib="gtk" version="4.0"/>
<object class="GtkDialog" id="dialog">
<property name="can_focus">True</property>
<property name="modal">True</property>
<property name="default_width">320</property>
<property name="default_height">260</property>
<property name="resizable">False</property>
<child>
<object class="GtkBox">
<property name="can_focus">False</property>
<property name="orientation">vertical</property>
<property name="margin-start">20</property>
<property name="margin-end">20</property>
<property name="margin-top">20</property>
<property name="margin-bottom">20</property>
<child>
<object class="GtkBox">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="orientation">vertical</property>
<child>
<object class="GtkLabel" id="default_encryption_warning_label">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="label" translatable="yes">You are opening a new conversation without having set end-to-end encryption by default.
It is strongly recommended to enable it to prevent your messages from being read by third parties.
Please select an option to start this conversation with. Choosing one of the encryption methods will also set it as default in the global settings.
</property>
<property name="wrap">True</property>
</object>
</child>
<child>
<object class="GtkCheckButton" id="omemo">
<property name="label" translatable="yes">OMEMO (automatic setup)</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
</object>
</child>
<child>
<object class="GtkCheckButton" id="openpgp">
<property name="label" translatable="yes">OpenPGP (external setup required)</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
<property name="group">omemo</property>
</object>
</child>
<child>
<object class="GtkCheckButton" id="none">
<property name="label" translatable="yes">Unencrypted</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
<property name="group">omemo</property>
</object>
</child>
<child>
<object class="GtkButton" id="accept_button">
<property name="label" translatable="yes">Apply</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">True</property>
<property name="margin-top">20</property>
<style>
<class name="text-button"/>
<class name="suggested-action"/>
</style>
</object>
</child>
</object>
</child>
</object>
</child>
</object>
</interface>

View file

@ -11,13 +11,11 @@
<file>conversation_content_view/item_metadata_header.ui</file> <file>conversation_content_view/item_metadata_header.ui</file>
<file>conversation_content_view/view.ui</file> <file>conversation_content_view/view.ui</file>
<file>conversation_details.ui</file> <file>conversation_details.ui</file>
<file>conversation_details.css</file>
<file>conversation_item_widget.ui</file> <file>conversation_item_widget.ui</file>
<file>conversation_list_titlebar.ui</file> <file>conversation_list_titlebar.ui</file>
<file>conversation_list_titlebar_csd.ui</file> <file>conversation_list_titlebar_csd.ui</file>
<file>conversation_row.ui</file> <file>conversation_row.ui</file>
<file>conversation_view.ui</file> <file>conversation_view.ui</file>
<file>default_encryption_dialog.ui</file>
<file>dino-conversation-list-placeholder-arrow.svg</file> <file>dino-conversation-list-placeholder-arrow.svg</file>
<file>file_default_widget.ui</file> <file>file_default_widget.ui</file>
<file>file_send_overlay.ui</file> <file>file_send_overlay.ui</file>
@ -51,8 +49,6 @@
<file>icons/scalable/status/dino-status-chat.svg</file> <file>icons/scalable/status/dino-status-chat.svg</file>
<file>icons/scalable/status/dino-status-dnd.svg</file> <file>icons/scalable/status/dino-status-dnd.svg</file>
<file>icons/scalable/status/dino-status-online.svg</file> <file>icons/scalable/status/dino-status-online.svg</file>
<file>icons/scalable/status/dino-status-online-bright.svg</file>
<file>icons/scalable/status/dino-status-offline.svg</file>
<file>icons/scalable/status/dino-tick-symbolic.svg</file> <file>icons/scalable/status/dino-tick-symbolic.svg</file>
<file>icons/scalable/status/dino-video-off-symbolic.svg</file> <file>icons/scalable/status/dino-video-off-symbolic.svg</file>
<file>icons/scalable/status/dino-video-symbolic.svg</file> <file>icons/scalable/status/dino-video-symbolic.svg</file>

View file

@ -1,11 +0,0 @@
<?xml version="1.0"?>
<svg width="99.999997" height="99.999997" xmlns="http://www.w3.org/2000/svg" xmlns:svg="http://www.w3.org/2000/svg" version="1.1">
<g class="layer">
<title>Layer 1</title>
<g id="svg_1">
<circle cx="50" cy="51" fill="#ff0000" fill-rule="evenodd" id="svg_2" r="50" stroke="#ff0000"/>
<path d="m20,51l60,0" fill="#ff0000" id="svg_3" stroke="#ff0000" stroke-linecap="round" stroke-width="15"/>
</g>
</g>
</svg>

Before

Width:  |  Height:  |  Size: 444 B

View file

@ -1,10 +0,0 @@
<?xml version="1.0"?>
<svg width="99.999997" height="99.999997" xmlns="http://www.w3.org/2000/svg" xmlns:svg="http://www.w3.org/2000/svg" version="1.1">
<g class="layer">
<title>Layer 1</title>
<g id="svg_1">
<circle cx="50" cy="50" fill="#00ff00" fill-rule="evenodd" id="svg_2" r="50"/>
</g>
</g>
</svg>

Before

Width:  |  Height:  |  Size: 316 B

View file

@ -1,10 +1,6 @@
<?xml version="1.0"?> <?xml version="1.0" encoding="UTF-8"?>
<svg width="99.999997" height="99.999997" xmlns="http://www.w3.org/2000/svg" xmlns:svg="http://www.w3.org/2000/svg" version="1.1"> <svg width="28.222mm" height="28.222mm" version="1.1" viewBox="0 0 99.999997 99.999997" xmlns="http://www.w3.org/2000/svg">
<g transform="translate(-62.857 -678.08)">
<g class="layer"> <circle cx="112.86" cy="728.08" r="50" style="fill-rule:evenodd;fill:#81c784"/>
<title>Layer 1</title>
<g id="svg_1">
<circle cx="50" cy="50" fill="#00ff00" fill-rule="evenodd" id="svg_2" r="50"/>
</g>
</g> </g>
</svg> </svg>

Before

Width:  |  Height:  |  Size: 316 B

After

Width:  |  Height:  |  Size: 302 B

View file

@ -4,7 +4,7 @@
<launchable type="desktop-id">im.dino.Dino.desktop</launchable> <launchable type="desktop-id">im.dino.Dino.desktop</launchable>
<metadata_license>CC0-1.0</metadata_license> <metadata_license>CC0-1.0</metadata_license>
<project_license>GPL-3.0+</project_license> <project_license>GPL-3.0+</project_license>
<name>Dino+</name> <name>Dino</name>
<summary>Modern XMPP Chat Client</summary> <summary>Modern XMPP Chat Client</summary>
<summary xml:lang="zh_TW">現代化的 XMPP 用戶端聊天軟件</summary> <summary xml:lang="zh_TW">現代化的 XMPP 用戶端聊天軟件</summary>
<summary xml:lang="zh_CN">现代 XMPP 聊天客户端</summary> <summary xml:lang="zh_CN">现代 XMPP 聊天客户端</summary>

View file

@ -4,7 +4,7 @@
<launchable type="desktop-id">im.dino.Dino.desktop</launchable> <launchable type="desktop-id">im.dino.Dino.desktop</launchable>
<metadata_license>CC0-1.0</metadata_license> <metadata_license>CC0-1.0</metadata_license>
<project_license>GPL-3.0+</project_license> <project_license>GPL-3.0+</project_license>
<name>Dino+</name> <name>Dino</name>
<summary>Modern XMPP Chat Client</summary> <summary>Modern XMPP Chat Client</summary>
<description> <description>
<p>Dino is a modern open-source chat client for the desktop. It focuses on providing a clean and reliable Jabber/XMPP experience while having your privacy in mind.</p> <p>Dino is a modern open-source chat client for the desktop. It focuses on providing a clean and reliable Jabber/XMPP experience while having your privacy in mind.</p>

View file

@ -6,7 +6,7 @@
<property name="margin-start">7</property> <property name="margin-start">7</property>
<property name="margin-bottom">3</property> <property name="margin-bottom">3</property>
<property name="margin-end">7</property> <property name="margin-end">7</property>
<property name="column-spacing">5</property> <property name="column-spacing">10</property>
<child> <child>
<object class="DinoUiAvatarPicture" id="picture"> <object class="DinoUiAvatarPicture" id="picture">
<property name="height-request">30</property> <property name="height-request">30</property>
@ -15,8 +15,7 @@
</child> </child>
<child> <child>
<object class="GtkLabel" id="name_label"> <object class="GtkLabel" id="name_label">
<property name="width-chars">20</property> <property name="max_width_chars">1</property>
<property name="max-width-chars">40</property>
<property name="ellipsize">end</property> <property name="ellipsize">end</property>
<property name="hexpand">1</property> <property name="hexpand">1</property>
<property name="xalign">0</property> <property name="xalign">0</property>

View file

@ -7,49 +7,6 @@
<property name="search-enabled">False</property> <property name="search-enabled">False</property>
<child> <child>
<object class="AdwPreferencesPage"> <object class="AdwPreferencesPage">
<child>
<object class="AdwPreferencesGroup">
<child>
<object class="AdwActionRow">
<property name="title" translatable="yes">Default encryption for 1-on-1 chats</property>
<child>
<object class="GtkBox" id="default_encryption_box">
<child>
<object class="GtkCheckButton" id="encryption_radio_undecided">
<property name="label" translatable="yes">Ask</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
<property name="group">encryption_radio_undecided</property>
</object>
</child>
<child>
<object class="GtkCheckButton" id="encryption_radio_omemo">
<property name="label" translatable="yes">OMEMO</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
<property name="group">encryption_radio_undecided</property>
</object>
</child>
<child>
<object class="GtkCheckButton" id="encryption_radio_openpgp">
<property name="label" translatable="yes">OpenPGP</property>
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="receives_default">False</property>
<property name="active">False</property>
<property name="group">encryption_radio_undecided</property>
</object>
</child>
</object>
</child>
</object>
</child>
</object>
</child>
<child> <child>
<object class="AdwPreferencesGroup"> <object class="AdwPreferencesGroup">
<child> <child>
@ -111,51 +68,6 @@
</child> </child>
</object> </object>
</child> </child>
<child>
<object class="AdwPreferencesGroup">
<child>
<object class="AdwActionRow">
<property name="title" translatable="yes">_Display send button</property>
<property name="use-underline">True</property>
<property name="activatable-widget">send_button_switch</property>
<child type="suffix">
<object class="GtkSwitch" id="send_button_switch">
<property name="valign">center</property>
</object>
</child>
</object>
</child>
<child>
<object class="AdwActionRow">
<property name="title" translatable="yes">_Use Enter key to start a new line</property>
<property name="subtitle" translatable="yes">If disabled, use Shift+Enter to start a new line</property>
<property name="use-underline">True</property>
<property name="activatable-widget">enter_newline_switch</property>
<child type="suffix">
<object class="GtkSwitch" id="enter_newline_switch">
<property name="valign">center</property>
</object>
</child>
</object>
</child>
</object>
</child>
<child>
<object class="AdwPreferencesGroup">
<child>
<object class="AdwActionRow">
<property name="title" translatable="yes">_Dark theme</property>
<property name="use-underline">True</property>
<property name="activatable-widget">dark_theme</property>
<child type="suffix">
<object class="GtkSwitch" id="dark_theme">
<property name="valign">center</property>
</object>
</child>
</object>
</child>
</object>
</child>
</object> </object>
</child> </child>
</template> </template>

View file

@ -49,11 +49,7 @@ window.dino-main .dino-conversation .message-box.highlight:not(.highlight-once)
} }
window.dino-main .dino-conversation .message-box { window.dino-main .dino-conversation .message-box {
padding: 3px 15px 3px 15px; padding: 6px 15px 6px 15px;
}
window.dino-main .dino-conversation .has-skeleton {
margin-top: 10px;
} }
window.dino-main .dino-conversation .message-box:not(.has-skeleton) { window.dino-main .dino-conversation .message-box:not(.has-skeleton) {

View file

@ -104,8 +104,6 @@ sources += gnome.compile_resources(
c_args = [ c_args = [
'-DG_LOG_DOMAIN="dino"', '-DG_LOG_DOMAIN="dino"',
'-DGETTEXT_PACKAGE="dino"', '-DGETTEXT_PACKAGE="dino"',
'-D_POSIX_C_SOURCE=1',
'-D_FILE_OFFSET_BITS=64',
'-DLOCALE_INSTALL_DIR="@0@"'.format(get_option('prefix') / get_option('localedir')), '-DLOCALE_INSTALL_DIR="@0@"'.format(get_option('prefix') / get_option('localedir')),
] ]
vala_args = [ vala_args = [
@ -120,15 +118,7 @@ endif
if dep_gtk4.version() == 'unknown' or dep_gtk4.version().version_compare('>=4.8') if dep_gtk4.version() == 'unknown' or dep_gtk4.version().version_compare('>=4.8')
vala_args += ['-D', 'GTK_4_8'] vala_args += ['-D', 'GTK_4_8']
endif endif
exe_dino = executable('dino', sources, c_args: c_args, vala_args: vala_args, dependencies: dependencies, install: true)
link_args = []
if host_machine.system() == 'windows'
link_args += ['-Wl,--export-all-symbols']
windows = import('windows')
sources += windows.compile_resources('dino-info.rc')
endif
exe_dino = executable('dino', sources, c_args: c_args, vala_args: vala_args, dependencies: dependencies, install: true, link_args: link_args, win_subsystem: 'windows')
install_data('data/icons/scalable/apps/im.dino.Dino-symbolic.svg', install_dir: get_option('datadir') / 'hicolor/symbolic/apps') install_data('data/icons/scalable/apps/im.dino.Dino-symbolic.svg', install_dir: get_option('datadir') / 'hicolor/symbolic/apps')
install_data('data/icons/scalable/apps/im.dino.Dino.svg', install_dir: get_option('datadir') / 'hicolor/scalable/apps') install_data('data/icons/scalable/apps/im.dino.Dino.svg', install_dir: get_option('datadir') / 'hicolor/scalable/apps')

View file

@ -533,10 +533,12 @@ msgid "Local Settings"
msgstr "" msgstr ""
#: main/src/ui/contact_details/settings_provider.vala:28 #: main/src/ui/contact_details/settings_provider.vala:28
#: main/data/settings_dialog.ui:23
msgid "Send typing notifications" msgid "Send typing notifications"
msgstr "" msgstr ""
#: main/src/ui/contact_details/settings_provider.vala:33 #: main/src/ui/contact_details/settings_provider.vala:33
#: main/data/settings_dialog.ui:32
msgid "Send read receipts" msgid "Send read receipts"
msgstr "" msgstr ""
@ -1010,48 +1012,16 @@ msgstr ""
msgid "Add Contact" msgid "Add Contact"
msgstr "" msgstr ""
#: main/data/settings_dialog.ui:14 #: main/data/settings_dialog.ui:41
msgid "Default encryption for 1-on-1 chats"
msgstr ""
#: main/data/settings_dialog.ui:19
msgid "Ask"
msgstr ""
#: main/data/settings_dialog.ui:57
msgid "Send _Typing Notifications"
msgstr ""
#: main/data/settings_dialog.ui:85
msgid "_Notifications"
msgstr ""
#: main/data/settings_dialog.ui:69
msgid "Send _Read Receipts"
msgstr ""
#: main/data/settings_dialog.ui:86
msgid "Notify when a new message arrives" msgid "Notify when a new message arrives"
msgstr "" msgstr ""
#: main/data/settings_dialog.ui:102 #: main/data/settings_dialog.ui:50
msgid "_Convert Smileys to Emoji" msgid "Convert smileys to emojis"
msgstr "" msgstr ""
#: main/data/settings_dialog.ui:118 #: main/data/settings_dialog.ui:59
msgid "_Display send button" msgid "Check spelling"
msgstr ""
#: main/data/settings_dialog.ui:130
msgid "_Use Enter key to start a new line"
msgstr ""
#: main/data/settings_dialog.ui:131
msgid "If disabled, use Shift+Enter to start a new line"
msgstr ""
#: main/data/settings_dialog.ui:147
msgid "_Dark theme"
msgstr "" msgstr ""
#: main/data/im.dino.Dino.appdata.xml.in:7 #: main/data/im.dino.Dino.appdata.xml.in:7

View file

@ -47,7 +47,6 @@ msgstr "доставка не удалась"
#: main/src/ui/chat_input/encryption_button.vala:26 #: main/src/ui/chat_input/encryption_button.vala:26
#: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:163 #: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:163
#: main/data/menu_encryption.ui:14 #: main/data/menu_encryption.ui:14
#: main/data/default_encryption_dialog.ui:57
msgid "Unencrypted" msgid "Unencrypted"
msgstr "Не зашифровано" msgstr "Не зашифровано"
@ -563,7 +562,7 @@ msgstr "Уведомления"
#: main/src/ui/contact_details/settings_provider.vala:55 #: main/src/ui/contact_details/settings_provider.vala:55
msgid "Pin conversation" msgid "Pin conversation"
msgstr "Закрепить беседу" msgstr ""
#: main/src/ui/contact_details/settings_provider.vala:55 #: main/src/ui/contact_details/settings_provider.vala:55
msgid "Pins the conversation to the top of the conversation list" msgid "Pins the conversation to the top of the conversation list"
@ -826,16 +825,16 @@ msgstr "Прочтите %s, чтобы узнать о процессе авт
#: main/src/ui/conversation_content_view/message_widget.vala:213 #: main/src/ui/conversation_content_view/message_widget.vala:213
msgid "Edit message" msgid "Edit message"
msgstr "Редактировать сообщение" msgstr ""
#: main/src/ui/conversation_content_view/reactions_widget.vala:102 #: main/src/ui/conversation_content_view/reactions_widget.vala:102
msgid "You" msgid "You"
msgstr "Вы" msgstr ""
#: main/src/ui/conversation_content_view/reactions_widget.vala:128 #: main/src/ui/conversation_content_view/reactions_widget.vala:128
#: main/src/ui/conversation_content_view/item_actions.vala:8 #: main/src/ui/conversation_content_view/item_actions.vala:8
msgid "Add reaction" msgid "Add reaction"
msgstr "Добавить реакцию" msgstr ""
#: main/src/ui/conversation_content_view/file_image_widget.vala:53 #: main/src/ui/conversation_content_view/file_image_widget.vala:53
#: main/src/ui/conversation_content_view/file_default_widget.vala:57 #: main/src/ui/conversation_content_view/file_default_widget.vala:57
@ -947,7 +946,7 @@ msgstr "несколько секунд"
#: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:191 #: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:191
msgid "Delivered" msgid "Delivered"
msgstr "Доставленo" msgstr ""
#: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:195 #: main/src/ui/conversation_content_view/conversation_item_skeleton.vala:195
msgid "Read" msgid "Read"
@ -959,15 +958,15 @@ msgstr "Этот контакт хочет добавить вас в свой
#: main/src/ui/conversation_content_view/item_actions.vala:21 #: main/src/ui/conversation_content_view/item_actions.vala:21
msgid "This conversation does not support reactions." msgid "This conversation does not support reactions."
msgstr "Этот разговор не поддерживает реакции." msgstr ""
#: main/src/ui/conversation_content_view/item_actions.vala:33 #: main/src/ui/conversation_content_view/item_actions.vala:33
msgid "Reply" msgid "Reply"
msgstr "Ответить" msgstr ""
#: main/src/ui/conversation_content_view/item_actions.vala:43 #: main/src/ui/conversation_content_view/item_actions.vala:43
msgid "This conversation does not support replies." msgid "This conversation does not support replies."
msgstr "Этот разговор не поддерживает ответы" msgstr ""
#: main/src/ui/conversation_content_view/file_default_widget.vala:64 #: main/src/ui/conversation_content_view/file_default_widget.vala:64
#, c-format #, c-format
@ -1006,7 +1005,7 @@ msgstr "Тут пустовато"
#: main/data/unified_main_content.ui:47 #: main/data/unified_main_content.ui:47
msgid "Click + to start a chat or join a channel" msgid "Click + to start a chat or join a channel"
msgstr "Нажмите +, чтобы начать чат или присоединиться к каналу" msgstr ""
#: main/data/add_conversation/conference_details_fragment.ui:20 #: main/data/add_conversation/conference_details_fragment.ui:20
#: main/data/add_conversation/add_groupchat_dialog.ui:40 #: main/data/add_conversation/add_groupchat_dialog.ui:40
@ -1028,49 +1027,17 @@ msgstr "Псевдоним"
msgid "Add Contact" msgid "Add Contact"
msgstr "Добавить контакт" msgstr "Добавить контакт"
#: main/data/settings_dialog.ui:14 #: main/data/settings_dialog.ui:41
msgid "Default encryption for 1-on-1 chats"
msgstr "Шифрование по умолчанию для 1-на-1 чатов"
#: main/data/settings_dialog.ui:19
msgid "Ask"
msgstr "Спрашивать"
#: main/data/settings_dialog.ui:57
msgid "Send _Typing Notifications"
msgstr "Отправлять уведомления при наборе сообщения"
#: main/data/settings_dialog.ui:85
msgid "_Notifications"
msgstr "Уведомления"
#: main/data/settings_dialog.ui:69
msgid "Send _Read Receipts"
msgstr "Отправлять уведомления о прочтении"
#: main/data/settings_dialog.ui:86
msgid "Notify when a new message arrives" msgid "Notify when a new message arrives"
msgstr "Уведомлять о новых сообщениях" msgstr "Уведомлять о новых сообщениях"
#: main/data/settings_dialog.ui:102 #: main/data/settings_dialog.ui:50
msgid "_Convert Smileys to Emoji" msgid "Convert smileys to emojis"
msgstr "Превращать смайлы в эмодзи" msgstr "Превращать смайлы в эмодзи"
#: main/data/settings_dialog.ui:118 #: main/data/settings_dialog.ui:59
msgid "_Display send button" msgid "Check spelling"
msgstr "Показывать кнопку отправки" msgstr "Проверка орфографии"
#: main/data/settings_dialog.ui:130
msgid "_Use Enter key to start a new line"
msgstr "Использовать Enter для вставки новой строки"
#: main/data/settings_dialog.ui:131
msgid "If disabled, use Shift+Enter to start a new line"
msgstr "Если опция недоступна, используйте Shift+Enter для вставки новой строки"
#: main/data/settings_dialog.ui:147
msgid "_Dark theme"
msgstr "Тёмная тема"
#: main/data/im.dino.Dino.appdata.xml.in:7 #: main/data/im.dino.Dino.appdata.xml.in:7
msgid "Modern XMPP Chat Client" msgid "Modern XMPP Chat Client"
@ -1217,28 +1184,6 @@ msgstr "Новый пароль"
msgid "Confirm new password" msgid "Confirm new password"
msgstr "Подтверждение пароля" msgstr "Подтверждение пароля"
#: main/data/default_encryption_dialog.ui :27
msgid ""
"You are opening a new conversation without having set end-to-end encryption by default.\n\n"
"It is strongly recommended to enable it to prevent your messages from being read by third parties.\n\n"
"Please select an option to start this conversation with. Choosing one of the encryption methods will also set it as default in the global settings.\n"
msgstr ""
"Вы начинаете новый диалог без установления настроек сквозного шифрования по умолчанию.\n\n"
"Крайне рекомендуется включить его, чтобы исключить перехват ваших сообщений злоумышленниками.\n\n"
"Пожалуйста, выберите с какими настройками шифрования начать этот диалог. Выбирая одну из них, вы установите эту же настройку глобально по умолчанию для других диалогов.\n"
#: main/data/default_encryption_dialog.ui :38
msgid "OMEMO (automatic setup)"
msgstr "OMEMO (настроится автоматически)"
#: main/data/default_encryption_dialog.ui :47
msgid "OpenPGP (external setup required)"
msgstr "OpenPGP (потребует ручной настройки)"
#: main/data/default_encryption_dialog.ui :67
msgid "Apply"
msgstr "Применить"
#~ msgid "Click here to start a conversation or join a channel." #~ msgid "Click here to start a conversation or join a channel."
#~ msgstr "Нажмите здесь, чтобы начать беседу или присоединиться к каналу." #~ msgstr "Нажмите здесь, чтобы начать беседу или присоединиться к каналу."

View file

@ -34,13 +34,7 @@ protected class RosterList {
foreach (ulong handler_id in handler_ids) stream_interactor.get_module(RosterManager.IDENTITY).disconnect(handler_id); foreach (ulong handler_id in handler_ids) stream_interactor.get_module(RosterManager.IDENTITY).disconnect(handler_id);
}); });
foreach (Account a in accounts) { foreach (Account a in accounts) fetch_roster_items(a);
ListRow own_account_row = new ListRow.from_jid(stream_interactor, a.bare_jid, a, accounts.size > 1);
ListBoxRow own_account_lbrow = new ListBoxRow() { child = own_account_row };
list_box.append(own_account_lbrow);
fetch_roster_items(a);
}
} }
private void on_removed_roster_item(Account account, Jid jid, Roster.Item roster_item) { private void on_removed_roster_item(Account account, Jid jid, Roster.Item roster_item) {
@ -53,7 +47,7 @@ protected class RosterList {
private void on_updated_roster_item(Account account, Jid jid, Roster.Item roster_item) { private void on_updated_roster_item(Account account, Jid jid, Roster.Item roster_item) {
on_removed_roster_item(account, jid, roster_item); on_removed_roster_item(account, jid, roster_item);
ListRow row = new ListRow.from_jid(stream_interactor, roster_item.jid, account, accounts.size > 1); ListRow row = new ListRow.from_jid(stream_interactor, roster_item.jid, account, accounts.size > 1);
ListBoxRow list_box_row = new ListBoxRow() { child = row }; ListBoxRow list_box_row = new ListBoxRow() { child=row };
rows[account][jid] = list_box_row; rows[account][jid] = list_box_row;
list_box.append(list_box_row); list_box.append(list_box_row);
list_box.invalidate_sort(); list_box.invalidate_sort();

View file

@ -29,10 +29,7 @@ public class Dino.Ui.Application : Adw.Application, Dino.Application {
public Application() throws Error { public Application() throws Error {
Object(application_id: "im.dino.Dino", flags: ApplicationFlags.HANDLES_OPEN); Object(application_id: "im.dino.Dino", flags: ApplicationFlags.HANDLES_OPEN);
init();
var style_manager = Adw.StyleManager.get_default();
bool system_dark_theme = style_manager.system_supports_color_schemes && style_manager.dark;
init(system_dark_theme);
Environment.set_application_name("Dino"); Environment.set_application_name("Dino");
Window.set_default_icon_name("im.dino.Dino"); Window.set_default_icon_name("im.dino.Dino");
@ -71,19 +68,6 @@ public class Dino.Ui.Application : Adw.Application, Dino.Application {
} }
} }
}); });
settings.dark_theme_update.connect((is_dark) => {
var manager = Adw.StyleManager.get_default();
if (is_dark != manager.dark) {
if (is_dark) {
manager.set_color_scheme(Adw.ColorScheme.FORCE_DARK);
} else {
manager.set_color_scheme(Adw.ColorScheme.FORCE_LIGHT);
}
}
});
settings.dark_theme_update(settings.dark_theme); // Change theme at startup.
}); });
activate.connect(() => { activate.connect(() => {
@ -275,21 +259,21 @@ public class Dino.Ui.Application : Adw.Application, Dino.Application {
#if Adw_1_2 #if Adw_1_2
Adw.AboutWindow about_window = new Adw.AboutWindow(); Adw.AboutWindow about_window = new Adw.AboutWindow();
about_window.application_icon = "im.dino.Dino"; about_window.application_icon = "im.dino.Dino";
about_window.application_name = "Dino+"; about_window.application_name = "Dino";
about_window.issue_url = "https://github.com/mxlgv/dino/issues"; about_window.issue_url = "https://github.com/dino/dino/issues";
#else #else
Gtk.AboutDialog about_window = new Gtk.AboutDialog(); Gtk.AboutDialog about_window = new Gtk.AboutDialog();
about_window.logo_icon_name = "im.dino.Dino"; about_window.logo_icon_name = "im.dino.Dino";
about_window.program_name = "Dino+"; about_window.program_name = "Dino";
about_window.website_label = "dino.im"; about_window.website_label = "dino.im";
#endif #endif
about_window.destroy_with_parent = true; about_window.destroy_with_parent = true;
about_window.transient_for = window; about_window.transient_for = window;
about_window.modal = true; about_window.modal = true;
about_window.title = _("About Dino+"); about_window.title = _("About Dino");
about_window.version = version; about_window.version = version;
about_window.website = "https://github.com/mxlgv/dino/"; about_window.website = "https://dino.im/";
about_window.copyright = "Copyright © 2016-2024 - Dino and Dino+ Teams"; about_window.copyright = "Copyright © 2016-2023 - Dino Team";
about_window.license_type = License.GPL_3_0; about_window.license_type = License.GPL_3_0;
if (!use_csd()) { if (!use_csd()) {

View file

@ -103,7 +103,7 @@ public class ChatInputController : Object {
private void on_encryption_changed(Encryption encryption) { private void on_encryption_changed(Encryption encryption) {
reset_input_field_status(); reset_input_field_status();
if (encryption == Encryption.NONE || encryption == Encryption.UNKNOWN) return; if (encryption == Encryption.NONE) return;
Application app = GLib.Application.get_default() as Application; Application app = GLib.Application.get_default() as Application;
var encryption_entry = app.plugin_registry.encryption_list_entries[encryption]; var encryption_entry = app.plugin_registry.encryption_list_entries[encryption];

View file

@ -101,10 +101,7 @@ public class ChatTextView : Box {
return true; return true;
} }
Dino.Entities.Settings settings = Dino.Application.get_default().settings; if ((state & ModifierType.SHIFT_MASK) > 0) {
if ((state & ModifierType.SHIFT_MASK) > 0
|| settings.enter_newline) {
text_view.buffer.insert_at_cursor("\n", 1); text_view.buffer.insert_at_cursor("\n", 1);
} else if (text_view.buffer.text.strip() != "") { } else if (text_view.buffer.text.strip() != "") {
send_text(); send_text();

View file

@ -107,7 +107,7 @@ public class OccupantsTabCompletor {
Gee.List<string> ret = generate_completions_from_messages(prefix); Gee.List<string> ret = generate_completions_from_messages(prefix);
// Then, suggest other nicks in alphabetical order // Then, suggest other nicks in alphabetical order
Gee.List<Jid>? occupants = stream_interactor.get_module(MucManager.IDENTITY).get_other_members(conversation.counterpart, conversation.account); Gee.List<Jid>? occupants = stream_interactor.get_module(MucManager.IDENTITY).get_other_occupants(conversation.counterpart, conversation.account);
Gee.List<string> filtered_occupants = new ArrayList<string>(); Gee.List<string> filtered_occupants = new ArrayList<string>();
if (occupants != null) { if (occupants != null) {
foreach (Jid jid in occupants) { foreach (Jid jid in occupants) {

View file

@ -27,7 +27,6 @@ public class View : Box {
[GtkChild] public unowned MenuButton encryption_button; [GtkChild] public unowned MenuButton encryption_button;
[GtkChild] public unowned Separator file_separator; [GtkChild] public unowned Separator file_separator;
[GtkChild] public unowned Label chat_input_status; [GtkChild] public unowned Label chat_input_status;
[GtkChild] public unowned Button send_button;
public EncryptionButton encryption_widget; public EncryptionButton encryption_widget;
@ -48,27 +47,6 @@ public class View : Box {
Util.force_css(frame, "* { border-radius: 3px; }"); Util.force_css(frame, "* { border-radius: 3px; }");
Dino.Entities.Settings settings = Dino.Application.get_default().settings;
chat_text_view.text_view.buffer.changed.connect(() => {
if (chat_text_view.text_view.buffer.text != "") {
send_button.sensitive = true;
}
else {
send_button.sensitive = false;
}
});
send_button.visible = settings.send_button;
settings.send_button_update.connect((visible) => {
send_button.visible = visible;
});
send_button.clicked.connect(() => {
chat_text_view.send_text();
});
return this; return this;
} }

View file

@ -28,21 +28,9 @@ public class HistoryProvider : Plugins.ContactDetailsProvider, Object {
entity_info.has_feature.begin(conversation.account, conversation.counterpart, Xmpp.MessageArchiveManagement.NS_URI, (_, res) => { entity_info.has_feature.begin(conversation.account, conversation.counterpart, Xmpp.MessageArchiveManagement.NS_URI, (_, res) => {
bool can_do_mam = entity_info.has_feature.end(res); bool can_do_mam = entity_info.has_feature.end(res);
if (can_do_mam) { if (can_do_mam) {
Button resync_button = new Button.with_label(RESYNC_LABEL); Button resync_button = new Button.with_label(RESYNC_LABEL);
Stack resync_stack = new Stack(); contact_details.add("Permissions", RESYNC_DESC_LABEL, "", resync_button);
Gtk.Spinner spinner = new Gtk.Spinner();
resync_stack.visible = true;
contact_details.add("Permissions", RESYNC_DESC_LABEL, "", resync_stack);
resync_stack.add_child(spinner);
resync_stack.add_child(resync_button);
resync_stack.set_visible_child(resync_button);
resync_button.clicked.connect(() => { resync_button.clicked.connect(() => {
resync_stack.set_visible_child(spinner);
spinner.start();
if (!sync_cancellables.has_key(conversation.account)) { if (!sync_cancellables.has_key(conversation.account)) {
sync_cancellables[conversation.account] = new HashMap<Jid, Cancellable>(); sync_cancellables[conversation.account] = new HashMap<Jid, Cancellable>();
} }
@ -52,8 +40,6 @@ public class HistoryProvider : Plugins.ContactDetailsProvider, Object {
var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync; var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync;
history_sync.fetch_history.begin(conversation.account, conversation.counterpart.bare_jid, sync_cancellables[conversation.account][conversation.counterpart.bare_jid], (_, res) => { history_sync.fetch_history.begin(conversation.account, conversation.counterpart.bare_jid, sync_cancellables[conversation.account][conversation.counterpart.bare_jid], (_, res) => {
history_sync.fetch_everything.end(res); history_sync.fetch_everything.end(res);
spinner.stop();
resync_stack.set_visible_child(resync_button);
sync_cancellables[conversation.account].unset(conversation.counterpart.bare_jid); sync_cancellables[conversation.account].unset(conversation.counterpart.bare_jid);
}); });
} }

View file

@ -41,9 +41,9 @@ public class ContentProvider : ContentItemCollection, Object {
return ret; return ret;
} }
public Gee.List<ContentMetaItem> populate_before(Conversation conversation, ContentItem before_item, int n, bool request_from_server = true) { public Gee.List<ContentMetaItem> populate_before(Conversation conversation, ContentItem before_item, int n) {
Gee.List<ContentMetaItem> ret = new ArrayList<ContentMetaItem>(); Gee.List<ContentMetaItem> ret = new ArrayList<ContentMetaItem>();
Gee.List<ContentItem> items = stream_interactor.get_module(ContentItemStore.IDENTITY).get_before(conversation, before_item, n, request_from_server); Gee.List<ContentItem> items = stream_interactor.get_module(ContentItemStore.IDENTITY).get_before(conversation, before_item, n);
foreach (ContentItem item in items) { foreach (ContentItem item in items) {
ret.add(create_content_meta_item(item)); ret.add(create_content_meta_item(item));
} }

View file

@ -44,12 +44,6 @@ public class ConversationView : Widget, Plugins.ConversationItemCollection, Plug
ContentMetaItem? current_meta_item = null; ContentMetaItem? current_meta_item = null;
double last_y = -1; double last_y = -1;
private void on_history_loaded(Conversation conversation, ContentItem item, int count) {
// We received new messages from the server
// Load them from the DB, but do not make new request to the server
load_earlier_messages(false);
}
construct { construct {
this.layout_manager = new BinLayout(); this.layout_manager = new BinLayout();
@ -84,9 +78,6 @@ public class ConversationView : Widget, Plugins.ConversationItemCollection, Plug
scrolled.vadjustment.notify["page-size"].connect(on_upper_notify); scrolled.vadjustment.notify["page-size"].connect(on_upper_notify);
scrolled.vadjustment.notify["value"].connect(on_value_notify); scrolled.vadjustment.notify["value"].connect(on_value_notify);
var content_item_store = stream_interactor.get_module(ContentItemStore.IDENTITY);
content_item_store.history_loaded.connect(on_history_loaded);
content_populator = new ContentProvider(stream_interactor); content_populator = new ContentProvider(stream_interactor);
subscription_notification = new SubscriptionNotitication(stream_interactor); subscription_notification = new SubscriptionNotitication(stream_interactor);
@ -561,42 +552,17 @@ public class ConversationView : Widget, Plugins.ConversationItemCollection, Plug
} }
} }
private void load_earlier_messages(bool request_from_server = true) { private void load_earlier_messages() {
was_value = scrolled.vadjustment.value; was_value = scrolled.vadjustment.value;
debug("loading earlier messages"); if (!reloading_mutex.trylock()) return;
if (content_items.size > 0) {
if (!reloading_mutex.trylock()) { Gee.List<ContentMetaItem> items = content_populator.populate_before(conversation, ((ContentMetaItem) content_items.first()).content_item, 20);
return;
}
if (content_items.size == 0) {
// List is empty, try load local data
Gee.List<ContentMetaItem> items = content_populator.populate_latest(conversation, 20);
debug("inserting new messages, size: %d", items.size);
foreach (ContentMetaItem item in items) {
do_insert_item(item);
}
if (request_from_server) {
// If the list is empty also try to load data from the server
var history_sync = stream_interactor.get_module(MessageProcessor.IDENTITY).history_sync;
history_sync.fetch_conversation_data.begin(conversation, new DateTime.now(), (_, res) => {
history_sync.fetch_conversation_data.end(res);
// Request finished, reload messages
load_earlier_messages(false);
});
}
} else if (content_items.size > 0) {
// List is not empty, fetch data before the latest available message
Gee.List<ContentMetaItem> items = content_populator.populate_before(conversation, ((ContentMetaItem) content_items.first()).content_item, 20, request_from_server);
debug("inserting new messages, size: %d", items.size);
foreach (ContentMetaItem item in items) { foreach (ContentMetaItem item in items) {
do_insert_item(item); do_insert_item(item);
} }
} else {
reloading_mutex.unlock();
} }
reloading_mutex.unlock();
} }
private void load_later_messages() { private void load_later_messages() {

View file

@ -39,8 +39,7 @@ public class FileDefaultWidget : Box {
}); });
} }
public void update_file_info(string? mime_type, uint64 transferred_bytes, public void update_file_info(string? mime_type, FileTransfer.State state, long size) {
bool direction, FileTransfer.State state, long size) {
this.state = state; this.state = state;
spinner.stop(); // A hidden spinning spinner still uses CPU. Deactivate asap spinner.stop(); // A hidden spinning spinner still uses CPU. Deactivate asap
@ -50,7 +49,7 @@ public class FileDefaultWidget : Box {
switch (state) { switch (state) {
case FileTransfer.State.COMPLETE: case FileTransfer.State.COMPLETE:
mime_label.label = _("%s offered: %s").printf(mime_description, get_size_string(size)); mime_label.label = mime_description;
image_stack.set_visible_child_name("content_type_image"); image_stack.set_visible_child_name("content_type_image");
// Create a menu // Create a menu
@ -62,17 +61,7 @@ public class FileDefaultWidget : Box {
popover_menu.closed.connect(on_pointer_left); popover_menu.closed.connect(on_pointer_left);
break; break;
case FileTransfer.State.IN_PROGRESS: case FileTransfer.State.IN_PROGRESS:
uint progress = 0; mime_label.label = _("Downloading %s…").printf(get_size_string(size));
if (size > 0)
progress = (uint)((transferred_bytes * (uint64)100) / (uint64)size);
if (direction == FileTransfer.DIRECTION_SENT) {
mime_label.label = _("Uploading %s (%u%%)…").printf(get_size_string(size), progress);
}
else {
mime_label.label = _("Downloading %s (%u%%)…").printf(get_size_string(size), progress);
}
spinner.start(); spinner.start();
image_stack.set_visible_child_name("spinner"); image_stack.set_visible_child_name("spinner");

View file

@ -73,9 +73,6 @@ public class FileImageWidget : Box {
image_overlay_toolbar.visible = false; image_overlay_toolbar.visible = false;
}); });
// Set tooltip to display the file name on hover
image.set_tooltip_text(file_name);
this.append(overlay); this.append(overlay);
} }
} }

View file

@ -89,9 +89,7 @@ public class FileWidget : SizeRequestBox {
} }
private async void update_widget() { private async void update_widget() {
if (show_image() && state != State.IMAGE if (show_image() && state != State.IMAGE) {
&& file_transfer.state == FileTransfer.State.COMPLETE
&& file_transfer.get_file().query_exists()) {
var content_bak = content; var content_bak = content;
FileImageWidget file_image_widget = null; FileImageWidget file_image_widget = null;
@ -110,8 +108,7 @@ public class FileWidget : SizeRequestBox {
} catch (Error e) { } } catch (Error e) { }
} }
if (state != State.DEFAULT || if (!show_image() && state != State.DEFAULT) {
(file_transfer.state == FileTransfer.State.COMPLETE && !file_transfer.get_file().query_exists())) {
if (content != null) this.remove(content); if (content != null) this.remove(content);
FileDefaultWidget default_file_widget = new FileDefaultWidget(); FileDefaultWidget default_file_widget = new FileDefaultWidget();
default_widget_controller = new FileDefaultWidgetController(default_file_widget); default_widget_controller = new FileDefaultWidgetController(default_file_widget);
@ -178,16 +175,10 @@ public class FileWidgetController : Object {
} }
private void open_file() { private void open_file() {
if (file_transfer.get_file().query_exists()) { try{
try { Dino.Util.launch_default_for_uri(file_transfer.get_file().get_uri());
Dino.Util.launch_default_for_uri(file_transfer.get_file().get_uri()); } catch (Error err) {
} catch (Error err) { warning("Failed to open %s - %s", file_transfer.get_file().get_uri(), err.message);
warning("Failed to open %s - %s", file_transfer.get_file().get_uri(), err.message);
}
} else {
warning("File %s does not exist", file_transfer.get_file().get_uri());
file_transfer.state = FileTransfer.State.NOT_STARTED;
widget.activate_action("file.download", null);
} }
} }
@ -224,7 +215,6 @@ public class FileDefaultWidgetController : Object {
private FileTransfer? file_transfer; private FileTransfer? file_transfer;
public string file_transfer_state { get; set; } public string file_transfer_state { get; set; }
public string file_transfer_mime_type { get; set; } public string file_transfer_mime_type { get; set; }
public uint64 file_transfer_transferred_bytes { get; set; }
private FileTransfer.State state; private FileTransfer.State state;
@ -235,7 +225,6 @@ public class FileDefaultWidgetController : Object {
this.notify["file-transfer-state"].connect(update_file_info); this.notify["file-transfer-state"].connect(update_file_info);
this.notify["file-transfer-mime-type"].connect(update_file_info); this.notify["file-transfer-mime-type"].connect(update_file_info);
this.notify["file-transfer-transferred-bytes"].connect(update_file_info);
} }
public void set_file_transfer(FileTransfer file_transfer) { public void set_file_transfer(FileTransfer file_transfer) {
@ -245,19 +234,13 @@ public class FileDefaultWidgetController : Object {
file_transfer.bind_property("state", this, "file-transfer-state"); file_transfer.bind_property("state", this, "file-transfer-state");
file_transfer.bind_property("mime-type", this, "file-transfer-mime-type"); file_transfer.bind_property("mime-type", this, "file-transfer-mime-type");
file_transfer.bind_property("transferred-bytes", this, "file-transfer-transferred-bytes");
update_file_info(); update_file_info();
} }
private void update_file_info() { private void update_file_info() {
state = file_transfer.state; state = file_transfer.state;
if (state == FileTransfer.State.COMPLETE && !file_transfer.get_file().query_exists()) { widget.update_file_info(file_transfer.mime_type, file_transfer.state, file_transfer.size);
state = FileTransfer.State.NOT_STARTED;
file_transfer.state = FileTransfer.State.NOT_STARTED;
}
widget.update_file_info(file_transfer.mime_type, file_transfer.transferred_bytes,
file_transfer.direction, state, file_transfer.size);
} }
private void on_clicked() { private void on_clicked() {
@ -266,11 +249,10 @@ public class FileDefaultWidgetController : Object {
widget.activate_action("file.open", null); widget.activate_action("file.open", null);
break; break;
case FileTransfer.State.NOT_STARTED: case FileTransfer.State.NOT_STARTED:
case FileTransfer.State.FAILED:
widget.activate_action("file.download", null); widget.activate_action("file.download", null);
break; break;
default: default:
// Clicking doesn't do anything in IN_PROGRESS state // Clicking doesn't do anything in FAILED and IN_PROGRESS states
break; break;
} }
} }

View file

@ -14,7 +14,6 @@ namespace Dino.Ui.ConversationDetails {
if (conversation.type_ == Conversation.Type.GROUPCHAT) { if (conversation.type_ == Conversation.Type.GROUPCHAT) {
stream_interactor.get_module(MucManager.IDENTITY).get_config_form.begin(conversation.account, conversation.counterpart, (_, res) => { stream_interactor.get_module(MucManager.IDENTITY).get_config_form.begin(conversation.account, conversation.counterpart, (_, res) => {
model.data_form = stream_interactor.get_module(MucManager.IDENTITY).get_config_form.end(res); model.data_form = stream_interactor.get_module(MucManager.IDENTITY).get_config_form.end(res);
if (model.data_form == null) return;
model.data_form_bak = model.data_form.stanza_node.to_string(); model.data_form_bak = model.data_form.stanza_node.to_string();
}); });
} }

View file

@ -18,6 +18,10 @@ public class ConversationSelectorRow : ListBoxRow {
[GtkChild] protected unowned Label nick_label; [GtkChild] protected unowned Label nick_label;
[GtkChild] protected unowned Label message_label; [GtkChild] protected unowned Label message_label;
[GtkChild] protected unowned Label unread_count_label; [GtkChild] protected unowned Label unread_count_label;
[GtkChild] protected unowned Button x_button;
[GtkChild] protected unowned Revealer time_revealer;
[GtkChild] protected unowned Revealer xbutton_revealer;
[GtkChild] protected unowned Revealer top_row_revealer;
[GtkChild] protected unowned Image pinned_image; [GtkChild] protected unowned Image pinned_image;
[GtkChild] public unowned Revealer main_revealer; [GtkChild] public unowned Revealer main_revealer;
@ -94,6 +98,9 @@ public class ConversationSelectorRow : ListBoxRow {
last_content_item = stream_interactor.get_module(ContentItemStore.IDENTITY).get_latest(conversation); last_content_item = stream_interactor.get_module(ContentItemStore.IDENTITY).get_latest(conversation);
x_button.clicked.connect(() => {
stream_interactor.get_module(ConversationManager.IDENTITY).close_conversation(conversation);
});
picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).set_conversation(conversation); picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).set_conversation(conversation);
conversation.notify["read-up-to-item"].connect(() => update_read()); conversation.notify["read-up-to-item"].connect(() => update_read());
conversation.notify["pinned"].connect(() => { update_pinned_icon(); }); conversation.notify["pinned"].connect(() => { update_pinned_icon(); });
@ -263,6 +270,19 @@ public class ConversationSelectorRow : ListBoxRow {
} }
} }
public override void state_flags_changed(StateFlags flags) {
StateFlags curr_flags = get_state_flags();
if ((curr_flags & StateFlags.PRELIGHT) != 0) {
time_revealer.set_reveal_child(false);
top_row_revealer.set_reveal_child(false);
xbutton_revealer.set_reveal_child(true);
} else {
time_revealer.set_reveal_child(true);
top_row_revealer.set_reveal_child(true);
xbutton_revealer.set_reveal_child(false);
}
}
private static Regex dino_resource_regex = /^dino\.[a-f0-9]{8}$/; private static Regex dino_resource_regex = /^dino\.[a-f0-9]{8}$/;
private Widget generate_tooltip() { private Widget generate_tooltip() {

View file

@ -11,41 +11,29 @@ class MenuEntry : Plugins.ConversationTitlebarEntry, Object {
StreamInteractor stream_interactor; StreamInteractor stream_interactor;
private Conversation? conversation; private Conversation? conversation;
MenuButton button = new MenuButton() { icon_name="view-more-symbolic" }; Button button = new Button() { icon_name="view-more-symbolic" };
public MenuEntry(StreamInteractor stream_interactor) { public MenuEntry(StreamInteractor stream_interactor) {
this.stream_interactor = stream_interactor; this.stream_interactor = stream_interactor;
Menu menu_model = new Menu(); button.clicked.connect(on_clicked);
menu_model.append(_("Conversation Details"), "conversation.details");
menu_model.append(_("Close Conversation"), "conversation.close");
Gtk.PopoverMenu popover_menu = new Gtk.PopoverMenu.from_model(menu_model);
button.popover = popover_menu;
SimpleActionGroup action_group = new SimpleActionGroup();
SimpleAction details_action = new SimpleAction("details", null);
details_action.activate.connect((parameter) => {
open_conversation_details();
});
action_group.insert(details_action);
SimpleAction close_action = new SimpleAction("close", null);
close_action.activate.connect((parameter) => {
stream_interactor.get_module(ConversationManager.IDENTITY).close_conversation(conversation);
});
action_group.insert(close_action);
button.insert_action_group("conversation", action_group);
} }
public new void set_conversation(Conversation conversation) { public new void set_conversation(Conversation conversation) {
button.sensitive = true; button.sensitive = true;
this.conversation = conversation; this.conversation = conversation;
if (conversation.type_ == Conversation.Type.GROUPCHAT) {
button.tooltip_text = Util.string_if_tooltips_active("Channel details");
} else {
button.tooltip_text = Util.string_if_tooltips_active("Conversation details");
}
} }
public new void unset_conversation() { public new void unset_conversation() {
button.sensitive = false; button.sensitive = false;
} }
private void open_conversation_details() { private void on_clicked() {
var conversation_details = ConversationDetails.setup_dialog(conversation, stream_interactor, (Window)button.get_root()); var conversation_details = ConversationDetails.setup_dialog(conversation, stream_interactor, (Window)button.get_root());
conversation_details.present(); conversation_details.present();
} }

View file

@ -114,61 +114,6 @@ public class ConversationViewController : Object {
((Gtk.Window)view.get_root()).add_shortcut(shortcut); ((Gtk.Window)view.get_root()).add_shortcut(shortcut);
} }
private void update_conversation_encryption(Conversation? conversation) {
if (conversation == null) {
return;
}
if (conversation.type_ == Conversation.Type.CHAT && conversation.encryption == UNKNOWN) {
Dino.Entities.Settings settings = Dino.Application.get_default().settings;
if (settings.default_encryption == UNKNOWN) {
var selection_dialog_builder = new Builder.from_resource("/im/dino/Dino/default_encryption_dialog.ui");
var selection_dialog = selection_dialog_builder.get_object("dialog") as Dialog;
var accept_button = selection_dialog_builder.get_object("accept_button") as Button;
var omemo_radio = selection_dialog_builder.get_object("omemo") as CheckButton;
var openpgp_radio = selection_dialog_builder.get_object("openpgp") as CheckButton;
var none_radio = selection_dialog_builder.get_object("none") as CheckButton;
selection_dialog.set_transient_for(view.get_root() as Window);
Encryption selected_default = UNKNOWN;
accept_button.sensitive = false;
omemo_radio.toggled.connect(() => {
accept_button.sensitive = true;
});
openpgp_radio.toggled.connect(() => {
accept_button.sensitive = true;
});
none_radio.toggled.connect(() => {
accept_button.sensitive = true;
});
accept_button.clicked.connect(() => {
if (omemo_radio.active) {selected_default = OMEMO;}
else if (openpgp_radio.active) {selected_default = PGP;}
else if (none_radio.active) {selected_default = NONE;}
selection_dialog.response(selected_default);
selection_dialog.close();
});
selection_dialog.response.connect((response_id) => {
if (response_id >= 0) {
conversation.encryption = response_id;
}
});
selection_dialog.show();
}
else {
conversation.encryption = settings.default_encryption;
}
}
}
public void select_conversation(Conversation? conversation, bool default_initialize_conversation) { public void select_conversation(Conversation? conversation, bool default_initialize_conversation) {
if (this.conversation != null) { if (this.conversation != null) {
conversation.notify["encryption"].disconnect(update_file_upload_status); conversation.notify["encryption"].disconnect(update_file_upload_status);
@ -179,8 +124,6 @@ public class ConversationViewController : Object {
this.conversation = conversation; this.conversation = conversation;
update_conversation_encryption(conversation);
// Set list model onto list view // Set list model onto list view
// Dino.Application app = GLib.Application.get_default() as Dino.Application; // Dino.Application app = GLib.Application.get_default() as Dino.Application;
// var map_list_model = get_conversation_content_model(new ContentItemMetaModel(app.db, conversation, stream_interactor), stream_interactor); // var map_list_model = get_conversation_content_model(new ContentItemMetaModel(app.db, conversation, stream_interactor), stream_interactor);

View file

@ -81,8 +81,7 @@ public class FileSendOverlay {
if (widget == null) { if (widget == null) {
FileDefaultWidget default_widget = new FileDefaultWidget(); FileDefaultWidget default_widget = new FileDefaultWidget();
default_widget.name_label.label = file_name; default_widget.name_label.label = file_name;
default_widget.update_file_info(mime_type, 0, FileTransfer.DIRECTION_SENT, default_widget.update_file_info(mime_type, FileTransfer.State.COMPLETE, (long)file_info.get_size());
FileTransfer.State.COMPLETE, (long)file_info.get_size());
widget = default_widget; widget = default_widget;
} }

View file

@ -87,10 +87,8 @@ namespace Dino.Ui{
string ret = yield stream_interactor.get_module(Register.IDENTITY).change_password(account, new_pw_input); string ret = yield stream_interactor.get_module(Register.IDENTITY).change_password(account, new_pw_input);
change_password_button.sensitive = true; change_password_button.sensitive = true;
change_password_stack.visible_child_name = "label"; change_password_stack.visible_child_name = "label";
if (ret == null) { if (ret == null)
account.password = new_pw_input;
close(); close();
}
change_password_error_label.label = ret; change_password_error_label.label = ret;

View file

@ -44,11 +44,7 @@ public class Dialog : Gtk.Dialog {
}); });
image_button.clicked.connect(show_select_avatar); image_button.clicked.connect(show_select_avatar);
alias_hybrid.entry.changed.connect(() => { selected_account.alias = alias_hybrid.text; }); alias_hybrid.entry.changed.connect(() => { selected_account.alias = alias_hybrid.text; });
password_hybrid.entry.changed.connect(() => { password_hybrid.entry.changed.connect(() => { selected_account.password = password_hybrid.text; });
if (password_hybrid.text != "************") {
selected_account.password = password_hybrid.text;
}
});
password_change_btn.clicked.connect(show_change_psswd_dialog); password_change_btn.clicked.connect(show_change_psswd_dialog);
Util.LabelHybridGroup label_hybrid_group = new Util.LabelHybridGroup(); Util.LabelHybridGroup label_hybrid_group = new Util.LabelHybridGroup();
@ -72,7 +68,6 @@ public class Dialog : Gtk.Dialog {
settings_list.attach(widget, 1, row_index, 2); settings_list.attach(widget, 1, row_index, 2);
row_index++; row_index++;
password_hybrid.text = "************";
} }
} }
@ -197,7 +192,6 @@ public class Dialog : Gtk.Dialog {
} }
private void populate_grid_data(Account account) { private void populate_grid_data(Account account) {
active_switch.sensitive = false;
active_switch.state_set.disconnect(change_account_state); active_switch.state_set.disconnect(change_account_state);
picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).add_participant(new Conversation(account.bare_jid, account, Conversation.Type.CHAT), account.bare_jid); picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).add_participant(new Conversation(account.bare_jid, account, Conversation.Type.CHAT), account.bare_jid);
@ -206,7 +200,7 @@ public class Dialog : Gtk.Dialog {
alias_hybrid.text = account.alias ?? ""; alias_hybrid.text = account.alias ?? "";
password_hybrid.entry.input_purpose = InputPurpose.PASSWORD; password_hybrid.entry.input_purpose = InputPurpose.PASSWORD;
password_hybrid.text = account.password;
update_status_label(account); update_status_label(account);
@ -228,14 +222,11 @@ public class Dialog : Gtk.Dialog {
ConnectionManager.ConnectionState state = stream_interactor.connection_manager.get_state(account); ConnectionManager.ConnectionState state = stream_interactor.connection_manager.get_state(account);
switch (state) { switch (state) {
case ConnectionManager.ConnectionState.CONNECTING: case ConnectionManager.ConnectionState.CONNECTING:
active_switch.sensitive = false;
state_label.label = _("Connecting…"); break; state_label.label = _("Connecting…"); break;
case ConnectionManager.ConnectionState.CONNECTED: case ConnectionManager.ConnectionState.CONNECTED:
active_switch.sensitive = true;
password_change_btn.sensitive = true; password_change_btn.sensitive = true;
state_label.label = _("Connected"); break; state_label.label = _("Connected"); break;
case ConnectionManager.ConnectionState.DISCONNECTED: case ConnectionManager.ConnectionState.DISCONNECTED:
active_switch.sensitive = true;
password_change_btn.sensitive = false; password_change_btn.sensitive = false;
state_label.label = _("Disconnected"); break; state_label.label = _("Disconnected"); break;
} }

View file

@ -17,9 +17,7 @@ public class List : Box {
private Conversation conversation; private Conversation conversation;
private string[]? filter_values; private string[]? filter_values;
private HashMap<Jid, Widget> rows = new HashMap<Jid, Widget>(Jid.hash_func, Jid.equals_func);
// List of all chat members with corresponding widgets
private HashMap<string, Widget> rows = new HashMap<string, Widget>();
public HashMap<Widget, ListRow> row_wrappers = new HashMap<Widget, ListRow>(); public HashMap<Widget, ListRow> row_wrappers = new HashMap<Widget, ListRow>();
public List(StreamInteractor stream_interactor, Conversation conversation) { public List(StreamInteractor stream_interactor, Conversation conversation) {
@ -30,44 +28,17 @@ public class List : Box {
search_entry.search_changed.connect(refilter); search_entry.search_changed.connect(refilter);
stream_interactor.get_module(PresenceManager.IDENTITY).show_received.connect(on_show_received); stream_interactor.get_module(PresenceManager.IDENTITY).show_received.connect(on_show_received);
stream_interactor.get_module(PresenceManager.IDENTITY).received_offline_presence.connect(on_quit_received); stream_interactor.get_module(PresenceManager.IDENTITY).received_offline_presence.connect(on_received_offline_presence);
initialize_for_conversation(conversation); initialize_for_conversation(conversation);
} }
public bool get_status(Jid jid, Account account) {
Gee.List<Jid>? full_jids = stream_interactor.get_module(PresenceManager.IDENTITY).get_full_jids(jid, account);
debug("Get presence status for %s", jid.bare_jid.to_string());
string presence_str = null;
if (full_jids != null){
// Iterate over all connected devices
for (int i = 0; i < full_jids.size; i++) {
Jid full_jid = full_jids[i];
presence_str = stream_interactor.get_module(PresenceManager.IDENTITY).get_last_show(full_jid, account);
switch(presence_str) {
case "online": {
// Return online status if user is online on at least one device
return true;
}
}
}
} else {
return false;
}
return false;
}
public void initialize_for_conversation(Conversation conversation) { public void initialize_for_conversation(Conversation conversation) {
this.conversation = conversation; this.conversation = conversation;
Gee.List<Jid>? occupants = stream_interactor.get_module(MucManager.IDENTITY).get_occupants(conversation.counterpart, conversation.account);
var identity = stream_interactor.get_module(MucManager.IDENTITY); if (occupants != null) {
Gee.List<Jid>? members = identity.get_members(conversation.counterpart, conversation.account); foreach (Jid occupant in occupants) {
if (members != null) { add_occupant(occupant);
// Add all members and their status to the list
foreach (Jid member in members) {
add_member(member); //use add_member_with_status if you want to get online/offline statuses
} }
} }
list_box.invalidate_filter(); list_box.invalidate_filter();
@ -82,135 +53,38 @@ public class List : Box {
list_box.invalidate_filter(); list_box.invalidate_filter();
} }
public void add_member_with_status(Jid jid, bool online) { public void add_occupant(Jid jid) {
// HACK:
// Here we track members based on their names (not jids)
// Sometimes the same member can be referenced with different jids, for example:
// When initializing the conversation (see initialize_for_conversation function),
// we reference members like this:
// test_user@test_domain (using a local part, without a resource)
// However when updating status, we get the jid in the following format
// local_domain@test_domain/test_user (using a resource)
string member_name = null;
if (jid.resourcepart != null) {
member_name = jid.resourcepart;
} else {
member_name = jid.localpart;
}
if (member_name == null) {
return;
}
if (!rows.has_key(member_name)) {
debug("adding new member %s", jid.to_string());
debug("local %s", jid.localpart);
debug("domain %s", jid.domainpart);
debug("resource %s", jid.resourcepart);
var row_wrapper = new ListRow(stream_interactor, conversation, jid);
var widget = row_wrapper.get_widget();
if (online) {
row_wrapper.set_online();
} else {
row_wrapper.set_offline();
}
row_wrappers[widget] = row_wrapper;
rows[member_name] = widget;
list_box.append(widget);
}
}
public void add_member(Jid jid) {
var row_wrapper = new ListRow(stream_interactor, conversation, jid); var row_wrapper = new ListRow(stream_interactor, conversation, jid);
var widget = row_wrapper.get_widget(); var widget = row_wrapper.get_widget();
string member_name = null;
if (jid.resourcepart != null) {
member_name = jid.resourcepart;
} else {
member_name = jid.localpart;
}
if (member_name == null) {
return;
}
row_wrappers[widget] = row_wrapper; row_wrappers[widget] = row_wrapper;
rows[member_name] = widget; rows[jid] = widget;
list_box.append(widget); list_box.append(widget);
} }
public void remove_member(Jid jid) { public void remove_occupant(Jid jid) {
var member_name = jid.resourcepart; list_box.remove(rows[jid]);
if (member_name == null) { rows.unset(jid);
return;
}
list_box.remove(rows[member_name]);
rows.unset(member_name);
} }
private void on_received_offline_presence(Jid jid, Account account) { private void on_received_offline_presence(Jid jid, Account account) {
if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) { if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) {
var member_name = jid.resourcepart; if (rows.has_key(jid)) {
if (member_name == null) { remove_occupant(jid);
return;
} }
if (rows.has_key(member_name)) {
row_wrappers[rows[member_name]].set_offline();
debug("%s is now offline", jid.to_string());
}
list_box.invalidate_filter();
}
}
private void on_quit_received(Jid jid, Account account) {
if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) {
var member_name = jid.resourcepart;
if (member_name == null) {
return;
}
if (rows.has_key(member_name)) {
remove_member(jid);
}
list_box.invalidate_filter();
}
}
private void on_received_online_presence(Jid jid, Account account) {
if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) {
var member_name = jid.resourcepart;
if (member_name == null) {
return;
}
if (!rows.has_key(member_name)) {
add_member_with_status(jid, true);
}
row_wrappers[rows[member_name]].set_online();
debug("%s is now online", jid.to_string());
list_box.invalidate_filter(); list_box.invalidate_filter();
} }
} }
private void on_show_received(Jid jid, Account account) { private void on_show_received(Jid jid, Account account) {
if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) { if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) {
var member_name = jid.resourcepart; if (!rows.has_key(jid)) {
if (member_name == null) { add_occupant(jid);
return;
}
if (!rows.has_key(member_name)) {
add_member(jid);
} }
list_box.invalidate_filter(); list_box.invalidate_filter();
} }
} }
private void header(ListBoxRow row, ListBoxRow? before_row) { private void header(ListBoxRow row, ListBoxRow? before_row) {
ListRow row_wrapper1 = row_wrappers[row.get_child()]; ListRow row_wrapper1 = row_wrappers[row.get_child()];
Xmpp.Xep.Muc.Affiliation? a1 = stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper1.jid, row_wrapper1.conversation.account); Xmpp.Xep.Muc.Affiliation? a1 = stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper1.jid, row_wrapper1.conversation.account);

View file

@ -10,6 +10,7 @@ public class ListRow : Object {
private Grid main_grid; private Grid main_grid;
private AvatarPicture picture; private AvatarPicture picture;
public Label name_label; public Label name_label;
public Conversation? conversation; public Conversation? conversation;
public Jid? jid; public Jid? jid;
@ -18,9 +19,6 @@ public class ListRow : Object {
main_grid = (Grid) builder.get_object("main_grid"); main_grid = (Grid) builder.get_object("main_grid");
picture = (AvatarPicture) builder.get_object("picture"); picture = (AvatarPicture) builder.get_object("picture");
name_label = (Label) builder.get_object("name_label"); name_label = (Label) builder.get_object("name_label");
main_grid.set_column_spacing(10);
main_grid.set_column_homogeneous(false);
main_grid.set_baseline_row(1);
} }
public ListRow(StreamInteractor stream_interactor, Conversation conversation, Jid jid) { public ListRow(StreamInteractor stream_interactor, Conversation conversation, Jid jid) {
@ -39,12 +37,6 @@ public class ListRow : Object {
public Widget get_widget() { public Widget get_widget() {
return main_grid; return main_grid;
} }
public void set_online() {
}
public void set_offline() {
}
} }
} }

View file

@ -1,5 +1,4 @@
using Gtk; using Gtk;
using Dino.Entities;
namespace Dino.Ui { namespace Dino.Ui {
@ -10,12 +9,6 @@ class SettingsDialog : Adw.PreferencesWindow {
[GtkChild] private unowned Switch marker_switch; [GtkChild] private unowned Switch marker_switch;
[GtkChild] private unowned Switch notification_switch; [GtkChild] private unowned Switch notification_switch;
[GtkChild] private unowned Switch emoji_switch; [GtkChild] private unowned Switch emoji_switch;
[GtkChild] private unowned CheckButton encryption_radio_undecided;
[GtkChild] private unowned CheckButton encryption_radio_omemo;
[GtkChild] private unowned CheckButton encryption_radio_openpgp;
[GtkChild] private unowned Switch send_button_switch;
[GtkChild] private unowned Switch enter_newline_switch;
[GtkChild] private unowned Switch dark_theme;
Dino.Entities.Settings settings = Dino.Application.get_default().settings; Dino.Entities.Settings settings = Dino.Application.get_default().settings;
@ -26,48 +19,11 @@ class SettingsDialog : Adw.PreferencesWindow {
marker_switch.active = settings.send_marker; marker_switch.active = settings.send_marker;
notification_switch.active = settings.notifications; notification_switch.active = settings.notifications;
emoji_switch.active = settings.convert_utf8_smileys; emoji_switch.active = settings.convert_utf8_smileys;
encryption_radio_undecided.active = settings.default_encryption == Encryption.UNKNOWN;
encryption_radio_omemo.active = settings.default_encryption == Encryption.OMEMO;
encryption_radio_openpgp.active = settings.default_encryption == Encryption.PGP;
send_button_switch.active = settings.send_button;
enter_newline_switch.active = settings.enter_newline;
enter_newline_switch.sensitive = settings.send_button;
dark_theme.active = settings.dark_theme;
typing_switch.notify["active"].connect(() => { settings.send_typing = typing_switch.active; } ); typing_switch.notify["active"].connect(() => { settings.send_typing = typing_switch.active; } );
marker_switch.notify["active"].connect(() => { settings.send_marker = marker_switch.active; } ); marker_switch.notify["active"].connect(() => { settings.send_marker = marker_switch.active; } );
notification_switch.notify["active"].connect(() => { settings.notifications = notification_switch.active; } ); notification_switch.notify["active"].connect(() => { settings.notifications = notification_switch.active; } );
emoji_switch.notify["active"].connect(() => { settings.convert_utf8_smileys = emoji_switch.active; }); emoji_switch.notify["active"].connect(() => { settings.convert_utf8_smileys = emoji_switch.active; });
encryption_radio_undecided.notify["active"].connect(() => {
if (encryption_radio_undecided.active) {
settings.default_encryption = Encryption.UNKNOWN;
}
});
encryption_radio_omemo.notify["active"].connect(() => {
if (encryption_radio_omemo.active) {
settings.default_encryption = Encryption.OMEMO;
}
});
encryption_radio_openpgp.notify["active"].connect(() => {
if (encryption_radio_openpgp.active) {
settings.default_encryption = Encryption.PGP;
}
});
send_button_switch.notify["active"].connect(() => { settings.send_button = send_button_switch.active; });
enter_newline_switch.notify["active"].connect(() => { settings.enter_newline = enter_newline_switch.active; });
settings.send_button_update.connect((visible) => {
enter_newline_switch.sensitive = visible;
if (visible == false) {
enter_newline_switch.active = visible;
}
});
dark_theme.notify["active"].connect(() => { settings.dark_theme = dark_theme.active; });
} }
} }

View file

@ -1,4 +1,4 @@
project('xmpp-vala', 'c', 'cpp', 'vala', version:'0.1') project('xmpp-vala', 'c', 'cpp', 'vala')
fs = import('fs') fs = import('fs')
gnome = import('gnome') gnome = import('gnome')
@ -8,62 +8,20 @@ python = import('python')
# plugin_crypto is enabled if any of the crypto plugins is enabled, auto if # plugin_crypto is enabled if any of the crypto plugins is enabled, auto if
# none of them are explicitly enabled but at least one is set to auto, or # none of them are explicitly enabled but at least one is set to auto, or
# disabled if all of them are disabled. # disabled if all of them are disabled.
# plugin_crypto = get_option('plugin-ice')
# On Windows, it's always required because we need it for glib-networking. foreach plugin : ['plugin-ice', 'plugin-omemo', 'plugin-rtp']
if host_machine.system() == 'windows' if get_option(plugin).enabled() and not plugin_crypto.enabled()
plugin_crypto = true plugin_crypto = get_option(plugin)
add_project_arguments('-D', '_WIN32', language: 'vala') elif get_option(plugin).allowed() and not plugin_crypto.allowed()
else plugin_crypto = get_option(plugin)
plugin_crypto = get_option('plugin-ice')
foreach plugin : ['plugin-ice', 'plugin-omemo', 'plugin-rtp']
if get_option(plugin).enabled() and not plugin_crypto.enabled()
plugin_crypto = get_option(plugin)
elif get_option(plugin).allowed() and not plugin_crypto.allowed()
plugin_crypto = get_option(plugin)
endif
endforeach
endif
if get_option('crypto-backend') == 'auto'
# Prefer libgcrypt/gnutls over openssl because glib-networking is usually
# built with gnutls anyway.
dep_libgcrypt = dependency('libgcrypt', required: false)
dep_gnutls = dependency('gnutls', required: false)
if dep_libgcrypt.found() and dep_libgcrypt.found()
dep_libgcrypt_or_openssl = dep_libgcrypt
dep_gnutls_or_openssl = dep_gnutls
crypto_backend = 'gnutls'
else
dep_openssl = dependency('openssl', disabler: true, required: plugin_crypto)
dep_libgcrypt_or_openssl = dep_openssl
dep_gnutls_or_openssl = dep_openssl
crypto_backend = 'openssl'
endif endif
elif get_option('crypto-backend') == 'openssl' endforeach
dep_openssl = dependency('openssl', disabler: true, required: plugin_crypto)
dep_libgcrypt_or_openssl = dep_openssl
dep_gnutls_or_openssl = dep_openssl
crypto_backend = 'openssl'
elif get_option('crypto-backend') == 'gnutls'
dep_libgcrypt = dependency('libgcrypt', disabler: true, required: plugin_crypto)
dep_gnutls = dependency('gnutls', disabler: true, required: get_option('plugin-ice'))
dep_libgcrypt_or_openssl = dep_libgcrypt
dep_gnutls_or_openssl = dep_gnutls
crypto_backend = 'gnutls'
endif
if get_option('use-soup2')
libsoup_version = '2.4'
else
libsoup_version = '3.0'
endif
cc = meson.get_compiler('c')
dep_gdk_pixbuf = dependency('gdk-pixbuf-2.0') dep_gdk_pixbuf = dependency('gdk-pixbuf-2.0')
dep_gee = dependency('gee-0.8') dep_gee = dependency('gee-0.8')
dep_gio = dependency('gio-2.0') dep_gio = dependency('gio-2.0')
dep_glib = dependency('glib-2.0') dep_glib = dependency('glib-2.0')
dep_gnutls = dependency('gnutls', disabler: true, required: get_option('plugin-ice'))
dep_gmodule = dependency('gmodule-2.0') dep_gmodule = dependency('gmodule-2.0')
dep_gpgme = dependency('gpgme', disabler: true, required: get_option('plugin-openpgp')) dep_gpgme = dependency('gpgme', disabler: true, required: get_option('plugin-openpgp'))
dep_gstreamer = dependency('gstreamer-1.0', disabler: true, required: get_option('plugin-rtp')) dep_gstreamer = dependency('gstreamer-1.0', disabler: true, required: get_option('plugin-rtp'))
@ -75,20 +33,18 @@ dep_gtk4 = dependency('gtk4')
dep_icu_uc = dependency('icu-uc') dep_icu_uc = dependency('icu-uc')
dep_libadwaita = dependency('libadwaita-1') dep_libadwaita = dependency('libadwaita-1')
dep_libcanberra = dependency('libcanberra', disabler: true, required: get_option('plugin-notification-sound')) dep_libcanberra = dependency('libcanberra', disabler: true, required: get_option('plugin-notification-sound'))
dep_libgcrypt = dependency('libgcrypt', disabler: true, required: plugin_crypto)
dep_libqrencode = dependency('libqrencode', disabler: true, required: get_option('plugin-omemo')) dep_libqrencode = dependency('libqrencode', disabler: true, required: get_option('plugin-omemo'))
dep_libsrtp2 = dependency('libsrtp2', disabler: true, required: plugin_crypto) dep_libsrtp2 = dependency('libsrtp2', disabler: true, required: plugin_crypto)
# libsignal-protocol-c has a history of breaking compatibility on the patch level # libsignal-protocol-c has a history of breaking compatibility on the patch level
# we'll have to check compatibility for every new release # we'll have to check compatibility for every new release
# distro maintainers may update this dependency after compatibility tests # distro maintainers may update this dependency after compatibility tests
dep_libsignal_protocol_c = dependency('libsignal-protocol-c', version: ['>=2.3.2', '<2.3.4'], disabler: true, required: get_option('plugin-omemo')) dep_libsignal_protocol_c = dependency('libsignal-protocol-c', version: ['>=2.3.2', '<2.3.4'], disabler: true, required: get_option('plugin-omemo'))
dep_libsoup = dependency('libsoup-@0@'.format(libsoup_version), disabler: true, required: get_option('plugin-http-files')) dep_libsoup = dependency('libsoup-3.0', disabler: true, required: get_option('plugin-http-files'))
dep_nice = dependency('nice', version: '>=0.1.15', disabler: true, required: get_option('plugin-ice')) dep_nice = dependency('nice', version: '>=0.1.15', disabler: true, required: get_option('plugin-ice'))
dep_m = cc.find_library('m', required: false) dep_m = meson.get_compiler('c').find_library('m', required: false)
dep_sqlite3 = dependency('sqlite3', version: '>=3.24') dep_sqlite3 = dependency('sqlite3', version: '>=3.24')
dep_webrtc_audio_processing = dependency('webrtc-audio-processing', version: ['>=0.2', '<0.4'], required: get_option('plugin-rtp-webrtc-audio-processing'))
dep_gstreamer_bad = dependency('gstreamer-plugins-bad-1.0', disabler: true, required: get_option('plugin-rtp-webrtc-audio-processing'))
gstpluginsdir = dep_gstreamer_bad.get_variable('pluginsdir')
dep_webrtcdsp = cc.find_library('gstwebrtcdsp', dirs: gstpluginsdir, disabler: true, required: get_option('plugin-rtp-webrtc-audio-processing'))
prog_git = find_program('git', required: false) prog_git = find_program('git', required: false)
prog_python = python.find_installation() prog_python = python.find_installation()

View file

@ -1,6 +1,4 @@
option('plugindir', type: 'string', value: 'dino/plugins', description: 'Dino plugin directory inside libdir') option('plugindir', type: 'string', value: 'lib/dino/plugins', description: 'Dino plugin directory')
option('crypto-backend', type: 'combo', choices: ['auto', 'openssl', 'gnutls'], value: 'auto', description: 'Preferred crypto backend')
option('plugin-http-files', type: 'feature', description: 'HTTP file upload') option('plugin-http-files', type: 'feature', description: 'HTTP file upload')
option('plugin-ice', type: 'feature', description: '') option('plugin-ice', type: 'feature', description: '')
@ -14,6 +12,3 @@ option('plugin-rtp-msdk', type: 'feature', value: 'disabled', description: 'Inte
option('plugin-rtp-vaapi', type: 'feature', value: 'disabled', description: 'Video Acceleration API') option('plugin-rtp-vaapi', type: 'feature', value: 'disabled', description: 'Video Acceleration API')
option('plugin-rtp-vp9', type: 'feature', value: 'disabled', description: 'VP9 codec') option('plugin-rtp-vp9', type: 'feature', value: 'disabled', description: 'VP9 codec')
option('plugin-rtp-webrtc-audio-processing', type: 'feature', description: 'Voice preprocessing') option('plugin-rtp-webrtc-audio-processing', type: 'feature', description: 'Voice preprocessing')
option('use-soup2', type: 'boolean', value: false, description: 'Use libsoup version 2 instead of 3')
option('with-wasapi', type: 'boolean', value: true, description: 'Use wasapi insted of directsound on windows')

View file

@ -15,10 +15,8 @@ sources = files(
'src/register_plugin.vala', 'src/register_plugin.vala',
) )
vala_args = [] vala_args = [
if dep_libsoup.version().version_compare('>=3.0') '--define=SOUP_3_0',
vala_args += ['--define=SOUP_3_0'] ]
endif lib_http_files = shared_library('http-files', sources, name_prefix: '', vala_args: vala_args, dependencies: dependencies, install: true, install_dir: get_option('libdir') / 'dino/plugins')
lib_http_files = shared_library('http-files', sources, name_prefix: '', vala_args: vala_args, dependencies: dependencies, kwargs: install_options)
dep_http_files = declare_dependency(link_with: lib_http_files, include_directories: include_directories('.')) dep_http_files = declare_dependency(link_with: lib_http_files, include_directories: include_directories('.'))

View file

@ -103,18 +103,6 @@ public class HttpFileSender : FileSender, Object {
put_message.wrote_headers.connect(() => transfer_more_bytes(file_transfer.input_stream, put_message.request_body)); put_message.wrote_headers.connect(() => transfer_more_bytes(file_transfer.input_stream, put_message.request_body));
put_message.wrote_chunk.connect(() => transfer_more_bytes(file_transfer.input_stream, put_message.request_body)); put_message.wrote_chunk.connect(() => transfer_more_bytes(file_transfer.input_stream, put_message.request_body));
#endif #endif
file_transfer.transferred_bytes = 0;
put_message.wrote_body_data.connect((chunk) => {
if (file_transfer.size != 0) {
#if SOUP_3_0
file_transfer.transferred_bytes += chunk;
#else
file_transfer.transferred_bytes += chunk.length;
#endif
}
});
foreach (var entry in file_send_data.headers.entries) { foreach (var entry in file_send_data.headers.entries) {
put_message.request_headers.append(entry.key, entry.value); put_message.request_headers.append(entry.key, entry.value);
} }

View file

@ -1,11 +1,3 @@
if crypto_backend == 'openssl'
if get_option('plugin-ice').enabled()
error('plugin-ice does not work with openssl backend yet')
else
warning('plugin-ice is not enabled!')
subdir_done()
endif
endif
dependencies = [ dependencies = [
dep_crypto_vala, dep_crypto_vala,
dep_dino, dep_dino,
@ -13,7 +5,7 @@ dependencies = [
dep_gee, dep_gee,
dep_glib, dep_glib,
dep_gmodule, dep_gmodule,
dep_gnutls_or_openssl, dep_gnutls,
dep_nice, dep_nice,
dep_qlite, dep_qlite,
dep_xmpp_vala, dep_xmpp_vala,
@ -32,5 +24,5 @@ c_args = [
vala_args = [ vala_args = [
'--vapidir', meson.current_source_dir() / 'vapi', '--vapidir', meson.current_source_dir() / 'vapi',
] ]
lib_ice = shared_library('ice', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, dependencies: dependencies, kwargs: install_options) lib_ice = shared_library('ice', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, dependencies: dependencies, install: true, install_dir: get_option('libdir') / 'dino/plugins')
dep_ice = declare_dependency(link_with: lib_ice, include_directories: include_directories('.')) dep_ice = declare_dependency(link_with: lib_ice, include_directories: include_directories('.'))

View file

@ -6,21 +6,6 @@ using Xmpp.Xep;
private extern const size_t NICE_ADDRESS_STRING_LEN; private extern const size_t NICE_ADDRESS_STRING_LEN;
public class Dino.Plugins.Ice.Plugin : RootInterface, Object { public class Dino.Plugins.Ice.Plugin : RootInterface, Object {
private const int64 delay_min = 300; // 10mn
private const int64 delay_max = (int64) uint.MAX;
private class TimerPayload {
public Account account { get; set; }
public uint timeout_handle_id;
public TimerPayload(Account account, uint timeout_handle_id) {
this.account = account;
this.timeout_handle_id = timeout_handle_id;
}
}
private HashMap<XmppStream, TimerPayload> timeouts = new HashMap<XmppStream, TimerPayload>(XmppStream.hash_func, XmppStream.equals_func);
public Dino.Application app; public Dino.Application app;
public void registered(Dino.Application app) { public void registered(Dino.Application app) {
@ -37,10 +22,10 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object {
stream.get_module(JingleRawUdp.Module.IDENTITY).set_local_ip_address_handler(get_local_ip_addresses); stream.get_module(JingleRawUdp.Module.IDENTITY).set_local_ip_address_handler(get_local_ip_addresses);
} }
}); });
app.stream_interactor.connection_manager.connection_state_changed.connect(on_connection_state_changed); app.stream_interactor.stream_negotiated.connect(on_stream_negotiated);
} }
private async void external_discovery_refresh_services(Account account, XmppStream stream) { private async void on_stream_negotiated(Account account, XmppStream stream) {
Module? ice_udp_module = stream.get_module(JingleIceUdp.Module.IDENTITY) as Module; Module? ice_udp_module = stream.get_module(JingleIceUdp.Module.IDENTITY) as Module;
if (ice_udp_module == null) return; if (ice_udp_module == null) return;
Gee.List<Xep.ExternalServiceDiscovery.Service> services = yield ExternalServiceDiscovery.request_services(stream); Gee.List<Xep.ExternalServiceDiscovery.Service> services = yield ExternalServiceDiscovery.request_services(stream);
@ -60,28 +45,6 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object {
} }
} }
} }
if (ice_udp_module.turn_service != null) {
DateTime? expires = ice_udp_module.turn_service.expires;
if (expires != null) {
int64 delay = (expires.to_unix() - new DateTime.now_utc().to_unix()) / 2;
if (delay >= delay_min && delay <= delay_max) {
debug("Next server external service discovery in %lds (because of TURN credentials' expiry time)", (long) delay);
uint timeout_handle_id = Timeout.add_seconds((uint) delay, () => {
on_timeout(stream);
return false;
});
timeouts[stream] = new TimerPayload(account, timeout_handle_id);
timeouts[stream].account = account;
timeouts[stream].timeout_handle_id = timeout_handle_id;
} else {
warning("Bogus TURN credentials' expiry time (delay value = %ld), *not* planning next service discovery", (long) delay);
}
}
}
if (ice_udp_module.stun_ip == null) { if (ice_udp_module.stun_ip == null) {
InetAddress ip = yield lookup_ipv4_addess("stun.dino.im"); InetAddress ip = yield lookup_ipv4_addess("stun.dino.im");
if (ip == null) return; if (ip == null) return;
@ -93,31 +56,6 @@ public class Dino.Plugins.Ice.Plugin : RootInterface, Object {
} }
} }
public void on_timeout(XmppStream stream) {
if (!timeouts.has_key(stream)) return;
TimerPayload pl = timeouts[stream];
timeouts.unset(stream);
external_discovery_refresh_services.begin(pl.account, stream);
}
public void on_connection_state_changed(Account account, ConnectionManager.ConnectionState state) {
switch(state)
{
case ConnectionManager.ConnectionState.DISCONNECTED:
XmppStream? stream = app.stream_interactor.connection_manager.get_stream(account);
if (stream == null) return;
if (!timeouts.has_key(stream)) return;
Source.remove(timeouts[stream].timeout_handle_id);
timeouts.unset(stream);
break;
case ConnectionManager.ConnectionState.CONNECTED:
XmppStream? stream = app.stream_interactor.connection_manager.get_stream(account);
external_discovery_refresh_services(account, stream);
break;
}
}
public void shutdown() { public void shutdown() {
// Nothing to do // Nothing to do
} }

View file

@ -1,14 +1,6 @@
install_options = {'install': true, 'install_dir': get_option('libdir') / get_option('plugindir')}
subdir('http-files') subdir('http-files')
subdir('ice') subdir('ice')
subdir('notification-sound') subdir('notification-sound')
subdir('omemo') subdir('omemo')
subdir('openpgp') subdir('openpgp')
subdir('rtp') subdir('rtp')
if host_machine.system() == 'windows'
subdir('win32-fonts')
subdir('windows-notification')
else
subdir('phone-ringer')
endif

View file

@ -15,5 +15,5 @@ sources = files(
vala_args = [ vala_args = [
'--vapidir', meson.current_source_dir() / 'vapi', '--vapidir', meson.current_source_dir() / 'vapi',
] ]
lib_notification_sound = shared_library('notification-sound', sources, name_prefix: '', vala_args: vala_args, dependencies: dependencies, kwargs: install_options) lib_notification_sound = shared_library('notification-sound', sources, name_prefix: '', vala_args: vala_args, dependencies: dependencies, install: true, install_dir: get_option('libdir') / 'dino/plugins')
dep_notification_sound = declare_dependency(link_with: lib_notification_sound, include_directories: include_directories('.')) dep_notification_sound = declare_dependency(link_with: lib_notification_sound, include_directories: include_directories('.'))

View file

@ -90,11 +90,9 @@ GENERATE_VAPI
omemo omemo
GENERATE_HEADER GENERATE_HEADER
omemo omemo
DEFINITIONS
GCRYPT
) )
add_definitions(${VALA_CFLAGS} -DGETTEXT_PACKAGE=\"${GETTEXT_PACKAGE}\" -DLOCALE_INSTALL_DIR=\"${LOCALE_INSTALL_DIR}\" -DG_LOG_DOMAIN="OMEMO" -DGCRYPT) add_definitions(${VALA_CFLAGS} -DGETTEXT_PACKAGE=\"${GETTEXT_PACKAGE}\" -DLOCALE_INSTALL_DIR=\"${LOCALE_INSTALL_DIR}\" -DG_LOG_DOMAIN="OMEMO")
add_library(omemo SHARED ${OMEMO_VALA_C} ${OMEMO_GRESOURCES_TARGET} ${CMAKE_CURRENT_SOURCE_DIR}/src/signal/signal_helper.c) add_library(omemo SHARED ${OMEMO_VALA_C} ${OMEMO_GRESOURCES_TARGET} ${CMAKE_CURRENT_SOURCE_DIR}/src/signal/signal_helper.c)
add_dependencies(omemo ${GETTEXT_PACKAGE}-translations) add_dependencies(omemo ${GETTEXT_PACKAGE}-translations)
target_include_directories(omemo PUBLIC src) target_include_directories(omemo PUBLIC src)
@ -104,7 +102,7 @@ set_target_properties(omemo PROPERTIES LIBRARY_OUTPUT_DIRECTORY ${CMAKE_BINARY_D
install(TARGETS omemo ${PLUGIN_INSTALL}) install(TARGETS omemo ${PLUGIN_INSTALL})
if(BUILD_TESTING) if(BUILD_TESTS)
vala_precompile(OMEMO_TEST_VALA_C vala_precompile(OMEMO_TEST_VALA_C
SOURCES SOURCES
"tests/signal/common.vala" "tests/signal/common.vala"
@ -127,5 +125,4 @@ if(BUILD_TESTING)
add_executable(omemo-test ${OMEMO_TEST_VALA_C}) add_executable(omemo-test ${OMEMO_TEST_VALA_C})
add_dependencies(omemo-test omemo) add_dependencies(omemo-test omemo)
target_link_libraries(omemo-test omemo ${OMEMO_PACKAGES}) target_link_libraries(omemo-test omemo ${OMEMO_PACKAGES})
add_test(NAME omemo COMMAND omemo-test) endif(BUILD_TESTS)
endif(BUILD_TESTING)

View file

@ -6,7 +6,7 @@ dependencies = [
dep_glib, dep_glib,
dep_gmodule, dep_gmodule,
dep_gtk4, dep_gtk4,
dep_libgcrypt_or_openssl, dep_libgcrypt,
dep_libqrencode, dep_libqrencode,
dep_libsignal_protocol_c, dep_libsignal_protocol_c,
dep_qlite, dep_qlite,
@ -61,24 +61,8 @@ c_args = [
'-DGETTEXT_PACKAGE="dino-omemo"', '-DGETTEXT_PACKAGE="dino-omemo"',
'-DLOCALE_INSTALL_DIR="@0@"'.format(get_option('prefix') / get_option('localedir')), '-DLOCALE_INSTALL_DIR="@0@"'.format(get_option('prefix') / get_option('localedir')),
] ]
if crypto_backend == 'gnutls'
c_args += ['-DGCRYPT']
endif
vala_args = [ vala_args = [
'--vapidir', meson.current_source_dir() / 'vapi', '--vapidir', meson.current_source_dir() / 'vapi',
] ]
if crypto_backend == 'gnutls' lib_omemo = shared_library('omemo', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, include_directories: include_directories('src'), dependencies: dependencies, install: true, install_dir: get_option('libdir') / 'dino/plugins')
vala_args += ['-D', 'GCRYPT']
endif
lib_omemo = shared_library('omemo', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, include_directories: include_directories('src'), dependencies: dependencies, kwargs: install_options)
dep_omemo = declare_dependency(link_with: lib_omemo, include_directories: include_directories('.')) dep_omemo = declare_dependency(link_with: lib_omemo, include_directories: include_directories('.'))
sources = files(
'tests/signal/common.vala',
'tests/signal/curve25519.vala',
'tests/signal/hkdf.vala',
'tests/signal/session_builder.vala',
'tests/signal/testcase.vala',
)
test_omemo = executable('test_omemo', sources, vala_args: vala_args, include_directories: include_directories('src'), dependencies: dependencies + [dep_omemo])
test('omemo', test_omemo)

View file

@ -59,7 +59,7 @@ public class OmemoFileDecryptor : FileDecryptor, Object {
file_transfer.encryption = Encryption.OMEMO; file_transfer.encryption = Encryption.OMEMO;
debug("Decrypting file %s from %s", file_transfer.file_name, file_transfer.server_file_name); debug("Decrypting file %s from %s", file_transfer.file_name, file_transfer.server_file_name);
SymmetricCipher cipher = new SymmetricCipher.decryption("AES-GCM"); SymmetricCipher cipher = new SymmetricCipher("AES-GCM");
cipher.set_key(key); cipher.set_key(key);
cipher.set_iv(iv); cipher.set_iv(iv);
return new ConverterInputStream(encrypted_stream, new SymmetricCipherDecrypter((owned) cipher, 16)); return new ConverterInputStream(encrypted_stream, new SymmetricCipherDecrypter((owned) cipher, 16));

View file

@ -31,7 +31,7 @@ public class OmemoFileEncryptor : Dino.FileEncryptor, Object {
uint8[] key = new uint8[KEY_SIZE]; uint8[] key = new uint8[KEY_SIZE];
Plugin.get_context().randomize(key); Plugin.get_context().randomize(key);
SymmetricCipher cipher = new SymmetricCipher.encryption("AES-GCM"); SymmetricCipher cipher = new SymmetricCipher("AES-GCM");
cipher.set_key(key); cipher.set_key(key);
cipher.set_iv(iv); cipher.set_iv(iv);

View file

@ -99,13 +99,13 @@ public class AesGcmCipher : Jet.Cipher, Object {
return new Jet.TransportSecret(key, iv); return new Jet.TransportSecret(key, iv);
} }
public InputStream wrap_input_stream(InputStream input, Jet.TransportSecret secret) requires (secret.transport_key.length == key_size) { public InputStream wrap_input_stream(InputStream input, Jet.TransportSecret secret) requires (secret.transport_key.length == key_size) {
SymmetricCipher cipher = new SymmetricCipher.decryption("AES-GCM"); SymmetricCipher cipher = new SymmetricCipher("AES-GCM");
cipher.set_key(secret.transport_key); cipher.set_key(secret.transport_key);
cipher.set_iv(secret.initialization_vector); cipher.set_iv(secret.initialization_vector);
return new ConverterInputStream(input, new SymmetricCipherDecrypter((owned) cipher, 16)); return new ConverterInputStream(input, new SymmetricCipherDecrypter((owned) cipher, 16));
} }
public OutputStream wrap_output_stream(OutputStream output, Jet.TransportSecret secret) requires (secret.transport_key.length == key_size) { public OutputStream wrap_output_stream(OutputStream output, Jet.TransportSecret secret) requires (secret.transport_key.length == key_size) {
Crypto.SymmetricCipher cipher = new SymmetricCipher.encryption("AES-GCM"); Crypto.SymmetricCipher cipher = new SymmetricCipher("AES-GCM");
cipher.set_key(secret.transport_key); cipher.set_key(secret.transport_key);
cipher.set_iv(secret.initialization_vector); cipher.set_iv(secret.initialization_vector);
return new ConverterOutputStream(output, new SymmetricCipherEncrypter((owned) cipher, 16)); return new ConverterOutputStream(output, new SymmetricCipherEncrypter((owned) cipher, 16));

View file

@ -12,7 +12,7 @@ public class Bundle {
assert(Plugin.ensure_context()); assert(Plugin.ensure_context());
} }
public int32 signed_pre_key_id { get { public int32 signed_pre_key_id { owned get {
if (node == null) return -1; if (node == null) return -1;
string? id = ((!)node).get_deep_attribute("signedPreKeyPublic", "signedPreKeyId"); string? id = ((!)node).get_deep_attribute("signedPreKeyPublic", "signedPreKeyId");
if (id == null) return -1; if (id == null) return -1;
@ -69,7 +69,7 @@ public class Bundle {
this.node = node; this.node = node;
} }
public int32 key_id { get { public int32 key_id { owned get {
return int.parse(node.get_attribute("preKeyId") ?? "-1"); return int.parse(node.get_attribute("preKeyId") ?? "-1");
}} }}

View file

@ -1,7 +1,7 @@
namespace Signal { namespace Signal {
public class Context { public class Context {
public NativeContext native_context; internal NativeContext native_context;
private RecMutex mutex = RecMutex(); private RecMutex mutex = RecMutex();
static void locking_function_lock(void* user_data) { static void locking_function_lock(void* user_data) {

View file

@ -1,11 +1,6 @@
#include "signal_helper.h" #include "signal_helper.h"
#ifdef GCRYPT
#include <gcrypt.h> #include <gcrypt.h>
#else
#include <openssl/evp.h>
#include <openssl/rand.h>
#endif
signal_type_base* signal_type_ref_vapi(void* instance) { signal_type_base* signal_type_ref_vapi(void* instance) {
g_return_val_if_fail(instance != NULL, NULL); g_return_val_if_fail(instance != NULL, NULL);
@ -70,31 +65,16 @@ void signal_protocol_address_set_device_id(signal_protocol_address* self, int32_
} }
int signal_vala_randomize(uint8_t *data, size_t len) { int signal_vala_randomize(uint8_t *data, size_t len) {
#ifdef GCRYPT
gcry_randomize(data, len, GCRY_STRONG_RANDOM); gcry_randomize(data, len, GCRY_STRONG_RANDOM);
return SG_SUCCESS; return SG_SUCCESS;
#else
return RAND_bytes(data, len) == 1 ? SG_SUCCESS : SG_ERR_UNKNOWN;
#endif
} }
int signal_vala_random_generator(uint8_t *data, size_t len, void *user_data) { int signal_vala_random_generator(uint8_t *data, size_t len, void *user_data) {
#ifdef GCRYPT
gcry_randomize(data, len, GCRY_STRONG_RANDOM); gcry_randomize(data, len, GCRY_STRONG_RANDOM);
return SG_SUCCESS; return SG_SUCCESS;
#else
return RAND_bytes(data, len) == 1 ? SG_SUCCESS : SG_ERR_UNKNOWN;
#endif
} }
#ifndef GCRYPT
struct SIGNAL_VALA_HMAC_CTX {
EVP_PKEY *pkey;
EVP_MD_CTX *ctx;
};
#endif
int signal_vala_hmac_sha256_init(void **hmac_context, const uint8_t *key, size_t key_len, void *user_data) { int signal_vala_hmac_sha256_init(void **hmac_context, const uint8_t *key, size_t key_len, void *user_data) {
#ifdef GCRYPT
gcry_mac_hd_t* ctx = malloc(sizeof(gcry_mac_hd_t)); gcry_mac_hd_t* ctx = malloc(sizeof(gcry_mac_hd_t));
if (!ctx) return SG_ERR_NOMEM; if (!ctx) return SG_ERR_NOMEM;
@ -111,49 +91,17 @@ int signal_vala_hmac_sha256_init(void **hmac_context, const uint8_t *key, size_t
*hmac_context = ctx; *hmac_context = ctx;
return SG_SUCCESS; return SG_SUCCESS;
#else
EVP_PKEY *pkey = EVP_PKEY_new_raw_private_key(EVP_PKEY_HMAC, NULL, key, key_len);
if (!pkey) {
return SG_ERR_NOMEM;
}
EVP_MD_CTX *ctx = EVP_MD_CTX_new();
if (!ctx) {
EVP_PKEY_free(pkey);
return SG_ERR_NOMEM;
}
if (EVP_DigestSignInit(ctx, NULL, EVP_sha256(), NULL, pkey) != 1) {
EVP_MD_CTX_free(ctx);
EVP_PKEY_free(pkey);
return SG_ERR_UNKNOWN;
}
struct SIGNAL_VALA_HMAC_CTX *hmac_ctx = malloc(sizeof(*hmac_ctx));
hmac_ctx->pkey = pkey;
hmac_ctx->ctx = ctx;
*hmac_context = hmac_ctx;
return SG_SUCCESS;
#endif
} }
int signal_vala_hmac_sha256_update(void *hmac_context, const uint8_t *data, size_t data_len, void *user_data) { int signal_vala_hmac_sha256_update(void *hmac_context, const uint8_t *data, size_t data_len, void *user_data) {
#ifdef GCRYPT
gcry_mac_hd_t* ctx = hmac_context; gcry_mac_hd_t* ctx = hmac_context;
if (gcry_mac_write(*ctx, data, data_len)) return SG_ERR_UNKNOWN; if (gcry_mac_write(*ctx, data, data_len)) return SG_ERR_UNKNOWN;
return SG_SUCCESS; return SG_SUCCESS;
#else
struct SIGNAL_VALA_HMAC_CTX *hmac_ctx = hmac_context;
if (EVP_DigestSignUpdate(hmac_ctx->ctx, data, data_len) != 1) {
return SG_ERR_UNKNOWN;
}
return SG_SUCCESS;
#endif
} }
int signal_vala_hmac_sha256_final(void *hmac_context, signal_buffer **output, void *user_data) { int signal_vala_hmac_sha256_final(void *hmac_context, signal_buffer **output, void *user_data) {
#ifdef GCRYPT
size_t len = gcry_mac_get_algo_maclen(GCRY_MAC_HMAC_SHA256); size_t len = gcry_mac_get_algo_maclen(GCRY_MAC_HMAC_SHA256);
uint8_t md[len]; uint8_t md[len];
gcry_mac_hd_t* ctx = hmac_context; gcry_mac_hd_t* ctx = hmac_context;
@ -166,49 +114,17 @@ int signal_vala_hmac_sha256_final(void *hmac_context, signal_buffer **output, vo
*output = output_buffer; *output = output_buffer;
return SG_SUCCESS; return SG_SUCCESS;
#else
size_t len;
struct SIGNAL_VALA_HMAC_CTX *hmac_ctx = hmac_context;
if (EVP_DigestSignFinal(hmac_ctx->ctx, NULL, &len) != 1) {
return SG_ERR_UNKNOWN;
}
signal_buffer *output_buffer = signal_buffer_alloc(len);
if (!output_buffer) {
return SG_ERR_NOMEM;
}
size_t another_len = len;
if (EVP_DigestSignFinal(hmac_ctx->ctx, signal_buffer_data(output_buffer), &another_len) != 1) {
signal_buffer_free(output_buffer);
return SG_ERR_UNKNOWN;
}
if (another_len != len) {
signal_buffer_free(output_buffer);
return SG_ERR_UNKNOWN;
}
*output = output_buffer;
return SG_SUCCESS;
#endif
} }
void signal_vala_hmac_sha256_cleanup(void *hmac_context, void *user_data) { void signal_vala_hmac_sha256_cleanup(void *hmac_context, void *user_data) {
#ifdef GCRYPT
gcry_mac_hd_t* ctx = hmac_context; gcry_mac_hd_t* ctx = hmac_context;
if (ctx) { if (ctx) {
gcry_mac_close(*ctx); gcry_mac_close(*ctx);
free(ctx); free(ctx);
} }
#else
struct SIGNAL_VALA_HMAC_CTX *hmac_ctx = hmac_context;
if (hmac_ctx) {
EVP_MD_CTX_free(hmac_ctx->ctx);
EVP_PKEY_free(hmac_ctx->pkey);
free(hmac_ctx);
}
#endif
} }
int signal_vala_sha512_digest_init(void **digest_context, void *user_data) { int signal_vala_sha512_digest_init(void **digest_context, void *user_data) {
#ifdef GCRYPT
gcry_md_hd_t* ctx = malloc(sizeof(gcry_mac_hd_t)); gcry_md_hd_t* ctx = malloc(sizeof(gcry_mac_hd_t));
if (!ctx) return SG_ERR_NOMEM; if (!ctx) return SG_ERR_NOMEM;
@ -220,38 +136,17 @@ int signal_vala_sha512_digest_init(void **digest_context, void *user_data) {
*digest_context = ctx; *digest_context = ctx;
return SG_SUCCESS; return SG_SUCCESS;
#else
EVP_MD_CTX *ctx = EVP_MD_CTX_new();
if (!ctx) {
return SG_ERR_NOMEM;
}
if (EVP_DigestInit_ex(ctx, EVP_sha512(), NULL) != 1) {
EVP_MD_CTX_free(ctx);
return SG_ERR_UNKNOWN;
}
*digest_context = ctx;
return SG_SUCCESS;
#endif
} }
int signal_vala_sha512_digest_update(void *digest_context, const uint8_t *data, size_t data_len, void *user_data) { int signal_vala_sha512_digest_update(void *digest_context, const uint8_t *data, size_t data_len, void *user_data) {
#ifdef GCRYPT
gcry_md_hd_t* ctx = digest_context; gcry_md_hd_t* ctx = digest_context;
gcry_md_write(*ctx, data, data_len); gcry_md_write(*ctx, data, data_len);
return SG_SUCCESS; return SG_SUCCESS;
#else
EVP_MD_CTX *ctx = digest_context;
if (EVP_DigestUpdate(ctx, data, data_len) != 1) {
return SG_ERR_UNKNOWN;
}
return SG_SUCCESS;
#endif
} }
int signal_vala_sha512_digest_final(void *digest_context, signal_buffer **output, void *user_data) { int signal_vala_sha512_digest_final(void *digest_context, signal_buffer **output, void *user_data) {
#ifdef GCRYPT
size_t len = gcry_md_get_algo_dlen(GCRY_MD_SHA512); size_t len = gcry_md_get_algo_dlen(GCRY_MD_SHA512);
gcry_md_hd_t* ctx = digest_context; gcry_md_hd_t* ctx = digest_context;
@ -267,43 +162,17 @@ int signal_vala_sha512_digest_final(void *digest_context, signal_buffer **output
*output = output_buffer; *output = output_buffer;
return SG_SUCCESS; return SG_SUCCESS;
#else
EVP_MD_CTX *ctx = digest_context;
size_t len = EVP_MD_size(EVP_sha512());
signal_buffer *output_buffer = signal_buffer_alloc(len);
if (!output_buffer) {
return SG_ERR_NOMEM;
}
if (EVP_DigestSignFinal(ctx, signal_buffer_data(output_buffer), &len) != 1) {
signal_buffer_free(output_buffer);
return SG_ERR_UNKNOWN;
}
if (len != EVP_MD_size(EVP_sha512())) {
signal_buffer_free(output_buffer);
return SG_ERR_UNKNOWN;
}
*output = output_buffer;
return SG_SUCCESS;
#endif
} }
void signal_vala_sha512_digest_cleanup(void *digest_context, void *user_data) { void signal_vala_sha512_digest_cleanup(void *digest_context, void *user_data) {
#ifdef GCRYPT
gcry_md_hd_t* ctx = digest_context; gcry_md_hd_t* ctx = digest_context;
if (ctx) { if (ctx) {
gcry_md_close(*ctx); gcry_md_close(*ctx);
free(ctx); free(ctx);
} }
#else
EVP_MD_CTX *ctx = digest_context;
if (ctx) {
EVP_MD_CTX_free(ctx);
}
#endif
} }
#ifdef GCRYPT const int aes_cipher(int cipher, size_t key_len, int* algo, int* mode) {
static int aes_cipher(int cipher, size_t key_len, int* algo, int* mode) {
switch (key_len) { switch (key_len) {
case 16: case 16:
*algo = GCRY_CIPHER_AES128; *algo = GCRY_CIPHER_AES128;
@ -332,35 +201,6 @@ static int aes_cipher(int cipher, size_t key_len, int* algo, int* mode) {
} }
return SG_SUCCESS; return SG_SUCCESS;
} }
#else
static const EVP_CIPHER *aes_cipher(int cipher, size_t key_len) {
switch (cipher) {
case SG_CIPHER_AES_CBC_PKCS5:
switch (key_len) {
case 16: return EVP_aes_128_cbc();
case 24: return EVP_aes_192_cbc();
case 32: return EVP_aes_256_cbc();
}
break;
case SG_CIPHER_AES_CTR_NOPADDING:
switch (key_len) {
case 16: return EVP_aes_128_ctr();
case 24: return EVP_aes_192_ctr();
case 32: return EVP_aes_256_ctr();
}
break;
case SG_CIPHER_AES_GCM_NOPADDING:
switch (key_len) {
case 16: return EVP_aes_128_gcm();
case 24: return EVP_aes_192_gcm();
case 32: return EVP_aes_256_gcm();
}
break;
}
return NULL;
}
#endif
int signal_vala_encrypt(signal_buffer **output, int signal_vala_encrypt(signal_buffer **output,
int cipher, int cipher,
@ -368,7 +208,6 @@ int signal_vala_encrypt(signal_buffer **output,
const uint8_t *iv, size_t iv_len, const uint8_t *iv, size_t iv_len,
const uint8_t *plaintext, size_t plaintext_len, const uint8_t *plaintext, size_t plaintext_len,
void *user_data) { void *user_data) {
#ifdef GCRYPT
int algo, mode, error_code = SG_ERR_UNKNOWN; int algo, mode, error_code = SG_ERR_UNKNOWN;
if (aes_cipher(cipher, key_len, &algo, &mode)) return SG_ERR_INVAL; if (aes_cipher(cipher, key_len, &algo, &mode)) return SG_ERR_INVAL;
@ -440,97 +279,6 @@ no_error:
gcry_cipher_close(ctx); gcry_cipher_close(ctx);
return SG_SUCCESS; return SG_SUCCESS;
#else
int result = 0;
uint8_t *out_buf = NULL;
const EVP_CIPHER *evp_cipher = aes_cipher(cipher, key_len);
if (!evp_cipher) {
// fprintf(stderr, "invalid AES mode or key size: %zu\n", key_len);
return SG_ERR_INVAL;
}
if (plaintext_len > INT_MAX - EVP_CIPHER_block_size(evp_cipher)) {
// fprintf(stderr, "invalid plaintext length: %zu\n", plaintext_len);
return SG_ERR_UNKNOWN;
}
EVP_CIPHER_CTX *ctx = EVP_CIPHER_CTX_new();
int buf_extra = 0;
if (cipher == SG_CIPHER_AES_GCM_NOPADDING) {
// In GCM mode we use the last 16 bytes as auth tag
buf_extra += 16;
if (EVP_EncryptInit_ex(ctx, evp_cipher, NULL, NULL, NULL) != 1) {
// fprintf(stderr, "cannot initialize cipher\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_SET_IVLEN, iv_len, NULL) != 1) {
// fprintf(stderr, "cannot set iv size\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (EVP_EncryptInit_ex(ctx, NULL, NULL, key, iv) != 1) {
// fprintf(stderr, "cannot set key/iv\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
} else {
// TODO: set ivlen?
if (EVP_EncryptInit_ex(ctx, evp_cipher, 0, key, iv) != 1) {
// fprintf(stderr, "cannot initialize cipher\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
if (cipher == SG_CIPHER_AES_CTR_NOPADDING || cipher == SG_CIPHER_AES_GCM_NOPADDING) {
if (EVP_CIPHER_CTX_set_padding(ctx, 0) != 1) {
// fprintf(stderr, "cannot set padding\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
out_buf = malloc(plaintext_len + EVP_CIPHER_block_size(evp_cipher) + buf_extra);
if (!out_buf) {
// fprintf(stderr, "cannot allocate output buffer\n");
result = SG_ERR_NOMEM;
goto complete;
}
int out_len = 0;
if (EVP_EncryptUpdate(ctx, out_buf, &out_len, plaintext, plaintext_len) != 1) {
// fprintf(stderr, "cannot encrypt plaintext\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
int final_len = 0;
if (EVP_EncryptFinal_ex(ctx, out_buf + out_len, &final_len) != 1) {
// fprintf(stderr, "cannot finish encrypting plaintext\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (cipher == SG_CIPHER_AES_GCM_NOPADDING) {
if (EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_GET_TAG, 16, out_buf + out_len + final_len) != 1) {
// fprintf(stderr, "cannot get tag\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
*output = signal_buffer_create(out_buf, out_len + final_len + buf_extra);
complete:
EVP_CIPHER_CTX_free(ctx);
if (out_buf) {
free(out_buf);
}
return result;
#endif
} }
int signal_vala_decrypt(signal_buffer **output, int signal_vala_decrypt(signal_buffer **output,
@ -539,7 +287,6 @@ int signal_vala_decrypt(signal_buffer **output,
const uint8_t *iv, size_t iv_len, const uint8_t *iv, size_t iv_len,
const uint8_t *ciphertext, size_t ciphertext_len, const uint8_t *ciphertext, size_t ciphertext_len,
void *user_data) { void *user_data) {
#ifdef GCRYPT
int algo, mode, error_code = SG_ERR_UNKNOWN; int algo, mode, error_code = SG_ERR_UNKNOWN;
*output = 0; *output = 0;
if (aes_cipher(cipher, key_len, &algo, &mode)) return SG_ERR_INVAL; if (aes_cipher(cipher, key_len, &algo, &mode)) return SG_ERR_INVAL;
@ -605,104 +352,11 @@ no_error:
gcry_cipher_close(ctx); gcry_cipher_close(ctx);
return SG_SUCCESS; return SG_SUCCESS;
#else
int result = 0;
uint8_t *out_buf = NULL;
const EVP_CIPHER *evp_cipher = aes_cipher(cipher, key_len);
if (!evp_cipher) {
// fprintf(stderr, "invalid AES mode or key size: %zu\n", key_len);
return SG_ERR_INVAL;
}
if (ciphertext_len > INT_MAX - EVP_CIPHER_block_size(evp_cipher)) {
// fprintf(stderr, "invalid ciphertext length: %zu\n", ciphertext_len);
return SG_ERR_UNKNOWN;
}
EVP_CIPHER_CTX *ctx = EVP_CIPHER_CTX_new();
if (cipher == SG_CIPHER_AES_GCM_NOPADDING) {
// In GCM mode we use the last 16 bytes as auth tag
ciphertext_len -= 16;
if (EVP_DecryptInit_ex(ctx, evp_cipher, NULL, NULL, NULL) != 1) {
// fprintf(stderr, "cannot initialize cipher\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_SET_IVLEN, iv_len, NULL) != 1) {
// fprintf(stderr, "cannot set iv size\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (EVP_DecryptInit_ex(ctx, NULL, NULL, key, iv) != 1) {
// fprintf(stderr, "cannot set key/iv\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
} else {
// TODO: set ivlen?
if (EVP_DecryptInit_ex(ctx, evp_cipher, 0, key, iv) != 1) {
// fprintf(stderr, "cannot initialize cipher\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
if (cipher == SG_CIPHER_AES_CTR_NOPADDING || cipher == SG_CIPHER_AES_GCM_NOPADDING) {
if (EVP_CIPHER_CTX_set_padding(ctx, 0) != 1) {
// fprintf(stderr, "cannot set padding\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
out_buf = malloc(ciphertext_len + EVP_CIPHER_block_size(evp_cipher));
if (!out_buf) {
// fprintf(stderr, "cannot allocate output buffer\n");
result = SG_ERR_NOMEM;
goto complete;
}
int out_len = 0;
if (EVP_DecryptUpdate(ctx, out_buf, &out_len, ciphertext, ciphertext_len) != 1) {
// fprintf(stderr, "cannot decrypt ciphertext\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
if (cipher == SG_CIPHER_AES_GCM_NOPADDING) {
if (EVP_CIPHER_CTX_ctrl(ctx, EVP_CTRL_GCM_SET_TAG, 16, (void *)(ciphertext + ciphertext_len)) != 1) {
// fprintf(stderr, "cannot set tag\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
}
int final_len = 0;
if (EVP_DecryptFinal_ex(ctx, out_buf + out_len, &final_len) != 1) {
// fprintf(stderr, "cannot finish decrypting ciphertexts\n");
result = SG_ERR_UNKNOWN;
goto complete;
}
*output = signal_buffer_create(out_buf, out_len + final_len);
complete:
EVP_CIPHER_CTX_free(ctx);
if (out_buf) {
free(out_buf);
}
return result;
#endif
} }
void setup_signal_vala_crypto_provider(signal_context *context) void setup_signal_vala_crypto_provider(signal_context *context)
{ {
#ifdef GCRYPT
gcry_check_version(NULL); gcry_check_version(NULL);
#endif
signal_crypto_provider provider = { signal_crypto_provider provider = {
.random_func = signal_vala_random_generator, .random_func = signal_vala_random_generator,

View file

@ -107,7 +107,7 @@ public class Store : Object {
public PreKeyStore pre_key_store { get; set; default = new SimplePreKeyStore(); } public PreKeyStore pre_key_store { get; set; default = new SimplePreKeyStore(); }
public SignedPreKeyStore signed_pre_key_store { get; set; default = new SimpleSignedPreKeyStore(); } public SignedPreKeyStore signed_pre_key_store { get; set; default = new SimpleSignedPreKeyStore(); }
public uint32 local_registration_id { get { return identity_key_store.local_registration_id; } } public uint32 local_registration_id { get { return identity_key_store.local_registration_id; } }
public NativeStoreContext native_context {get { return native_store_context_; }} internal NativeStoreContext native_context {get { return native_store_context_; }}
private NativeStoreContext native_store_context_; private NativeStoreContext native_store_context_;
static int iks_get_identity_key_pair(out Buffer public_data, out Buffer private_data, void* user_data) { static int iks_get_identity_key_pair(out Buffer public_data, out Buffer private_data, void* user_data) {

View file

@ -154,7 +154,7 @@ public class BadMessagesWidget : Box {
} else if (conversation.type_ == Conversation.Type.GROUPCHAT) { } else if (conversation.type_ == Conversation.Type.GROUPCHAT) {
who = jid.to_string(); who = jid.to_string();
// `jid` is a real JID. In MUCs, try to show nicks instead (given that the JID is currently online) // `jid` is a real JID. In MUCs, try to show nicks instead (given that the JID is currently online)
var occupants = plugin.app.stream_interactor.get_module(MucManager.IDENTITY).get_members(conversation.counterpart, conversation.account); var occupants = plugin.app.stream_interactor.get_module(MucManager.IDENTITY).get_occupants(conversation.counterpart, conversation.account);
if (occupants == null) return; if (occupants == null) return;
foreach (Jid occupant in occupants) { foreach (Jid occupant in occupants) {
if (jid.equals_bare(plugin.app.stream_interactor.get_module(MucManager.IDENTITY).get_real_jid(occupant, conversation.account))) { if (jid.equals_bare(plugin.app.stream_interactor.get_module(MucManager.IDENTITY).get_real_jid(occupant, conversation.account))) {

View file

@ -39,5 +39,5 @@ c_args = [
vala_args = [ vala_args = [
'--vapidir', meson.current_source_dir() / 'vapi', '--vapidir', meson.current_source_dir() / 'vapi',
] ]
lib_openpgp = shared_library('openpgp', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, include_directories: include_directories('src'), dependencies: dependencies, kwargs: install_options) lib_openpgp = shared_library('openpgp', sources, name_prefix: '', c_args: c_args, vala_args: vala_args, include_directories: include_directories('src'), dependencies: dependencies, install: true, install_dir: get_option('libdir') / 'dino/plugins')
dep_openpgp = declare_dependency(link_with: lib_openpgp, include_directories: include_directories('.')) dep_openpgp = declare_dependency(link_with: lib_openpgp, include_directories: include_directories('.'))

View file

@ -1,10 +1,9 @@
msgid "" msgid ""
msgstr "" msgstr ""
"Project-Id-Version: PACKAGE VERSION\n" "Project-Id-Version: dino-openpgp-0.0\n"
"Report-Msgid-Bugs-To: \n" "Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2024-06-09 22:16+0200\n" "POT-Creation-Date: 2023-02-07 21:31+0100\n"
"PO-Revision-Date: 2024-06-09 22:39+0200\n" "PO-Revision-Date: 2020-04-16 20:11+0000\n"
"Last-Translator: eerielili \n"
"Language-Team: German <https://hosted.weblate.org/projects/dino/plugin-" "Language-Team: German <https://hosted.weblate.org/projects/dino/plugin-"
"openpgp/de/>\n" "openpgp/de/>\n"
"Language: de\n" "Language: de\n"
@ -12,11 +11,11 @@ msgstr ""
"Content-Type: text/plain; charset=UTF-8\n" "Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n" "Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=n != 1;\n" "Plural-Forms: nplurals=2; plural=n != 1;\n"
"X-Generator: Poedit 3.4.2\n" "X-Generator: Weblate 4.0.1-dev\n"
#: plugins/openpgp/src/account_settings_entry.vala:68 #: plugins/openpgp/src/account_settings_entry.vala:68
#: plugins/openpgp/src/account_settings_entry.vala:72 #: plugins/openpgp/src/account_settings_entry.vala:72
#: plugins/openpgp/src/account_settings_entry.vala:113 #: plugins/openpgp/src/account_settings_entry.vala:101
msgid "Key publishing disabled" msgid "Key publishing disabled"
msgstr "Schlüsselveröffentlichung deaktiviert" msgstr "Schlüsselveröffentlichung deaktiviert"
@ -25,40 +24,18 @@ msgid "Error in GnuPG"
msgstr "Fehler in GnuPG" msgstr "Fehler in GnuPG"
#: plugins/openpgp/src/account_settings_entry.vala:72 #: plugins/openpgp/src/account_settings_entry.vala:72
msgid "" msgid "No keys available. Generate one!"
"No keys available. Generate one or check if your keys aren't expired or " msgstr "Keine Schlüssel vorhanden. Erzeuge einen!"
"revoked!"
msgstr ""
"Keine Schlüssel vorhanden. Erstellen Sie einen oder prüfen Sie,"
"ob Ihre Schlüssel nicht abgelaufen sind oder widerrufen!"
#: plugins/openpgp/src/account_settings_entry.vala:101
#: plugins/openpgp/src/account_settings_entry.vala:95
msgid "expired!"
msgstr "abgelaufen"
#: plugins/openpgp/src/account_settings_entry.vala:95
msgid "revoked!"
msgstr "widerrufen!"
#: plugins/openpgp/src/account_settings_entry.vala:96
msgid "Attention required!"
msgstr "Achtung!"
#: plugins/openpgp/src/account_settings_entry.vala:96
#, c-format
msgid "Your key %s is %s"
msgstr "Ihr Schlüssel %s is %s"
#: plugins/openpgp/src/account_settings_entry.vala:113
msgid "Select key" msgid "Select key"
msgstr "Wähle einen Schlüssel" msgstr "Wähle einen Schlüssel"
#: plugins/openpgp/src/account_settings_entry.vala:126 #: plugins/openpgp/src/account_settings_entry.vala:114
msgid "Loading…" msgid "Loading…"
msgstr "Lade…" msgstr "Lade…"
#: plugins/openpgp/src/account_settings_entry.vala:126 #: plugins/openpgp/src/account_settings_entry.vala:114
msgid "Querying GnuPG" msgid "Querying GnuPG"
msgstr "Frage GnuPG ab" msgstr "Frage GnuPG ab"
@ -69,3 +46,6 @@ msgstr "Schlüssel nicht im Schlüsselbund"
#: plugins/openpgp/src/contact_details_provider.vala:30 #: plugins/openpgp/src/contact_details_provider.vala:30
msgid "Encryption" msgid "Encryption"
msgstr "Verschlüsselung" msgstr "Verschlüsselung"
#~ msgid "OpenPGP"
#~ msgstr "OpenPGP"

View file

@ -1,68 +1,52 @@
# SOME DESCRIPTIVE TITLE.
# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
# This file is distributed under the same license as the PACKAGE package.
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
#
#, fuzzy
msgid "" msgid ""
msgstr "" msgstr ""
"Project-Id-Version: PACKAGE VERSION+\n" "Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2024-06-09 22:16+0200\n" "Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2024-06-09 22:18+0200\n" "POT-Creation-Date: 2023-02-07 21:31+0100\n"
"Last-Translator: \n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Language-Team: Dino+\n" "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language: en\n" "Language-Team: LANGUAGE <LL@li.org>\n"
"Language: \n"
"MIME-Version: 1.0\n" "MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n" "Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n" "Content-Transfer-Encoding: 8bit\n"
"X-Generator: Poedit 3.4.2\n"
"X-Poedit-Basepath: ../src\n"
"X-Poedit-KeywordsList: _(\n"
"X-Poedit-SourceCharset: UTF-8\n"
"X-Poedit-SearchPath-0: .\n"
#: account_settings_entry.vala:68 account_settings_entry.vala:72 #: plugins/openpgp/src/account_settings_entry.vala:68
#: account_settings_entry.vala:113 #: plugins/openpgp/src/account_settings_entry.vala:72
#: plugins/openpgp/src/account_settings_entry.vala:101
msgid "Key publishing disabled" msgid "Key publishing disabled"
msgstr "" msgstr ""
#: account_settings_entry.vala:68 #: plugins/openpgp/src/account_settings_entry.vala:68
msgid "Error in GnuPG" msgid "Error in GnuPG"
msgstr "" msgstr ""
#: account_settings_entry.vala:72 #: plugins/openpgp/src/account_settings_entry.vala:72
msgid "" msgid "No keys available. Generate one!"
"No keys available. Generate one or check if your keys aren't expired or "
"revoked!"
msgstr "" msgstr ""
#: account_settings_entry.vala:95 #: plugins/openpgp/src/account_settings_entry.vala:101
msgid "expired!"
msgstr ""
#: account_settings_entry.vala:95
msgid "revoked!"
msgstr ""
#: account_settings_entry.vala:96
msgid "Attention required!"
msgstr ""
#: account_settings_entry.vala:96
#, c-format
msgid "Your key %s is %s"
msgstr ""
#: account_settings_entry.vala:113
msgid "Select key" msgid "Select key"
msgstr "" msgstr ""
#: account_settings_entry.vala:126 #: plugins/openpgp/src/account_settings_entry.vala:114
msgid "Loading…" msgid "Loading…"
msgstr "" msgstr ""
#: account_settings_entry.vala:126 #: plugins/openpgp/src/account_settings_entry.vala:114
msgid "Querying GnuPG" msgid "Querying GnuPG"
msgstr "" msgstr ""
#: contact_details_provider.vala:28 #: plugins/openpgp/src/contact_details_provider.vala:28
msgid "Key not in keychain" msgid "Key not in keychain"
msgstr "" msgstr ""
#: contact_details_provider.vala:30 #: plugins/openpgp/src/contact_details_provider.vala:30
msgid "Encryption" msgid "Encryption"
msgstr "" msgstr ""

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