Commit cddf6b58 authored by Grigory Fedorov's avatar Grigory Fedorov

Merge branch 'feature/otr_fix' into develop

parents d5d31d71 a26b1857
......@@ -74,22 +74,17 @@ import java.util.concurrent.ThreadFactory;
* @author alexander.ivanov
*/
public class OTRManager implements OtrEngineHost, OtrEngineListener,
OnLoadListener, OnAccountAddedListener, OnAccountRemovedListener,
OnCloseListener {
OnLoadListener, OnAccountAddedListener, OnAccountRemovedListener, OnCloseListener {
private final static OTRManager instance;
private static Map<SecurityOtrMode, OtrPolicy> POLICIES;
static {
POLICIES = new HashMap<SettingsManager.SecurityOtrMode, OtrPolicy>();
POLICIES.put(SecurityOtrMode.disabled, new OtrPolicy(
OtrPolicy.NEVER));
POLICIES.put(SecurityOtrMode.manual, new OtrPolicy(
OtrPolicy.OTRL_POLICY_MANUAL & ~OtrPolicy.ALLOW_V1));
POLICIES.put(SecurityOtrMode.auto, new OtrPolicy(
OtrPolicy.OPPORTUNISTIC & ~OtrPolicy.ALLOW_V1));
POLICIES.put(SecurityOtrMode.required, new OtrPolicy(
OtrPolicy.OTRL_POLICY_ALWAYS & ~OtrPolicy.ALLOW_V1));
POLICIES = new HashMap<>();
POLICIES.put(SecurityOtrMode.disabled, new OtrPolicy(OtrPolicy.NEVER));
POLICIES.put(SecurityOtrMode.manual, new OtrPolicy(OtrPolicy.OTRL_POLICY_MANUAL & ~OtrPolicy.ALLOW_V1));
POLICIES.put(SecurityOtrMode.auto, new OtrPolicy(OtrPolicy.OPPORTUNISTIC & ~OtrPolicy.ALLOW_V1));
POLICIES.put(SecurityOtrMode.required, new OtrPolicy(OtrPolicy.OTRL_POLICY_ALWAYS & ~OtrPolicy.ALLOW_V1));
}
static {
......@@ -104,8 +99,7 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
*/
private final NestedNestedMaps<String, Boolean> fingerprints;
/**
* Fingerprint of encrypted or encrypted and verified session for user in
* account.
* Fingerprint of encrypted or encrypted and verified session for user in account.
*/
private final NestedMap<String> actives;
/**
......@@ -122,21 +116,17 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
private final ExecutorService keyPairGenerator;
private OTRManager() {
smRequestProvider = new EntityNotificationProvider<SMRequest>(
R.drawable.ic_stat_ic_help_black);
smProgressProvider = new EntityNotificationProvider<SMProgress>(
R.drawable.ic_stat_ic_play_circle_fill);
smRequestProvider = new EntityNotificationProvider<>(R.drawable.ic_stat_ic_help_black);
smProgressProvider = new EntityNotificationProvider<>(R.drawable.ic_stat_ic_play_circle_fill);
smProgressProvider.setCanClearNotifications(false);
fingerprints = new NestedNestedMaps<String, Boolean>();
actives = new NestedMap<String>();
finished = new NestedMap<Boolean>();
sessions = new NestedMap<Session>();
keyPairGenerator = Executors
.newSingleThreadExecutor(new ThreadFactory() {
fingerprints = new NestedNestedMaps<>();
actives = new NestedMap<>();
finished = new NestedMap<>();
sessions = new NestedMap<>();
keyPairGenerator = Executors.newSingleThreadExecutor(new ThreadFactory() {
@Override
public Thread newThread(Runnable runnable) {
Thread thread = new Thread(runnable,
"Key pair generator service");
Thread thread = new Thread(runnable, "Key pair generator service");
thread.setPriority(Thread.MIN_PRIORITY);
thread.setDaemon(true);
return thread;
......@@ -150,7 +140,7 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
@Override
public void onLoad() {
final NestedNestedMaps<String, Boolean> fingerprints = new NestedNestedMaps<String, Boolean>();
final NestedNestedMaps<String, Boolean> fingerprints = new NestedNestedMaps<>();
Cursor cursor = OTRTable.getInstance().list();
try {
if (cursor.moveToFirst()) {
......@@ -175,133 +165,120 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
private void onLoaded(NestedNestedMaps<String, Boolean> fingerprints) {
this.fingerprints.addAll(fingerprints);
NotificationManager.getInstance().registerNotificationProvider(
smRequestProvider);
NotificationManager.getInstance().registerNotificationProvider(
smProgressProvider);
NotificationManager.getInstance().registerNotificationProvider(smRequestProvider);
NotificationManager.getInstance().registerNotificationProvider(smProgressProvider);
}
private Session getOrCreateSession(String account, String user) {
Session session = sessions.get(account, user);
if (session != null)
return session;
AccountItem accountItem = AccountManager.getInstance().getAccount(
account);
session = new Session(new SessionID(account, user,
accountItem == null ? "" : accountItem.getConnectionSettings()
.getProtocol().toString()), this);
session.addOtrEngineListener(this);
sessions.put(account, user, session);
return session;
}
public void startSession(String account, String user)
throws NetworkException {
public void startSession(String account, String user) throws NetworkException {
LogManager.i(this, "Starting session for " + user);
try {
getOrCreateSession(account, user).startSession();
} catch (OtrException e) {
throw new NetworkException(R.string.OTR_ERROR, e);
}
LogManager.i(this, "Started session for " + user);
}
public void refreshSession(String account, String user)
throws NetworkException {
public void refreshSession(String account, String user) throws NetworkException {
LogManager.i(this, "Refreshing session for " + user);
try {
getOrCreateSession(account, user).refreshSession();
} catch (OtrException e) {
throw new NetworkException(R.string.OTR_ERROR, e);
}
LogManager.i(this, "Refreshed session for " + user);
}
public void endSession(String account, String user) throws NetworkException {
LogManager.i(this, "Ending session for " + user);
try {
getOrCreateSession(account, user).endSession();
} catch (OtrException e) {
throw new NetworkException(R.string.OTR_ERROR, e);
}
AbstractChat abstractChat = MessageManager.getInstance().getChat(
account, user);
MessageArchiveManager.getInstance().setSaveMode(account, user,
abstractChat.getThreadId(), SaveMode.body);
SSNManager.getInstance().setSessionOtrMode(account, user,
abstractChat.getThreadId(), OtrMode.concede);
}
private void injectMessage(String account, String user, String msg)
throws OtrException {
AbstractChat abstractChat = MessageManager.getInstance().getChat(
account, user);
AbstractChat abstractChat = MessageManager.getInstance().getChat(account, user);
MessageArchiveManager.getInstance().setSaveMode(account, user, abstractChat.getThreadId(), SaveMode.body);
SSNManager.getInstance().setSessionOtrMode(account, user, abstractChat.getThreadId(), OtrMode.concede);
LogManager.i(this, "Ended session for " + user);
}
private Session getOrCreateSession(String account, String user) {
Session session = sessions.get(account, user);
if (session != null) {
LogManager.i(this, "Found session with id " + session.getSessionID() + " with status " + session.getSessionStatus() + " for user " + user);
return session;
}
LogManager.i(this, "Creating new session for " + user);
AccountItem accountItem = AccountManager.getInstance().getAccount(account);
session = new Session(new SessionID(account, user,
accountItem == null ? "" : accountItem.getConnectionSettings().getProtocol().toString()), this);
session.addOtrEngineListener(this);
sessions.put(account, user, session);
return session;
}
@Override
public void injectMessage(SessionID sessionID, String msg) throws OtrException {
injectMessage(sessionID.getAccountID(), sessionID.getUserID(), msg);
}
private void injectMessage(String account, String user, String msg) throws OtrException {
LogManager.i(this, "injectMessage. user: " + user + " message: " + msg);
AbstractChat abstractChat = MessageManager.getInstance().getChat(account, user);
try {
MessageArchiveManager.getInstance().setSaveMode(account, user,
abstractChat.getThreadId(), SaveMode.fls);
MessageArchiveManager.getInstance().setSaveMode(account, user, abstractChat.getThreadId(), SaveMode.fls);
} catch (NetworkException e) {
throw new OtrException(e);
}
SSNManager.getInstance().setSessionOtrMode(account, user,
abstractChat.getThreadId(), OtrMode.prefer);
SSNManager.getInstance().setSessionOtrMode(account, user, abstractChat.getThreadId(), OtrMode.prefer);
try {
ConnectionManager.getInstance().sendPacket(
abstractChat.getAccount(),
abstractChat.createMessagePacket(msg));
ConnectionManager.getInstance()
.sendPacket(abstractChat.getAccount(), abstractChat.createMessagePacket(msg));
} catch (NetworkException e) {
throw new OtrException(e);
}
}
@Override
public void injectMessage(SessionID sessionID, String msg)
throws OtrException {
injectMessage(sessionID.getAccountID(), sessionID.getUserID(), msg);
public void unreadableMessageReceived(SessionID sessionID) throws OtrException {
LogManager.i(this, "unreadableMessageReceived");
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_unreadable);
}
/**
* Creates new action in specified chat.
*
* @param account
* @param user
* @param text
* @param action
*/
private void newAction(String account, String user, String text,
ChatAction action) {
MessageManager.getInstance().getChat(account, user)
.newAction(null, text, action);
}
@Override
public void unreadableMessageReceived(SessionID sessionID)
throws OtrException {
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_unreadable);
private void newAction(String account, String user, String text, ChatAction action) {
LogManager.i(this, "newAction. text: " + text + " action " + action);
MessageManager.getInstance().getChat(account, user).newAction(null, text, action);
}
@Override
public String getReplyForUnreadableMessage(SessionID sessionID) {
return Application.getInstance().getString(
R.string.otr_unreadable_message);
return Application.getInstance().getString(R.string.otr_unreadable_message);
}
@Override
public void unencryptedMessageReceived(SessionID sessionID, String msg)
throws OtrException {
public void unencryptedMessageReceived(SessionID sessionID, String msg) throws OtrException {
LogManager.i(this, "unencrypted Message Received. " + msg);
throw new OtrException(new OTRUnencryptedException(msg));
}
@Override
public void showError(SessionID sessionID, String error)
throws OtrException {
newAction(sessionID.getAccountID(), sessionID.getUserID(), error,
ChatAction.otr_error);
public void showError(SessionID sessionID, String error) throws OtrException {
LogManager.i(this, "ShowError: " + error);
newAction(sessionID.getAccountID(), sessionID.getUserID(), error, ChatAction.otr_error);
}
@Override
public void smpError(SessionID sessionID, int tlvType, boolean cheated)
throws OtrException {
public void smpError(SessionID sessionID, int tlvType, boolean cheated) throws OtrException {
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
cheated ? ChatAction.otr_smp_cheated
: ChatAction.otr_smp_failed);
if (cheated)
cheated ? ChatAction.otr_smp_cheated : ChatAction.otr_smp_failed);
if (cheated) {
removeSMProgress(sessionID.getAccountID(), sessionID.getUserID());
}
}
@Override
......@@ -312,18 +289,14 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
@Override
public void finishedSessionMessage(SessionID sessionID, String msgText) throws OtrException {
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_finished_session);
throw new OtrException(
new IllegalStateException(
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_finished_session);
throw new OtrException(new IllegalStateException(
"Prevent from null to be returned. Just process it as regular exception."));
}
@Override
public void requireEncryptedMessage(SessionID sessionID, String msgText)
throws OtrException {
throw new OtrException(
new IllegalStateException(
public void requireEncryptedMessage(SessionID sessionID, String msgText) throws OtrException {
throw new OtrException(new IllegalStateException(
"Prevent from null to be returned. Just process it as regular exception."));
}
......@@ -333,11 +306,10 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
}
private KeyPair getLocalKeyPair(String account) throws OtrException {
KeyPair keyPair = AccountManager.getInstance().getAccount(account)
.getKeyPair();
if (keyPair == null)
throw new OtrException(new IllegalStateException(
"KeyPair is not ready, yet."));
KeyPair keyPair = AccountManager.getInstance().getAccount(account).getKeyPair();
if (keyPair == null) {
throw new OtrException(new IllegalStateException("KeyPair is not ready, yet."));
}
return keyPair;
}
......@@ -350,9 +322,11 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
public void sessionStatusChanged(SessionID sessionID) {
removeSMRequest(sessionID.getAccountID(), sessionID.getUserID());
removeSMProgress(sessionID.getAccountID(), sessionID.getUserID());
Session session = sessions.get(sessionID.getAccountID(),
sessionID.getUserID());
Session session = sessions.get(sessionID.getAccountID(), sessionID.getUserID());
SessionStatus sStatus = session.getSessionStatus();
LogManager.i(this, "session status changed " + sessionID.getUserID() + " status: " + sStatus);
if (sStatus == SessionStatus.ENCRYPTED) {
finished.remove(sessionID.getAccountID(), sessionID.getUserID());
PublicKey remotePublicKey = session.getRemotePublicKey();
......@@ -364,66 +338,48 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
value = null;
}
if (value != null) {
actives.put(sessionID.getAccountID(), sessionID.getUserID(),
value);
if (fingerprints.get(sessionID.getAccountID(),
sessionID.getUserID(), value) == null) {
fingerprints.put(sessionID.getAccountID(),
sessionID.getUserID(), value, false);
requestToWrite(sessionID.getAccountID(),
sessionID.getUserID(), value, false);
actives.put(sessionID.getAccountID(), sessionID.getUserID(), value);
if (fingerprints.get(sessionID.getAccountID(), sessionID.getUserID(), value) == null) {
fingerprints.put(sessionID.getAccountID(), sessionID.getUserID(), value, false);
requestToWrite(sessionID.getAccountID(), sessionID.getUserID(), value, false);
}
}
newAction(
sessionID.getAccountID(),
sessionID.getUserID(),
null,
isVerified(sessionID.getAccountID(), sessionID.getUserID()) ? ChatAction.otr_verified
: ChatAction.otr_encryption);
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, isVerified(sessionID.getAccountID(),
sessionID.getUserID()) ? ChatAction.otr_verified : ChatAction.otr_encryption);
MessageManager.getInstance()
.getChat(sessionID.getAccountID(), sessionID.getUserID())
.sendMessages();
.getChat(sessionID.getAccountID(), sessionID.getUserID()).sendMessages();
} else if (sStatus == SessionStatus.PLAINTEXT) {
actives.remove(sessionID.getAccountID(), sessionID.getUserID());
sessions.remove(sessionID.getAccountID(), sessionID.getUserID());
finished.remove(sessionID.getAccountID(), sessionID.getUserID());
try {
session.endSession();
} catch (OtrException e) {
LogManager.exception(this, e);
}
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_plain);
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_plain);
} else if (sStatus == SessionStatus.FINISHED) {
actives.remove(sessionID.getAccountID(), sessionID.getUserID());
sessions.remove(sessionID.getAccountID(), sessionID.getUserID());
finished.put(sessionID.getAccountID(), sessionID.getUserID(), true);
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_finish);
} else
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_finish);
} else {
throw new IllegalStateException();
RosterManager.getInstance().onContactChanged(sessionID.getAccountID(),
sessionID.getUserID());
}
RosterManager.getInstance().onContactChanged(sessionID.getAccountID(), sessionID.getUserID());
}
@Override
public void askForSecret(SessionID sessionID, InstanceTag receiverTag, String question) {
smRequestProvider.add(
new SMRequest(sessionID.getAccountID(), sessionID.getUserID(),
question), true);
smRequestProvider.add(new SMRequest(sessionID.getAccountID(), sessionID.getUserID(), question), true);
}
/**
* Transform outgoing message before sending.
*
* @param account
* @param user
* @param content
* @return
* @throws OtrException
*/
public String transformSending(String account, String user, String content)
throws OtrException {
String parts[] = getOrCreateSession(account, user)
.transformSending(content, null);
public String transformSending(String account, String user, String content) throws OtrException {
LogManager.i(this, "transform outgoing message... " + user);
String parts[] = getOrCreateSession(account, user).transformSending(content, null);
if (BuildConfig.DEBUG && parts.length != 1) {
throw new RuntimeException(
"We do not use fragmentation, so there must be only one otr fragment.");
......@@ -433,18 +389,14 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
/**
* Transform incoming message after receiving.
*
* @param account
* @param user
* @param content
* @return
* @throws OtrException
*/
public String transformReceiving(String account, String user, String content)
throws OtrException {
public String transformReceiving(String account, String user, String content) throws OtrException {
LogManager.i(this, "transform incoming message... " + content);
Session session = getOrCreateSession(account, user);
try {
return session.transformReceiving(content);
String s = session.transformReceiving(content);
LogManager.i(this, "transformed incoming message: " + s + " session status: " + session.getSessionStatus());
return s;
} catch (UnsupportedOperationException e) {
throw new OtrException(e);
}
......@@ -452,73 +404,65 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
public SecurityLevel getSecurityLevel(String account, String user) {
if (actives.get(account, user) == null) {
if (finished.get(account, user) == null)
if (finished.get(account, user) == null) {
return SecurityLevel.plain;
else
} else {
return SecurityLevel.finished;
}
} else {
if (isVerified(account, user))
if (isVerified(account, user)) {
return SecurityLevel.verified;
else
} else {
return SecurityLevel.encrypted;
}
}
}
public boolean isVerified(String account, String user) {
String active = actives.get(account, user);
if (active == null)
if (active == null) {
return false;
}
Boolean value = fingerprints.get(account, user, active);
return value != null && value;
}
private void setVerifyWithoutNotification(String account, String user,
String fingerprint, boolean value) {
private void setVerifyWithoutNotification(String account, String user, String fingerprint, boolean value) {
fingerprints.put(account, user, fingerprint, value);
requestToWrite(account, user, fingerprint, value);
}
/**
* Set whether fingerprint was verified. Add action to the chat history.
*
* @param account
* @param user
* @param fingerprint
* @param value
*/
public void setVerify(String account, String user, String fingerprint,
boolean value) {
public void setVerify(String account, String user, String fingerprint, boolean value) {
setVerifyWithoutNotification(account, user, fingerprint, value);
if (value)
if (value) {
newAction(account, user, null, ChatAction.otr_smp_verified);
else if (actives.get(account, user) != null)
} else if (actives.get(account, user) != null) {
newAction(account, user, null, ChatAction.otr_encryption);
}
}
private void setVerify(SessionID sessionID, boolean value) {
String active = actives.get(sessionID.getAccountID(),
sessionID.getUserID());
String active = actives.get(sessionID.getAccountID(), sessionID.getUserID());
if (active == null) {
LogManager.exception(this, new IllegalStateException(
"There is no active fingerprint"));
LogManager.exception(this, new IllegalStateException("There is no active fingerprint"));
return;
}
setVerifyWithoutNotification(sessionID.getAccountID(),
sessionID.getUserID(), active, value);
setVerifyWithoutNotification(sessionID.getAccountID(), sessionID.getUserID(), active, value);
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
value ? ChatAction.otr_smp_verified
: ChatAction.otr_smp_unverified);
RosterManager.getInstance().onContactChanged(sessionID.getAccountID(),
sessionID.getUserID());
value ? ChatAction.otr_smp_verified : ChatAction.otr_smp_unverified);
RosterManager.getInstance().onContactChanged(sessionID.getAccountID(), sessionID.getUserID());
}
@Override
public void verify(SessionID sessionID, String fingerprint, boolean approved) {
if (approved)
if (approved) {
setVerify(sessionID, true);
else if (isVerified(sessionID.getAccountID(), sessionID.getUserID()))
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_smp_not_approved);
} else if (isVerified(sessionID.getAccountID(), sessionID.getUserID())) {
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_smp_not_approved);
}
removeSMProgress(sessionID.getAccountID(), sessionID.getUserID());
}
......@@ -534,10 +478,7 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
public String getLocalFingerprint(String account) {
try {
return OtrCryptoEngine.getFingerprint(getLocalKeyPair(
account).getPublic());
} catch (OtrCryptoException e) {
LogManager.exception(this, e);
return OtrCryptoEngine.getFingerprint(getLocalKeyPair(account).getPublic());
} catch (OtrException e) {
LogManager.exception(this, e);
}
......@@ -546,9 +487,7 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
@Override
public byte[] getLocalFingerprintRaw(SessionID sessionID) {
return SerializationUtils
.hexStringToByteArray(getLocalFingerprint(sessionID
.getAccountID()));
return SerializationUtils.hexStringToByteArray(getLocalFingerprint(sessionID.getAccountID()));
}
@Override
......@@ -558,15 +497,9 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
/**
* Respond using SM protocol.
*
* @param account
* @param user
* @param question
* @param secret
* @throws NetworkException
*/
public void respondSmp(String account, String user, String question,
String secret) throws NetworkException {
public void respondSmp(String account, String user, String question, String secret) throws NetworkException {
LogManager.i(this, "responding smp... " + user);
removeSMRequest(account, user);
addSMProgress(account, user);
try {
......@@ -578,15 +511,9 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
/**
* Initiate request using SM protocol.
*
* @param account
* @param user
* @param question
* @param secret
* @throws NetworkException
*/
public void initSmp(String account, String user, String question,
String secret) throws NetworkException {
public void initSmp(String account, String user, String question, String secret) throws NetworkException {
LogManager.i(this, "initializing smp... " + user);
removeSMRequest(account, user);
addSMProgress(account, user);
try {
......@@ -598,12 +525,9 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
/**
* Abort SM negotiation.
*
* @param account
* @param user
* @throws NetworkException
*/
public void abortSmp(String account, String user) throws NetworkException {
LogManager.i(this, "aborting smp... " + user);
removeSMRequest(account, user);
removeSMProgress(account, user);
try {
......@@ -627,13 +551,13 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
@Override
public void onAccountAdded(final AccountItem accountItem) {
if (accountItem.getKeyPair() != null)
if (accountItem.getKeyPair() != null) {
return;
}
keyPairGenerator.execute(new Runnable() {
@Override
public void run() {
LogManager.i(this, "KeyPair generation started for "
+ accountItem.getAccount());
LogManager.i(this, "KeyPair generation started for " + accountItem.getAccount());
final KeyPair keyPair;
try {
keyPair = KeyPairGenerator.getInstance("DSA").genKeyPair();
......@@ -649,12 +573,10 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
Application.getInstance().runOnUiThread(new Runnable() {
@Override
public void run() {
LogManager.i(this, "KeyPair generation finished for "
+ accountItem.getAccount());
if (AccountManager.getInstance().getAccount(
accountItem.getAccount()) != null)
AccountManager.getInstance().setKeyPair(
accountItem.getAccount(), keyPair);
LogManager.i(this, "KeyPair generation finished for " + accountItem.getAccount());
if (AccountManager.getInstance().getAccount(accountItem.getAccount()) != null) {
AccountManager.getInstance().setKeyPair(accountItem.getAccount(), keyPair);
}
}
});
}
......@@ -671,32 +593,28 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
/**
* Save chat specific otr settings.
*
* @param account
* @param user
* @param fingerprint
* @param verified
*/
private void requestToWrite(final String account, final String user,
final String fingerprint, final boolean verified) {
Application.getInstance().runInBackground(new Runnable() {
@Override
public void run() {
OTRTable.getInstance().write(account, user, fingerprint,
verified);
OTRTable.getInstance().write(account, user, fingerprint, verified);
}
});
}
private void endAllSessions() {
NestedMap<String> entities = new NestedMap<String>();
LogManager.i(this, "End all sessions");
NestedMap<String> entities = new NestedMap<>();
entities.addAll(actives);
for (Entry<String> entry : entities)
for (Entry<String> entry : entities) {
try {
endSession(entry.getFirst(), entry.getSecond());
} catch (NetworkException e) {
LogManager.exception(this, e);
}
}
}
@Override
......@@ -705,8 +623,9 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
}
public void onSettingsChanged() {
if (SettingsManager.securityOtrMode() == SecurityOtrMode.disabled)
if (SettingsManager.securityOtrMode() == SecurityOtrMode.disabled) {
endAllSessions();
}
}
@Override
......@@ -725,8 +644,7 @@ public class OTRManager implements OtrEngineHost, OtrEngineListener,
public void messageFromAnotherInstanceReceived(SessionID sessionID) {
LogManager.i(this, "Message from another instance received on SessionID "
+ sessionID + ". Restarting OTR session for this user.");
newAction(sessionID.getAccountID(), sessionID.getUserID(), null,
ChatAction.otr_unreadable);
newAction(sessionID.getAccountID(), sessionID.getUserID(), null, ChatAction.otr_unreadable);
}
@Override
......
......@@ -31,8 +31,7 @@ public class SMProgress extends BaseEntity implements EntityNotificationItem {
@Override
public Intent getIntent() {
return QuestionViewer.createCancelIntent(
Application.getInstance(), account, user);
return QuestionViewer.createCancelIntent(Application.getInstance(), account, user);
}
@Override
......@@ -43,8 +42,7 @@ public class SMProgress extends BaseEntity implements EntityNotificationItem {
@Override
public String getText() {
return Application.getInstance().getString(
R.string.otr_verification_in_progress);
return Application.getInstance().getString(R.string.otr_verification_in_progress);
}
}
......@@ -35,8 +35,7 @@ public class SMRequest extends BaseEntity implements EntityNotificationItem {
@Override
public Intent getIntent() {
return QuestionViewer.createIntent(
Application.getInstance(), account, user, question != null,
true, question);
Application.getInstance(), account, user, question != null, true, question);
}
@Override
......
......@@ -14,14 +14,6 @@
*/
package com.xabber.android.data.message;
import net.java.otr4j.OtrException;
import org.jivesoftware.smack.packet.Message;
import org.jivesoftware.smack.packet.Message.Type;
import org.jivesoftware.smack.packet.Packet;
import org.jivesoftware.smack.packet.Presence;
import org.jivesoftware.smackx.packet.MUCUser;
import com.xabber.android.data.LogManager;
import com.xabber.android.data.NetworkException;
import com.xabber.android.data.SettingsManager;
......@@ -35,6 +27,14 @@ import com.xabber.xmpp.archive.SaveMode;
import com.xabber.xmpp.delay.Delay;
import com.xabber.xmpp.muc.MUC;
import net.java.otr4j.OtrException;
import org.jivesoftware.smack.packet.Message;
import org.jivesoftware.smack.packet.Message.Type;
import org.jivesoftware.smack.packet.Packet;
import org.jivesoftware.smack.packet.Presence;
import org.jivesoftware.smackx.packet.MUCUser;
/**
* Represents normal chat.
*
......@@ -141,8 +141,7 @@ public class RegularChat extends AbstractChat {
updateThreadId(thread);
boolean unencrypted = false;
try {
text = OTRManager.getInstance().transformReceiving(account,
user, text);
text = OTRManager.getInstance().transformReceiving(account, user, text);
} catch (OtrException e) {
if (e.getCause() instanceof OTRUnencryptedException) {
text = ((OTRUnencryptedException) e.getCause()).getText();
......@@ -154,7 +153,7 @@ public class RegularChat extends AbstractChat {
}
}
// System message received.
if (text == null)
if (text == null || text.trim().equals(""))
return true;
if (!"".equals(resource))
this.resource = resource;
......
......@@ -356,21 +356,29 @@ public class ChatViewer extends ManagedActivity implements OnChatChangedListener
chatScrollIndicatorAdapter.update(chatViewerAdapter.getActiveChats());
selectPage();
} else {
updateRegisteredChats();
updateRegisteredRecentChatsFragments();
updateStatusBar();
for (ChatViewerFragment chat : registeredChats) {
if (chat.isEqual(selectedChat) && incoming) {
chat.playIncomingAnimation();
if (chat.isEqual(selectedChat)) {
chat.updateChat();
if (incoming) {
chat.playIncomingAnimation();
}
}
}
updateRegisteredRecentChatsFragments();
updateStatusBar();
}
}
@Override
public void onContactsChanged(Collection<BaseEntity> entities) {
updateRegisteredChats();
for (BaseEntity contact : entities) {
for (ChatViewerFragment chat : registeredChats) {
if (chat.isEqual(contact)) {
chat.updateChat();
}
}
}
updateRegisteredRecentChatsFragments();
updateStatusBar();
}
......
......@@ -23,6 +23,7 @@ import android.widget.ImageView;
import android.widget.TextView;
import com.xabber.android.R;
import com.xabber.android.data.LogManager;
import com.xabber.android.data.SettingsManager;
import com.xabber.android.data.account.AccountItem;
import com.xabber.android.data.account.AccountManager;
......@@ -139,6 +140,7 @@ public class ChatMessageAdapter extends RecyclerView.Adapter<RecyclerView.ViewHo
incomingMessage.messageBalloon.setVisibility(View.GONE);
incomingMessage.messageTime.setVisibility(View.GONE);
incomingMessage.avatar.setVisibility(View.GONE);
LogManager.w(this, "Empty message! Hidden, but need to correct");
} else {
incomingMessage.messageBalloon.setVisibility(View.VISIBLE);
incomingMessage.messageTime.setVisibility(View.VISIBLE);
......@@ -285,7 +287,7 @@ public class ChatMessageAdapter extends RecyclerView.Adapter<RecyclerView.ViewHo
return null;
}
public static class BasicMessage extends RecyclerView.ViewHolder {
public static class BasicMessage extends RecyclerView.ViewHolder {
public TextView messageText;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment