magic.endpoint.services
15.8.2
See the version list below for details.
dotnet add package magic.endpoint.services --version 15.8.2
NuGet\Install-Package magic.endpoint.services -Version 15.8.2
<PackageReference Include="magic.endpoint.services" Version="15.8.2" />
paket add magic.endpoint.services --version 15.8.2
#r "nuget: magic.endpoint.services, 15.8.2"
// Install magic.endpoint.services as a Cake Addin #addin nuget:?package=magic.endpoint.services&version=15.8.2 // Install magic.endpoint.services as a Cake Tool #tool nuget:?package=magic.endpoint.services&version=15.8.2
magic.endpoint - How Hyperlambda endpoints are resolved
magic.endpoint is a dynamic endpoint URL controller, allowing you to declare endpoints that are dynamically
resolved using your IHttpExecutorAsync
service implementation. The default implementation of this interface,
is the class called HttpExecutorAsync
, and the rest of this file will be focused on documenting this
implementation, since it is the default service implementation for magic.endpoint - Although, technically, you
could exchange this with your own implementation if you wish, completely changing the behaviour of the library
if you wish to for instance resolve endpoints to Python, Ruby, or any other dynamic programming language
implementation, and you have some means to execute such code from within a .Net 6 environment.
The resolver will be invoked for all relative URLs starting with "magic/", for the following verbs.
GET
POST
PUT
DELETE
PATCH
The default service implementation will resolve everything after the "magic/" parts in the given URL, to a Hyperlambda file assumed to be found relatively beneath your "/files/" folder - Although, exactly where you physically put your files on disc, can be configured through your "appsettings.json" file. The HTTP verb is assumed to be the last parts of your filename, before its extension, implying an HTTP request such as the following.
GET magic/modules/foo/bar
Will resolve to the following physical file on disc.
files/modules/foo/bar.get.hl
Only the "magic" part of your URL is rewritten before the verb is appended to the URL, and finally the extension ".hl" appended. Then the file is loaded and parsed as Hyperlambda, and whatever arguments you pass in, either as query parameters or as your JSON payload is appended into your resulting lambda node's [.arguments] node as arguments to your Hyperlambda file invocation. The resolver will never return files directly, but is only able to execute Hyperlambda files, so by default there is no way to get static files, unless you create a Hyperlambda endpoint that returns a static file somehow.
The default resolver will only allow the client to resolve files inside your "/files/modules/"
folder and "/files/system/" folder. This allows you to safely keep files that parts of your system
relies upon inside your dynamic "/files/" folder, without accidentally creating endpoints clients can
resolve, resulting in breaches in your security. Only characters a-z, 0-9 and '-', '_' and '/' are legal
characters for the resolvers, and only lowercase characters to avoid file system incompatibilities between
Linux and Windows. There is one exception to this rule though, which is that the resolver will resolve
files and folder starting out with a period (.) character, since this is necessary to allow for having
"hidden files" being resolved as endpoints - Which is a requirement to make things such as
Apple's ".well-known" endpoints being resolved.
Below is probably the simplest HTTP endpoint you could create. Save the following Hyperlambda in a
file at the path of /modules/tutorials/foo.get.hl
using for instance your Magic
"Hyper IDE" menu item.
return
result:Hello from Magic Backend
Then invoke the endpoint using the GET verb with the following URL.
http://localhost:5000/magic/modules/tutorials/foo
Hyperlambda endpoints and arguments
The default IHttpExecutorAsync
implementation can explicitly declare what arguments the file can
legally accept, and if an argument is given during invocation that the file doesn't allow for, an
exception will be thrown and the file will never be executed. This allows you to declare what
arguments your Hyperlambda file can accept, and avoid having anything but arguments explicitly
declared in your Hyperlambda file from being sent into your endpoint during invocation of your
HTTP endpoint. An example Hyperlambda file taking two arguments can be found below.
.arguments
arg1:string
arg2:int
strings.concat
get-value:x:@.arguments/*/arg1
.:" - "
get-value:x:@.arguments/*/arg2
unwrap:x:+/*
return
result:x:@strings.concat
If you save this file on disc as /files/modules/tutorials/foo2.get.hl
, you can invoke it as follows
using the HTTP GET verb - Assuming your backend is running on localhost at port 5000.
http://localhost:5000/magic/modules/tutorials/foo2?arg1=howdy&arg2=5
JSON payloads and form URL encoded payloads are automatically converted to lambda/nodes - And query parameters are treated indiscriminately the same way as JSON payloads - Except of course, query parameters cannot pass in complex graph objects, but only simply key/value arguments. Only POST, PUT and PATCH endpoints can handle payloads. If you supply a payload to a GET or DELETE endpoint, an exception will be thrown, and an error returned to the caller.
To allow for any arguments to your files, simply ommit the [.arguments] node
in your Hyperlambda althogether, or supply an [.arguments] node and set its value to *
.
Alternatively, you can also partially ignore arguments sanity checking of individual nodes,
by setting their values to *
, such as the following illustrates.
.arguments
arg1:string
arg2:date
arg3:*
In the above arguments declaration, [arg1] and [arg2] will be sanity checked, and input converted
to string
or date
(DateTime) - But the [arg3] parts will be completely ignored, allowing the caller
to invoke it with anything as arg3
during invocation - Including complete graph JSON objects, assuming
the above declaration is for a PUT
, POST
or PATCH
Hyperlambda file. The '*' value for an argument also turn
off all conversion, implying everything will be given to your lambda object with the JSON type the argument
was passed in as.
All arguments declared are considered optional, and the file will still resolve if the argument is not given,
except of course the argument won't exist in the [.arguments] node. However, no argument not found
in your [.arguments] declaration can be provided during invocations, assuming you choose to declare
an [.arguments] collection in your Hyperlambda endpoint file, and you don't set its value to *
.
To declare what type your arguments can be, set the value of the argument declaration node to the Hyperlambda type value inside of your arguments declaration, such as illustrated above. Arguments will be converted if possible, to the type declaration in your argument's declaration. If no conversion is possible, an exception will be thrown. Although the sanity check will check graph objects, passed in as JSON payloads, it has its restrictions, such as not being able to sanity check complex objects passed in as arrays, etc. If you need stronger sanity checking of your arguments, you will have to manually check your more complex graph objects yourself in your own Hyperlambda files.
Also realise that if the value originates from a payload, as in from a PUT, PATCH or POST JSON object
for instance, these types of objects might contain null values. If they do, no conversion will be attempted,
and internally within your endpoint's Hyperlambda code, you might therefor expect to see for instance
long
values being in fact null, even though technically these are not nullable types in .Net.
Accepted Content-Type values for Hyperlambda endpoints
The POST, PUT and PATCH endpoints can intelligently handle any of the following Content-Types.
application/json
application/x-json
application/www-form-urlencoded
application/x-www-form-urlencoded
multipart/form-data
JSON types of payloads are fairly well described above, and URL encoded form payloads are handled the exact same way, except of course the [.arguments] node is built from URL encoded values instead of JSON - However, internally this is transparent for you, and JSON, query parameters, URL encoded forms, and "multipart/form-data" can be interchanged 100% transparently from your code's perspective - Except "multipart/form-data" might have [file] arguments wrapping streams that you need to handle separately as such. File attachments will be passed into your endpoint as follows.
.arguments
file
name:filename-on-client.txt
stream:[raw Stream object here]
All other types of payloads will be passed in as the raw stream, not trying to read from it in any ways, allowing you to intercept reading with things such as authentication, authorisation, logic of where to persist content, etc. To understand how you can handle these streams, check out the "magic.lambda.io" project's documentation, and specifically the [io.stream.xxx] slots.
Extending the Hyperlambda Content-Type request and response resolver
The Content-Type resolver/parser is extendible, allowing you to change its behaviour by providing your own callback that will be invoked for some specific Content-Type value provided. This is useful if you want to be able to for instance handle "text/xml" or "text/csv" types of request/response objects, and intelligently and automatically create an argument collection from it. Below is example code illustrating how to create your own HTTP request resolver for the MIME type of "application/x-foo".
EndpointController.RegisterContentType("application/x-foo", async (signaler, request) =>
{
var args = new Node();
/* ... Create some sort of collection of arguments and put into args node here ... */
return args;
});
Notice - The argument sanity checking will still be invoked with a custom handler, implying your Content-Type handler and the [.arguments] declaration in your Hyperlambda file still needs to agree upon the arguments, and if a non-valid argument is specified to a Hyperlambda file, an exception will be thrown. Also notice that registering a custom Content-Type is not thread safe, and should be done as you start your application, and not during its life time. You can also provide your own HTTP response resolver that will be invoked given some specified Content-Type from your Hyperlambda file. This is done in a similar manner using something resembling the following.
EndpointController.RegisterContentType("application/x-foo", (response) =>
{
/* ... Return some sort of IActionResult here ... */
return new ObjectResult(response.Content) { StatusCode = response.Result };
});
The above method should also exclusively be used during startup, and not later, since it is not thread safe. The above method assumes you register your Content-Type handlers as your application is starting.
Hyperlambda endpoints and meta information
Due to the semantic structure of Hyperlambda, retrieving meta information from your HTTP endpoints using this module is very easy. The project has one slot called [endpoints.list] that returns meta information about all your endpoints. This slot again can be invoked using the following URL.
http://localhost:5000/magic/system/endpoints/list
This endpoint/slot will semantically traverse your endpoints, recursively loading up all Hyperlambda files from disc that are resolved from a valid URL, and return meta information about the file/endpoint back to the caller. This allows the system to easily figure out things such as the following about your endpoints.
- What is the endpoint's HTTP VERB
- What is the endpoint's URL
- What arguments can the endpoint handle
- Has the file been given a friendly description, through a [.description] node
- Etc ...
This slot/endpoint is what allows you to see meta information about all your HTTP REST endpoints in the "Endpoints" menu item in the Magic dashboard for instance. The return value from this slot/endpoint again, is what's used as some sort of frontend is being generated using the Magic dashboard.
Extending the meta data retrieval process
You can extend the meta data retrieval process by
invoking ListEndpoints.AddMetaDataResolver
, and pass in your own function. This class can be
found in the magic.endpoint.services.slots
namespace.
The AddMetaDataResolver
method takes one function object, which will be invoked for every file
the meta generator is trying to create meta data for, with the complete lambda
, verb
and args
of your endpoint. This allows you to semantically traverse the lambda/args nodes, and append
any amount of (additional) meta information you wish - Allowing you to extend the generating
of meta data, if you have some sort of general custom Hyperlambda module, creating custom
HTTP endpoints of some sort.
This function will be invoked for every single Hyperlambda file in your system, every time meta data is retrieved, so you might want to ensure it executes in a fairly short amount of time, not clogging the server or HTTP endpoint meta generating process in any ways.
Slots related to endpoints and the HTTP context
In addition to the meta retrieval endpoint described above, the module contains the following slots.
- [server.ip] - Returns the IP address of the server itself
- [response.status.set] - Sets the status code (e.g. 404) on the response object
- [request.cookies.list] - Lists all HTTP request cookies
- [request.cookies.get] - Returns the value of a cookie sent by the request
- [response.cookies.set] - Creates a cookie that will be returned to the client over the response
- [request.headers.list] - Lists all HTTP request headers sent by the request
- [request.headers.get] - Returns a single HTTP header associated with the request
- [request.ip] - Returns the IP address of the HTTP request
- [request.url] - Returns the relative URL associated with the request, without its magic/ prefix
- [request.host] - Returns the host name associated with the request
- [request.scheme] - Returns the scheme associated with the request
- [response.headers.set] - Adds an HTTP header to the response object
- [mime.add] - Associates a file extension with a MIME type
Changing your Hyperlambda endpoint's response type
Unless you explicitly change the Content-Type
of your response object, by using
the [response.headers.set] slot, a Content-Type of application/json
will be assumed,
and this header will be added to the resulting HTTP response object. If you wish to override
this behavious and return plain text for instance, you could create an endpoint containing
the following.
response.headers.set
Content-Type:text/plain
return:Hello from Magic Backend
If you intend to return anything but JSON, you must set the Content-Type
header, because
the resolver will by default try to serialize your content as JSON, and obviously fail unless it is
valid JSON.
You can also return stream objects using for instance the [return-value] slot, at which point
ASP.NET Core will automatically stream your content back over the response object, and Dispose
your stream automatically for you afterwards. This allows you to for instance return large files back
to the client without loading them into memory first. If you do this, you'll have to change
your Content-Type
accordingly.
Hyperlambda and cookies
Since cookies have more parameters than just a simple key/value declaration, the [response.cookies.set] slot takes the following arguments.
- [value] - The string content of your cookie
- [expires] - Absolute expiration date of your cookie, as a Hyperlambda
date
value - [http-only] - Boolean value declaring whether or not the cookie should only be accessible on the server
- [secure] - Boolean value declaring whether or not cookie should only be transmitted from the client to the server over a secure (https) connection
- [domain] - Domain value of your cookie
- [path] - Path value of your cookie
- [same-site] - Same-site value of your cookie
Only the [value] from above is mandatory. To delete a cookie on the client, set the expiration date to a value in the past.
How to use [mime.add]
This slots associates a file extension with a MIME type. Notice, it will override previous associations if existing. Example usage can be found below.
mime.add:py
.:application/python
Then later when the endpoint resolver is returning files ending with ".py", it will return these with
a Content-Type
of "application/python".
Hyperlambda code behind files
The resolver will resolve anything not starting out with /magic/
as a static file, optionally applied
as a mixin file having a Hyperlambda code behind file for mixing in dynamic content with any ".html"
files. This allows you to render HTML, CSS, JavaScript and "whatever", with the ability to dynamically
render parts of your HTML files using Hyperlambda. This logic relies upon the [io.file.mixin] slot
from the "magic.lambda.io" project. If you create two files such as follows and put both of these
files in your "/etc/www/" folder, you can see this logic in action.
index.html
<html>
<head>
<title>Hell world</title>
</head>
<body>
<h1>Hello world</h1>
<p>
Hello there Thomas Hansen, {{*/.calculate}}
</p>
</body>
</html>
index.hl
.calculate
math.add
.:int:2
.:int:2
return:x:-
The above will substitute your {{*/.calculate}}
parts with the result of invoking your [.calculate] lambda
object, resulting in 4. To understand how this works, you need to read about the [io.file.mixin] slot in
the "magic.lambda.io" project, and realise that the above will actually transform to the following as the
mixin logic is executed.
io.file.mixin:/etc/www/index.html
.calculate
math.add
.:int:2
.:int:2
return:x:-
This allows you to serve dynamically rendered HTML files, where parts of your HTML is substituted with the result of invoking some lambda object. If you have an HTML file without a Hyperlambda code behind file, it will be served as a static file. CSS files, JavaScript files, and images will also be served as static files.
Notice, interceptor files will be executed as normally, allowing you to apply interceptor files similarly to how you apply these with your "/magic/" endpoints. In addition, any file called "default.html" having a Hyperlambda counterpart will be used for default URL resolving if no explicit URL is found, allowing you to handle dynamica URLs with this file.
Project website for magic.endpoint
The source code for this repository can be found at github.com/polterguy/magic.endpoint, and you can provide feedback, provide bug reports, etc at the same place.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- magic.endpoint.contracts (>= 15.8.2)
- magic.node.extensions (>= 15.8.2)
- magic.signals.contracts (>= 15.8.2)
NuGet packages (1)
Showing the top 1 NuGet packages that depend on magic.endpoint.services:
Package | Downloads |
---|---|
magic.library
Helper project for Magic to wire up everything easily by simply adding one package, and invoking two simple methods. When using Magic, this is (probably) the only package you should actually add, since this package pulls in everything else you'll need automatically, and wires up everything sanely by default. To use package go to https://polterguy.github.io |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
17.2.0 | 448 | 1/22/2024 |
17.1.7 | 404 | 1/12/2024 |
17.1.6 | 373 | 1/11/2024 |
17.1.5 | 404 | 1/5/2024 |
17.0.1 | 447 | 1/1/2024 |
17.0.0 | 613 | 12/14/2023 |
16.11.6 | 503 | 11/14/2023 |
16.11.5 | 462 | 11/12/2023 |
16.9.0 | 609 | 10/9/2023 |
16.7.0 | 966 | 7/11/2023 |
16.4.1 | 834 | 7/2/2023 |
16.4.0 | 845 | 6/22/2023 |
16.3.1 | 763 | 6/7/2023 |
16.3.0 | 716 | 5/28/2023 |
16.1.10 | 977 | 5/1/2023 |
16.1.9 | 692 | 4/30/2023 |
15.10.1 | 840 | 4/13/2023 |
15.9.40 | 822 | 4/3/2023 |
15.9.1 | 860 | 3/26/2023 |
15.9.0 | 846 | 3/24/2023 |
15.8.2 | 896 | 3/20/2023 |
15.7.0 | 786 | 3/6/2023 |
15.5.0 | 1,958 | 1/28/2023 |
15.2.0 | 1,085 | 1/18/2023 |
15.1.0 | 1,588 | 12/28/2022 |
14.5.7 | 1,127 | 12/13/2022 |
14.5.5 | 1,179 | 12/6/2022 |
14.5.1 | 1,094 | 11/23/2022 |
14.5.0 | 1,020 | 11/18/2022 |
14.4.5 | 1,121 | 10/22/2022 |
14.4.1 | 1,166 | 10/22/2022 |
14.4.0 | 1,067 | 10/17/2022 |
14.3.6 | 1,277 | 10/7/2022 |
14.3.1 | 1,325 | 9/12/2022 |
14.3.0 | 1,056 | 9/10/2022 |
14.1.3 | 1,282 | 8/7/2022 |
14.1.2 | 1,040 | 8/7/2022 |
14.1.1 | 1,055 | 8/7/2022 |
14.0.14 | 1,083 | 7/26/2022 |
14.0.12 | 1,025 | 7/24/2022 |
14.0.11 | 1,042 | 7/23/2022 |
14.0.10 | 1,046 | 7/23/2022 |
14.0.9 | 1,058 | 7/23/2022 |
14.0.8 | 1,139 | 7/17/2022 |
14.0.5 | 1,182 | 7/11/2022 |
14.0.4 | 1,168 | 7/6/2022 |
14.0.3 | 1,112 | 7/2/2022 |
14.0.2 | 1,098 | 7/2/2022 |
14.0.0 | 1,290 | 6/25/2022 |
13.4.2 | 2,215 | 6/4/2022 |
13.4.0 | 1,153 | 5/31/2022 |
13.3.6 | 1,507 | 5/14/2022 |
13.3.4 | 1,190 | 5/9/2022 |
13.3.0 | 1,372 | 5/1/2022 |
13.2.0 | 1,580 | 4/21/2022 |
13.1.0 | 1,410 | 4/7/2022 |
13.0.0 | 1,116 | 4/5/2022 |
11.0.5 | 1,841 | 3/2/2022 |
11.0.4 | 1,188 | 2/22/2022 |
11.0.3 | 1,180 | 2/6/2022 |
11.0.2 | 1,190 | 2/6/2022 |
11.0.1 | 895 | 2/5/2022 |
11.0.0 | 1,183 | 2/5/2022 |
10.0.21 | 1,165 | 1/28/2022 |
10.0.20 | 1,160 | 1/27/2022 |
10.0.19 | 1,181 | 1/23/2022 |
10.0.18 | 1,113 | 1/17/2022 |
10.0.15 | 1,325 | 12/31/2021 |
10.0.14 | 952 | 12/28/2021 |
10.0.7 | 1,847 | 12/22/2021 |
10.0.6 | 936 | 12/19/2021 |
10.0.5 | 987 | 12/18/2021 |
9.9.9 | 1,509 | 11/29/2021 |
9.9.5 | 1,058 | 11/23/2021 |
9.9.3 | 1,085 | 11/9/2021 |
9.9.2 | 1,028 | 11/4/2021 |
9.9.0 | 1,123 | 10/30/2021 |
9.8.9 | 1,105 | 10/29/2021 |
9.8.7 | 1,052 | 10/27/2021 |
9.8.6 | 1,088 | 10/27/2021 |
9.8.5 | 1,066 | 10/26/2021 |
9.8.0 | 1,772 | 10/20/2021 |
9.7.9 | 990 | 10/19/2021 |
9.7.5 | 1,877 | 10/14/2021 |
9.7.2 | 787 | 10/14/2021 |
9.7.1 | 944 | 10/14/2021 |
9.7.0 | 1,080 | 10/9/2021 |
9.6.7 | 1,277 | 8/26/2021 |
9.6.6 | 1,145 | 8/14/2021 |
9.6.4 | 1,155 | 8/11/2021 |
9.6.3 | 1,011 | 8/10/2021 |
9.6.2 | 1,282 | 8/8/2021 |
9.6.1 | 991 | 8/8/2021 |
9.5.0 | 2,141 | 7/9/2021 |
9.2.4 | 3,245 | 6/6/2021 |
9.2.1 | 1,595 | 6/1/2021 |
9.2.0 | 1,158 | 5/26/2021 |
9.1.9 | 1,020 | 5/5/2021 |
9.1.8 | 1,119 | 5/5/2021 |
9.1.7 | 1,042 | 5/3/2021 |
9.1.4 | 1,010 | 4/21/2021 |
9.1.0 | 1,386 | 4/14/2021 |
9.0.0 | 1,284 | 4/5/2021 |
8.9.9 | 1,514 | 3/30/2021 |
8.9.3 | 2,029 | 3/19/2021 |
8.9.2 | 1,548 | 1/29/2021 |
8.9.1 | 1,556 | 1/24/2021 |
8.9.0 | 1,702 | 1/22/2021 |
8.6.9 | 3,517 | 11/8/2020 |
8.6.6 | 2,436 | 11/2/2020 |
8.6.1 | 3,799 | 10/29/2020 |
8.6.0 | 1,660 | 10/28/2020 |
8.5.0 | 2,393 | 10/23/2020 |
8.4.3 | 3,293 | 10/17/2020 |
8.4.2 | 1,792 | 10/16/2020 |
8.4.1 | 2,447 | 10/15/2020 |
8.4.0 | 1,778 | 10/13/2020 |
8.3.1 | 3,125 | 10/5/2020 |
8.3.0 | 1,774 | 10/3/2020 |
8.2.3 | 1,720 | 10/1/2020 |
8.2.2 | 1,885 | 9/26/2020 |
8.2.1 | 1,843 | 9/25/2020 |
8.2.0 | 1,789 | 9/25/2020 |
8.1.19 | 3,747 | 9/21/2020 |
8.1.18 | 2,551 | 9/14/2020 |
8.1.17 | 3,172 | 9/13/2020 |
8.1.16 | 1,804 | 9/13/2020 |
8.1.15 | 1,760 | 9/12/2020 |
8.1.11 | 3,007 | 9/11/2020 |
8.1.10 | 1,829 | 9/6/2020 |
8.1.9 | 1,779 | 9/3/2020 |
8.1.8 | 1,808 | 9/2/2020 |
8.1.7 | 1,643 | 8/28/2020 |
8.1.4 | 1,708 | 8/25/2020 |
8.1.3 | 1,844 | 8/18/2020 |
8.1.2 | 1,737 | 8/16/2020 |
8.1.1 | 1,813 | 8/15/2020 |
8.1.0 | 1,153 | 8/15/2020 |
8.0.1 | 3,174 | 8/7/2020 |
8.0.0 | 1,668 | 8/7/2020 |
7.0.1 | 1,894 | 6/28/2020 |
7.0.0 | 1,754 | 6/28/2020 |
5.0.4 | 2,584 | 5/30/2020 |
5.0.3 | 3,897 | 2/26/2020 |
5.0.2 | 1,867 | 2/26/2020 |
5.0.1 | 1,848 | 2/25/2020 |
5.0.0 | 2,448 | 2/25/2020 |
4.1.2 | 1,796 | 2/23/2020 |
4.1.1 | 2,539 | 2/22/2020 |
4.1.0 | 1,792 | 2/22/2020 |
4.0.9 | 2,396 | 2/21/2020 |
4.0.8 | 1,868 | 2/7/2020 |
4.0.7 | 1,731 | 2/7/2020 |
4.0.5 | 1,799 | 2/7/2020 |
4.0.4 | 2,568 | 1/27/2020 |
4.0.3 | 1,817 | 1/27/2020 |
4.0.2 | 1,912 | 1/16/2020 |
4.0.1 | 1,960 | 1/11/2020 |
4.0.0 | 1,881 | 1/5/2020 |
3.1.1 | 1,894 | 12/17/2019 |
3.1.0 | 5,991 | 11/10/2019 |
3.0.0 | 4,300 | 10/23/2019 |
2.0.4 | 3,678 | 10/19/2019 |
2.0.3 | 5,042 | 10/17/2019 |
2.0.2 | 1,740 | 10/15/2019 |
2.0.1 | 2,300 | 10/14/2019 |
2.0.0 | 1,540 | 10/13/2019 |
1.2.1 | 1,903 | 10/11/2019 |
1.1.9 | 2,282 | 10/10/2019 |
1.1.8 | 1,631 | 10/10/2019 |
1.1.7 | 1,678 | 10/9/2019 |
1.1.6 | 1,588 | 10/7/2019 |
1.1.5 | 1,680 | 10/6/2019 |
1.1.4 | 1,276 | 10/6/2019 |