
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. In the following, I've used [] to denote multiple values, rather than necessarily an array.The operations/methods a Hub should supportRegistration1 private-id = register*(hub-secret) 2 declareAppMetaData(private-id, metadata[]) 3 declareMessages(private-id, mtypes[]) Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The application provides the hub with a secret key that is only easily knowable by applications running under the user's uid. This is to make it harder for unauthorized applications to register by (for example) port scanning. The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, and is kept secret from other applications to avoid application-spoofing. Each application will also be assigned a public-id that is used to identify it to other applications (see below). Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=app://1234567-891 is. Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 10) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." 4 unregister(private-id) When an application closes, it should inform the hub.Messaging5 msgId = send(sender-private-id, recipient-public-Id, message) 6 msgId = broadcast(sender-private-id, message) 5) sends a message from sender to recipient using the hub-supplied ids for addressing. The sender's private-id is obtained on registration, while the recipient's public-id is obtained by interrogating the hub (see below). 6) sends the message to everyone (if they've elected to receive that message's mtype). The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages. we need to elaborate on this msgId idea a bit These messages are asynchronous in the sense that they return as soon as the hub has received the message. Hubs will disregard messages with unknown or invalid private-ids. The contents of message are specified on ApplicationsMessagingMsgContent.Application and hub metadata7 hub-public-id = getHubId() We can treat the hub itself as just another application that can be messaged. However, you do need to know which application it is, and this operation does the job. 8 public-ids[] = getApplicationIds() Tell me the hub-assigned public ids of all the registered applications. 9 public-ids[] = getApplicationsUnderstanding(mtype) Tell me the hub-assigned public ids of the all the registered applications that understand messages with this mtype. 10 mtypes[] = getUnderstoodMessagesForApplication(public-id) Tell me the mtypes of the messages this application wants to receive. 11 metadata[] = getAppMetaData(public-id) Return the metadata for that application. 12 public-id = getPublicId(private-id) An application will use its private-id for all communication with the hub, but it might occasionally need to know its public-id.The operations a client application should support13 receive(recipient-publid-id, sender-public-id, msgId, message) Receive a message from this sender.Application metadataIn PLASTIC, the hub stores the human-readable name of each application, in addition to its generated id. Additionally other metadata, such as a logo and description is available by using some standard messages. I propose to move all of this to the hub so it is accessible by environments that cannot receive response messages, and tidy it up. I suggest that the metadata is a struct mapping keys to values that has certain reserved keys: ivoa.name ivoa.description ivoa.ivorn ivoa.logoURL The prefix ivoa. would be reserved, but application authors are free to add new data to this struct. This would allow users such as Mike to define (e.g.) iraf.worker=1 to filter the connected applications.-- JohnTaylor - 10 Apr 2007 | 
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. In the following, I've used [] to denote multiple values, rather than necessarily an array.The operations/methods a Hub should supportRegistration1 private-id = register*(hub-secret) 2 declareAppMetaData(private-id, metadata[]) 3 declareMessages(private-id, mtypes[]) Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The application provides the hub with a secret key that is only easily knowable by applications running under the user's uid. This is to make it harder for unauthorized applications to register by (for example) port scanning. The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, and is kept secret from other applications to avoid application-spoofing. Each application will also be assigned a public-id that is used to identify it to other applications (see below). Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=app://1234567-891 is. Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 10) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." 4 unregister(private-id) When an application closes, it should inform the hub.Messaging5 msgId = send(sender-private-id, recipient-public-Id, message) 6 msgId = broadcast(sender-private-id, message) 5) sends a message from sender to recipient using the hub-supplied ids for addressing. The sender's private-id is obtained on registration, while the recipient's public-id is obtained by interrogating the hub (see below). 6) sends the message to everyone (if they've elected to receive that message's mtype). The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages. we need to elaborate on this msgId idea a bit These messages are asynchronous in the sense that they return as soon as the hub has received the message. Hubs will disregard messages with unknown or invalid private-ids. The contents of message are specified on ApplicationsMessagingMsgContent.Application and hub metadata7 hub-public-id = getHubId() We can treat the hub itself as just another application that can be messaged. However, you do need to know which application it is, and this operation does the job. 8 public-ids[] = getApplicationIds() Tell me the hub-assigned public ids of all the registered applications. 9 public-ids[] = getApplicationsUnderstanding(mtype) Tell me the hub-assigned public ids of the all the registered applications that understand messages with this mtype. 10 mtypes[] = getUnderstoodMessagesForApplication(public-id) Tell me the mtypes of the messages this application wants to receive. 11 metadata[] = getAppMetaData(public-id) Return the metadata for that application. 12 public-id = getPublicId(private-id) An application will use its private-id for all communication with the hub, but it might occasionally need to know its public-id.The operations a client application should support | ||||||||
| Changed: | ||||||||
| < < | 13 receive(sender-public-id, msgId, message) | |||||||
| > > | 13 receive(recipient-publid-id, sender-public-id, msgId, message) | |||||||
| Receive a message from this sender. Application metadataIn PLASTIC, the hub stores the human-readable name of each application, in addition to its generated id. Additionally other metadata, such as a logo and description is available by using some standard messages. I propose to move all of this to the hub so it is accessible by environments that cannot receive response messages, and tidy it up. I suggest that the metadata is a struct mapping keys to values that has certain reserved keys: ivoa.name ivoa.description ivoa.ivorn ivoa.logoURL The prefix ivoa. would be reserved, but application authors are free to add new data to this struct. This would allow users such as Mike to define (e.g.) iraf.worker=1 to filter the connected applications.<-- 
 | ||||||||
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. In the following, I've used [] to denote multiple values, rather than necessarily an array.The operations/methods a Hub should supportRegistration1 private-id = register*(hub-secret) 2 declareAppMetaData(private-id, metadata[]) 3 declareMessages(private-id, mtypes[]) Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The application provides the hub with a secret key that is only easily knowable by applications running under the user's uid. This is to make it harder for unauthorized applications to register by (for example) port scanning. The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, and is kept secret from other applications to avoid application-spoofing. Each application will also be assigned a public-id that is used to identify it to other applications (see below). Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=app://1234567-891 is. Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 10) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." 4 unregister(private-id) When an application closes, it should inform the hub.Messaging5 msgId = send(sender-private-id, recipient-public-Id, message) 6 msgId = broadcast(sender-private-id, message) 5) sends a message from sender to recipient using the hub-supplied ids for addressing. The sender's private-id is obtained on registration, while the recipient's public-id is obtained by interrogating the hub (see below). 6) sends the message to everyone (if they've elected to receive that message's mtype). The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages. we need to elaborate on this msgId idea a bit These messages are asynchronous in the sense that they return as soon as the hub has received the message. Hubs will disregard messages with unknown or invalid private-ids. | ||||||||
| Changed: | ||||||||
| < < | The contents of message are specified on ApplicationsMessagingMessageContent. | |||||||
| > > | The contents of message are specified on ApplicationsMessagingMsgContent. | |||||||
| Application and hub metadata7 hub-public-id = getHubId() We can treat the hub itself as just another application that can be messaged. However, you do need to know which application it is, and this operation does the job. 8 public-ids[] = getApplicationIds() Tell me the hub-assigned public ids of all the registered applications. 9 public-ids[] = getApplicationsUnderstanding(mtype) Tell me the hub-assigned public ids of the all the registered applications that understand messages with this mtype. 10 mtypes[] = getUnderstoodMessagesForApplication(public-id) Tell me the mtypes of the messages this application wants to receive. 11 metadata[] = getAppMetaData(public-id) Return the metadata for that application. 12 public-id = getPublicId(private-id) An application will use its private-id for all communication with the hub, but it might occasionally need to know its public-id.The operations a client application should support13 receive(sender-public-id, msgId, message) Receive a message from this sender.Application metadataIn PLASTIC, the hub stores the human-readable name of each application, in addition to its generated id. Additionally other metadata, such as a logo and description is available by using some standard messages. I propose to move all of this to the hub so it is accessible by environments that cannot receive response messages, and tidy it up. I suggest that the metadata is a struct mapping keys to values that has certain reserved keys: ivoa.name ivoa.description ivoa.ivorn ivoa.logoURL The prefix ivoa. would be reserved, but application authors are free to add new data to this struct. This would allow users such as Mike to define (e.g.) iraf.worker=1 to filter the connected applications.<-- 
 | ||||||||
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. In the following, I've used [] to denote multiple values, rather than necessarily an array.The operations/methods a Hub should supportRegistration | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 1 private-id = register*(hub-secret) | |||||||
| Deleted: | ||||||||
| < < | 
 | |||||||
| Added: | ||||||||
| > > | 2 declareAppMetaData(private-id, metadata[]) 3 declareMessages(private-id, mtypes[]) | |||||||
| Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The application provides the hub with a secret key that is only easily knowable by applications running under the user's uid. This is to make it harder for unauthorized applications to register by (for example) port scanning. | ||||||||
| Changed: | ||||||||
| < < | The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, but is kept secret from other applications to avoid application-spoofing. Each application will also be provided with a public-id that is used to identify it to other applications (see below). | |||||||
| > > | The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, and is kept secret from other applications to avoid application-spoofing. Each application will also be assigned a public-id that is used to identify it to other applications (see below). | |||||||
| Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=app://1234567-891 is. | ||||||||
| Changed: | ||||||||
| < < | Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 12) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." | |||||||
| > > | Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 10) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." | |||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 4 unregister(private-id) | |||||||
| When an application closes, it should inform the hub. Messaging | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 5 msgId = send(sender-private-id, recipient-public-Id, message) | |||||||
| Deleted: | ||||||||
| < < | 
 | |||||||
| Added: | ||||||||
| > > | 6 msgId = broadcast(sender-private-id, message) | |||||||
| 5) sends a message from sender to recipient using the hub-supplied ids for addressing. The sender's private-id is obtained on registration, while the recipient's public-id is obtained by interrogating the hub (see below). 6) sends the message to everyone (if they've elected to receive that message's mtype). The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages. | ||||||||
| Changed: | ||||||||
| < < | These messages are asynchronous in the sense that they return as soon as the hub has received the message. The contents of message are specified on ApplicationsMessagingMessageContent. | |||||||
| > > | we need to elaborate on this msgId idea a bit | |||||||
| Added: | ||||||||
| > > | These messages are asynchronous in the sense that they return as soon as the hub has received the message. Hubs will disregard messages with unknown or invalid private-ids. | |||||||
| Added: | ||||||||
| > > | The contents of message are specified on ApplicationsMessagingMessageContent. | |||||||
| Application and hub metadata | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 7 hub-public-id = getHubId() | |||||||
| We can treat the hub itself as just another application that can be messaged. However, you do need to know which application it is, and this operation does the job. | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 8 public-ids[] = getApplicationIds() | |||||||
| Tell me the hub-assigned public ids of all the registered applications. | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 9 public-ids[] = getApplicationsUnderstanding(mtype) | |||||||
| Tell me the hub-assigned public ids of the all the registered applications that understand messages with this mtype. | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 10 mtypes[] = getUnderstoodMessagesForApplication(public-id) | |||||||
| Tell me the mtypes of the messages this application wants to receive. | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 11 metadata[] = getAppMetaData(public-id) | |||||||
| Return the metadata for that application. | ||||||||
| Added: | ||||||||
| > > | 12 public-id = getPublicId(private-id) | |||||||
| Added: | ||||||||
| > > | An application will use its private-id for all communication with the hub, but it might occasionally need to know its public-id. | |||||||
| The operations a client application should support | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 13 receive(sender-public-id, msgId, message) | |||||||
| Receive a message from this sender. | ||||||||
| Added: | ||||||||
| > > | Application metadataIn PLASTIC, the hub stores the human-readable name of each application, in addition to its generated id. Additionally other metadata, such as a logo and description is available by using some standard messages. I propose to move all of this to the hub so it is accessible by environments that cannot receive response messages, and tidy it up. I suggest that the metadata is a struct mapping keys to values that has certain reserved keys: ivoa.name ivoa.description ivoa.ivorn ivoa.logoURL The prefix ivoa. would be reserved, but application authors are free to add new data to this struct. This would allow users such as Mike to define (e.g.) iraf.worker=1 to filter the connected applications. | |||||||
| <-- 
 | ||||||||
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. In the following, I've used [] to denote multiple values, rather than necessarily an array. | ||||||||
| Deleted: | ||||||||
| < < | The operations a client application should support
 | |||||||
