Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
O
Openfire
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
Openfire
Commits
8bb250ef
Commit
8bb250ef
authored
Dec 13, 2015
by
Dave Cridland
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #448 from surevine/xep-0198-fixes
OF-1003 OF-1002 Make XEP-0198 thread safer
parents
f4a4d27c
e812f6c2
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
72 additions
and
43 deletions
+72
-43
SessionManager.java
src/java/org/jivesoftware/openfire/SessionManager.java
+1
-16
LocalClientSession.java
...org/jivesoftware/openfire/session/LocalClientSession.java
+1
-9
StreamManager.java
...jivesoftware/openfire/streammanagement/StreamManager.java
+70
-18
No files found.
src/java/org/jivesoftware/openfire/SessionManager.java
View file @
8bb250ef
...
...
@@ -1269,22 +1269,7 @@ public class SessionManager extends BasicModule implements ClusterEventListener/
router
.
route
(
presence
);
}
// Re-deliver unacknowledged stanzas from broken stream (XEP-0198)
if
(
session
.
getStreamManager
().
isEnabled
())
{
session
.
getStreamManager
().
setEnabled
(
false
);
// Avoid concurrent usage.
Deque
<
StreamManager
.
UnackedPacket
>
unacknowledgedStanzas
=
session
.
getStreamManager
().
getUnacknowledgedServerStanzas
();
if
(!
unacknowledgedStanzas
.
isEmpty
())
{
for
(
StreamManager
.
UnackedPacket
unacked
:
unacknowledgedStanzas
)
{
if
(
unacked
.
packet
instanceof
Message
)
{
Message
m
=
(
Message
)
unacked
.
packet
;
Element
delayInformation
=
m
.
addChildElement
(
"delay"
,
"urn:xmpp:delay"
);
delayInformation
.
addAttribute
(
"stamp"
,
XMPPDateTimeFormat
.
format
(
unacked
.
timestamp
));
delayInformation
.
addAttribute
(
"from"
,
serverAddress
.
toBareJID
());
}
router
.
route
(
unacked
.
packet
);
}
}
}
session
.
getStreamManager
().
onClose
(
router
,
serverAddress
);
}
finally
{
// Remove the session
...
...
src/java/org/jivesoftware/openfire/session/LocalClientSession.java
View file @
8bb250ef
...
...
@@ -814,15 +814,7 @@ public class LocalClientSession extends LocalSession implements ClientSession {
public
void
deliver
(
Packet
packet
)
throws
UnauthorizedException
{
conn
.
deliver
(
packet
);
if
(
streamManager
.
isEnabled
())
{
streamManager
.
incrementServerSentStanzas
();
// Temporarily store packet until delivery confirmed
streamManager
.
getUnacknowledgedServerStanzas
().
addLast
(
new
StreamManager
.
UnackedPacket
(
new
Date
(),
packet
.
createCopy
()));
if
(
getNumServerPackets
()
%
JiveGlobals
.
getLongProperty
(
"stream.management.requestFrequency"
,
5
)
==
0
)
{
streamManager
.
sendServerRequest
();
}
}
streamManager
.
sentStanza
(
packet
);
}
@Override
...
...
src/java/org/jivesoftware/openfire/streammanagement/StreamManager.java
View file @
8bb250ef
...
...
@@ -6,6 +6,13 @@ import java.util.LinkedList;
import
org.dom4j.Element
;
import
org.jivesoftware.openfire.Connection
;
import
org.jivesoftware.openfire.PacketRouter
;
import
org.jivesoftware.util.JiveGlobals
;
import
org.jivesoftware.util.XMPPDateTimeFormat
;
import
org.slf4j.Logger
;
import
org.slf4j.LoggerFactory
;
import
org.xmpp.packet.JID
;
import
org.xmpp.packet.Message
;
import
org.xmpp.packet.Packet
;
import
org.xmpp.packet.PacketError
;
...
...
@@ -16,6 +23,7 @@ import org.xmpp.packet.PacketError;
* @author jonnyheavey
*/
public
class
StreamManager
{
private
static
final
Logger
Log
=
LoggerFactory
.
getLogger
(
StreamManager
.
class
);
public
static
class
UnackedPacket
{
public
final
Date
timestamp
;
public
final
Packet
packet
;
...
...
@@ -90,7 +98,7 @@ public class StreamManager {
/**
* Sends XEP-0198 request <r /> to client from server
*/
p
ublic
void
sendServerRequest
()
{
p
rivate
void
sendServerRequest
()
{
if
(
isEnabled
())
{
String
request
=
String
.
format
(
"<r xmlns='%s' />"
,
getNamespace
());
getConnection
().
deliverRawText
(
request
);
...
...
@@ -115,27 +123,71 @@ public class StreamManager {
*/
public
void
processClientAcknowledgement
(
Element
ack
)
{
if
(
isEnabled
())
{
if
(
ack
.
attribute
(
"h"
)
!=
null
)
{
long
count
=
Long
.
valueOf
(
ack
.
attributeValue
(
"h"
));
// Remove stanzas from temporary storage as now acknowledged
Deque
<
UnackedPacket
>
unacknowledgedStanzas
=
getUnacknowledgedServerStanzas
();
long
i
=
getClientProcessedStanzas
();
if
(
count
<
i
)
{
synchronized
(
this
)
{
if
(
ack
.
attribute
(
"h"
)
!=
null
)
{
long
count
=
Long
.
valueOf
(
ack
.
attributeValue
(
"h"
));
// Remove stanzas from temporary storage as now acknowledged
Deque
<
UnackedPacket
>
unacknowledgedStanzas
=
getUnacknowledgedServerStanzas
();
long
i
=
getClientProcessedStanzas
();
Log
.
debug
(
"Ack: h={} mine={} length={}"
,
count
,
i
,
unacknowledgedStanzas
.
size
());
if
(
count
<
i
)
{
/* Consider rollover? */
if
(
i
>
mask
)
{
while
(
count
<
i
)
{
count
+=
mask
+
1
;
}
}
}
while
(
i
<
count
)
{
unacknowledgedStanzas
.
removeFirst
();
i
++;
Log
.
debug
(
"Maybe rollover"
);
if
(
i
>
mask
)
{
while
(
count
<
i
)
{
Log
.
debug
(
"Rolling..."
);
count
+=
mask
+
1
;
}
}
}
while
(
i
<
count
)
{
unacknowledgedStanzas
.
removeFirst
();
i
++;
Log
.
debug
(
"In Ack: h={} mine={} length={}"
,
count
,
i
,
unacknowledgedStanzas
.
size
());
}
setClientProcessedStanzas
(
count
);
}
}
}
}
setClientProcessedStanzas
(
count
);
public
void
sentStanza
(
Packet
packet
)
{
if
(
isEnabled
())
{
synchronized
(
this
)
{
incrementServerSentStanzas
();
// Temporarily store packet until delivery confirmed
getUnacknowledgedServerStanzas
().
addLast
(
new
StreamManager
.
UnackedPacket
(
new
Date
(),
packet
.
createCopy
()));
Log
.
debug
(
"Added stanza of type {}, now {} / {}"
,
packet
.
getClass
().
getName
(),
getServerSentStanzas
(),
getUnacknowledgedServerStanzas
().
size
());
}
if
(
getServerSentStanzas
()
%
JiveGlobals
.
getLongProperty
(
"stream.management.requestFrequency"
,
5
)
==
0
)
{
sendServerRequest
();
}
}
}
public
void
onClose
(
PacketRouter
router
,
JID
serverAddress
)
{
// Re-deliver unacknowledged stanzas from broken stream (XEP-0198)
if
(
isEnabled
())
{
setEnabled
(
false
);
// Avoid concurrent usage.
synchronized
(
this
)
{
Deque
<
StreamManager
.
UnackedPacket
>
unacknowledgedStanzas
=
getUnacknowledgedServerStanzas
();
if
(!
unacknowledgedStanzas
.
isEmpty
())
{
for
(
StreamManager
.
UnackedPacket
unacked
:
unacknowledgedStanzas
)
{
if
(
unacked
.
packet
instanceof
Message
)
{
Message
m
=
(
Message
)
unacked
.
packet
;
Element
delayInformation
=
m
.
addChildElement
(
"delay"
,
"urn:xmpp:delay"
);
delayInformation
.
addAttribute
(
"stamp"
,
XMPPDateTimeFormat
.
format
(
unacked
.
timestamp
));
delayInformation
.
addAttribute
(
"from"
,
serverAddress
.
toBareJID
());
}
router
.
route
(
unacked
.
packet
);
}
}
}
}
}
/**
...
...
@@ -160,7 +212,7 @@ public class StreamManager {
* manager belongs to.
* @param enabled
*/
public
void
setEnabled
(
boolean
enabled
)
{
synchronized
public
void
setEnabled
(
boolean
enabled
)
{
this
.
enabled
=
enabled
;
if
(
enabled
)
{
...
...
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