Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
X
xabber-android
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Administrator
xabber-android
Commits
cddf6b58
Commit
cddf6b58
authored
Jun 05, 2015
by
Grigory Fedorov
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'feature/otr_fix' into develop
parents
d5d31d71
a26b1857
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
186 additions
and
262 deletions
+186
-262
OTRManager.java
...ava/com/xabber/android/data/extension/otr/OTRManager.java
+155
-237
SMProgress.java
...ava/com/xabber/android/data/extension/otr/SMProgress.java
+2
-4
SMRequest.java
...java/com/xabber/android/data/extension/otr/SMRequest.java
+1
-2
RegularChat.java
...ain/java/com/xabber/android/data/message/RegularChat.java
+10
-11
ChatViewer.java
app/src/main/java/com/xabber/android/ui/ChatViewer.java
+15
-7
ChatMessageAdapter.java
...ava/com/xabber/android/ui/adapter/ChatMessageAdapter.java
+3
-1
No files found.
app/src/main/java/com/xabber/android/data/extension/otr/OTRManager.java
View file @
cddf6b58
...
...
@@ -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
...
...
app/src/main/java/com/xabber/android/data/extension/otr/SMProgress.java
View file @
cddf6b58
...
...
@@ -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
);
}
}
app/src/main/java/com/xabber/android/data/extension/otr/SMRequest.java
View file @
cddf6b58
...
...
@@ -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
...
...
app/src/main/java/com/xabber/android/data/message/RegularChat.java
View file @
cddf6b58
...
...
@@ -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
;
...
...
app/src/main/java/com/xabber/android/ui/ChatViewer.java
View file @
cddf6b58
...
...
@@ -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
();
}
...
...
app/src/main/java/com/xabber/android/ui/adapter/ChatMessageAdapter.java
View file @
cddf6b58
...
...
@@ -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
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment