1diameter_app(3) Erlang Module Definition diameter_app(3)
2
3
4
6 diameter_app - Callback module of a Diameter application.
7
9 A diameter service as started by diameter:start_service/2 configures
10 one of more Diameter applications, each of whose configuration speci‐
11 fies a callback that handles messages specific to the application. The
12 messages and AVPs of the application are defined in a dictionary file
13 whose format is documented in diameter_dict(4) while the callback mod‐
14 ule is documented here. The callback module implements the Diameter ap‐
15 plication-specific functionality of a service.
16
17 A callback module must export all of the functions documented below.
18 The functions themselves are of three distinct flavours:
19
20 * peer_up/3 and peer_down/3 signal the attainment or loss of connec‐
21 tivity with a Diameter peer.
22
23 * pick_peer/4, prepare_request/3, prepare_retransmit/3, handle_an‐
24 swer/4 and handle_error/4 are (or may be) called as a consequence
25 of a call to diameter:call/4 to send an outgoing Diameter request
26 message.
27
28 * handle_request/3 is called in response to an incoming Diameter re‐
29 quest message.
30
31 The arities for the the callback functions here assume no extra argu‐
32 ments. All functions will also be passed any extra arguments configured
33 with the callback module itself when calling diameter:start_service/2
34 and, for the call-specific callbacks, any extra arguments passed to di‐
35 ameter:call/4.
36
38 capabilities() = #diameter_caps{}:
39 A record containing the identities of the local Diameter node and
40 the remote Diameter peer having an established transport connec‐
41 tion, as well as the capabilities as determined by capabilities ex‐
42 change. Each field of the record is a 2-tuple consisting of values
43 for the (local) host and (remote) peer. Optional or possibly multi‐
44 ple values are encoded as lists of values, mandatory values as the
45 bare value.
46
47 message() = diameter_codec:message():
48 The representation of a Diameter message as passed to diame‐
49 ter:call/4 or returned from a handle_request/3 callback.
50
51 packet() = diameter_codec:packet():
52 A container for incoming and outgoing Diameter messages that's
53 passed through encode/decode and transport. Fields should not be
54 set in return values except as documented.
55
56 peer_ref() = term():
57 A term identifying a transport connection with a Diameter peer.
58
59 peer() = {peer_ref(), capabilities()}:
60 A tuple representing a Diameter peer connection.
61
62 state() = term():
63 The state maintained by the application callback functions
64 peer_up/3, peer_down/3 and (optionally) pick_peer/4. The initial
65 state is configured in the call to diameter:start_service/2 that
66 configures the application on a service. Callback functions return‐
67 ing a state are evaluated in a common service-specific process
68 while those not returning state are evaluated in a request-specific
69 process.
70
72 Mod:peer_up(SvcName, Peer, State) -> NewState
73
74 Types:
75
76 SvcName = diameter:service_name()
77 Peer = peer()
78 State = NewState = state()
79
80 Invoked to signal the availability of a peer connection on the
81 local Erlang node. In particular, capabilities exchange with the
82 peer has indicated support for the application in question, the
83 RFC 3539 watchdog state machine for the connection has reached
84 state OKAY and Diameter messages can be both sent and received.
85
86 Note:
87 A watchdog state machine can reach state OKAY from state SUSPECT
88 without a new capabilities exchange taking place. A new trans‐
89 port connection (and capabilities exchange) results in a new
90 peer_ref().
91
92
93 Note:
94 There is no requirement that a callback return before incoming
95 requests are received: handle_request/3 callbacks must be han‐
96 dled independently of peer_up/3 and peer_down/3.
97
98
99 Mod:peer_down(SvcName, Peer, State) -> NewState
100
101 Types:
102
103 SvcName = diameter:service_name()
104 Peer = peer()
105 State = NewState = state()
106
107 Invoked to signal that a peer connection on the local Erlang
108 node is no longer available following a previous call to
109 peer_up/3. In particular, that the RFC 3539 watchdog state ma‐
110 chine for the connection has left state OKAY and the peer will
111 no longer be a candidate in pick_peer/4 callbacks.
112
113 Mod:pick_peer(LocalCandidates, RemoteCandidates, SvcName, State) -> Se‐
114 lection | false
115
116 Types:
117
118 LocalCandidates = RemoteCandidates = [peer()]
119 SvcName = diameter:service_name()
120 State = NewState = state()
121 Selection = {ok, Peer} | {Peer, NewState}
122 Peer = peer() | false
123
124 Invoked as a consequence of a call to diameter:call/4 to select
125 a destination peer for an outgoing request. The return value in‐
126 dicates the selected peer.
127
128 The candidate lists contain only those peers that have adver‐
129 tised support for the Diameter application in question during
130 capabilities exchange, that have not be excluded by a filter op‐
131 tion in the call to diameter:call/4 and whose watchdog state ma‐
132 chine is in the OKAY state. The order of the elements is unspec‐
133 ified except that any peers whose Origin-Host and Origin-Realm
134 matches that of the outgoing request (in the sense of a {filter,
135 {all, [host, realm]}} option to diameter:call/4) will be placed
136 at the head of the list. LocalCandidates contains peers whose
137 transport process resides on the local Erlang node while Remote‐
138 Candidates contains peers that have been communicated from other
139 nodes by services of the same name.
140
141 A callback that returns a peer() will be followed by a pre‐
142 pare_request/3 callback and, if the latter indicates that the
143 request should be sent, by either handle_answer/4 or handle_er‐
144 ror/4 depending on whether or not an answer message is received
145 from the peer. If the transport becomes unavailable after pre‐
146 pare_request/3 then a new pick_peer/4 callback may take place to
147 failover to an alternate peer, after which prepare_retransmit/3
148 takes the place of prepare_request/3 in resending the request.
149 There is no guarantee that a pick_peer/4 callback to select an
150 alternate peer will be followed by any additional callbacks
151 since a retransmission to an alternate peer is abandoned if an
152 answer is received from a previously selected peer.
153
154 The return values false and {false, State} (that is, NewState =
155 State) are equivalent, as are {ok, Peer} and {Peer, State}.
156
157 Note:
158 The diameter:service_opt() use_shared_peers determines whether
159 or not a service uses peers shared from other nodes. If not then
160 RemoteCandidates is the empty list.
161
162
163 Warning:
164 The return value {Peer, NewState} is only allowed if the Diame‐
165 ter application in question was configured with the diameter:ap‐
166 plication_opt() {call_mutates_state, true}. Otherwise, the State
167 argument is always the initial value as configured on the appli‐
168 cation, not any subsequent value returned by a peer_up/3 or
169 peer_down/3 callback.
170
171
172 Mod:prepare_request(Packet, SvcName, Peer) -> Action
173
174 Types:
175
176 Packet = packet()
177 SvcName = diameter:service_name()
178 Peer = peer()
179 Action = Send | Discard | {eval_packet, Action, PostF}
180 Send = {send, packet() | message()}
181 Discard = {discard, Reason} | discard
182 PostF = diameter:eval()}
183
184 Invoked to return a request for encoding and transport. Allows
185 the sender to use the selected peer's capabilities to modify the
186 outgoing request. Many implementations may simply want to return
187 {send, Packet}
188
189 A returned packet() should set the request to be encoded in its
190 msg field and can set the transport_data field in order to pass
191 information to the transport process. Extra arguments passed to
192 diameter:call/4 can be used to communicate transport (or any
193 other) data to the callback.
194
195 A returned packet() can set the header field to a #diame‐
196 ter_header{} to specify values that should be preserved in the
197 outgoing request, values otherwise being those in the header
198 record contained in Packet. A returned length, cmd_code or ap‐
199 plication_id is ignored.
200
201 A returned PostF will be evaluated on any encoded #diame‐
202 ter_packet{} prior to transmission, the bin field containing the
203 encoded binary. The return value is ignored.
204
205 Returning {discard, Reason} causes the request to be aborted and
206 the diameter:call/4 for which the callback has taken place to
207 return {error, Reason}. Returning discard is equivalent to re‐
208 turning {discard, discarded}.
209
210 Mod:prepare_retransmit(Packet, SvcName, Peer) -> Action
211
212 Types:
213
214 Packet = packet()
215 SvcName = diameter:service_name()
216 Peer = peer()
217 Action = Send | Discard | {eval_packet, Action, PostF}
218 Send = {send, packet() | message()}
219 Discard = {discard, Reason} | discard
220 PostF = diameter:eval()}
221
222 Invoked to return a request for encoding and retransmission. Has
223 the same role as prepare_request/3 in the case that a peer con‐
224 nection is lost an an alternate peer selected but the argument
225 packet() is as returned by the initial prepare_request/3.
226
227 Returning {discard, Reason} causes the request to be aborted and
228 a handle_error/4 callback to take place with Reason as initial
229 argument. Returning discard is equivalent to returning {discard,
230 discarded}.
231
232 Mod:handle_answer(Packet, Request, SvcName, Peer) -> Result
233
234 Types:
235
236 Packet = packet()
237 Request = message()
238 SvcName = diameter:service_name()
239 Peer = peer()
240 Result = term()
241
242 Invoked when an answer message is received from a peer. The re‐
243 turn value is returned from diameter:call/4 unless the detach
244 option was specified.
245
246 The decoded answer record and undecoded binary are in the msg
247 and bin fields of the argument packet() respectively. Request is
248 the outgoing request message as was returned from prepare_re‐
249 quest/3 or prepare_retransmit/3.
250
251 For any given call to diameter:call/4 there is at most one han‐
252 dle_answer/4 callback: any duplicate answer (due to retransmis‐
253 sion or otherwise) is discarded. Similarly, only one of han‐
254 dle_answer/4 or handle_error/4 is called.
255
256 By default, an incoming answer message that cannot be success‐
257 fully decoded causes the request process to fail, causing diame‐
258 ter:call/4 to return {error, failure} unless the detach option
259 was specified. In particular, there is no handle_error/4 call‐
260 back in this case. The diameter:application_opt() answer_errors
261 can be set to change this behaviour.
262
263 Mod:handle_error(Reason, Request, SvcName, Peer) -> Result
264
265 Types:
266
267 Reason = timeout | failover | term()
268 Request = message()
269 SvcName = diameter:service_name()
270 Peer = peer()
271 Result = term()
272
273 Invoked when an error occurs before an answer message is re‐
274 ceived in response to an outgoing request. The return value is
275 returned from diameter:call/4 unless the detach option was spec‐
276 ified.
277
278 Reason timeout indicates that an answer message has not been re‐
279 ceived within the time specified with the corresponding diame‐
280 ter:call_opt(). Reason failover indicates that the transport
281 connection to the peer to which the request has been sent has
282 become unavailable and that not alternate peer was not selected.
283
284 Mod:handle_request(Packet, SvcName, Peer) -> Action
285
286 Types:
287
288 Packet = packet()
289 SvcName = term()
290 Peer = peer()
291 Action = Reply | {relay, [Opt]} | discard |
292 {eval|eval_packet, Action, PostF}
293 Reply = {reply, packet() | message()} | {answer_message,
294 3000..3999|5000..5999} | {protocol_error, 3000..3999}
295 Opt = diameter:call_opt()
296 PostF = diameter:eval()
297
298 Invoked when a request message is received from a peer. The ap‐
299 plication in which the callback takes place (that is, the call‐
300 back module as configured with diameter:start_service/2) is de‐
301 termined by the Application Identifier in the header of the in‐
302 coming request message, the selected module being the one whose
303 corresponding dictionary declares itself as defining either the
304 application in question or the Relay application.
305
306 The argument packet() has the following signature.
307
308 #diameter_packet{header = #diameter_header{},
309 avps = [#diameter_avp{}],
310 msg = record() | undefined,
311 errors = [Unsigned32() | {Unsigned32(), #diameter_avp{}}],
312 bin = binary(),
313 transport_data = term()}
314
315
316 The msg field will be undefined in case the request has been re‐
317 ceived in the relay application. Otherwise it contains the
318 record representing the request as outlined in diameter_dict(4).
319
320 The errors field specifies any results codes identifying errors
321 found while decoding the request. This is used to set Result-
322 Code and/or Failed-AVP in a returned answer unless the callback
323 returns a #diameter_packet{} whose errors field is set to either
324 a non-empty list of its own, in which case this list is used in‐
325 stead, or the atom false to disable any setting of Result-Code
326 and Failed-AVP. Note that the errors detected by diameter are of
327 the 3xxx and 5xxx series, Protocol Errors and Permanent Failures
328 respectively. The errors list is empty if the request has been
329 received in the relay application.
330
331 The transport_data field contains an arbitrary term passed into
332 diameter from the transport module in question, or the atom un‐
333 defined if the transport specified no data. The term is pre‐
334 served if a message() is returned but must be set explicitly in
335 a returned packet().
336
337 The semantics of each of the possible return values are as fol‐
338 lows.
339
340 {reply, packet() | message()}:
341 Send the specified answer message to the peer. In the case
342 of a packet(), the message to be sent must be set in the msg
343 field and the header field can be set to a #diame‐
344 ter_header{} to specify values that should be preserved in
345 the outgoing answer, appropriate values otherwise being set
346 by diameter.
347
348 {answer_message, 3000..3999|5000..5999}:
349 Send an answer message to the peer containing the specified
350 Result-Code. Equivalent to
351
352 {reply, ['answer-message' | Avps]
353
354
355 where Avps sets the Origin-Host, Origin-Realm, the specified
356 Result-Code and (if the request contained one) Session-Id
357 AVPs, and possibly Failed-AVP as described below.
358
359 Returning a value other than 3xxx or 5xxx will cause the re‐
360 quest process in question to fail, as will returning a 5xxx
361 value if the peer connection in question has been configured
362 with the RFC 3588 common dictionary diame‐
363 ter_gen_base_rfc3588. (Since RFC 3588 only allows 3xxx val‐
364 ues in an answer-message.)
365
366 When returning 5xxx, Failed-AVP will be populated with the
367 AVP of the first matching Result-Code/AVP pair in the errors
368 field of the argument packet(), if found. If this is not ap‐
369 propriate then an answer-message should be constructed ex‐
370 plicitly and returned in a reply tuple instead.
371
372 {relay, Opts}:
373 Relay a request to another peer in the role of a Diameter
374 relay agent. If a routing loop is detected then the request
375 is answered with 3005 (DIAMETER_LOOP_DETECTED). Otherwise a
376 Route-Record AVP (containing the sending peer's Origin-Host)
377 is added to the request and pick_peer/4 and subsequent call‐
378 backs take place just as if diameter:call/4 had been called
379 explicitly. The End-to-End Identifier of the incoming re‐
380 quest is preserved in the header of the relayed request.
381
382 The returned Opts should not specify detach. A subsequent
383 handle_answer/4 callback for the relayed request must return
384 its first argument, the packet() containing the answer mes‐
385 sage. Note that the extra option can be specified to supply
386 arguments that can distinguish the relay case from others if
387 so desired. Any other return value (for example, from a han‐
388 dle_error/4 callback) causes the request to be answered with
389 3002 (DIAMETER_UNABLE_TO_DELIVER).
390
391 discard:
392 Discard the request. No answer message is sent to the peer.
393
394 {eval, Action, PostF}:
395 Handle the request as if Action has been returned and then
396 evaluate PostF in the request process. The return value is
397 ignored.
398
399 {eval_packet, Action, PostF}:
400 Like eval but evaluate PostF on any encoded #diame‐
401 ter_packet{} prior to transmission, the bin field containing
402 the encoded binary. The return value is ignored.
403
404 {protocol_error, 3000..3999}:
405 Equivalent to {answer_message, 3000..3999}.
406
407 Note:
408 Requests containing errors may be answered by diameter, without
409 a callback taking place, depending on the value of the diame‐
410 ter:application_opt() request_errors.
411
412
413
414
415Ericsson AB diameter 2.3 diameter_app(3)