| The operations/methods a Hub should support | ||||||||
| Deleted: | ||||||||
| < < | ||||||||
| Registration | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 
 | |||||||
| Changed: | ||||||||
| < < | Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The hub provides a session-unique id that is used in subsequent communications. This differs slightly from Mike's model where the application's human-readable-name is used for addressing. Early on we canvassed opinion amongst our users as to whether they'd ever want to run two instances of the same program simultaneously, they said yes, so I prefer the unique-id mechanism, even though it's less user-friendly. Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. | |||||||
| > > | Once an application has located a hub (by means as yet undecided), its first step is to register with it - this needs only be done once per session per application. The application provides the hub with a secret key that is only easily knowable by applications running under the user's uid. This is to make it harder for unauthorized applications to register by (for example) port scanning. | |||||||
| Changed: | ||||||||
| < < | Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=vom://1234567-891 is. (with operation 2b) | |||||||
| > > | The hub provides the application with a session-unique private-id that is used to authenticate subsequent communications, but is kept secret from other applications to avoid application-spoofing. Each application will also be provided with a public-id that is used to identify it to other applications (see below). | |||||||
| Changed: | ||||||||
| < < | Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 12) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type" [if we do, I hope we make a better job of it than we did in PLASTIC...which was a mess.] | |||||||
| > > | Note that asterisk in the register operation. This denotes that there could be several different register operations, one for each "wire protocol". An application would choose the register operation appropriate to the way it wanted to communicate. | |||||||
| Changed: | ||||||||
| < < | In PLASTIC, these 3 operations are rolled into 1, but I feel it's better to separate them. | |||||||
| > > | Operation 2) is for the application to tell the hub about itself: a human-readable name, and maybe an icon, description etc.... I see this as being more contentious: we could instead simply use messages between applications to achieve the same thing. The advantage of having it as an operation in the protocol is for those environments that can send messages but can't easily receive them. For instance, I could send a message from the python commandline to Topcat, and all Topcat would have to identify me would be the hub-generated unique id, which isn't very friendly. If I'd also declared some metadata to the hub, then Topcat could interrogate the hub to find out just who senderId=app://1234567-891 is. | |||||||
| Added: | ||||||||
| > > | Operation 3) is for the application to tell the hub which message types it wishes to receive. The hub won't send it messages it can't understand, and applications can interrogate the hub (using operation 12) to determine the appropriate messages to send. We might need a mechanism to say "send me messages of any type", or "send me messages matching this pattern." | |||||||
| Added: | ||||||||
| > > | 
 | |||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | When an application closes, it should inform the hub. | |||||||
