Miscellaneous¶
Of course, there’s a lot more to Cripts than URLs, headers and connections. This chapter in the Cripts documentation covers a variety of topics that don’t fit into the their own chapter.
Errors¶
Often it’s useful to be able to abort a client transaction prematurely, and return an error to the client. Cripts provid making this easy.
Note
Explicitly forcing an HTTP error overrides any other response status that may have been set.
Function |
Description |
---|---|
|
Sets the response to the status code, and force the request to error. |
|
Get the current response status for the request. |
|
Sets an explicit reason message with the status code. TBD |
Example:
do_remap()
{
borrow req = cripts::Client::Request::get();
if (req["X-Header"] == "yes") {
cripts::Error::Status::Set(403);
}
// Do more stuff here
if (cripts::Error::status::Get() != 403) {
// Do even more stuff here if we're not in error state
}
}
ATS transactions are generally hidden within Cripts, but for power users, the
transaction
object provides access to the underlying transaction. In this object,
the following functions are available:
Function |
Description |
---|---|
|
Disables a future callback in this Cript, for this transaction. |
|
Has the transaction been aborted. |
|
Returns the cache lookup status for the transaction. |
When disabling a callback, use the following names:
Callback |
Description |
---|---|
|
The |
|
The |
|
The |
|
The |
|
The |
|
The |
|
The |
Finally, the transaction
object also provides access to these ATS objects, which can be used
with the lower level ATS APIs:
Object |
Description |
---|---|
|
The TSHttpTxn pointer. |
|
TSHttpSsn pointer. |
The transaction
object is a global available everywhere, just like the instance
object.
Example usage to turn off a particular hook conditionally:
do_remap()
{
static borrow req = cripts::Client::Request::get();
if (req["X-Header"] == "yes") {
transaction.DisableCallback(cripts::Callback::DO_READ_RESPONSE);
}
}
Note
Disabling callbacks like this is an optimization, avoiding for the hook to be called at all. It can be particularly useful when the decision to run the hook is made early in the Cript.
Time¶
Cripts has encapsulated some common time-related functions in the core. At the
moment only the localtime is available, via the cripts::Time::Local
object and its
Now()
method. The Now()
method returns the current time as an object
with the following functions:
Function |
Description |
---|---|
|
Returns the number of seconds since the Unix epoch (00:00:00 UTC, January 1, 1970). |
|
Returns the year. |
|
Returns the month (1-12). |
|
Returns the day of the month (1-31). |
|
Returns the hour (0-23). |
|
Returns the minute (0-59). |
|
Returns the second (0-59). |
|
Returns the day of the week (0-6, Sunday is 0). |
|
Returns the day of the year (0-365). |
The time as returned by Now()
can also be used directly in comparisons with previous or future
times.
Plugins¶
While Cripts provides a lot of functionality out of the box, there are times when you want to continue using existing remap plugins conditionally. Cripts allows you to load and run existing remap plugins from within your Cripts. This opens up new possibilities for your existing plugins, as you gain the full power of Cript to decide when to run such plugins.
Setting up existing remap plugins must be done in the do_create_instance()
hook. The instantiated remap plugins must be added to the instance object for the
Cript, using the addPlugin()
method. Here’s an example to run the rate limiting
plugin based on the client request headers:
do_create_instance()
{
instance.AddPlugin("my_ratelimit", "rate_limit.so", {"--limit=300", "--error=429"});
}
do_remap()
{
static borrow plugin = instance.plugins["my_ratelimit"];
borrow req = cripts::Client::Request::Get();
if (req["X-Header"] == "yes") {
plugin.RunRemap();
}
}
Note
The name of the plugin instance, as specified to AddPlugin()
, must be
unique across all Cripts.
Files¶
In same cases, albeit not likely, you may need to read lines of text from A file. Cripts of course allows this to be done with C or C++ standard file APIs, but we also provide a few convenience functions to make this easier.
The cripts::File
object encapsulates the common C++ files operations. For convenience,
and being such a common use case, reading a single line from a file is provided
by the cripts::File::Line::Reader
object. Some examples:
do_remap()
{
static const cripts::File::Path p1("/tmp/foo");
static const cripts::File::Path p2("/tmp/secret.txt");
if (cripts::File::Status(p1).Type() == cripts::File::Type::regular) {
resp["X-Foo-Exists"] = "yes";
} else {
resp["X-Foo-Exists"] = "no";
}
string secret = cripts::File::Line::Reader(p2);
CDebug("Read secret = {}", secret);
}
UUID¶
Cripts supports generating a few different UUID (Universally Unique Identifier), for
different purposes. The UUID
class provides the following objects:
Object |
Description |
---|---|
|
Returns a UUID for the running process (changes on ATS startup). |
|
Returns a completely unique UUID for the server and transacion. |
|
Returns a unique id for this request. |
Using the UUID
object is simple, via the Get()
method. Here’s an example:
do_remap()
{
static borrow req = cripts::Client::Request::Get();
resp["X-UUID"] = cripts::UUID::Unique::Get();
}
Metrics¶
Cripts metrics are built directly on top of the atomic core ATS metrics. As such, they not only work the same as the core metrics, but they are also as efficient. There are two types of metrics:
Metric |
Description |
---|---|
|
A simple counter, which can only be incremented. |
|
A gauge, which can be incremented and decremented, and set to a value. |
Example:
do_create_instance()
{
instance.metrics[0] = cripts::Metrics::Counter::Create("cript.example1.instance_calls");
}
do_remap()
{
static auto plugin_metric = cripts::Metrics::Counter("cript.example1.plugin_calls");
plugin_metric.Increment();
instance.metrics[0]->Increment();
}
A cripts::Metric::Gauge
can also be set via the Setter()
method.