Note that the following elements in the code examples below must be replaced by actual values from Franca:
// "<Attribute>" the Franca name of the attribute
// "<AttributeType>" the Franca name of the attribute type
// "<broadcast>" the Franca name of the broadcast, starting with a lowercase letter
// "<Broadcast>" the Franca name of the broadcast, starting with capital letter
// "BroadcastFilter<Attribute>" Attribute is the Franca attributes name
// "<Filter>" the Franca name of the broadcast filter
// "<interface>" the Franca interface name, starting with a lowercase letter
// "<Interface>" the Franca interface name, starting with capital letter
// "<method>" the Franca method name, starting with a lowercase letter
// "<Method>" the Franca method name, starting with capital letter
// "<OutputType>" the Franca broadcast output type name
// "<Package>" the Franca package name
// "<ProviderDomain>" the provider domain name used by provider and client
// "<ReturnType>" the Franca return type name
The Franca <Package>
will be transformed to the Java package joynr.<Package>
.
The Franca <TypeCollection>
will be transformed to the Java package joynr.<Package>.<TypeCollection>
.
Any Franca complex type <TypeCollection>.<Type>
will result in the creation of a class joynr.<Package>.<TypeCollection>.<Type>
(see above).
The same <Type>
will be used for all elements in the event that this type is used as an element of other complex types, as a method input or output argument, or as a broadcast output argument.
Getter and Setter methods will be created for any element of a struct type. Also a standard constructor, full arguments constructor and object argument constructor will be created automatically.
The Franca <Interface>
will be used as a prefix to create the following Java classes or interfaces:
public abstract class joynr.<Package>.<Interface>AbstractProvider
public interface joynr.<Package>.<Interface>Async
public interface joynr.<Package>.<Interface>BroadcastInterface
public interface joynr.<Package>.<Interface>
public abstract class joynr.<Package>.<Interface><Broadcast>BroadcastFilter
public interface joynr.<Package>.<Interface>Provider
public interface joynr.<Package>.<Interface>Proxy
public interface joynr.<Package>.<Interface>SubscriptionInterface
public interface joynr.<Package>.<Interface>Sync
A java joynr application inherits from AbstractJoynrApplication
class and contains at least a main()
, run()
and shutdown()
method.
The following base imports are required for a Java Consumer application:
import io.joynr.arbitration.ArbitrationStrategy;
import io.joynr.arbitration.DiscoveryQos;
import io.joynr.arbitration.DiscoveryScope;
import io.joynr.exceptions.DiscoveryException;
import io.joynr.exceptions.JoynrCommunicationException;
import io.joynr.exceptions.JoynrRuntimeException;
import io.joynr.messaging.MessagingPropertyKeys;
import io.joynr.messaging.MessagingQos;
import io.joynr.proxy.Callback;
import io.joynr.proxy.Future;
import io.joynr.proxy.ProxyBuilder;
import io.joynr.runtime.AbstractJoynrApplication;
import io.joynr.runtime.JoynrApplication;
import io.joynr.runtime.JoynrApplicationModule;
import io.joynr.runtime.JoynrInjectorFactory;
import java.io.IOException;
import java.util.Properties;
import joynr.<Package>.<Interface>Proxy;
import com.google.inject.Inject;
import com.google.inject.name.Named;
// required imports
...
public class MyConsumerApplication extends AbstractJoynrApplication {
@Inject
@Named(APP_CONFIG_PROVIDER_DOMAIN)
private String providerDomain;
private <Interface>Proxy <interface>Proxy;
public static void main(String[] args) throws IOException {
// initialization, perhaps including setting the domain
// for this instance of the application.
}
public void run() {
// main application logic
}
public void shutdown() {
// unregister and cleanup
}
}
The main()
method must setup the configuration (provider domain etc.) and create the JoynrApplication
instance by instantiating a new JoynrApplicationModule
. Then the run()
method of the consumer application can be called to do the work.
As a prerequisite, the provider and consumer domain need to be defined using Properties
as shown below.
public static void main(String[] args) throws IOException {
String providerDomain = "<ProviderDomain>";
Properties joynrConfig = new Properties();
joynrConfig.setProperty(MessagingPropertyKeys.PERSISTENCE_FILE, STATIC_PERSISTENCE_FILE);
joynrConfig.setProperty(PROPERTY_JOYNR_DOMAIN_LOCAL, "my_local_domain");
Properties appConfig = new Properties();
appConfig.setProperty(APP_CONFIG_PROVIDER_DOMAIN, providerDomain);
JoynrApplication myConsumerApp = new JoynrInjectorFactory(joynrConfig).createApplication(
new JoynrApplicationModule(MyApplication.class, appConfig));
myConsumerApp.run();
myConsumerApp.shutdown();
}
The class DiscoveryQos
configures how the search for a provider will be handled. It has the following members:
- discoveryTimeout Timeout for discovery process (milliseconds), afterwards triggers DiscoveryException
- cacheMaxAge Defines the maximum allowed age of cached entries (milliseconds), only younger entries will be considered. If no suitable providers are found, then depending on the discoveryScope, a remote global lookup may be triggered.
- arbitrationStrategy The arbitration strategy (see below)
- customParameters special parameters, that must match, e.g. keyword (see below)
- retryInterval The time to wait between discovery retries after encountering a discovery error.
- discoveryScope default: LOCAL_AND_GLOBAL (details see below)
The discoveryScope defines, whether a suitable provider will be searched only in the local capabilities directory or also in the global one.
Available values are as follows:
- LOCAL_ONLY Only entries from local capability directory will be searched
- LOCAL_THEN_GLOBAL Entries will be taken from local capabilities directory, unless no such entries exist, in which case global entries will be looked at as well.
- LOCAL_AND_GLOBAL Entries will be taken from local capabilities directory and from global capabilities directory.
- GLOBAL_ONLY Only the global entries will be looked at.
Whenever global entries are involved, they are first searched in the local cache. In case no global entries are found in the cache, a remote lookup is triggered.
The enumeration ArbitrationStrategy
defines special options to select a Provider:
- NotSet (not allowed in the app, otherwise arbitration will throw DiscoveryException)
- FixedChannel (also see FixedParticipantArbitrator)
- Keyword Only entries that have a matching keyword will be considered
- HighestPriority Entries will be considered according to priority
The priority is set by the provider through the call providerQos.setPriority()
.
Class ArbitrationConstants
provides keys for the key-value pair for the custom Parameters of discoveryScope:
- PRIORITY_PARAMETER (apparently not implemented as of now)
- KEYWORD_PARAMETER
- FIXEDPARTICIPANT_KEYWORD
Example for Keyword arbitration strategy:
discoveryQos.addCustomParameter(ArbitrationConstants.KEYWORD_PARAMETER, "keyword");
Example for FixedChannel arbitration strategy:
discoveryQos.addCustomParameter(ArbitrationConstants.FIXED_PARTICIPANT_KEYWORD, participantId);
Example for the creation of a DiscoveryQos class object:
DiscoveryQos discoveryQos = new DiscoveryQos();
discoveryQos.setDiscoveryTimeout(10000); // optional, default 30000
discoveryQos.setCacheMaxAge(Long.MAX_VALUE); // optional, default 0
discoveryQos.setArbitrationStrategy(ArbitrationStrategy.HighestPriority); // default HP
discoveryQos.addCustomParameter(key, value); // optional, default none
discoveryQos.setProviderMustSupportOnChange(true); // optional, default false
discoveryQos.setRetryInterval(1000); // optional, default 1000
discoveryQos.setDiscoveryScope(DiscoveryScope.LOCAL_AND_GLOBAL); // optional, default as stated
The MesssagingQos
class defines the roundtrip timeout for RPC requests in milliseconds.
If no specific setting is given, the default is 60 seconds.
Example:
long ttl_ms = 60000;
MessagingQos messagingQos = new MessagingQos(ttl_ms);
Inside the run()
method, the consumer application instance must create one proxy per used Franca interface in order to be able to
- call its methods (RPC) either synchronously or asynchronously
- subscribe or unsubscribe to its attributes or update a subscription
- subscribe or unsubscribe to its broadcasts or update a subscription
In case no suitable provider can be found during discovery, a DiscoveryException
is thrown.
In case of communication errors, a JoynrCommunicationException
is thrown.
@Override
public void run() {
DiscoveryQos discoveryQos = new DiscoveryQos();
// the qos can be fine tuned here by calling setters
ProxyBuilder<<Interface>Proxy> proxyBuilder =
runtime.getProxyBuilder(providerDomain, <Interface>Proxy.class);
try {
<interface>Proxy = proxyBuilder.
setMessagingQos(new MessagingQos()).
setDiscoveryQos(discoveryQos).
build();
// call methods, subscribe to broadcasts etc.
// enter some event loop
} catch (DiscoveryException e) {
// no provider found
} catch (JoynrCommunicationException e) {
// could not send message
}
}
While the provider executes the call asynchronously in any case, the consumer will wait until the call is finished, i.e. the thread will be blocked. Note that the message order on Joynr RPCs will not be preserved.
Example for calls with single return parameter:
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public void run() {
// setup proxy named <interface>Proxy
...
try {
<ReturnType> retval;
retval = <interface>Proxy.<method>([inputVal1, ..., inputValN]);
} catch (JoynrRuntimeException e) {
// error handling
}
}
In case of multiple return parameters the parameters will be wrapped into a class named
<Method>Returned
. Each parameter value is available through a public member variable inside this class.
Example:
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public void run() {
// setup proxy named <interface>Proxy
...
try {
<Method>Returned retval;
retval = <interface>Proxy.<method>([inputVal1, ..., inputValN]);
// handle return parameters
// retval.<returnParameter1>
// ...
// retval.<returnParameterN>
} catch (JoynrRuntimeException e) {
// error handling
}
}
For methods which are modelled with error enumerations, additionally, ApplicationExceptions have to be caught. The ApplicationException serves as container for the actual error enumeration which can be retrieved by calling e.getError().
Using asynchronous method calls allows the current thread to continue its work. For this purpose a callback has to be provided for the API call in order to receive the result and error respectively. Note the current thread will still be blocked until the Joynr message is internally set up and serialized. It will then be enqueued and handled by a Joynr Middleware thread.
The message order on Joynr RPCs will not be preserved.
If no return type exists, the term Void
is used instead.
Example for calls with single return parameter:
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public class MyCallback implements Callback<<ReturnType>> {
void onSuccess(<ReturnType> result) {
// handle result
}
void onFailure(JoynrRuntimeException error) {
// handle error
}
}
...
public void run() {
// setup proxy named "<interface>Proxy"
...
public Future<<ReturnType>> future;
MyCallback myCallback = new MyCallback();
future = <interface>Proxy.<method>(
myCallback,
[inputVal1, ..., inputValN]
);
try {
long timeoutInMilliseconds;
// set timeout value here
<ReturnType> result = future.get(timeOutInMilliseconds);
} catch (InterruptedException|JoynrRuntimeException e) {
// handle error
}
...
}
If the Franca model includes error enums, then the Callback will also need to implement onFailure for the modeled error:
@Override
public void onFailure(<Method>ErrorEnum errorEnum) {
switch (errorEnum) {
case <ENUM_LITERAL_A>:
break;
case <ENUM_LITERAL_B>:
break;
default:
// handle default error case
break;
}
}
In case of multiple return parameters the parameters will be wrapped into a class named
<Method>Returned
. Each parameter value is available through a public member variable inside this class.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection.<Type>;
...
public class MyCallback implements Callback<<Method>Returned> {
void onSuccess(<Method>Returned result) {
// handle result
}
void onFailure(JoynrRuntimeException error) {
// handle error
}
}
...
public void run() {
// setup proxy named "<interface>Proxy"
...
public Future<<Method>Returned> future;
MyCallback myCallback = new MyCallback();
future = <interface>Proxy.<method>(
myCallback,
[inputVal1, ..., inputValN]
);
try {
long timeoutInMilliseconds;
// set timeout value here
<Method>Returned result = future.get(timeOutInMilliseconds);
// handle return parameters
// result.<returnParameter1>
// ...
// result.<returnParameterN>
} catch (InterruptedException|JoynrRuntimeException e) {
// handle error
}
...
}
If the Franca model includes error enums, then the Callback will also need to implement onFailure for the modeled error:
@Override
public void onFailure(<Method>ErrorEnum errorEnum) {
switch (errorEnum) {
case <ENUM_LITERAL_A>:
break;
case <ENUM_LITERAL_B>:
break;
default:
// handle default error case
break;
}
}
The abstract class SubscriptionQos
has the following members:
- expiry date Absolute Time until notifications will be send (milliseconds)
- publicationTtl Lifespan of a notification (milliseconds), it will be deleted afterwards
The class PeriodicSubscriptionQos
inherits from SubscriptionQos
and has the following additional members:
- period defines how long to wait before sending an update even if the value did not change
- alertAfterInterval Timeout for notifications, afterwards a missed publication notification will be raised (milliseconds)
This class can be used for subscriptions to attributes.
Note that updates will be send only based on the specified interval and not based on changes of the attribute.
The class OnChangeSubscriptionQos
inherits from SubscriptionQos
and has the following additional members:
- minimum Interval Minimum time to wait between successive notifications (milliseconds)
This class should be used for subscriptions to broadcasts. It can also be used for subscriptions to attributes if no periodic update is required.
The class OnChangeWithKeepAliveSubscriptionQos
inherits from OnChangeSubscriptionQos
and has the following additional members:
- maximum Interval Maximum time to wait between notifications, if value has not changed
- alertAfterInterval Timeout for notifications, afterwards a missed publication notification will be raised (milliseconds)
This class can be used for subscriptions to attributes. Updates will then be sent based both periodically and after a change (i.e. this acts like a combination of PeriodicSubscriptionQos
and OnChangeSubscriptionQos
).
Using it for subscriptions to broadcasts is theoretically possible because of inheritance but makes no sense (in this case the additional members will be ignored).
Attribute subscription - depending on the subscription quality of service settings used - informs the application either periodically and / or on change of an attribute about the current value. The subscriptionId returned by the call can be used later to update the subscription or to unsubscribe from it. To receive the subscription, a callback has to be provided.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
import joynr.OnChangeWithKeepAliveSubscriptionQos;
import io.joynr.pubsub.subscription.AttributeSubscriptionAdapter;
...
public void run() {
// setup proxy named "<interface>Proxy"
...
// Quality of service settings can be provided as either
// "PeriodicSubscriptionQos",
// "OnChangeSubscriptionQos" or "OnChangeWithKeepAliveSubscriptionQos" class
// referenced by <QosClass> below
<QosClass> qos = new <QosClass>(... parameters ...);
...
try {
subscriptionId = <interface>Proxy.subscribeTo<Attribute>(
new AttributeSubscriptionAdapter<AttributeType>() {
@Override
public void onReceive(<AttributeType> value) {
// handle info
}
@Override
public void onError(JoynrRuntimeException e) {
// handle subscription error, e.g. missed publication
}
},
qos
);
} catch (JoynrRuntimeException e) {
// handle error
}
}
The subscribeTo method can also be used to update an existing subscription, when the subscriptionId is given as additional parameter as follows:
try {
subscriptionId = <interface>Proxy.subscribeTo<Attribute>(
new AttributeSubscriptionAdapter<AttributeType>() {
@Override
public void onReceive(<AttributeType> value) {
// handle info
}
@Override
public void onError(JoynrRuntimeException e) {
// handle subscription error, e.g. missed publication
}
},
qos,
subscriptionId
);
} catch (JoynrRuntimeException e) {
// handle error
}
Unsubscribing from an attribute subscription also requires the subscriptionId returned by the ealier subscribeTo call.
// for required imports see subscription info
...
public void run() {
// setup proxy named "<interface>Proxy"
...
private String subscriptionId;
...
try {
// subscriptionId must have been assigned by previous call
...
<interface>Proxy.unsubscribeFrom<Attribute>(subscriptionId);
} catch (JoynrRuntimeException e) {
// handle error
}
}
Broadcast subscription informs the application in case a broadcast is fired from provider side and returns the output values via callback. The subscriptionId returned by the call can be used later to update the subscription or to unsubscribe from it.
import joynr.OnChangeSubscriptionQos;
...
// for any Franca broadcast named "<Broadcast>" used
import joynr.<Package>.<Interface>BroadcastInterface.<Broadcast>BroadcastAdapter;
...
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public void run() {
// setup proxy named "<interface>Proxy"
...
private String subscriptionId;
...
try {
int minInterval;
long expiryDate;
int publicationTtl;
...
// provide values for minInterval, expiryDate, publicationTtl here
...
OnChangeSubscriptionQos qos =
new OnChangeSubscriptionQos(minInterval, expiryDate, publicationTtl);
...
subscriptionId = <interface>Proxy.subscribeTo<Broadcast>Broadcast(
new <Broadcast>BroadcastAdapter() {
@Override
public void onReceive(... OutputParameters ...) {
// handle info
}
@Override
public void onError() {
// handle subscription error, e.g. missed info
}
},
qos
);
...
} catch (DiscoveryException e) {
// handle error
} catch (JoynrCommunicationExceptin e) {
// handle error
}
...
}
The subscribeTo method can also be used to update an existing subscription, when the subscriptionId is given as additional parameter as follows:
subscriptionId = <interface>Proxy.subscribeTo<Broadcast>Broadcast(
new <Broadcast>BroadcastAdapter() {
@Override
public void onReceive(... OutputParameters ...) {
// handle info
}
@Override
public void onError() {
// handle subscription error, e.g. missed info
}
},
qos,
subscriptionId
);
Selective Broadcasts use filter logic implemented by the provider and filter parameters set by the consumer to send only those broadcasts from the provider to the consumer that pass the filter. The broadcast output values are passed to the consumer via callback.
The subscriptionId returned by the call can be used later to update the subscription or to unsubscribe from it.
In addition to the normal broadcast subscription, the filter parameters for this broadcast must be created and initialized as additional parameters to the subscribeTo
method. These filter parameters are used to receive only those broadcasts matching the provided filter criteria.
import joynr.OnChangeSubscriptionQos;
...
// for any Franca broadcast named "<Broadcast>" used
import joynr.<Package>.<Interface>BroadcastInterface.<Broadcast>BroadcastAdapter;
import joynr.<Package>.<Interface>BroadcastInterface.<Broadcast>FilterParameters;
...
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public void run() {
// setup proxy named "<interface>Proxy"
...
private String subscriptionId;
...
try {
int minInterval;
long expiryDate;
int publicationTtl;
...
// provide values for minInterval, expiryDate, publicationTtl here
...
OnChangeSubscriptionQos qos =
new OnChangeSubscriptionQos(minInterval, expiryDate, publicationTtl);
<Broadcast>FilterParameters filter = new <Broadcast>FilterParameters();
// foreach BroadcastFilterAttribute of that filter
filter.setBroadcastFilter<Attribute>(value);
...
subscriptionId = <interface>Proxy.subscribeTo<Broadcast>Broadcast(
new <Broadcast>BroadcastAdapter() {
@Override
public void onReceive(... OutputParameters ...) {
// handle info
}
@Override
public void onError() {
// handle subscription error, e.g. missed info
}
},
qos,
filter
);
...
} catch (DiscoveryException e) {
// handle error
} catch (JoynrCommunicationExceptin e) {
// handle error
}
...
}
The subscribeTo method can also be used to update an existing subscription, when the subscriptionId is given as additional parameter as follows:
subscriptionId = <interface>Proxy.subscribeTo<Broadcast>Broadcast(
new <Broadcast>BroadcastAdapter() {
@Override
public void onReceive(... OutputParameters ...) {
// handle info
}
@Override
public void onError() {
// handle subscription error, e.g. missed info
}
},
qos,
filter,
subscriptionId
);
Unsubscribing from a broadcast subscription requires the subscriptionId returned by the earlier subscribe call.
public void run() {
// setup proxy named "<interface>Proxy"
...
private String subscriptionId;
...
try {
<interface>Proxy.unsubscribeFrom<Broadcast>Broadcast(subscriptionId);
...
} catch (DiscoveryException e) {
// handle error
} catch (JoynrCommunicationExceptin e) {
// handle error
}
...
}
The shutdown method should be called on exit of the application. Inside the shutdown()
method , the consumer should unsubscribe from any attributes and broadcasts it was subscribed to and
terminate the instance.
@Override
public void shutdown() {
// for all proxies
if (<interface>Proxy != null) {
if (subscribed) {
// unsubscribe from attributes
// unsubscribe from broadcasts
}
}
runtime.shutdown(true);
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
// handle error
}
System.exit(0);
}
The Java Provider mainly consists of the following classes:
- A generic Provider Application Class
- One Provider Class for each Franca interface to be supported
The provider application class is used to register a provider class for each Franca interface to be supported.
import io.joynr.accesscontrol.StaticDomainAccessControlProvisioning;
import io.joynr.accesscontrol.StaticDomainAccessControlProvisioningModule;
import io.joynr.exceptions.JoynrRuntimeException;
import io.joynr.messaging.MessagingPropertyKeys;
import io.joynr.runtime.AbstractJoynrApplication;
import io.joynr.runtime.JoynrApplication;
import io.joynr.runtime.JoynrApplicationModule;
import io.joynr.runtime.JoynrInjectorFactory;
import java.io.IOException;
import java.util.Properties;
import com.google.inject.Inject;
import com.google.inject.Module;
import edu.umd.cs.findbugs.annotations.SuppressWarnings;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectMapper.DefaultTyping;
The class must extend AbstractJoynrApplication
and can theoretically serve multiple Franca
interfaces.
For each Franca interface implemented, the providing application creates an instance of
My<Interface>Provider
, which implements the service for that particular interface, and
registers it as a capability at the Joynr Middleware.
The example below shows the code for one interface:
package myPackage;
...
// required imports
...
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public class MyProviderApplication extends AbstractJoynrApplication {
private static final String AUTH_TOKEN = "MyProvider_authToken";
public static final String STATIC_PERSISTENCE_FILE = "provider-joynr.properties";
private My<Interface>Provider <interface>provider = null;
public static void main(String[] args) {
// ...
}
@Override
public void run() {
// ...
}
@Override
public void shutdown() {
// ...
}
}
public static void main(String[] args) {
String localDomain = "<ProviderDomain>";
Properties joynrConfig = new Properties();
joynrConfig.setProperty(MessagingPropertyKeys.PERSISTENCE_FILE, STATIC_PERSISTENCE_FILE);
joynrConfig.setProperty(PROPERTY_JOYNR_DOMAIN_LOCAL, localDomain);
Properties appConfig = new Properties();
provisionAccessControl(joynrConfig, localDomain);
JoynrApplication joynrApplication =
new JoynrInjectorFactory(joynrConfig,
new StaticDomainAccessControlProvisioningModule()).createApplication(
new JoynrApplicationModule(MyProviderApplication.class, appConfig)
);
joynrApplication.run();
joynrApplication.shutdown();
}
The run method registers the interface specific provider class instance as capability. From that time on, the provider will be reachable from outside and react on incoming requests (e.g. method RPC etc.). It can be found by consumers through Discovery. Any specific broadcast filters must be added prior to registry.
@Override
public void run() {
<interface>provider = new My<Interface>Provider();
// for any filter of a broadcast with filter
<interface>provider.addBroadcastFilter(new <Filter>BroadcastFilter());
runtime.registerProvider(localDomain, <interface>provider);
// loop here
}
The shutdown
method should be called on exit of the application. It should cleanly unregister
any capabilities the application had registered earlier.
@Override
@SuppressWarnings(value = "DM_EXIT", justification = "WORKAROUND to be removed")
public void shutdown() {
if (<interface>provider != null) {
try {
runtime.unregisterProvider(localDomain, <interface>provider);
} catch (JoynrRuntimeException e) {
// handle error
}
}
runtime.shutdown(true);
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
// do nothing; exiting application
}
System.exit(0);
}
The following allows anyone to access interface:
private static void provisionAccessControl(Properties properties, String domain) throws Exception {
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.enableDefaultTypingAsProperty(DefaultTyping.JAVA_LANG_OBJECT, "_typeName");
MasterAccessControlEntry newMasterAccessControlEntry = new MasterAccessControlEntry(
"*",
domain,
MyProvider.INTERFACE_NAME,
TrustLevel.LOW,
Arrays.asList(TrustLevel.LOW),
TrustLevel.LOW,
Arrays.asList(TrustLevel.LOW),
"*",
Permission.YES,
Arrays.asList(Permission.YES)
);
MasterAccessControlEntry[] provisionedAccessControlEntries = { newMasterAccessControlEntry };
String provisionedAccessControlEntriesAsJson = objectMapper.writeValueAsString(provisionedAccessControlEntries);
properties.setProperty(StaticDomainAccessControlProvisioning.PROPERTY_PROVISIONED_MASTER_ACCESSCONTROLENTRIES,
provisionedAccessControlEntriesAsJson);
}
The provider class implements the attributes, methods and broadcasts of a particular Franca interface.
The following Joynr Java imports are required:
import io.joynr.provider.Deferred;
import io.joynr.provider.DeferredVoid;
import io.joynr.provider.Promise;
import joynr.<Package>.<Interface>AbstractProvider;
The ProviderQos
has the following members:
- customParameters e.g. the key-value for the arbitration strategy Keyword during discovery
- providerVersion the version of the provider
- priority the priority used for arbitration strategy HighestPriority during discovery
- scope the Provider scope (see below), used in discovery
- supportsOnChangeSubscriptions whether the provider supports subscriptions on changes
The ProviderScope can be
- LOCAL The provider will be registered in the local capability directory
- GLOBAL The provider will be registered in the local and global capability directory
Example:
ProviderQos providerQos = new ProviderQos();
providerQos.setCustomParameters(customParameters);
providerQos.setProviderVersion(1);
providerQos.setPriority(100);
providerQos.setScope(ProviderScope.GLOBAL);
providerQos.setSupportsOnChangeSubscriptions(true);
The provider class must extend the generated class <Interface>AbstractProvider
and implement
getter methods for each Franca attribute and a method for each method of the Franca interface. In
order to send broadcasts the generated code of the super class <Interface>AbstractProvider
can be used.
package myPackage;
...
// required imports
...
public class My<Interface>Provider extends <Interface>AbstractProvider {
// member variables realizing the Franca interfaces Attributes go here, if any
<AttributeType> <Attribute>;
...
// default constructor
public My<Interface>Provider() {
// set the priority, used for (default) arbitration by highest priority
long priorityValue;
...
// set priorityValue
...
providerQos.setPriority(priorityValue);
...
// initialize members and attributes here, if any
}
...
// foreach Franca interface "<Attribute>" provide a getter method
...
// foreach Franca interface "<method>" provide an implementation
...
// foreach Franca "<broadcast>" you can use the provided method to fire the event
...
}
The asynchronous getter methods return the current value of an attribute. Since the current thread is blocked while the getter runs, activity should be kept as short as possible. In most cases, when a simple element is returned, the method can resolve the Promise immediately. However, if longer activity is required, it should be done in the background and the deferred should also be resolved by a background thread.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
@Override
public Promise<Deferred<<AttributeType>>> get<Attribute>() {
Deferred<<AttributeType>> deferred = new Deferred<<AttributeType>>();
<AttributeType> value;
...
// start some activity to get the value
// if complex, execute this asynchronously;
// once the value is available, resolve the Promise
// may be run from background thread, if required
deferred.resolve(value);
// if an error occurs, the Deferred can be rejected with a ProviderRuntimeException
deferred.reject(new ProviderRuntimeException(<errorMessage>));
...
// from current thread
return new Promise<Deferred<<AttributeType>>>(deferred);
}
Since the current thread is blocked while the setter runs, activity should be kept as short as possible. In most cases, when a simple element is returned, the method can resolve the Promise immediately. However, if longer activity is required, it should be done in the background and the deferred should also be resolved by a background thread.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
@Override
public Promise<DeferredVoid> set<Attribute>(<AttributeType> <attribute>) {
DeferredVoid deferred = new DeferredVoid();
...
// start some activity to set the value
// if complex, execute this asynchronously;
// once the value is set, resolve the Promise
// may be run from background thread, if required
deferred.resolve();
// if an error occurs, the Deferred can be rejected with a ProviderRuntimeException
deferred.reject(new ProviderRuntimeException(<errorMessage>));
...
// from current thread
return new Promise<DeferredVoid>(deferred);
}
The provider should always implement RPC calls asynchronously in order to not block the main thread longer than required. Also it needs to take care not to overload the server, e.g. it must not accept unlimited amount of RPC requests causing background activity. After exceeding a limit, further calls should be rejected until the number of outstanding activities falls below the limit again.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
@Override
public Promise<Deferred<<ReturnType>>> <method>(... parameters ...) {
Deferred<<ReturnType>> deferred = new Deferred<<ReturnType>>();
<ReturnType> returnValue;
...
// start some activity to perform the task;
// if complex, execute this asynchronously by background thread;
// once the task is finished, resolve the Promise providing
// the returnValue, if any (see following line).
deferred.resolve(returnValue);
// For methods which are modelled with error enumerations, the Promise can be rejected with such
// an error enumeration. It is then wrapped in an ApplicationException which serves as container
// for the actual error enumeration.
deferred.reject(<ErrorEnum>.<VALUE>);
// If no errors are modelled, the Deferred can be rejected with a ProviderRuntimeException
deferred.reject(new ProviderRuntimeException(<errorMessage>));
...
// from current thread
return new Promise<Deferred<<ReturnType>>>(deferred);
}
Firing a broadcast blocks the current thread until the message is serialized.
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public void fire<Broadcast>Event {
<OutputValueType1> outputValue1;
...
<OutputValueTypeN> outputValueN;
...
// setup outputValue(s)
...
// use the method provided by generators to send the broadcast
fire<Broadcast>(outputValue1, ... , outputValueN);
}
In contrast to unfiltered broadcasts, to realize selective (filtered) broadcasts, the filter logic has to be implemented and registered by the provider. If multiple filters are registered on the same provider and broadcast, all filters are applied in a chain and the broadcast is only delivered if all filters in the chain return true.
A broadcast filter class implements a filtering function called filter()
which returns a
boolean value indicating whether the broadcast should be delivered. The input parameters of the
filter()
method reflect the output values of the broadcast.
import joynr.<Package>.<Interface>BroadcastInterface.<Broadcast>BroadcastFilterParameters;
import joynr.<Package>.<Interface><Broadcast>BroadcastFilter;
// for any Franca type named "<Type>" used
import joynr.<Package>.<TypeCollection>.<Type>;
...
public class <Filter>BroadcastFilter extends <Interface><Broadcast>BroadcastFilter {
...
@Override
public boolean filter(
<OutputValueType1> outputValue1,
...
<OutputValueTypeN> outputValueN,
<Broadcast>BroadcastFilterParameters filterParameters
) {
boolean returnValue;
...
// calculate result
...
return returnValue;
}
}