| Messaging | ||||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | 
 | |||||||
| Added: | ||||||||
| > > | 5) sends a message from sender to recipient using the hub-supplied ids for addressing. The sender's private-id is obtained on registration, while the recipient's public-id is obtained by interrogating the hub (see below). 6) sends the message to everyone (if they've elected to receive that message's mtype). The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages. | |||||||
| These messages are asynchronous in the sense that they return as soon as the hub has received the message. The contents of message are specified on ApplicationsMessagingMessageContent. | ||||||||
| Deleted: | ||||||||
| < < | Application metadata | |||||||
| Changed: | ||||||||
| < < | 
 | |||||||
| > > | Application and hub metadata | |||||||
| Deleted: | ||||||||
| < < | 
 | |||||||
| Changed: | ||||||||
| < < | ||||||||
| > > | 
 | |||||||
| Deleted: | ||||||||
| < < | ===============================================1) id= register*()
2) declareAppMetaData(id, metadata)
3) declareMessages(id, mtypes[])
4) unregister(id)
When an application closes, it should inform the hub.
5) msgId = sendAsynch(senderId, recipientId, mtype, args[])
6) msgId = broadcastAsynch(senderId, mtype, args[])
OK, this is where the fun happens.  5) sends a message from sender to recipient using the hub-supplied ids for addressing.  6) sends the message to everyone (if they've elected to receive that mtype).  The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages (in Mike's parlance).  In these operations the message argument list args[] is intended to be quite general. 
I'm explicitly writing out the parameters here...if you prefer you can write it:
5) msgId = sendAsynch(message)
where message wraps the whole lot.  At some point we need to decide which parameters are part of the message, and which are not.  My initial feeling is that mtype and args[] are part of the message, while the addressing info are not.  [We could also add the msgId (if one already exists for that "conversation") as one of the input params, or we could put it inside args[] - I'm not sure which is best.] 
Plastic devs will note another difference with Plastic - the above operations reflect the two most common modes of use that we've encountered in 18 months of experience: one-to-one, and one-to-all.  Plastic has a one-to-subsetOfAll mode, which is rarely used to send to more than one application and is extremely annoying. 
7) msgId, response = send(senderId, recipientId, mtype, args[])
8) msgId, responses[] = broadcast(senderId, mtype, args[])
These are like 5 and 6, but block and then return a response value.  This value can be quite general ( e.g. some kind of datastructure)
These are going to be contentious. Plastic allows a request-response style messaging pattern.  This is not exactly radical: SOAP does it, and MPI has blocking send modes (though they're not quite the same as this).  It is very useful in making the programmer's life easier when dealing with very quick informational messages such as "tell me what your application's icon is".  Yes, I know you could do this asynchronously and sit around waiting for a message to come back with the answer, but I feel that we should be prepared to sacrifice some purity of style to make the application programmer's life easier.  Furthermore, my putative python-command-line hacker cannot receive data back from applications any other way (or at least, not easily). 
However, I'd be interested to know what the community thinks.  Could including a request-response style seriously restrict our choices when it comes to picking wire protocols?
9) hubId = getHubId() | |||||||
| We can treat the hub itself as just another application that can be messaged. However, you do need to know which application it is, and this operation does the job. | ||||||||
| Changed: | ||||||||
| < < | 10) ids[] = getApplicationIds() | |||||||
| > > | 
 | |||||||
