The subrequest Statement
subrequest <type> {
[ statements ]
}
The subrequest
keyword creates a child request. The child request
is empty, and contains no attributes. Attributes in the child can be
copied from the parent via editing statements.
Please see the list syntax for a description of how to
refer to parent requests.
The subrequest
keyword allows the server to receive one type of
packet, and then to create a different type of packet. subrequest
is most commonly used in conjunction with the
call keyword to change packet types or
protocols, and then run a different virtual server. See the
"Operation" section below for more information.
- <type>
-
The type of the child request being created.
The <type> field is a either an enumerated packet name such as
::Access-Request
, or a protocol name followed by a packet name, such asdhcpv4::Discover
. Please see the protocol dictionaries for a complete list of packet types for each protocol.The <type> field cannot be a dynamic expansion.
The _<type> field can, however, be an attribute reference. When an attribute reference is used, the attribute must be of an integer type (8, 16, 32, or 64-bit), or of type
string
. Integer types are resolved to known / allowed values of thePacket-Type
attribute. String types are resolved to named values for thePacket-Type
attribute.When the <type> field is an attribute reference, it is not possible to change the dictionary.
- [ statements ]
-
The
unlang
commands which will be executed.
The child request executes the statements inside of the subrequest
section. When the child request has finished execution, it is freed.
The return code of the subrequest
section is the return code from
the last statement that was executed.
subrequest ::Disconnect-Request {
&User-Name := &parent.request.User-Name
&NAS-IP-Address := &parent.request.NAS-IP-Address
&NAS-Port := &parent.request.NAS-Port
&Acct-Session-Id := &parent.request.Acct-Session-Id
radius
}
Changing Protocols
The subrequest
keyword can also be used to change protocols. For
example, the server can receive a RADIUS Access-Request
, and then
create a DHCPv4 packet of type Discover
. Note that when the
protocol changes, the attributes in the subrequest
section are
parsed in the context of the new protocol.
subrequest dhcpv4::Discover {
&Your-IP-Address := &parent.request.Framed-IP-Address
...
}
Operation
The subrequest
keyword is an generalization of functionality which
was in version 3. In previous versions of the server, child requests
could be created only via the update coa
syntax. The subrequest
keyword is much more powerful than the previous functionality. It can
be used anywhere, and can create any kind of packet.
The main purpose of a subrequest
is to change packet types, or
protocols. For example, when a user is authenticated it is sometimes
useful to start an accounting session for that user immediately. In
normal operations, accounting sessions are started when an accounting
packet is received, but those packets may be delayed for seconds, or
may be missing altogether.
The solution to this problem is to start an accounting session immediately when the user is authenticated. However, any "ad hoc" way of starting a session is likely to miss some critical step which is done for normal accounting packets. The best approach, then, is to use the normal accounting process to create the user session.
However, if the incoming packet is Access-Request
, then it clearly
cannot be used as part of an accounting policy. The way to solve this
problem is subrequest
.
When the user is authenticated and the system determines it needs to
start an accounting session for the user, the unlang
policies can
take the following steps:
The result is a policy which cannot ever go wrong. The fake "start accounting" packet is run through exactly the same policies as a normal accounting packet. There is no way for the real and fake policies to get out of synchronization, as there is no difference between the two of them. There is only one policy for accounting packets: the real one.
Similar rules can be applied when the server receives
Accounting-Request
packets. If the policy decides that the user has
used "too much" data, the server can create a Disconnect-Request
packet, and disconnect the users session.
Or, the server can receive a DHCPv4 request, and then send a RADIUS
packet to an external RADIUS server for MAC address authentication.
The combination of subrequest
and call means
that the server can "mix and match" protocols in a way which is
difficult or impossible to do with other protocol servers.
Subrequests are Synchronous
Execution of the parent request is paused while the child request is running. The parent request continues execution once the child request has finished.
In some cases, it is useful to let the child request continue execution
independently of the parent request. In the example given above,
sending a Disconnect-Request
packet may result in a multiple-second
delay the NAS does not respond quickly, due to retransmits.
A child request can be made asynchronous by using the detach keyword. Once a child request is detached from the parent, the parent can continue execution independently of the child. However, once a child request is detached from the parent request, the child can no longer access any attributes in the parent.
subrequest ::Disconnect-Request {
&User-Name := &parent.request.User-Name
&NAS-IP-Address := &parent.request.NAS-IP-Address
&NAS-Port := &parent.request.NAS-Port
&Acct-Session-Id := &parent.request.Acct-Session-Id
detach
radius
}