2 Copyright (C) 2015-2020 AsamK and contributors
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>.
17 package org
.asamk
.signal
;
19 import net
.sourceforge
.argparse4j
.ArgumentParsers
;
20 import net
.sourceforge
.argparse4j
.impl
.Arguments
;
21 import net
.sourceforge
.argparse4j
.inf
.ArgumentParser
;
22 import net
.sourceforge
.argparse4j
.inf
.ArgumentParserException
;
23 import net
.sourceforge
.argparse4j
.inf
.MutuallyExclusiveGroup
;
24 import net
.sourceforge
.argparse4j
.inf
.Namespace
;
25 import net
.sourceforge
.argparse4j
.inf
.Subparser
;
26 import net
.sourceforge
.argparse4j
.inf
.Subparsers
;
28 import org
.asamk
.Signal
;
29 import org
.asamk
.signal
.commands
.Command
;
30 import org
.asamk
.signal
.commands
.Commands
;
31 import org
.asamk
.signal
.commands
.DbusCommand
;
32 import org
.asamk
.signal
.commands
.ExtendedDbusCommand
;
33 import org
.asamk
.signal
.commands
.LocalCommand
;
34 import org
.asamk
.signal
.commands
.ProvisioningCommand
;
35 import org
.asamk
.signal
.dbus
.DbusSignalImpl
;
36 import org
.asamk
.signal
.manager
.Manager
;
37 import org
.asamk
.signal
.manager
.ProvisioningManager
;
38 import org
.asamk
.signal
.manager
.ServiceConfig
;
39 import org
.asamk
.signal
.util
.IOUtils
;
40 import org
.asamk
.signal
.util
.SecurityProvider
;
41 import org
.bouncycastle
.jce
.provider
.BouncyCastleProvider
;
42 import org
.freedesktop
.dbus
.connections
.impl
.DBusConnection
;
43 import org
.freedesktop
.dbus
.exceptions
.DBusException
;
44 import org
.whispersystems
.signalservice
.api
.push
.exceptions
.AuthorizationFailedException
;
45 import org
.whispersystems
.signalservice
.api
.util
.PhoneNumberFormatter
;
46 import org
.whispersystems
.signalservice
.internal
.configuration
.SignalServiceConfiguration
;
49 import java
.io
.IOException
;
50 import java
.security
.Security
;
53 import static org
.whispersystems
.signalservice
.internal
.util
.Util
.isEmpty
;
57 public static void main(String
[] args
) {
58 installSecurityProviderWorkaround();
60 Namespace ns
= parseArgs(args
);
65 int res
= handleCommands(ns
);
69 public static void installSecurityProviderWorkaround() {
70 // Register our own security provider
71 Security
.insertProviderAt(new SecurityProvider(), 1);
72 Security
.addProvider(new BouncyCastleProvider());
75 private static int handleCommands(Namespace ns
) {
76 final String username
= ns
.getString("username");
78 if (ns
.getBoolean("dbus") || ns
.getBoolean("dbus_system")) {
80 DBusConnection
.DBusBusType busType
;
81 if (ns
.getBoolean("dbus_system")) {
82 busType
= DBusConnection
.DBusBusType
.SYSTEM
;
84 busType
= DBusConnection
.DBusBusType
.SESSION
;
86 try (DBusConnection dBusConn
= DBusConnection
.getConnection(busType
)) {
87 Signal ts
= dBusConn
.getRemoteObject(DbusConfig
.SIGNAL_BUSNAME
,
88 DbusConfig
.SIGNAL_OBJECTPATH
,
91 return handleCommands(ns
, ts
, dBusConn
);
93 } catch (UnsatisfiedLinkError e
) {
94 System
.err
.println("Missing native library dependency for dbus service: " + e
.getMessage());
96 } catch (DBusException
| IOException e
) {
101 String dataPath
= ns
.getString("config");
102 if (isEmpty(dataPath
)) {
103 dataPath
= getDefaultDataPath();
106 final SignalServiceConfiguration serviceConfiguration
= ServiceConfig
.createDefaultServiceConfiguration(
107 BaseConfig
.USER_AGENT
);
109 if (username
== null) {
110 ProvisioningManager pm
= new ProvisioningManager(dataPath
, serviceConfiguration
, BaseConfig
.USER_AGENT
);
111 return handleCommands(ns
, pm
);
116 manager
= Manager
.init(username
, dataPath
, serviceConfiguration
, BaseConfig
.USER_AGENT
);
117 } catch (Throwable e
) {
118 System
.err
.println("Error loading state file: " + e
.getMessage());
122 try (Manager m
= manager
) {
124 m
.checkAccountState();
125 } catch (AuthorizationFailedException e
) {
126 if (!"register".equals(ns
.getString("command"))) {
127 // Register command should still be possible, if current authorization fails
128 System
.err
.println("Authorization failed, was the number registered elsewhere?");
131 } catch (IOException e
) {
132 System
.err
.println("Error while checking account: " + e
.getMessage());
136 return handleCommands(ns
, m
);
137 } catch (IOException e
) {
144 private static int handleCommands(Namespace ns
, Signal ts
, DBusConnection dBusConn
) {
145 String commandKey
= ns
.getString("command");
146 final Map
<String
, Command
> commands
= Commands
.getCommands();
147 if (commands
.containsKey(commandKey
)) {
148 Command command
= commands
.get(commandKey
);
150 if (command
instanceof ExtendedDbusCommand
) {
151 return ((ExtendedDbusCommand
) command
).handleCommand(ns
, ts
, dBusConn
);
152 } else if (command
instanceof DbusCommand
) {
153 return ((DbusCommand
) command
).handleCommand(ns
, ts
);
155 System
.err
.println(commandKey
+ " is not yet implemented via dbus");
162 private static int handleCommands(Namespace ns
, ProvisioningManager pm
) {
163 String commandKey
= ns
.getString("command");
164 final Map
<String
, Command
> commands
= Commands
.getCommands();
165 if (commands
.containsKey(commandKey
)) {
166 Command command
= commands
.get(commandKey
);
168 if (command
instanceof ProvisioningCommand
) {
169 return ((ProvisioningCommand
) command
).handleCommand(ns
, pm
);
171 System
.err
.println(commandKey
+ " only works with a username");
178 private static int handleCommands(Namespace ns
, Manager m
) {
179 String commandKey
= ns
.getString("command");
180 final Map
<String
, Command
> commands
= Commands
.getCommands();
181 if (commands
.containsKey(commandKey
)) {
182 Command command
= commands
.get(commandKey
);
184 if (command
instanceof LocalCommand
) {
185 return ((LocalCommand
) command
).handleCommand(ns
, m
);
186 } else if (command
instanceof DbusCommand
) {
187 return ((DbusCommand
) command
).handleCommand(ns
, new DbusSignalImpl(m
));
188 } else if (command
instanceof ExtendedDbusCommand
) {
189 System
.err
.println(commandKey
+ " only works via dbus");
197 * Uses $XDG_DATA_HOME/signal-cli if it exists, or if none of the legacy directories exist:
198 * - $HOME/.config/signal
199 * - $HOME/.config/textsecure
201 * @return the data directory to be used by signal-cli.
203 private static String
getDefaultDataPath() {
204 String dataPath
= IOUtils
.getDataHomeDir() + "/signal-cli";
205 if (new File(dataPath
).exists()) {
209 String legacySettingsPath
= System
.getProperty("user.home") + "/.config/signal";
210 if (new File(legacySettingsPath
).exists()) {
211 return legacySettingsPath
;
214 legacySettingsPath
= System
.getProperty("user.home") + "/.config/textsecure";
215 if (new File(legacySettingsPath
).exists()) {
216 return legacySettingsPath
;
222 private static Namespace
parseArgs(String
[] args
) {
223 ArgumentParser parser
= ArgumentParsers
.newFor("signal-cli")
226 .description("Commandline interface for Signal.")
227 .version(BaseConfig
.PROJECT_NAME
+ " " + BaseConfig
.PROJECT_VERSION
);
229 parser
.addArgument("-v", "--version").help("Show package version.").action(Arguments
.version());
230 parser
.addArgument("--config")
231 .help("Set the path, where to store the config (Default: $XDG_DATA_HOME/signal-cli , $HOME/.local/share/signal-cli).");
233 MutuallyExclusiveGroup mut
= parser
.addMutuallyExclusiveGroup();
234 mut
.addArgument("-u", "--username").help("Specify your phone number, that will be used for verification.");
235 mut
.addArgument("--dbus").help("Make request via user dbus.").action(Arguments
.storeTrue());
236 mut
.addArgument("--dbus-system").help("Make request via system dbus.").action(Arguments
.storeTrue());
238 Subparsers subparsers
= parser
.addSubparsers()
239 .title("subcommands")
241 .description("valid subcommands")
242 .help("additional help");
244 final Map
<String
, Command
> commands
= Commands
.getCommands();
245 for (Map
.Entry
<String
, Command
> entry
: commands
.entrySet()) {
246 Subparser subparser
= subparsers
.addParser(entry
.getKey());
247 entry
.getValue().attachToSubparser(subparser
);
252 ns
= parser
.parseArgs(args
);
253 } catch (ArgumentParserException e
) {
254 parser
.handleError(e
);
258 if ("link".equals(ns
.getString("command"))) {
259 if (ns
.getString("username") != null) {
261 System
.err
.println("You cannot specify a username (phone number) when linking");
264 } else if (!ns
.getBoolean("dbus") && !ns
.getBoolean("dbus_system")) {
265 if (ns
.getString("username") == null) {
267 System
.err
.println("You need to specify a username (phone number)");
270 if (!PhoneNumberFormatter
.isValidNumber(ns
.getString("username"), null)) {
271 System
.err
.println("Invalid username (phone number), make sure you include the country code.");
275 if (ns
.getList("recipient") != null && !ns
.getList("recipient").isEmpty() && ns
.getString("group") != null) {
276 System
.err
.println("You cannot specify recipients by phone number and groups at the same time");