| Changed: | ||||||||
| < < | Tell me the hub-assigned ids of all the registered applications. | |||||||
| > > | Tell me the hub-assigned public ids of all the registered applications. | |||||||
| Changed: | ||||||||
| < < | 11) ids[] = getApplicationsUnderstanding(mtype) | |||||||
| > > | 
 | |||||||
| Changed: | ||||||||
| < < | Tell me the hub-assigned ids of the all the registered applications that understand messages with this mtype. | |||||||
| > > | Tell me the hub-assigned public ids of the all the registered applications that understand messages with this mtype. | |||||||
| Changed: | ||||||||
| < < | 12) mtypes[] = getUnderstoodMessagesForApplication(id) | |||||||
| > > | 
 | |||||||
| Deleted: | ||||||||
| < < | Tell me what mtypes this particular application claims to support. | |||||||
| Changed: | ||||||||
| < < | Security | |||||||
| > > | Tell me the mtypes of the messages this application wants to receive. | |||||||
| Deleted: | ||||||||
| < < | The question of applications spoofing one another has come up a few times. While I don't really believe it's a serious risk (those astronomers - they're such jokers!), I set out below a simple modification to the above protocol that would make it more secure. This is based on proposals by Mark Taylor in discussions we had a year ago - if they don't make any sense then blame my faulty memory rather than Mark. | |||||||
| Deleted: | ||||||||
| < < | The register operation could be changed as follows: (id, application-secret) = register*(hub-secret) | |||||||
| Changed: | ||||||||
| < < | The hub-secret is a secret that can only be easily known by applications running under the user's uid and is intended to defeat other users on the same machine who might try ( e.g.) port scanning. For instance, it could be a random string written into the .ivoamsg file. The application-secret is a per-application secret that apps must keep track of and use along with their id to identify themselves. Thus, operations 2-8 would all include this as an extra parameter (though it wouldn't be transmitted to any receiving application by the hub.) This prevents Topcat pretending to be Aladin and vice versa (you guys!). | |||||||
| > > | 
 | |||||||
| Changed: | ||||||||
| < < | Well, I hope that you have time in between munching Easter Eggs to give me your thoughts on this. I'll try to get this up on the wiki so that we can keep track of modifications and suggestions. | |||||||
| > > | Return the metadata for that application. | |||||||
| Added: | ||||||||
| > > | The operations a client application should support
 | |||||||
| Changed: | ||||||||
| < < | ||||||||
| > > | Receive a message from this sender. | |||||||
| <-- 
 | ||||||||
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change. | ||||||||
| Added: | ||||||||
| > > | In the following, I've used [] to denote multiple values, rather than necessarily an array. | |||||||
| Added: | ||||||||
| > > | The operations a client application should support
 The operations/methods a Hub should support | |||||||
| Added: | ||||||||
| > > | Registration
 
 Messaging
 Application metadata
 ===============================================1) id= register*()
