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
4e33aa0d
Commit
4e33aa0d
authored
Mar 13, 2015
by
Dave Cridland
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #192 from nobelium/OF-858
OF-858: Auto create new nodes in PEP
parents
8e9fe62b
eacb38f9
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
99 additions
and
35 deletions
+99
-35
IQPEPHandler.java
src/java/org/jivesoftware/openfire/pep/IQPEPHandler.java
+9
-1
PubSubEngine.java
src/java/org/jivesoftware/openfire/pubsub/PubSubEngine.java
+90
-34
No files found.
src/java/org/jivesoftware/openfire/pep/IQPEPHandler.java
View file @
4e33aa0d
...
@@ -21,6 +21,7 @@
...
@@ -21,6 +21,7 @@
package
org
.
jivesoftware
.
openfire
.
pep
;
package
org
.
jivesoftware
.
openfire
.
pep
;
import
java.util.ArrayList
;
import
java.util.ArrayList
;
import
java.util.Arrays
;
import
java.util.Iterator
;
import
java.util.Iterator
;
import
java.util.Map
;
import
java.util.Map
;
import
java.util.concurrent.ExecutorService
;
import
java.util.concurrent.ExecutorService
;
...
@@ -278,7 +279,14 @@ public class IQPEPHandler extends IQHandler implements ServerIdentitiesProvider,
...
@@ -278,7 +279,14 @@ public class IQPEPHandler extends IQHandler implements ServerIdentitiesProvider,
* in the server's disco#info result (as per section 4 of XEP-0163).
* in the server's disco#info result (as per section 4 of XEP-0163).
*/
*/
public
Iterator
<
String
>
getFeatures
()
{
public
Iterator
<
String
>
getFeatures
()
{
return
XMPPServer
.
getInstance
().
getPubSubModule
().
getFeatures
(
null
,
null
,
null
);
Iterator
<
String
>
it
=
XMPPServer
.
getInstance
().
getPubSubModule
().
getFeatures
(
null
,
null
,
null
);
ArrayList
<
String
>
features
=
new
ArrayList
<
String
>();
while
(
it
.
hasNext
())
{
features
.
add
(
it
.
next
());
}
// Auto Creation of nodes is supported in PEP
features
.
add
(
"http://jabber.org/protocol/pubsub#auto-create"
);
return
features
.
iterator
();
}
}
...
...
src/java/org/jivesoftware/openfire/pubsub/PubSubEngine.java
View file @
4e33aa0d
...
@@ -39,6 +39,7 @@ import org.jivesoftware.openfire.RoutingTable;
...
@@ -39,6 +39,7 @@ import org.jivesoftware.openfire.RoutingTable;
import
org.jivesoftware.openfire.XMPPServer
;
import
org.jivesoftware.openfire.XMPPServer
;
import
org.jivesoftware.openfire.XMPPServerListener
;
import
org.jivesoftware.openfire.XMPPServerListener
;
import
org.jivesoftware.openfire.component.InternalComponentManager
;
import
org.jivesoftware.openfire.component.InternalComponentManager
;
import
org.jivesoftware.openfire.pep.PEPService
;
import
org.jivesoftware.openfire.pubsub.cluster.RefreshNodeTask
;
import
org.jivesoftware.openfire.pubsub.cluster.RefreshNodeTask
;
import
org.jivesoftware.openfire.pubsub.models.AccessModel
;
import
org.jivesoftware.openfire.pubsub.models.AccessModel
;
import
org.jivesoftware.openfire.user.UserManager
;
import
org.jivesoftware.openfire.user.UserManager
;
...
@@ -52,6 +53,7 @@ import org.xmpp.packet.IQ;
...
@@ -52,6 +53,7 @@ import org.xmpp.packet.IQ;
import
org.xmpp.packet.JID
;
import
org.xmpp.packet.JID
;
import
org.xmpp.packet.Message
;
import
org.xmpp.packet.Message
;
import
org.xmpp.packet.PacketError
;
import
org.xmpp.packet.PacketError
;
import
org.xmpp.packet.PacketError.Condition
;
import
org.xmpp.packet.Presence
;
import
org.xmpp.packet.Presence
;
/**
/**
...
@@ -315,8 +317,14 @@ public class PubSubEngine {
...
@@ -315,8 +317,14 @@ public class PubSubEngine {
private
void
publishItemsToNode
(
PubSubService
service
,
IQ
iq
,
Element
publishElement
)
{
private
void
publishItemsToNode
(
PubSubService
service
,
IQ
iq
,
Element
publishElement
)
{
String
nodeID
=
publishElement
.
attributeValue
(
"node"
);
String
nodeID
=
publishElement
.
attributeValue
(
"node"
);
Node
node
;
Node
node
;
JID
from
=
iq
.
getFrom
();
// TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet
JID
owner
=
from
.
asBareJID
();
if
(
nodeID
==
null
)
{
if
(
nodeID
==
null
)
{
// No node was specified. Return bad_request error
// XEP-0060 Section 7.2.3.3 - No node was specified. Return bad_request error
// This suggests that Instant nodes should not be auto-created
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
"nodeid-required"
,
"http://jabber.org/protocol/pubsub#errors"
));
"nodeid-required"
,
"http://jabber.org/protocol/pubsub#errors"
));
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
bad_request
,
pubsubError
);
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
bad_request
,
pubsubError
);
...
@@ -326,15 +334,29 @@ public class PubSubEngine {
...
@@ -326,15 +334,29 @@ public class PubSubEngine {
// Look for the specified node
// Look for the specified node
node
=
service
.
getNode
(
nodeID
);
node
=
service
.
getNode
(
nodeID
);
if
(
node
==
null
)
{
if
(
node
==
null
)
{
if
(
service
instanceof
PEPService
&&
service
.
isServiceAdmin
(
owner
)){
// If it is a PEP service & publisher is service owner -
// auto create nodes.
Element
childElement
=
iq
.
getChildElement
();
Element
createElement
=
publishElement
.
element
(
"publish"
);
CreateNodeResponse
response
=
createNodeHelper
(
service
,
iq
,
childElement
,
createElement
);
if
(
response
.
newNode
==
null
)
{
// New node creation failed. Since pep#auto-create is advertised
// in disco#info, node creation error should be sent to the client.
sendErrorPacket
(
iq
,
response
.
creationStatus
,
response
.
pubsubError
);
}
else
{
// Node creation succeeded, set node to newNode.
node
=
response
.
newNode
;
}
}
else
{
// Node does not exist. Return item-not-found error
// Node does not exist. Return item-not-found error
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
item_not_found
,
null
);
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
item_not_found
,
null
);
return
;
return
;
}
}
}
}
}
JID
from
=
iq
.
getFrom
();
// TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet
JID
owner
=
from
.
asBareJID
();
if
(!
node
.
getPublisherModel
().
canPublish
(
node
,
owner
)
&&
!
service
.
isServiceAdmin
(
owner
))
{
if
(!
node
.
getPublisherModel
().
canPublish
(
node
,
owner
)
&&
!
service
.
isServiceAdmin
(
owner
))
{
// Entity does not have sufficient privileges to publish to node
// Entity does not have sufficient privileges to publish to node
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
forbidden
,
null
);
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
forbidden
,
null
);
...
@@ -1084,13 +1106,61 @@ public class PubSubEngine {
...
@@ -1084,13 +1106,61 @@ public class PubSubEngine {
}
}
private
void
createNode
(
PubSubService
service
,
IQ
iq
,
Element
childElement
,
Element
createElement
)
{
private
void
createNode
(
PubSubService
service
,
IQ
iq
,
Element
childElement
,
Element
createElement
)
{
// Call createNodeHelper and get the node creation status.
CreateNodeResponse
response
=
createNodeHelper
(
service
,
iq
,
childElement
,
createElement
);
if
(
response
.
newNode
==
null
)
{
// New node creation failed
sendErrorPacket
(
iq
,
response
.
creationStatus
,
response
.
pubsubError
);
}
else
{
IQ
reply
=
IQ
.
createResultIQ
(
iq
);
Node
newNode
=
response
.
newNode
;
String
nodeID
=
createElement
.
attributeValue
(
"node"
);
// Include new nodeID if it has changed from the original nodeID
if
(!
newNode
.
getNodeID
().
equals
(
nodeID
))
{
Element
elem
=
reply
.
setChildElement
(
"pubsub"
,
"http://jabber.org/protocol/pubsub"
);
elem
.
addElement
(
"create"
).
addAttribute
(
"node"
,
newNode
.
getNodeID
());
}
router
.
route
(
reply
);
}
}
/**
* Response Object returned by createNodeHelper method
*/
private
class
CreateNodeResponse
{
public
final
PacketError
.
Condition
creationStatus
;
public
final
Node
newNode
;
public
final
Element
pubsubError
;
public
CreateNodeResponse
(
PacketError
.
Condition
creationStatus
,
Element
pubsubError
,
Node
newNode
)
{
this
.
creationStatus
=
creationStatus
;
this
.
newNode
=
newNode
;
this
.
pubsubError
=
pubsubError
;
}
}
/**
* Checks if the following conditions are satisfied and creates a node
* - Requester can create nodes
* - Instant node creation is enabled
* - Node does not already exist
* - New node configuration is valid
*
* NOTE: This method should not reply to the client
*
* @param service
* @param iq
* @param childElement
* @param createElement
* @return
*/
private
CreateNodeResponse
createNodeHelper
(
PubSubService
service
,
IQ
iq
,
Element
childElement
,
Element
createElement
)
{
// Get sender of the IQ packet
// Get sender of the IQ packet
JID
from
=
iq
.
getFrom
();
JID
from
=
iq
.
getFrom
();
// Verify that sender has permissions to create nodes
// Verify that sender has permissions to create nodes
if
(!
service
.
canCreateNode
(
from
)
||
(!
UserManager
.
getInstance
().
isRegisteredUser
(
from
)
&&
!
isComponent
(
from
))
)
{
if
(!
service
.
canCreateNode
(
from
)
||
(!
UserManager
.
getInstance
().
isRegisteredUser
(
from
)
&&
!
isComponent
(
from
))
)
{
// The user is not allowed to create nodes so return an error
// The user is not allowed to create nodes so return an error
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
forbidden
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
forbidden
,
null
,
null
);
return
;
}
}
DataForm
completedForm
=
null
;
DataForm
completedForm
=
null
;
CollectionNode
parentNode
=
null
;
CollectionNode
parentNode
=
null
;
...
@@ -1102,8 +1172,7 @@ public class PubSubEngine {
...
@@ -1102,8 +1172,7 @@ public class PubSubEngine {
// Instant nodes creation is not allowed so return an error
// Instant nodes creation is not allowed so return an error
Element
pubsubError
=
DocumentHelper
.
createElement
(
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
"nodeid-required"
,
"http://jabber.org/protocol/pubsub#errors"
));
QName
.
get
(
"nodeid-required"
,
"http://jabber.org/protocol/pubsub#errors"
));
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
not_acceptable
,
pubsubError
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
not_acceptable
,
pubsubError
,
null
);
return
;
}
}
do
{
do
{
// Create a new nodeID and make sure that the random generated string does not
// Create a new nodeID and make sure that the random generated string does not
...
@@ -1129,13 +1198,11 @@ public class PubSubEngine {
...
@@ -1129,13 +1198,11 @@ public class PubSubEngine {
Node
tempNode
=
service
.
getNode
(
parentNodeID
);
Node
tempNode
=
service
.
getNode
(
parentNodeID
);
if
(
tempNode
==
null
)
{
if
(
tempNode
==
null
)
{
// Requested parent node was not found so return an error
// Requested parent node was not found so return an error
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
item_not_found
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
item_not_found
,
null
,
null
);
return
;
}
}
else
if
(!
tempNode
.
isCollectionNode
())
{
else
if
(!
tempNode
.
isCollectionNode
())
{
// Requested parent node is not a collection node so return an error
// Requested parent node is not a collection node so return an error
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
not_acceptable
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
not_acceptable
,
null
,
null
);
return
;
}
}
parentNode
=
(
CollectionNode
)
tempNode
;
parentNode
=
(
CollectionNode
)
tempNode
;
}
}
...
@@ -1158,8 +1225,7 @@ public class PubSubEngine {
...
@@ -1158,8 +1225,7 @@ public class PubSubEngine {
Node
existingNode
=
service
.
getNode
(
newNodeID
);
Node
existingNode
=
service
.
getNode
(
newNodeID
);
if
(
existingNode
!=
null
)
{
if
(
existingNode
!=
null
)
{
// There is a conflict since a node with the same ID already exists
// There is a conflict since a node with the same ID already exists
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
conflict
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
conflict
,
null
,
null
);
return
;
}
}
if
(
collectionType
&&
!
service
.
isCollectionNodesSupported
())
{
if
(
collectionType
&&
!
service
.
isCollectionNodesSupported
())
{
...
@@ -1167,24 +1233,21 @@ public class PubSubEngine {
...
@@ -1167,24 +1233,21 @@ public class PubSubEngine {
Element
pubsubError
=
DocumentHelper
.
createElement
(
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
"unsupported"
,
"http://jabber.org/protocol/pubsub#errors"
));
QName
.
get
(
"unsupported"
,
"http://jabber.org/protocol/pubsub#errors"
));
pubsubError
.
addAttribute
(
"feature"
,
"collections"
);
pubsubError
.
addAttribute
(
"feature"
,
"collections"
);
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
feature_not_implemented
,
pubsubError
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
feature_not_implemented
,
pubsubError
,
null
);
return
;
}
}
if
(
parentNode
!=
null
&&
!
collectionType
)
{
if
(
parentNode
!=
null
&&
!
collectionType
)
{
// Check if requester is allowed to add a new leaf child node to the parent node
// Check if requester is allowed to add a new leaf child node to the parent node
if
(!
parentNode
.
isAssociationAllowed
(
from
))
{
if
(!
parentNode
.
isAssociationAllowed
(
from
))
{
// User is not allowed to add child leaf node to parent node. Return an error.
// User is not allowed to add child leaf node to parent node. Return an error.
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
forbidden
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
forbidden
,
null
,
null
);
return
;
}
}
// Check if number of child leaf nodes has not been exceeded
// Check if number of child leaf nodes has not been exceeded
if
(
parentNode
.
isMaxLeafNodeReached
())
{
if
(
parentNode
.
isMaxLeafNodeReached
())
{
// Max number of child leaf nodes has been reached. Return an error.
// Max number of child leaf nodes has been reached. Return an error.
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
"max-nodes-exceeded"
,
Element
pubsubError
=
DocumentHelper
.
createElement
(
QName
.
get
(
"max-nodes-exceeded"
,
"http://jabber.org/protocol/pubsub#errors"
));
"http://jabber.org/protocol/pubsub#errors"
));
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
conflict
,
pubsubError
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
conflict
,
pubsubError
,
null
);
return
;
}
}
}
}
...
@@ -1221,23 +1284,16 @@ public class PubSubEngine {
...
@@ -1221,23 +1284,16 @@ public class PubSubEngine {
}
}
if
(
conflict
)
{
if
(
conflict
)
{
// There is a conflict since a node with the same ID already exists
// There is a conflict since a node with the same ID already exists
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
conflict
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
conflict
,
null
,
null
);
}
}
else
{
else
{
// Return success to the node owner
// Return success to the node owner
IQ
reply
=
IQ
.
createResultIQ
(
iq
);
return
new
CreateNodeResponse
(
null
,
null
,
newNode
);
// Include new nodeID if it has changed from the original nodeID
if
(!
newNode
.
getNodeID
().
equals
(
nodeID
))
{
Element
elem
=
reply
.
setChildElement
(
"pubsub"
,
"http://jabber.org/protocol/pubsub"
);
elem
.
addElement
(
"create"
).
addAttribute
(
"node"
,
newNode
.
getNodeID
());
}
router
.
route
(
reply
);
}
}
}
}
catch
(
NotAcceptableException
e
)
{
catch
(
NotAcceptableException
e
)
{
// Node should have at least one owner. Return not-acceptable error.
// Node should have at least one owner. Return not-acceptable error.
sendErrorPacket
(
iq
,
PacketError
.
Condition
.
not_acceptable
,
null
);
return
new
CreateNodeResponse
(
PacketError
.
Condition
.
not_acceptable
,
null
,
null
);
}
}
}
}
...
...
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