2) declareAppMetaData(id, metadata)
3) declareMessages(id, mtypes[])
4) unregister(id)
When an application closes, it should inform the hub.
5) msgId = sendAsynch(senderId, recipientId, mtype, args[])
6) msgId = broadcastAsynch(senderId, mtype, args[])
OK, this is where the fun happens.  5) sends a message from sender to recipient using the hub-supplied ids for addressing.  6) sends the message to everyone (if they've elected to receive that mtype).  The hub returns a session-unique msgId that can be used in further comms, such as to tie reply messages up with request messages (in Mike's parlance).  In these operations the message argument list args[] is intended to be quite general. 
I'm explicitly writing out the parameters here...if you prefer you can write it:
5) msgId = sendAsynch(message)
where message wraps the whole lot.  At some point we need to decide which parameters are part of the message, and which are not.  My initial feeling is that mtype and args[] are part of the message, while the addressing info are not.  [We could also add the msgId (if one already exists for that "conversation") as one of the input params, or we could put it inside args[] - I'm not sure which is best.] 
Plastic devs will note another difference with Plastic - the above operations reflect the two most common modes of use that we've encountered in 18 months of experience: one-to-one, and one-to-all.  Plastic has a one-to-subsetOfAll mode, which is rarely used to send to more than one application and is extremely annoying. 
7) msgId, response = send(senderId, recipientId, mtype, args[])
8) msgId, responses[] = broadcast(senderId, mtype, args[])
These are like 5 and 6, but block and then return a response value.  This value can be quite general ( e.g. some kind of datastructure)
These are going to be contentious. Plastic allows a request-response style messaging pattern.  This is not exactly radical: SOAP does it, and MPI has blocking send modes (though they're not quite the same as this).  It is very useful in making the programmer's life easier when dealing with very quick informational messages such as "tell me what your application's icon is".  Yes, I know you could do this asynchronously and sit around waiting for a message to come back with the answer, but I feel that we should be prepared to sacrifice some purity of style to make the application programmer's life easier.  Furthermore, my putative python-command-line hacker cannot receive data back from applications any other way (or at least, not easily). 
However, I'd be interested to know what the community thinks.  Could including a request-response style seriously restrict our choices when it comes to picking wire protocols?
9) hubId = getHubId()
We can treat the hub itself as just another application that can be messaged.  However, you do need to know which application it is, and this operation does the job.
10) ids[] = getApplicationIds()
Tell me the hub-assigned ids of all the registered applications.
11) ids[] = getApplicationsUnderstanding(mtype)
Tell me the hub-assigned ids of the all the registered applications that understand messages with this mtype.
12) mtypes[] = getUnderstoodMessagesForApplication(id) 
Tell me what mtypes this particular application claims to support.
SecurityThe question of applications spoofing one another has come up a few times. While I don't really believe it's a serious risk (those astronomers - they're such jokers!), I set out below a simple modification to the above protocol that would make it more secure. This is based on proposals by Mark Taylor in discussions we had a year ago - if they don't make any sense then blame my faulty memory rather than Mark. The register operation could be changed as follows: (id, application-secret) = register*(hub-secret) The hub-secret is a secret that can only be easily known by applications running under the user's uid and is intended to defeat other users on the same machine who might try ( e.g.) port scanning. For instance, it could be a random string written into the .ivoamsg file. The application-secret is a per-application secret that apps must keep track of and use along with their id to identify themselves. Thus, operations 2-8 would all include this as an extra parameter (though it wouldn't be transmitted to any receiving application by the hub.) This prevents Topcat pretending to be Aladin and vice versa (you guys!). Well, I hope that you have time in between munching Easter Eggs to give me your thoughts on this. I'll try to get this up on the wiki so that we can keep track of modifications and suggestions. | |||||||
| <-- 
 | ||||||||
| 
 High Level ProtocolThis document describes the application messaging protocol in terms of the operations a Hub and a client application should support. It does not describe lower level details such as the wire protocol and parameter types. This document is still under discussion and will change.<-- 
 | 
 
  
  Copyright © 2008-2025 by the contributing authors. All material on this collaboration platform is the property of the contributing authors.
Copyright © 2008-2025 by the contributing authors. All material on this collaboration platform is the property of the contributing authors.