From 820aae2f178b23d1ac1b5562cf5b527f6a6f7325 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Mon, 22 Jul 2024 09:21:16 +0100 Subject: [PATCH 01/61] add month infinity --- docs/basics/datatypes.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/basics/datatypes.md b/docs/basics/datatypes.md index 864331017..16b293894 100644 --- a/docs/basics/datatypes.md +++ b/docs/basics/datatypes.md @@ -25,7 +25,7 @@ n c name sz literal null inf SQL Java .Net 10 c char 1 " " " " Character char 11 s symbol \` \` varchar 12 p timestamp 8 dateDtimespan 0Np 0Wp Timestamp DateTime (RW) -13 m month 4 2000.01m 0Nm +13 m month 4 2000.01m 0Nm 0Wm 14 d date 4 2000.01.01 0Nd 0Wd date Date 15 z datetime 8 dateTtime 0Nz 0wz timestamp Timestamp DateTime (RO) 16 n timespan 8 00:00:00.000000000 0Nn 0Wn Timespan TimeSpan From 6e8fd03bd552ac5daa912060d36ea48d359a615b Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Mon, 22 Jul 2024 10:46:00 +0100 Subject: [PATCH 02/61] amend at examples --- docs/ref/amend.md | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/docs/ref/amend.md b/docs/ref/amend.md index b869de3b6..ea2605faa 100644 --- a/docs/ref/amend.md +++ b/docs/ref/amend.md @@ -98,6 +98,25 @@ q).[(5 2.14; "abc"); 1 2; :; "x"] / replace at depth 2 "abx" ``` +### Amend At + +Indicies results are accumulated when repeated: + +```q +q)@[(0 1 2;1 2 3 4;7 8 9) ;1 1; 2*] +0 1 2 +4 8 12 16 / equates to 2*2*1 2 3 4 +7 8 9 +q)@[(0 1 2;1 2 3 4;7 8 9) ;0 1 2 1; 100*] +0 100 200 / equates to 100*0 1 2 +10000 20000 30000 40000 / equates to 100*100*1 2 3 4 +700 800 900 / equates to 100*7 8 9 +q)@[(0 1 2;1 2 3 4;7 8 9) ;0 1 2 1; {x*y};100] +0 100 200 / equates to {x*100}0 1 2 +10000 20000 30000 40000 / equates to {x*100}{x*100}1 2 3 4 +700 800 900 / equates to {x*100}7 8 9 +``` + ### Cross sections From 5a80cb124d53ab8edde3ad7f91df95e1168ef3ed Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Mon, 22 Jul 2024 12:25:41 +0100 Subject: [PATCH 03/61] improved details of .Q.hdpf --- docs/ref/dotq.md | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/docs/ref/dotq.md b/docs/ref/dotq.md index 02706b815..3570a49fb 100644 --- a/docs/ref/dotq.md +++ b/docs/ref/dotq.md @@ -992,7 +992,11 @@ q).Q.gz{0N!count x;x}[.Q.gz(9;10000#"helloworld")] .Q.hdpf[historicalport;directory;partition;`p#field] ``` -Saves all tables by calling `.Q.dpft`, clears tables, and sends reload message to HDB. +The function: + +* saves all tables to disk, by calling [`.Q.dpft`](#dpft-save-table) (saves as splayed tables to a partition) +* clears in-memory tables +* sends reload message to HDB, by opening a temporary connection and sending [`\l .`](../basics/syscmds.md#l-load-file-or-directory) ## `hg` (HTTP get) From eaa0a89903e45da92e5865edd19c9e10dd76693c Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 24 Jul 2024 12:37:16 +0100 Subject: [PATCH 04/61] add link for deferred sync --- docs/kb/load-balancing.md | 2 +- docs/wp/ipc/img/tcp-diagram.png | Bin 15044 -> 0 bytes docs/wp/ipc/index.md | 834 -------------------------------- 3 files changed, 1 insertion(+), 835 deletions(-) delete mode 100644 docs/wp/ipc/img/tcp-diagram.png delete mode 100644 docs/wp/ipc/index.md diff --git a/docs/kb/load-balancing.md b/docs/kb/load-balancing.md index b3778cde3..15ecf37f7 100644 --- a/docs/kb/load-balancing.md +++ b/docs/kb/load-balancing.md @@ -36,7 +36,7 @@ q)h "xs" 0 1 2 3 4 5 6 7 8 ``` -Asynchronous messages are forwarded to one of the secondary servers, transparently to the client. The code below issues an asynchronous request, then [blocks on the handle](../basics/ipc.md#async-blocking) waiting for a result to be returned. This is called _deferred synchronous_. +Asynchronous messages are forwarded to one of the secondary servers, transparently to the client. The code below issues an asynchronous request, then [blocks on the handle](../basics/ipc.md#async-blocking) waiting for a result to be returned. This is called [_deferred synchronous_](../basics/ipc.md#deferred-sync). ```q q)(neg h) "select sym,price from trade where size > 50000" ; h[] diff --git a/docs/wp/ipc/img/tcp-diagram.png b/docs/wp/ipc/img/tcp-diagram.png deleted file mode 100644 index 11fd268f542b41a6e3e6b8963229181df099d3dd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15044 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 diff --git a/docs/wp/ipc/index.md b/docs/wp/ipc/index.md deleted file mode 100644 index 97b6e9085..000000000 --- a/docs/wp/ipc/index.md +++ /dev/null @@ -1,834 +0,0 @@ ---- -title: Interprocess communications | Interfaces | q and kdb+ documentation -description: Core concepts related to IPC in kdb+, and the role of IPC in a kdb+ tick application -author: Katrina McCormack -date: April 2021 ---- -# Interprocess communications - -by [Katrina McCormack](#author) -{: .wp-author} - - -!!! summary - - A look at the mechanisms underlying interprocess communication in kdb+ begins with an overview of TCP connections and discusses applications for using Unix domain sockets across processes on a local host, and the use of TLS/SSL for more secure communication. It describes the main message handlers invoked in the process of opening a connection, running a query and closing a connection to a process, with various examples to illustrate each step. - - It then takes a closer look at how IPC is applied in a vanilla tickerplant application, from when the data is first received from an upstream feed process to how tables are saved at the end of the day. - - - -Microsoft defines Interprocess Communications (IPC) as the collective term for mechanisms facilitating communications and data sharing between applications. This paper explores some core concepts related to IPC in kdb+ before investigating the role of IPC in a kdb+ tick application. - - -## Core concepts - -### Set port - -A port is used by TCP as a communications endpoint: in other words, a point through which information flows to and from a process. - -A kdb+ process can be set to listen on a port in two ways. For this example, we will listen on port 1234 using either - -- `\p 1234` or `system"p 1234"` within the process -- `-p 1234` on the command line at process startup - -To stop listening on a port, set the server to listen on port 0: - -```q -q)\p -4567i -q).z.i -2493i -q)\p 5678 -q)\p -5678i -q)system"p 6789" -q)\p -6789i -q)\p 0 -``` - -It is possible to see this listening port using `lsof` (list all open files) to view TCP connections for this PID. If the port number is changed, using `lsof` will show the process listening on the new port only. - -```q -q).z.i -97082i -q)\p 4567 -q)\p 6789 -``` - -```bash -$ lsof -p 97082 -i tcp -a -$ lsof -p 97082 -i tcp -a -COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME -q 97082 katrina 4u IPv4 2140799 0t0 TCP *:4567 (LISTEN) - -$ lsof -p 97082 -i tcp -a -COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME -q 97082 katrina 4u IPv4 2149383 0t0 TCP *:6789 (LISTEN) -``` - -Port numbers between 0 and 1024 are called _system_ or _well known_ ports and are used by the application layer of the Internet Protocol suite for the establishment of host-to-host connectivity. Root privileges are required to open these ports. Ports 1024 to 49151 are called _user_ ports. Some of these may also be reserved; for example, port 8080 is commonly used as a server port. - -:fontawesome-brands-wikipedia-w: -[List of TCP and UDP port numbers](https://en.wikipedia.org/wiki/List_of_TCP_and_UDP_port_numbers "Wikipedia") - -Both system and user ports are used by transport protocols to indicate an application or service. - - -### Connecting to a process listening on a port - -When the client process executes the function `hopen` with a server’s details as its argument, it starts a connection and returns a positive integer used to identify the connection handle. This integer is assigned to a variable, `h` in the following examples demonstrating different connection protocols. - -### TCP - -`hopen` can be used to open a TCP handle to another process by passing the process host and port as argument: user and password can also be included. - -```q -q)h:hopen`:localhost:4567 -q)h"2+2" -4 -``` - -The hostname can be omitted when connecting to a process running on the same machine. - -```q -q)i:hopen`::4567 -q)i"2+2" -4 -``` - -This can also be written with the target process port number only: - -```q -q)j:hopen 4567 -q)j"2+2" -4 -``` - -It is also worth noting that for the purposes of this white paper we will be assigning open handles to variables; but this is not required. - -```q -q)hopen 4567 -4i -q)4"1+1" -2 -q)4"\\p" -4567i -``` - -Using `netstat` we can see the process listening on port 4567 and the established TCP connection between the client and server processes. - -```bash -$ netstat | grep 4567 -tcp 0 0 localhost:50254 localhost:4567 ESTABLISHED -tcp 0 0 localhost:4567 localhost:50254 ESTABLISHED -unix 3 [ ] DGRAM 14567 -``` - -It is also possible to open a ‘one-shot’ connection to another process. This will establish a connection for only as long as it takes to execute the query. Since V4.0 a one-shot query can be run with a timeout, as in the second example below. - -```q -q)`::4567"2+2" -4 -q)`::[(`::4567;100);"1+1"] -2 -``` - -It is possible for a kdb+ process to have too many open connections. -The max is defined by the system limit for protocol (operating system configurable). Prior to 4.1t 2023.09.15, the limit was hardcoded to 1022. -After the limit is reached, you see the error `'conn` on the server process. (All successfully opened connections remain open.) - - -```q -q)\p 5678 -q)'conn -``` - -```q -q)openCon:{hopen 5678} -q)do[2000;openCon[]] -'hop. OS reports: Connection reset by peer - [1] openCon:{hopen 5678} - ^ -``` - -`hopen` also accepts a timeout parameter in its argument. This will prevent the process hanging when attempting to connect to a process which is alive but unresponsive; for example, due to a long running query. - -```q -q)// system"sleep 30" running on port 4567 -q)l:hopen(`::4567;1) -'timeout - [0] l:hopen(`::4567;1) - ^ -``` - -Attempts to connect to a non-existent process signal an error. - -```q -q)l:hopen 5678 -'hop. OS reports: Connection refused - [0] l:hopen 5678 - ^ -``` - -Opening a connection to the current port will execute any queries on handle 0, the file descriptor for standard input. - -```q -q)\p 4567 -q)h:hopen 4567 -q)h -0i -q)h"0N!`hi" -`hi -`hi -``` - -`hopen` can also be used to open a handle to a file location. This is discussed below with regard to writing to a log file within a kdb+ tickerplant setup. - -:fontawesome-solid-book: -[`hopen`](../../ref/hopen.md) - - -### Unix Domain Socket - -A Unix Domain Socket ([UDS](../../basics/listening-port.md#unix-domain-socket)) can be used for connections between processes running on the same host. A UDS allows bidirectional data exchange between processes running on the same machine. They are similar to internet sockets (TCP/IP socket) but, rather than using a network protocol, all communication occurs entirely within the operating system kernel. Using a UDS to communicate when processes are on the same machine will avoid some checks and operations in the TCP/IP protocol, making them faster and lighter. There is no Windows equivalent. - -```q -q)h:hopen`:unix://4567 -q)h -4i -``` - -Use `netstat` to view this connection vs tcp - -```bash -// unix domain socket established -$ netstat |grep 4567 -unix 3 [ ] DGRAM 14567 -unix 3 [ ] STREAM CONNECTED 100811 @/tmp/kx.4567 -``` - -### TLS/SSL - -The TLS protocol is used to communicate across a network in a way designed to prevent eavesdropping and tampering. It is possible to initiate a TLS connection between kdb+ processes for secure communication. Note that a TLS connection requires certificates to be in place before initializing and it should currently only be considered for long-standing, latency-insensitive, low-throughput connections given high overheads. - -:fontawesome-solid-graduation-cap: -[Secure Socket Layer](../../kb/ssl.md) - - -### Negative ports - -A kdb+ process can be started in multithreaded input mode by setting a [negative port number](../../basics/listening-port.md#multi-threaded-port). A multithreaded process will use separate threads for every process that connects, which means that each client request can be executed on a separate CPU. - -Although secondary processes are used to farm queries out to multiple processes, they still have a single-threaded input queue. By using a negative port number, it is possible to multithread that queue too. - -```bash -$ q -p -4567 -KDB+ 4.0 2020.03.17 Copyright (C) 1993-2020 Kx Systems -l64/ 2(16)core 1959MB katrina ubuntu 127.0.1.1 EXPIRE 2021.05.05 kmccormack@kx.com .. - -q)\p --4567i -``` - -Connections can be opened to this process in the same way as described previously for positive port numbers. - -```q -q)h:hopen 4567 -q)h"1+1" -2 -q)h"a:2" -'noupdate: `. `a - [0] h"a:2" - ^ -``` - -:fontawesome-solid-graduation-cap: -[Multithreaded input](../../kb/multithreaded-input.md) - - -### IPC handlers - -The `.z` namespace is the main namespace used in kdb+ IPC programming. When a client sends a query via IPC, the message is [serialized](../../kb/serialization.md), sent, then deserialized on the server side after passing through a number of `.z` IPC handlers. This paper discusses the main handlersonly. - -:fontawesome-solid-book: -[`.z` namespace](../../ref/dotz.md) - -When a process attempts to open a connection to a kdb+ process, two main handlers are invoked; `.z.pw` and `.z.po`. - - -### `.z.pw` - -If [`.z.pw`](../../ref/dotz.md#zpw-validate-user) is set, it is the first handler invoked on the server when a client attempts to connect. This happens immediately after ['-u' checks](../..//basics/cmdline.md#-u-usr-pwd) if this option has been specified in the process command line. `.z.pw` is simply a function that can be used to perform custom validation so user and password info, as well as any other rules, can be validated as required by the application. - -By default `.z.pw` will return `1b`. The arguments passed to the function are user name (symbol) and password (string). These are optional in arguments to `hopen`. If the output of `.z.pw` is `1b`, the login can proceed (next stop `.z.po`). If it returns `0b`, login fails and the client gets an `access` error. - -:fontawesome-regular-map: -[Permissions with kdb+](../permissions/index.md "White paper") - -```q -q)// client process -q)h:hopen`::4567:katrina:password -``` - -```q -q)// server process -q)\p 4567 -q).z.pw:{[u;p] 0N!(u;p);1b} -q)(`katrina;"password") -``` - -If no username or password is passed to `hopen`, `u` and `p` are as below where the default username is the output of `.z.u` on the client process. - -```q -q)// client process -q).z.u -`katrina -q)hopen 4567 -4i -``` - -```q -q)// server process -q).z.pw:{[u;p] 0N!(u;p);1b} -q)(`katrina;"") -``` - - -### `.z.po` - -[`.z.po`](../../ref/dotz.md#zpo-open) (port open) is evaluated when a connection to a kdb+ process has been initialized and after it has been validated against `.z.pw` checks. Similar to `.z.pw`, `.z.po` will not be evaluated by default but only if it is assigned a user-defined function. Its argument is the handle to the connecting client process. This is typically used to build a dictionary of handles (`.z.w`) with session information such as [`.z.a`](../../ref/dotz.md#za-ip-address) (IP address) and [`.z.u`](../../ref/dotz.md#zu-user-id) (user). It is also commonly used, together with `.z.pc`, to track open connections to the process. - -```q -q).z.po:{0N!(x;.z.w;.z.a;.z.u)} -q)(7i;7i;2130706433i;`katrina) -``` - - -### Synchronous vs asynchronous communication - -Once we have established a connection, the next step is to query data available on the server process. This query can be either [synchronous or asynchronous](../../learn/startingkdb/ipc.md#synchronousasynchronous). - - -### Synchronous queries - -If the client queries the server synchronously, the client will be unavailable until the server responds with the result of the query or an error. The handle, the positive integer assigned to the variable `h` in this example, is used to send the query. - -When a query is sent synchronously any messages queued on this handle are sent and no incoming messages will be processed on any handle until a response to this sync query is received. - -```q -q)h:hopen 4567 -q)h"2+2" -4 -``` - -The basic method used to execute a query via IPC is sending the query as a string as in the above example. A function can also be executed on the server by passing a [parse tree](../../basics/parsetrees.md) to the handle: a list with the function as first item, followed by its arguments. - -To execute a function defined on the client side, simply pass the function name so it will be resolved before sending. To execute a function defined on the server, pass the function name as a symbol. - -```q -q)add:{x+2*y} -q)h:hopen 4567 -q)h"add" -{x+y} -q) -q)h(add;2;3) -8 -q)h(`add;2;3) -5 -``` - -If a synchronous query is interrupted, for example if the server process is killed, the client process will receive an error. In the example below the process running on port 5678 was killed before the query completed successfully. The variable `h` will still be assigned to the handle number but any further attempts to communicate across this handle will fail. - -```q -q)h:hopen 5678 -q) -q)h"system\"sleep 10\"" -'os - [0] h"system\"sleep 10\"" - ^ -q)h -4i -q)h"1+1" -'Cannot write to handle 4. OS reports: Bad file descriptor - [0] h"1+1" - ^ -``` - -The stale handle will be recycled if a new connection is made: `4` and therefore `h` could now point to a completely different process. - -It is possible to interrupt a long-running sync query with `kill -s INT *PID*`. As with the previous example, any subsequent attempt to communicate across this handle will fail. - -```q -q)h"system\"sleep 30\"" -'rcv handle: 4. OS reports: Interrupted system call - [0] h"system\"sleep 30\"" - ^ -q) -q)h"a" -'Cannot write to handle 4. OS reports: Bad file descriptor - [0] h"a" - ^ -``` - -!!! tip "Deferred response" - - [Deferred reponse](../../kb/deferred-response.md) with [`-30!`](../../basics/internal.md#-30x-deferred-response) allows a server to defer the response to a synchronous query, allowing other messages to be processed before responding. This is useful where synchronous messaging is necessary on the client side. - - An example implementation of deferred sync message handling is discussed in the blog [kdb+/q Insights: Deferred Response](https://kx.com/blog/kdb-q-insights-deferred-response/). - - -### Asynchronous queries - -A query can be sent asynchronously using a negative handle. An async query will not return a result and the client process does not wait. Async messages can be serialized and queued for sending, but the messages will not necessarily be dispatched immediately. Since the process is not waiting for a response, async querying is critical in situations where waiting for an unresponsive subscriber is unacceptable, e.g. in a tickerplant. - -```q -q)h:hopen 4567 -q)neg[h]"a:2+2" -q) -q)h"a" -``` - -### Flushing - -Outgoing aysnc messages are sent periodically on each iteration of the underlying process timer. This can cause messages to be queued such that it is necessary to flush all messages through a handle. This can be achieved as below: - -- execute `neg[handle][]` or `neg[handle](::)` -- send a synchronous message on the same handle: this will confirm execution as all messages are processed in the order they are sent - -:fontawesome-solid-book-open: -[Flushing](../../basics/ipc.md#flushing) - - -### Deferred synchronous - -[Deferred sync](../../kb/load-balancing.md) is when a message is sent asynchronously to the server using the negative handle and executes a function which includes an instruction to return the result though the handle to the client process (`.z.w`), again asynchronously. After the client sends its async request it blocks on the handle waiting for a result to be returned. - -```q -q)h:hopen 4567 -q)h"add" -{x+y+z} -q)h"proc" -{r:add . x;0N!r;neg[.z.w]({0N!x};r)} -q) -q)neg[h](`proc;1 2 3);res:h[]; -q)res -q)6 -``` - - -### Asynchronous callback - -In a kdb+ application there is generally a gateway process that will manage queries from multiple clients. In this situation it is not practical or useful to query data synchronously. Instead, it will use async callbacks. The initial async query from the client invokes a function on the server which will use `.z.w` (handle of the client) to return the result asynchronously to the client process. - -```q -q)// server process on port 4567 -q)\p 4567 -q)getLastPriceBySym:{select last price by sym from quote} -q)quote:([]date:.z.d; size:100?250; price:100?100.; sym:100?`a`b`c) -``` - -```q -q)// server process on port 4567 -q)// client process -q)h:hopen 4567 -q)onPriceReceived:{[x] -1 "Received async callback"; show x}; -q)neg[h]({neg[.z.w](`onPriceReceived;getLastPriceBySym x)};`) -q)Received async callback -sym| price ----| -------- -a | 79.20702 -b | 35.29016 -c | 43.7764 -``` - -:fontawesome-solid-graduation-cap: -[Callbacks](../../kb/callbacks.md) - - -### Broadcast - -Much of the overhead of sending a message via IPC is in serializing the data before sending. It is possible to ‘async broadcast’ the same message to multiple handles using the internal [`-25!`](../../basics/internal.md#-25x-async-broadcast) function. This serializes the message once and sends to all handles to reduce CPU and memory load. An error in publishing to any handle results in the message not being sent to any of the handles, regardless of the handle’s position in the list. - -```q -q)h1:hopen 5551 -q)h2:hopen 5552 -q)h3:hopen 5553 -q) -q)h1"a:1" -q)h3"a:1" -q) -q)-25!((h1;h2;h3);({c::a+1};`)) -q)h1"c" -2 -q)h3"c" -2 -q)// close process on 5552, try to assign 'd' -q)-25!((h1;h2;h3);({d::a+1};`)) -'5 is not an ipc handle - [0] -25!((h1;h2;h3);({d::a+1};`)) - ^ -q)h1"d" -'d - [0] h1"d" - ^ -q)h3"d" -'d - [0] h3"d" - ^ -``` - -This can be applied to a tickerplant publishing asynchronously to multiple subscribers: RDBs, chained tickerplants, or other processes performing realtime calculations. - -### `.z.pg` (get) and `.z.ps` (set) - -When the query reaches the server, it invokes a different message handler according to whether the query was sent synchronously ([`.z.pg`](../../ref/dotz.md#zpg-get) – get) or asynchronously ([`.z.ps`](../../ref/dotz.md#zps-set) – set). The return value from `.z.pg` is sent as the response message and the return value from `.z.ps` is ignored unless it is an error. If it is an error, the message is printed to the console of the process the query is being executed on. The error will not be visible to the querying process (the client). - -By default, `.z.pg` and `.z.ps` are equivalent to `{value x}` but are commonly edited to implement user-level permissioning. The default behavior of these (or any `.z.p*` handlers defined before `.q.k` is loaded) can be restored by `\x`. - -:fontawesome-regular-map: -[Permissions with kdb+](../permissions/index.md "White paper") - -Note that outside of the common applications discussed above, any value can be sent across a handle and `.z.pg` or `.z.ps` defined to evaluate it. - -```q -q)// server process -q)\p 4567 -q).z.pg:{99h=type x} -``` - -```q -q)// client process -q)h:hopen 4567 -q)h"1+1" -0b -q)h`a`b`c!`d`e`f -1b -``` - - -### Closing a connection - -Now the client has connected to the server and run the query, the last step is to close the handle. - -To close the handle inside a process, use [`hclose`](../../ref/hopen.md#hclose). - -```q -q)h:hopen 4567 -q)h -4i -q)h"2+2" -4 -q)hclose h -q)h"2+2" -'Cannot write to handle 4. OS reports: Bad file descriptor - [0] h"2+2" - ^ -``` - -This can also be used to close a handle from the server side. - -```q -q)// server process -q)// use .z.po to store a list of opened handles -q)l:() -q).z.po:{`l set l,.z.w} -q)l -q) -q)// connection opened from client -q)l -,6i -q)hclose 6 -``` - - -### `.z.pc` (close) - -Running `hclose[handle]` on the client will cause [`.z.pc`](../../ref/dotz.md#zpc-close) to be invoked on the server. Unlike the message handlers we have seen before, it is not possible to obtain information relating to the client process in `.z.pc` because at this point that connection no longer exists. The integer identifying the handle that was just closed is passed as argument to `.z.pc` and can be used together with `.z.po`, to track connections to the server process. - -Besides when a handle is closed gracefully using `hclose`, `.z.pc` is also invoked on the server if the client process is killed. - -```q -q).z.po:{0N!"connection opened, handle: ",string[.z.w]} -q).z.pc:{0N!("handle closed";x;.z.w)} -q) -q)"connection opened, handle: 6" -("handle closed";6i;0i) -``` - - -### Tracking open connections - -[`.z.H`](../../ref/dotz.md#zh-active-sockets) can be used to view active sockets, augmented by the internal function [-38!](../../basics/internal.md#-38x-socket-table) to view details of active connections. - -In addition `.z.po` and `.z.pc` can be used to track all connections from client processes. In the below example, we create a table in memory on the server side, keyed on handle. When a new connection is opened, a new row is added to the table; and when the connection is closed, the row can be deleted or preferably updated to reflect the new status. - -```q -q)// on the server side -q)\p 4567 -q)trackConnections:([handle:()]ip:();user:();status:()) -q).z.po:{`trackConnections upsert (.z.w;.z.a;.z.u;`OPEN)} -q).z.pc:{`trackConnections set update status:`CLOSED from trackConnections where handle=x} -``` - -```q -q)// on the client side -q)h:hopen 4567 -q)h"trackConnections" -handle| ip user status -------| ------------------------- -892 | 2130706433 katrina OPEN -q)hclose h -q) -q)h:hopen 4567 -q)h"trackConnections" -handle| ip user status -------| ------------------------- -892 | 2130706433 katrina CLOSED -736 | 2130706433 katrina OPEN -``` - -If the handle is closed from the server side, `trackConnections` is not updated by default, and `.z.pc` must be invoked manually. - -```q -q)trackConnections -handle| ip user status -------| ------------------------- -6 | 2130706433 katrina OPEN -q) -q)hclose 6 -q)trackConnections -handle| ip user status -------| ------------------------- -6 | 2130706433 katrina OPEN -q)6"2+2" -'Cannot write to handle 6. OS reports: Bad file descriptor - [0] 6"2+2" - ^ -q).z.pc[6] -`trackConnections -q)trackConnections -handle| ip user status -------| ------------------------- -6 | 2130706433 katrina CLOSED -``` - - -## Application of IPC in a kdb+ tick system - -:fontawesome-solid-graduation-cap: -[Realtime database](../../learn/startingkdb/tick.md) -
-:fontawesome-regular-map: -[Building real-time tick subscribers](../rt-tick/index.md) - -The core elements of this kdb+ tick setup are - -- A tickerplant -- A realtime database (RDB) -- A historical database (HDB) -- A source of data (feed) - -The discussion uses the KX tick code to explore how IPC is used in a [vanilla tick application](../../architecture/index.md). - -:fontawesome-brands-github: -[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) - -In the following section, code taken from the kdb+ tick scripts is flagged. - - -### Initialize tickerplant - -When a vanilla tickerplant is first started up a number of variables are assigned, the process port is set, and the `u.q` utilities and the table schema scripts are loaded. - -```q -//tick.q extract -system"l ",tickdir,"/u.q" -``` - -The tickerplant process is set to listen on port 5010 unless a port is specified on the command line at startup. - -```q -//tick.q extract -if[not system"p";system"p 5010"] -``` - -All messages published to a tickerplant are immediately logged to a file. In the event of a process crashing, this file can be used to replay all messages up to the point of failure. A handle (`.u.l`) is opened to the desired file (`.u.L`) using `hopen`. Messages sent to this handle are appended to the file. - -:fontawesome-regular-map: -[Data recovery](../data-recovery.md#recovery) - -```q -q).u.L:`:sampleLog -q).u.L set () -`:sampleLog -q).u.l:hopen .u.L -q).u.l enlist (`upd;`tab;([]"first record")); -q)get .u.L -`upd `tab +(,`x)!,"first record" -q).u.l enlist (`upd;`tab;([]"second record")) -q)get .u.L -`upd `tab +(,`x)!,"first record" -`upd `tab +(,`x)!,"second record" -``` - - -### Initialize RDB - -During RDB initialization, the functions below are invoked to initialize table schemas and, if necessary, replay data from the tickerplant log (tplog) to catch up to the current state. - -```q -//tick.q extract -/ connect to ticker plant for (schema;(logcount;log)) -.u.rep .(hopen `$tpport)"(.u.sub[`;`];`.u `i`L)" -``` - -The RDB process opens a handle to the tickerplant using `hopen`, runs a synchronous query to subscribe this RDB to all tables and all symbols available from the tickerplant, and return the table names and schemas to the rdb. The result is passed to `.u.rep` in order to initialize these table schemas in memory. We can split this sequence out as below: - -```q -q)h:hopen `$tpport -q)x:h"(.u.sub[`;`];`.u `i`L)" -q).u.rep . x -``` - - -### Publish data to a tickerplant - -In a vanilla tickerplant `.z.pg` and `.z.ps` use the default `{value x}` and `.u.upd` is defined as below: - -```q -//tick.q extract -\d .u -upd:{[t;x]t insert x;if[l;l enlist (`upd;t;x);j+:1]} -``` - -Data is first inserted into a table in memory, then logged to the tplog on disk such that running `{value x}` on any line will invoke the `upd` function with the table name and data as arguments. - -Data is published synchronously from a feedhandler to the tickerplant. This is not recommended for time-critical feeds with a large number of updates but can be used for feeds where you need confirmation that each message was received correctly. - - -### Managing tickerplant subscriptions - -In a tickerplant, current subscriptions are maintained in `.u.w`, an in-memory dictionary which stores each subscriber’s handle and subscription information. When a process subscribes to the tickerplant, the process handle and any symbol filters are added to `.u.w`. - -Below is an example of `.u.w` in a tickerplant with two tables: `trade` and `quote`. In this example the RDB is the only subscriber and it is subscribed to all symbols for both tables. - -```q -q).u.w -quote| 7i ` -trade| 7i ` -``` - -`.z.pc` is invoked when a handle to the process is closed. In the context of a tickerplant this means that a connection from a subscriber has closed and this subscription must be removed from `.u.w`. - -```q -//tick.q extract -\d .u -del:{w[x]_:w[x;;0]?y} -.z.pc:{del[;x]each t} -``` - -The function `.u.del` takes two arguments: `x`, the handle to the now closed subscriber, and `.u.t`, a global list of all tables defined in the tickerplant. This function will remove the handle from the subscription list for each of these tables. In this example, if the single RDB subscribing to the process is killed, `.u.w` will be empty. - -```q -q).u.w -quote| -trade| -``` - - -### Publishing from the tickerplant to subscribers - -Data is published from the tickerplant by invoking `.u.pub` on the tickerplant periodically on a timer. `.u.pub` takes two arguments: `t`, the table name, and `x`, the data (`value t`). - -```q -//tick.q extract -\d .u -pub:{[t;x]{[t;x;w]if[count x:sel[x]w 1;(neg first w)(`upd;t;x)]}[t;x]each w t} -``` - -If on a timer tick (whatever `\t` is set to, 1000ms by default) the count of the in-memory table `x` is greater than zero, these rows are published asynchronously to each handle subscribed to table `t` in `.u.w`. - -This publishing is considered the critical path of data in a low-latency tick system. As such, it is best practice to use async messaging so that no unnecessary delays in streaming data are caused by the process waiting for a response from a hanging or unresponsive subscriber. - -It is possible that a client subscribed to a tickerplant might not be processing the data it is being sent quickly enough, causing a backlog to form and the TCP buffer of the subscriber to fill. When this happens the pending messages will sit in an output queue in the memory space of the tickerplant process itself until the slow subscriber becomes available. It is possible to view a dictionary of open handles mapped to the size of messages queued for this handle using [`.z.W`](../../ref/dotz.md#zw-handles). In extreme cases the tickerplant memory footprint might grow to an unmanageable level, resulting in a [`wsfull` error](../../basics/errors.md). If writing logic for the tickerplant to remove a slow consumer to protect the tickerplant, `.z.pc` must be manually invoked to perform subscription cleanup after the bad client is kicked as shown in the _Tracking open connections_ example above. - -:fontawesome-regular-map: -[Disaster recovery](../disaster-recovery/index.md) - - -### End of day - -The timer has another important function within the tickerplant: the end-of-day rollover. The [`.z.ts`](../../ref/dotz.md#zts-timer) timer function takes the system’s date as an argument and passes it to `.u.ts`. Below we can see the function definitions of `endofday` and `ts` within the `.u` namespace. - -```q -//tick.q extract -endofday:{ - end d; - d+:1; - if[l;hclose l;l::0(`.u.ld;d)]}; - -ts:{if[d -[:fontawesome-solid-envelope:](mailto:kmccormack@kx.com) -  -[:fontawesome-brands-linkedin:](https://www.linkedin.com/in/katrina-mccormack-35379359/) - - - -## Notes - -TCP/IP - -: The Internet Protocol Suite, commonly known as TCP/IP, is a conceptual model and set of communications protocols which specifies how data is exchanged. It provides end-to-end communications that identify how the data should be broken into packets, addressed, transmitted, routed and received at the destination with little central management. - -TCP/IP sockets - -: The term _socket_ usually refers to a TCP socket. A socket is one end point of a two-way communication link. These network sockets allow communication between two different processes on the same or on different machines. These sockets are assumed to be associated with a specific socket address: the combination of an IP address and port number. The local process can communicate with another (foreign) process by sending data to or receiving data from the foreign socket address which will have its own associated socket. - -![TCP diagram](img/tcp-diagram.png) - -A process can refer to a socket using a file descriptor or handle, an abstract indicator used to access a file, or other resource. - - -### Major IPC-related releases - -version | date | feature ---------|------------|-------- -2.4 | 2012.03.31 | Added [Multi-threaded input](../../releases/ChangesIn2.4.md#multi-threaded-input) using a negative port number -| | [`.z.pw`](../../releases/ChangesIn2.4.md#zpw): username and password passed to `.z.pw` to enable option for custom validation -2.5 | 2012.03.31 | Added [`.z.W`](../../releases/ChangesIn2.5.md#zw) to return a dictionary of IPC handles with the total number of bytes in each output queue -2.6 | 2012.08.03 | Added [IPC compression](../../releases/ChangesIn2.6.md#ipc-compression) -| | [`.z.W`](../../releases/ChangesIn2.6.md#zw) updated to return the size in bytes of each messages in the input queue -2.7 | 2013.06.28 | Added an [IPC message validator](../../releases/ChangesIn2.7.md#ipc-message-validator) -[3.4](../../releases/ChangesIn3.4/) | 2019.06.03 | IPC message size limit raised from 2GB to 1TB -| | Added support for IPC via Unix Domain Sockets -| | Secure Sockets Layer(SSL)/Transport Layer Security (TLS) -| | Added async broadcast as `-25!(handles;msg)` -3.5 | 2020.02.13 | Added `hopen` timeout for [TLS](../../releases/ChangesIn3.5.md#ssltls) -3.6 | 2020.02.24 | [Deferred response](../../releases/ChangesIn3.6.md#deferred-response): a server process can now use `-30!x` to defer responding to a sync query From 1d95cd2fbfbee9483a602aea7a369cff8e286d16 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 24 Jul 2024 12:39:05 +0100 Subject: [PATCH 05/61] add tick details --- docs/architecture/index.md | 27 +--- docs/architecture/rq.md | 122 +++++++++++++++++++ docs/architecture/tickq.md | 244 +++++++++++++++++++++++++++++++++++++ docs/architecture/uq.md | 144 ++++++++++++++++++++++ docs/wp/index.md | 1 - mkdocs.yml | 8 +- 6 files changed, 520 insertions(+), 26 deletions(-) create mode 100644 docs/architecture/rq.md create mode 100644 docs/architecture/tickq.md create mode 100644 docs/architecture/uq.md diff --git a/docs/architecture/index.md b/docs/architecture/index.md index 7992e0f37..7426f63db 100644 --- a/docs/architecture/index.md +++ b/docs/architecture/index.md @@ -37,6 +37,8 @@ Manages subscriptions: adds and removes subscribers, and sends subscriber table Handles end-of-day (EOD) processing. +[`tick.q`](tickq.md) represents a tickerplant and is provided as a starting point for most environments. + !!! tip "Best practices for tickerplants" Tickerplants should be lightweight, not capturing data and using very little memory. @@ -67,6 +69,8 @@ At startup, the RDB sends a message to the tickerplant and receives a reply cont At end of day usually writes intraday data to the Historical Database, and sends it a new EOD message. +[`r.q`](rq.md) represents a tickerplant and is provided as a starting point for most environments. + !!! tip "Best practices for real-time databases" RDBs queried intraday should exploit attributes in their tables. For example, a trade table might be marked as sorted by time (`` `s#time``) and grouped by sym (`` `g#sym``). @@ -157,26 +161,3 @@ Can connect both the real-time and historical data to allow users to query acros [Query Routing: A kdb+ framework for a scalable, load balanced system](../wp/query-routing/index.md) -## :fontawesome-solid-hand-point-right: What next? - -:fontawesome-regular-map: -[Building real-time tick subscribers](../wp/rt-tick/index.md) -
-:fontawesome-regular-map: -[Data recovery for kdb+ tick](../wp/data-recovery.md) -
-:fontawesome-regular-map: -[Disaster-recovery planning for kdb+ tick systems](../wp/disaster-recovery/index.md) -
-:fontawesome-regular-map: -[Intraday writedown solutions](../wp/intraday-writedown/index.md) -
-:fontawesome-regular-map: -[Query routing: a kdb+ framework for a scalable load-balanced system](../wp/query-routing/index.md) -
-:fontawesome-regular-map: -[Order book: a kdb+ intraday storage and access methodology](../wp/order-book.md) -
-:fontawesome-regular-map: -[kdb+tick profiling for throughput optimization](../wp/tick-profiling.md) - diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md new file mode 100644 index 000000000..852a85489 --- /dev/null +++ b/docs/architecture/rq.md @@ -0,0 +1,122 @@ +--- +title: RBB (r.q) | Documentation for q and kdb+ +description: How to construct an RDB +keywords: kdb+, q, rdb, streaming +--- +# Real-time Database (RDB) using r.q + +`r.q` is available from :fontawesome-brands-github:[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) + +## Overview + +### Customization + +`r.q` provides a starting point to most environments. The source code is freely avaialble and can be tailered to individual needs. For example: + +#### Memory use + +The default RDB will store all of a days data in memory before end-of-day writting to disk. The host machines should be configured that all required resources +can handle the demands that may be made of them (both for today and the future). +Depending on when there may be periods of low/no activity, [garbage collection](../ref/dotq.md#gc-garbage-collect) could be deployed after clearing tables at end-of-day, or a system for intra-day writedowns. + +#### User queries + +A gateway process should control user queries and authorisation/authentication, using RDBs/RTEs/HDBs to retrieve the required information. +If known/common queries can be designed, the RDB can load additional scripts to pre-define functions a gateway can call. + +### End-of-day + +The end-of-day event is governed by the tickerplant process. The tickerplant calls the RDB [`.u.end`](#uend) function when this event occur. +The main end-of-day event for an RDB is to save todays data from memory to disk, clear its tables and cause the HDB to be aware of a new days dataset for it to access. + +!!! Note "[.u.rep](#urep) sets the HDB directory be the same as the tickerplant log file directory. This can be edited to use a different directory if required" + +### Recovery + +Using IPC, the RDB process can retrieve the current tickerplant log location and use via the [variables](tickq.md#variables) the tickerplant maintains. +The function [`.u.rep`](#urep) is then used to populate any tables from the log. + +!!! Note "The RDB should have tickerplant log available from a directory on the same machine. The RDB/tickerplant can be changed to reside on different hosts but this will entail the increased resource utilisation of transmitting the log file contents over the network." + +## Usage + +```bash +q tick/r.q [host1]:port1[:usr:pwd] [host2]:port2[:usr:pwd] [-p 5020] +``` + +| Parameter Name | Description | Default | +| ---- | ---- | --- | +| host1 | host running kdb+ instance that the RDB will subscribe to e.g. tickerplant host | localhost | +| port1 | port of kdb+ instance that the RDB will subscribe to e.g. tickerplant port | 5010 | +| host2 | host of kdb+ instance to inform at end-of-day, after data saved to disk e.g. HBD host | localhost | +| port2 | port of kdb+ instance to inform at end-of-day, after data saved to disk e.g. HBD port | 5012 | +| usr | username | <none> | +| pwd | password | <none> | +| -p | [listening port](../basics/cmdline.md#-p-listening-port) for client communications | <none> | + +!!! Note "Standard kdb+ [command line options](../basics/cmdline.md) may also be passed" + +## Variables + +| Name | Description | +| ---- | ---- | +| .u.x | Connection list. First element populated by [`host1`](#usage) (tickerplant), and second element populated by [`host2`](#usage) (HDB) | + +## Functions + +Functions are open source & open to customisation. + +### upd + +Called by external process to update table data. Defaults to [`insert`](../ref/insert.md) to insert/append data to a table. + +```q +upd[x;y] +``` +Where + +* `x` is a symbol atom naming a table +* `y` is table data to add to table `x`. + +### .u.end + +Perform end-of-day actions of saving tables to disk, clearing tables and running reload on HDB instance to make it aware of new day of data. + +```q +.u.end[x] +``` +Where x is the date that has ended. + +Actions performed: + +* finds all tables with the group attribute on the sym column +* calls [`.Q.dpft`](../ref/dotq.md#hdpf-save-tables), with params: + * HDB connection from [`.u.x`](#variables) (second element) + * current directory (note: directory was changed in [`.u.rep`](#urep)) + * date passed to this function (`x`) i.e. day that is ending + * `` `sym `` column +* re-apply group attribute to sym column for those tables found in first steps (as clearing the table removed grouped attribute) + +### .u.rep + +Initialise RDB by creating tables, which is then populated with any existing tickerplant log. Will set the HDB directory to use at end-of-day. + +```q +.u.rep[x;y] +``` +Where + +* `x` is a list of table details, each element a two item list + * symbol for table name + * schema table +* `y` is the tickerplant log details.log comprising of a two item list: + * a long for the log count (null represents no log) + * a file symbol for the location of the current tickerplant log (null represents no log) + +Actions performed: + +* tables are created using `x` +* if a tickerplant log file has been provided + * log file is replayed using [`-11!`](../basics/internal.md#-11-streaming-execute) to populate tables + * set the HDB directory by changing the working directory to the same directory as used by the log file (see [`tick.q`](tickq.md#usage) for details on how to alter the log file directory). + diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md new file mode 100644 index 000000000..1ffb700e2 --- /dev/null +++ b/docs/architecture/tickq.md @@ -0,0 +1,244 @@ +--- +title: tick.q | Documentation for q and kdb+ +description: How to construct a tickerplant process +keywords: hdb, kdb+, q, tick, tickerplant, streaming +--- +# Tickerplant (TP) using tick.q + +`tick.q` is available from :fontawesome-brands-github:[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) + +## Overview + +### Customization + +`tick.q` provides a starting point to most environments. The source code is freely avaialble and can be tailered to individual needs. + +### Schema file + +A schema file should be created to describe the data you plan to capture, by specifing the tables that will be populated by the tickerplant environment. +The [datatypes](../basics/datatypes.md) and [attributes](../ref/set-attribute.md) are denoted within the file as shown in this example: +```q +quote:([]time:`timespan$(); sym:`g#`symbol$(); bid:`float$(); ask:`float$(); bsize:`long$(); asize:`long$(); mode:`char$(); ex:`char$()) +trade:([]time:`timespan$(); sym:`g#`symbol$(); price:`float$(); size:`int$(); side:`char$()) +``` + +The default setup requires the first two columns to be `time` and `sym`. + +### Real-time vs Batch Mode + +_The mode is controlled via the [`-t`](#usage) command line parameter._ +Batch mode can alleviate CPU use on both the tickerplant and its subscribers by grouping together multiple ticks within the timer interval prior to sending/writing. +This will be at the expense of tickerplant memory (required memory to hold several ticks) and increased latency that may occur between adding to the batch and sending. +There is no ideal setting for all deployments as it depends on the frequency of the ticks received. +Real-time mode will process every tick as soon as they occur. + +!!! note "A feedhandler can be written to send messages comprising of multiple ticks to a tickerplant. In this situation real-time mode will already be processing batches of messages." + +### End-of-day + +The tickerplant watches for a change in the current day. +As the day ends, a new tickerplant log is created and the tickerplant informs all subscribed clients, via their `.u.end` function. +For example, a RDB may implement [`.u.end`](rq.md#uend) to write down all in-memory tables to disk which can then be consumed by a HDB. + +### Tickerplant Logs + +[Log files](../kb/logging.md) are created using the format `/` e.g. `tplog/sym2022.02.02`. +These record all published messages and permit recovery by downstream clients, by allowing them to replay messages they have missed. +The directory used should have enough space to record all published data. + +As end-day-day causes a file roll, a process should be put in place to remove old log files that are no longer required. + +!!! note "The tickerplant does not replay log files for clients, but exposes [log file details](#variables) to clients so they can access the current log file" + +### Publishing to a tickerplant + +Feed handlers publish ticks to the tickerplant using [IPC](../basics/ipc.md). These can be a kdb+ process or clients written in any number of different languages that use one of the available client APIs. +Each feed sends data to the tickerplant by calling the [`.u.upd`](#uupd) function. The call can include one or many ticks. For example, publishing from kdb+: + +```q +q)h:hopen 5010 / connect to TP on port 5010 of same host +q)neg[h](".u.upd";`trade;(.z.n;`APPL;35.65;100;`B)) / async publish single tick to a table called trade +q)neg[h](".u.upd";`trade;(10#.z.n;10?`MSFT`AMZN;10?10000f;10?100i;10?`B`S)) / async publish 10 ticks of some random data to a table called trade +... +``` + +### Subscribing to a tickerplant + +Clients, such as a RDB or RTE, can subscribe by calling [`.u.sub`](uq.md#usub) over [IPC](../basics/ipc.md). + +```q +q)h:hopen 5010 / connect to TP on port 5010 of same host +q)h".u.sub[`;`]" / subscribe to all updates +``` +```q +q)h:hopen 5010 / connect to TP on port 5010 of same host +q)h".u.sub[`trade;`MSFT.O`IBM.N]" / subscribe to updates to trade table that contain sym value of MSFT.O or IBM.N only +``` + +Clients should implement functions [`upd`](rq.md#upd) to receive updates, and [`.u.end`](rq.md#uend) to perform any end-of-day actions. + +## Usage + +```bash +q tick.q SRC DST [-p 5010] [-t 1000] [-o hours] +``` + +| Parameter Name | Description | Default | +| --- | --- | --- | +| SRC | schema filename, loaded using the format `tick/.q` | sym | +| DST | directory to be used by tickerplant logs. _No tickerplant log is created if no directory specified_ | <none> | +| -p | [listening port](../basics/cmdline.md#-p-listening-port) for client communications | 5010 | +| -t | [timer period](../basics/cmdline.md#-t-timer-ticks) in milliseconds. Use zero value to enable real-time mode, otherwise will operate in batch mode. | real-time mode (with timer of 1000ms) | +| -o | [utc offset](../basics/cmdline.md#-o-utc-offset) | localtime | + +!!! Note "Standard kdb+ [command line options](../basics/cmdline.md) may also be passed" + +## Variables + +| Name | Description | +| ---- | ---- | +| .u.w | Dictionary of registered clients interest in data being processed i.e. tables->(handle;syms) | +| .u.i | Msg count in log file | +| .u.j | Total msg count (log file plus those held in buffer) - used when in batch mode | +| .u.t | Table names | +| .u.L | TP log filename | +| .u.l | Handle to tp log file | +| .u.d | Current date | + +## Functions + +Functions are open source & open to customisation. + +### .u.endofday + +Performs end-of-day actions. + +```q +.u.endofday[] +``` + +Actions performed: + +* inform all subscribed clients (for example, RDB/RTE/etc) that the day is ending by calling [.u.end](uq.md#uend) +* increment current date ([`.u.d`](#variables)) to next day +* roll log if using tickerplant log, i.e. + * close current tickerplant log ([`.u.l`](#variables)) + * create a new tickerplant log file i.e set [`.u.l`](#variables), call [`.u.ld`](#uld) with new date + +### .u.tick + +Performs initialisation actions for the tickerplant. + +```q +.u.tick[x;y] +``` + +Where + +* `x` is the name of the schema file without the `.q` file extension i.e. [`SRC`](#usage) command line parameter +* `y` is the directory used to store tickerplant logs i.e. [`DST`](#usage) command line parameter + +Actions performed: + +* call [`.u.init[]`](uq.md#uinit) to initialise table info, [`.u.t`](#variables) and [`.u.w`](#variables) +* check first two columns in all tables of provided schema are called `time` and `sym` (throw `timesym` error if not) +* apply [`grouped`](../ref/set-attribute.md#grouped-and-parted) attribute to the sym column of all tables in provided schema +* set [`.u.d`](#variables) to current local date, using [`.z.D`](../ref/dotz.md#zt-zt-zd-zd-timedate-shortcuts) +* if a tickerplant log filename was provided: + * set [`.u.L`](#variables) with a temporary value of `` `:/.......... `` (will have date added in next step) + * create/initialise the log file by calling [`.u.ld`](#uld), passing [`.u.d`](#variables) (current local date) + * set [`.u.l`](#variables) to log file handle + +### .u.ld + +Initialise or reopen existing log file. + +```q +.u.ld[x] +``` + +Where `x` is current date. Returns handle of log file for that date. + +Actions performed: + +* using [`.u.L`](#variables), change last 10 chars to provided date and create log file if it doesnt yet exist +* set [`.u.i`](#variables) and [`.u.j`](#variables) to count of valid messages currently in log file +* if log file is found to be corrupt (size bigger than size of number of valid messages) an error will be returned +* open new/existing log file + +### .u.ts + +Given a date, runs end-of-day procedure if a new day has started. + +```q +.u.ts[x] +``` +Where x is a date. + +Compares date provided with [`.u.d`](#variables). If no change, no action taken. +If one day difference (i.e. a new day), [`.u.endofday`](#uendofday) is called. +More than one day results in an error and the kdb+ timer is cancelled. + +### .u.upd + +Update tickerplant with data to process/analyse. External processes call this to input data into the tickerplant. + +```q +.u.upd[x;y] +``` +Where + +* `x` is table name (sym) +* `y` is data for table `x` (list of column data, each element can be an atom or list) + +#### Batch Mode + +Add each recieved message to batch and record message to tickerplant log. Batch will be published on running timer. + +Actions performed: +* If the first element of `y` is not a timespan (or list of timespan) + * inspect [`.u.d`](#variables), if a new day has occured call [`.z.ts`](#batch-mode_1) + * add a new timespan column populated with the current local time ([`.z.P`](../ref/dotz.md#zp-local-timestamp)). If mutiple rows of data, all rows receive the same time. +* Add data to current batch (i.e. new data `y` inserted into table `x`), which will be published on batch timer [`.z.ts`](#batch-mode_1). +* If tickerplant log file created, write .u.upd call & params to the log and increment [`.u.j`](#variables) + + +#### Realtime Mode + +Publish each received message to all interested clients & record message to tickerplant log. + +Actions performed: + +* Checks if end-of-day procedure should be run by calling [`.u.ts`](#uts) with the current date +* If the first element of `y` is not a timespan (or list of timespan), add a new timespan column populated with the current local time ([`.z.P`](../ref/dotz.md#zp-local-timestamp)). If mutiple rows of data, all rows receive the same time. +* Retrieves the column names of table `x` +* Publish data to all interested clients, by calling [`.u.pub`](uq.md#upub) with table name `x` and table generated from `y` and column names. +* If tickerplant log file created, write .u.upd call & params to the log and increment [`.u.i`](#variables) so that data recovery is possible. + +### .z.ts + +Defines the action for the kdb+ timer callback function [`.z.ts`](../ref/dotz.md#zts-timer). + +The frequency of the timer was set on the [command line](#usage) ([`-t`](../basics/cmdline.md#-t-timer-ticks) command-line option or [`\t`](../basics/syscmds.md#t-timer) system command). + +#### Batch Mode + +Runs on system timer at specified interval. + +Actions performed: + +* For every table in [`.u.t`](#variables) + * publish data to all interested clients, by calling [`.u.pub`](uq.md#upub) with table name `x` and table generated from `y` and column names. + * reapply the grouped attribute to the sym column +* Update count of processed messages by setting [`u.i`](#variables) to [`u.j`](#variables) (the number of batched messages). +* Checks if end-of-day procedure should be run by calling [`.u.ts`](#uts) with the current date + +#### Realtime Mode + +If batch timer not specified, system timer is set to run every 1000 milliseconds to check if end-of-day has occured. +End-of-day is checked by calling [`.u.ts`](#uts), passing current local date ([`.z.D`](../ref/dotz.md#zt-zt-zd-zd-timedate-shortcuts)). + +### Pub/Sub functions + +`tick.q` also loads [`u.q`](uq.md) which enables all of its features within the tickerplant. + diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md new file mode 100644 index 000000000..7d5f42a5f --- /dev/null +++ b/docs/architecture/uq.md @@ -0,0 +1,144 @@ +--- +title: u.q | Documentation for q and kdb+ +description: How to construct systems from kdb+ processes +keywords: kdb+, q, tick, tickerplant, streaming +--- +# u.q + +`u.q` is available from :fontawesome-brands-github:[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) + +## Overview + +Contains functions to allow clients to subscribe to all or subsets of available data, publishing to interested clients & alerting clients to events (i.e. end-of-day). Tracks client subscription interest and removes a client subscription details on their disconnection. + +This script is loaded by other processes, for example [a tickerplant](tickq.md). + +## Variables + +| Name | Description | +| ---- | ---- | +| .u.w | Dictionary of registered client interest in data being processed i.e. (tables->(handle;syms) | +| .u.t | Table names | + +## Functions + +Functions are open source & open to customisation. + +### .u.init + +Initialise variables used to track registered clients. + +```q +.u.init[] +``` + +Initialises [variables](#variables) used to track client interest in data being published. + +### .u.del + +Delete subscriber from dictionary of known subscribers ([`.u.w`](#variables)) for given table + +```q +.u.del[x;y] +``` +Where + +* `x` is a table name +* `y` is the connection handle + +### .u.sel + +Select from table, given optional sym filter. Used to filter tables to clients who may not want everything from the table. + +```q +.u.sel[x;y] +``` +Where + +* `x` is a table +* `y` is a list of syms (can be empty list) + +returns the table `x`, which can be filtered by `y`. + +### .u.pub + +Publish updates to subscribers. + +```q +.u.pub[x;y] +``` +Where + +* `x` is table name (sym type) +* `y` is new data for table `x` (table type) + +Actions performed: + +* find interested client handles for table `x` and any filter they may have (using [`.u.w`](#variables)) +* for each client + * filter `y` using [`.u.sel`](#usel) (if client specifed a filter at subscription time) + * publish [asynchronously](../basics/ipc.md#async-message-set) to client, calling their `upd` function with parameters _table name_ and _table data_. + + +### .u.add + +Add client subscription interest in table with optional filter. + +```q +.u.add[x;y] +``` +Where +* `x` is a table name (sym) +* `y` is list of syms used to filter table data, with empty sym representing for all table data + +Actions performed: + +* uses [`.z.w`](../ref/dotz.md#zw-handle) to get current client handle. +* find any existing subscriptions to table `x` for client (using [`.u.w`](#variables)) + * if existing, update filter with union on `y` + * else a new entry is added to [`.u.w`](#variables) with client handle, `x` and `y`. + +Returns 2 element list. The first element is the table name. The second element depends on whether `x` refers to a keyed table. + +* If `x` is a keyed table, [`.u.sel`](#usel) is used to select from the keyed table the required syms +* otherwise returns an empty table `x` (i.e. schema definition of table), with the [grouped attribute](../ref/set-attribute.md#grouped-and-parted) applied to the sym column. + +### .u.sub + +Used by clients to register subscription interest. + +```q +.u.sub[x;y] +``` +Where + +* `x` is a table name (sym) +* `y` is list of syms used to filter table data, with empty sym representing for all table data + +If `x` is empty symbol, client is subscribed to all known tables using `y` criteria. This is achieved by calling .u.sub for each table in [`.u.t`](#variables). +It then returns a list of all the return values provided by .u.sub (i.e. a list of pairs comprising of table name and table definition). + +An error is returned if the table does not exist. + +For the subscribing client, any previous registered in the given tables are removed prior to reinstating new criteria provided i.e. calls [`.u.del`](#udel). + +Calls [`.u.add`](#uadd) to record the client subscription and passes the returned values to client. + +### .u.end + +Inform all registered clients that end-of-day has occured. + +```q +.u.end[x] +``` + +Where `x` is a date, representing the day that is ending. + +Iterates over all client handles via [`.u.w`](#variables) and asyncronously calls their `.u.end` function passing `x`. + +### .z.pc + +Implementation of [`.z.pc`](../ref/dotz.md#zpc-close) callback for connection close. + +Called when a client disconnects. The client handle provided is used to call [`.u.del`](#udel) for all tables. This ensures all subscriptions are removed for that client. + diff --git a/docs/wp/index.md b/docs/wp/index.md index f9b4fae26..9c8ed10fc 100644 --- a/docs/wp/index.md +++ b/docs/wp/index.md @@ -26,7 +26,6 @@ White papers are flagged in the navigation menus. ## :fontawesome-solid-handshake: Interfaces - [**Internet of Things with MQTT**](iot-mqtt/index.md)
Rian Ó Cuinneagáin, 2021.06 -- [**Interprocess communication**](ipc/index.md)
Katrina McCormack, 2021.04 - [**Publish/subscribe with the Solace event broker**](solace/index.md)
Himanshu Gupta, 2020.11 - [**Lightning tickerplants**: pay-per-ticker with micropayments on the Lightning network](lightning-tickerplants/index.md)
Jeremy Lucid, 2019.05 - [**C API for kdb+**](capi/index.md)
Jeremy Lucid, 2018.12 diff --git a/mkdocs.yml b/mkdocs.yml index bbfc99057..76d79783e 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -464,7 +464,6 @@ nav: - SSL/TLS: kb/ssl.md - HTTP: kb/http.md - WebSockets: kb/websockets.md - - Interprocess communication (WP): wp/ipc/index.md - Tools: - Code profiler: kb/profiler.md - Debugging: basics/debug.md @@ -523,7 +522,12 @@ nav: - Developer tools: devtools.md - FAQ: kb/faq-listbox.md - Streaming: - - General architecture: architecture/index.md + - General architecture: + - Environment: architecture/index.md + - kdb-tick: + - Tickerplant (tick.q): architecture/tickq.md + - u.q: architecture/uq.md + - RDB (r.q): architecture/rq.md - Alternative architecture: kb/kdb-tick.md - Alternative in-memory layouts: kb/alternative-in-memory-layouts.md - Corporate actions: kb/corporate-actions.md From 873bf8a70b2cc300100693805ffadf7c6e040cc7 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 24 Jul 2024 14:06:00 +0100 Subject: [PATCH 06/61] fix format --- docs/kb/kdb-tick.md | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/docs/kb/kdb-tick.md b/docs/kb/kdb-tick.md index 7539f1ee7..275f8adb2 100644 --- a/docs/kb/kdb-tick.md +++ b/docs/kb/kdb-tick.md @@ -35,6 +35,7 @@ q chainedtick.q [host]:port[:usr:pwd] [-p 5110] [-t N] ``` !!! note + chainedtick.q contains `\l tick/u.q` therefore has a dependancy on [`u.q`](https://github.com/KxSystems/kdb-tick/blob/master/tick/u.q) existing within the directory `tick`. If the primary tickerplant is running on the same host (port 5010), the following starts a chained tickerplant on port 5010 sending bulk updates, every 1000 milliseconds. @@ -99,8 +100,10 @@ is a potential replacement for the default RDB `w.q` connects to the tickerplant, but buffers requests. Each time the number of records in the buffer is equal to `MAXROWS`, it will write the records to disk. At day end, remaining data is flushed to disk, the database is sorted (on disk) and then moved to the appropriate date partition within the historical database. -!!! Note It is not recommended to query the task running `w.q` as it contains a small (and variable-sized) selection of records. -Although it wouldn’t be difficult to modify it to keep the last 5 minutes of data, for example, that sort of custom collection is probably better housed in a task running a [`c.q`](#cq)-like aggregation. +!!! note + + It is not recommended to query the task running `w.q` as it contains a small (and variable-sized) selection of records. + Although it wouldn’t be difficult to modify it to keep the last 5 minutes of data, for example, that sort of custom collection is probably better housed in a task running a [`c.q`](#cq)-like aggregation. Syntax: ```bash From 6c5954ea158eaf5b27bbf17e1314994e42e3d7aa Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 24 Jul 2024 15:08:45 +0100 Subject: [PATCH 07/61] tidy up --- docs/img/parallelism.jpg | Bin 49944 -> 0 bytes docs/kb/kdb-tick.md | 6 ++---- docs/kb/mt-primitives.md | 6 ++---- 3 files changed, 4 insertions(+), 8 deletions(-) delete mode 100644 docs/img/parallelism.jpg diff --git a/docs/img/parallelism.jpg b/docs/img/parallelism.jpg deleted file mode 100644 index 53f02b97c56cbf10b225ff4104264864ae30efe9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 49944 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 uP(5n|j)@~SxrY^@y`090eAlk`4jAs1 - To complement existing explicit parallel computation facilities ([`peach`](../ref/each.md)), kdb+ 4.0 introduces implicit, within-primitive parallelism. It is able to exploit internal parallelism of the hardware – in-memory, with modern multi-channel memory architectures, and on-disk, e.g. making use of SSD internal parallelism. ```q @@ -22,6 +18,8 @@ q)(s;r[0]%r;r:st[;"\\t:100 f a"]each s:1 4 16 32) 1082 262 131 95 / time, ms ``` +## Supported Primitives + The following primitives now use multiple threads where appropriate: ```txt From 5d53162a29334a2cc61d8f8b395fcd2cd50b6f54 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 24 Jul 2024 19:44:06 +0100 Subject: [PATCH 08/61] reduce file - dup source code, link to relevant info --- docs/kb/publish-subscribe.md | 122 ++++------------------------------- 1 file changed, 11 insertions(+), 111 deletions(-) diff --git a/docs/kb/publish-subscribe.md b/docs/kb/publish-subscribe.md index dcc0d87f7..7b5b37bc8 100644 --- a/docs/kb/publish-subscribe.md +++ b/docs/kb/publish-subscribe.md @@ -5,133 +5,33 @@ keywords: kdb+, publish, q, subscribe --- # Publish and subscribe +## Setup - - -:fontawesome-brands-github: -[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) -contains functionality to allow processes to publish data and subscribe to it. It is worth highlighting how the publish-and-subscribe code can be used by any process on a standalone basis. The pubsub functionality is supplied in the `u.q` script of kdb+tick. +[`.u.q`](../architecture/uq.md) contains functionality to allow processes to publish data and subscribe to it. +Although commonly used by tickerplants, It is worth highlighting how the publish-and-subscribe code can be used by any process on a standalone basis. To give the ability to publish data to any process, a few things need to be done: - load `u.q` - declare the tables to be published in the top level namespace. Each table must contain a column called `sym`, which acts as the single key field to which subscribers subscribe -- initialize by calling `.u.init[]` -- publish data by calling `.u.pub[table name; table data]` +- initialize by calling [`.u.init[]`](../architecture/uq.md#uinit) +- publish data by calling [`.u.pub[table name; table data]`](../architecture/uq.md#upub) -The list of tables that can be published and the processes currently subscribed are held in `.u.w`. When a client process closes a connection, it is removed from `.u.w`. +The list of tables that can be published and the processes currently subscribed are held in [`.u.w`](../architecture/uq.md#variables). -Subscriber processes must open a connection to the publisher and call `.u.sub[tablename;list_of_symbols_to_subscribe_to]`. +Subscriber processes must open a connection to the publisher and call [`.u.sub[tablename;list_of_symbols_to_subscribe_to]`](../architecture/uq.md#usub). -`.u.sub` can be called synchronously or asynchronously. If `.u.sub` is called synchronously, the table schema is returned to the client. If the table being subscribed to is a keyed table, then the current value for each subscribed `sym` is returned, assuming it is stored. Otherwise, an empty schema definition is returned. Specifying `` ` `` for either parameter of `.u.sub` means _all_ – all tables, all syms, or all tables and all syms. +If a subscriber calls `.u.sub` again, the current subscription will be overwritten either for all tables (if a wildcard is used) or the specified table. To add to a subscription (e.g. add more syms to a current subscription) the subscriber can call [`.u.add`](../architecture/uq.md#uadd)). -If a subscriber calls `.u.sub` again, the current subscription will be overwritten either for all tables (if a wildcard is used) or the specified table. To add to a subscription (e.g. add more syms to a current subscription) the subscriber can call `.u.add`. +## Example - -The example scripts below can be downloaded from GitHub. Each script should be run from the OS command prompt e.g. +The example scripts below can be downloaded from :fontawesome-brands-github:[KxSystems/cookbook/pubsub](https://github.com/KxSystems/cookbook/tree/master/pubsub). Each script should be run from the OS command prompt e.g. ```bash $ q publisher.q $ q subscriber.q ``` - :fontawesome-brands-github: - [KxSystems/cookbook/pubsub](https://github.com/KxSystems/cookbook/tree/master/pubsub) - - -## Publisher - -The code below will generate some random data and publish it periodically on a timer. - -```q -\d .testdata - -// set the port -@[system;"p 6812";{-2"Failed to set port to 6812: ",x, - ". Please ensure no other processes are running on that port", - " or change the port in both the publisher and subscriber scripts."; - exit 1}] - -// create some test data to be published -// this could also be read from a csv file (for example) -meterdata:([]sym:10000?200; reading:10000?500i) -griddata:([]sym:2000?100?`3; capacity:2000?100f; flowrate:2000?3000i) - -// utility functions to get the next set of data to publish -// get the next chunk of data, return to start once data set is exhausted -counts:`.testdata.meterdata`.testdata.griddata!0 0 -getdata:{[table;n] - res:`time xcols update time:.z.p from (counts[table];n) sublist value table; - counts[table]+:n; - if[count[value table]<=counts[table]; counts[table]:0]; - res} -getmeter:getdata[`.testdata.meterdata] -getgrid:getdata[`.testdata.griddata] - -\d . - -// the tables to be published - all must be in the top level namespace -// tables to be published require a sym column, which can be of any type -// apart from that, they can be anything you like -meter:([]time:`timestamp$(); sym:`long$(); reading:`int$()) -grid:([]time:`timestamp$(); sym:`symbol$(); capacity:`float$(); flowrate:`int$()) - -// load in u.q from tick -upath:"tick/u.q" -@[system;"l ",upath;{-2"Failed to load u.q from ",x," : ",y, - ". Please make sure u.q is accessible.", - " kdb+tick can be downloaded from https://github.com/KxSystems/kdb-tick"; - exit 2}[upath]] - -// initialise pubsub -// all tables in the top level namespace (`.) become publish-able -// tables that can be published can be seen in .u.w -.u.init[] - -// functions to publish data -// .u.pub takes the table name and table data -// there is no checking to ensure that the table being published matches -// the table schema defined at the top level -// that is left up to the programmer! -publishmeter:{.u.pub[`meter; .testdata.getmeter[x]]} -publishgrid:{.u.pub[`grid; .testdata.getgrid[x]]} - -// create timer function to randomly publish -// between 1 and 10 meter records, and between 1 and 5 grid records -.z.ts:{publishmeter[1+rand 10]; publishgrid[1+rand 5]} - -/- fire timer every 1 second -\t 1000 -``` - - -## Subscriber - -```q -// define upd function -// this is the function invoked when the publisher pushes data to it -upd:{[tabname;tabdata] show tabname; show tabdata} - -// open a handle to the publisher -h:@[hopen;`::6812;{-2"Failed to open connection to publisher on port 6812: ", - x,". Please ensure publisher is running"; - exit 1}] - -// subscribe to the required data -// .u.sub[tablename; list of instruments] -// ` is wildcard for all -h(`.u.sub;`;`) - -\ -Could also do (for example) - -Subscribe to 10 syms of meter data: -h(`.u.sub;`meter;`long$til 10) - -Add subscriptions -h(`.u.add;`meter;20 21 22) -``` - -## Running +The publisher will generate some random data and publish it periodically on a timer. The subscriber will receive data from the publisher and output it to the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. From 5c792edb61762e5e2532469eecfa762d6a0973ee Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Thu, 25 Jul 2024 17:08:08 +0100 Subject: [PATCH 09/61] add clarity, remove dups --- docs/architecture/tickq.md | 4 ++-- docs/wp/data-recovery.md | 31 +++++++------------------------ mkdocs.yml | 6 +++--- 3 files changed, 12 insertions(+), 29 deletions(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index 1ffb700e2..a636c46d5 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -200,7 +200,7 @@ Actions performed: * inspect [`.u.d`](#variables), if a new day has occured call [`.z.ts`](#batch-mode_1) * add a new timespan column populated with the current local time ([`.z.P`](../ref/dotz.md#zp-local-timestamp)). If mutiple rows of data, all rows receive the same time. * Add data to current batch (i.e. new data `y` inserted into table `x`), which will be published on batch timer [`.z.ts`](#batch-mode_1). -* If tickerplant log file created, write .u.upd call & params to the log and increment [`.u.j`](#variables) +* If tickerplant log file created, write upd function call & params to the log and increment [`.u.j`](#variables) so that an RDB can execute what was originally called during recovery. #### Realtime Mode @@ -213,7 +213,7 @@ Actions performed: * If the first element of `y` is not a timespan (or list of timespan), add a new timespan column populated with the current local time ([`.z.P`](../ref/dotz.md#zp-local-timestamp)). If mutiple rows of data, all rows receive the same time. * Retrieves the column names of table `x` * Publish data to all interested clients, by calling [`.u.pub`](uq.md#upub) with table name `x` and table generated from `y` and column names. -* If tickerplant log file created, write .u.upd call & params to the log and increment [`.u.i`](#variables) so that data recovery is possible. +* If tickerplant log file created, write upd function call & params to the log and increment [`.u.i`](#variables) so that an RDB can execute what was originally called during recovery ### .z.ts diff --git a/docs/wp/data-recovery.md b/docs/wp/data-recovery.md index 0ae4244e1..d083cef7c 100644 --- a/docs/wp/data-recovery.md +++ b/docs/wp/data-recovery.md @@ -39,16 +39,7 @@ accounts:([] time:`timespan$(); sym:`$(); curr:`$(); action:`$(); limit:`long$() ### Tick scripts -kdb+tick is freely available and contains a few short, yet powerful scripts. - -:fontawesome-brands-github: -[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) - -script | purpose ------------|-------- -`tick.q` | runs a standard tickerplant -`tick/r.q` | runs a standard real-time database -`tick/u.q` | contains functions for subscription and publication +:fontawesome-brands-github:[KxSystems/kdb-tick](https://github.com/KxSystems/kdb-tick) is freely available and contains a few short, yet powerful scripts. :fontawesome-regular-hand-point-right: Starting kdb+: [Realtime database](../learn/startingkdb/tick.md) @@ -74,22 +65,13 @@ Here, `functionname` and `tablename` are symbols, and `tabledata` is a row of da In a standard tick system, each kdb+ message will call a function named `upd`. Each process may have a different definition of this function. A TP will publish each time the `upd` function is called (if its timer is not set to batch the data), and an RDB will simply insert the data into the relevant table. - ## Recovery - ### Writing a tplog Should the TP fail, or be shut down for any period of time, no downstream subscriber will receive any published data for the period of its downtime. This data typically will not be recoverable. Thus it is imperative that the TP remain always running and available. -Every message that the tickerplant receives is written to a kdb+ binary file, called the tickerplant log file, or _tplog_. The tickerplant maintains some key variables which are important in the context of data recovery for subscribers. - -variable | purpose ----------|----------------- -`.u.l` | A handle to the log file which is created at startup. This is used to write each message to disk. -`.u.L` | Path to the log file. In a standard tickerplant, the name of the log file will be a combination of the first parameter passed to `tick.q` (the name of the schema file, generally `sym`) and the current date. -`.u.i` | Total count of messages in the log file. -`.u.j` | Total count of messages in the tickerplant: `.u.i` plus what is buffered in memory. +Every message that the tickerplant receives is written to a kdb+ binary file, called the tickerplant log file, or _tplog_. The tickerplant maintains some key [variables](../architecture/tickq.md#variables) which are important in the context of data recovery for subscribers. ```bash # start tickerplant @@ -102,7 +84,6 @@ q).u.l 376i q).u.i 0 ``` - The `upd` function is called each time a TP receives a message. Within this function, the TP will write the message to the tplog. ```q @@ -117,9 +98,11 @@ if[l; l enlist(`upd;t;x); j+:1] ### Replaying a tplog -Recovery of an RDB typically involves simply restarting the process. On startup, an RDB will subscribe to a TP and will receive information about the message count (`.u.i`) and location of the tplog (`.u.L`) in return. +Recovery of an RDB typically involves simply restarting the process. On startup, an RDB will subscribe to a TP and will receive information about the message count ([`.u.i`](../architecture/tickq.md#variables)) and location of the tplog ([`.u.L`](../architecture/tickq.md#variables)) in return. + +It will then replay this tplog to recover all the data that has passed through the TP up to that point in the day. The replay is achieved using [`-11!`](../basics/internal.md#-11-streaming-execute), the streaming replay function. -It will then replay this tplog to recover all the data that has passed through the TP up to that point in the day. The replay is achieved using `-11!`, the streaming replay function. This is called within `.u.rep`, which is executed when the RDB connects to the TP. +This is called within `.u.rep`, which is executed when the RDB connects to the TP. ```q //from r.q @@ -131,7 +114,7 @@ kdb+ messages were described above in [_kdb+ messages and upd function_](#kdb-me ### `-11!` functionality -`-11!` is an internal function that will read each message in a tplog in turn, running the function `functioname` on the `(tablename;tabledata)` parameters. +[`-11!`](../basics/internal.md#-11-streaming-execute) is an internal function that will read each message in a tplog in turn, running the function `functioname` on the `(tablename;tabledata)` parameters. In this section, we will focus on normal usage of the `-11!` function where the tplog is uncorrupted, and move on to discuss how to use it to recover from a corrupted tplog below in [_Replay errors_](#replay-errors). diff --git a/mkdocs.yml b/mkdocs.yml index 76d79783e..a3a4675e6 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -523,15 +523,15 @@ nav: - FAQ: kb/faq-listbox.md - Streaming: - General architecture: - - Environment: architecture/index.md + - Overview: architecture/index.md - kdb-tick: - Tickerplant (tick.q): architecture/tickq.md - - u.q: architecture/uq.md + - Tickerplant pub/sub (u.q): architecture/uq.md - RDB (r.q): architecture/rq.md - Alternative architecture: kb/kdb-tick.md - Alternative in-memory layouts: kb/alternative-in-memory-layouts.md - Corporate actions: kb/corporate-actions.md - - Data recovery: wp/data-recovery.md + - TP Log (data recovery): wp/data-recovery.md - Disaster recovery: wp/disaster-recovery/index.md - Gateway design: wp/gateway-design/index.md - Profiling: wp/tick-profiling.md From e3a88f846f89b9f946772c3675ae4620bc081894 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Fri, 26 Jul 2024 10:15:11 +0100 Subject: [PATCH 10/61] rename and add links --- docs/wp/rt-tick/index.md | 103 +++++++++++++++++++-------------------- mkdocs.yml | 2 +- 2 files changed, 51 insertions(+), 54 deletions(-) diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index e891aa0e2..fd36c1504 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -1,16 +1,16 @@ --- -title: Building real-time tick subscribers | kdb+ and q documentation -description: How to build a custom real-time tick subscriber +title: Building real-time tick engines | kdb+ and q documentation +description: How to build a custom real-time tick engine author: Nathan Perrem date: August 2014 keywords: kdb+, q, real-time, subscribe, tick --- -# Building real-time tick subscribers +# Building real-time engines by [Nathan Perrem](#author) {: .wp-author} -The purpose of this white paper is to help q developers who wish to build their own custom real-time tick subscribers. KX provides kdb+tick, a tick capture system which includes the core q code for the tickerplant process (`tick.q`) and the vanilla real-time subscriber process (`r.q`), known as the real-time database. This vanilla real-time process subscribes to all tables and to all symbols on the tickerplant. This process has very simple behavior upon incoming updates – it simply inserts these records to the end of the corresponding table. This may be perfectly useful to some clients, however what if the client requires more interesting functionality? For example, the client may need to build or maintain their queries or analytics in real time. How would one take `r.q` and modify it to achieve said behavior? This white paper attempts to help with this task. It breaks down into the following broad sections: +The purpose of this white paper is to help q developers who wish to build their own custom real-time engine. KX provides kdb+tick, a tick capture system which includes the core q code for the tickerplant process ([`tick.q`](../../architecture/tickq.md)) and the vanilla real-time engine process ([`r.q`](../../architecture/rq.md)), known as the real-time database (RDB). This vanilla real-time process subscribes to all tables and to all symbols on the tickerplant. This process has very simple behavior upon incoming updates – it simply inserts these records to the end of the corresponding table. This may be perfectly useful to some clients, however what if the client requires more interesting functionality? For example, the client may need to build or maintain their queries or analytics in real time. How would one take `r.q` and modify it to achieve said behavior? This white paper attempts to help with this task. It breaks down into the following broad sections: 1. Explain the existing code and principles behind `r.q`. 2. Use `r.q` as a template to build some sample real-time analytic @@ -19,19 +19,16 @@ The purpose of this white paper is to help q developers who wish to build their It is hoped this white paper will help dispel any notion of tick being a black box product which cannot be adapted to the requirements of the real-time data consumer. All tests were run using kdb+ V3.1 (2013.09.19) on Windows. -The tickerplant and real-time database scripts can be obtained from GitHub. - -:fontawesome-brands-github: -[KxSystems/kdb+tick](https://github.com/KxSystems/kdb-tick) +The tickerplant and real-time database scripts can be obtained from :fontawesome-brands-github:[KxSystems/kdb+tick](https://github.com/KxSystems/kdb-tick) The tickerplant and real-time database scripts used are dated 2014.03.12 and 2008.09.09 respectively. These are the most up-to-date versions as of the writing of this white paper. -This paper is focused on the real-time database and custom real-time subscribers. However, some background will be provided on the other key processes in this environment. +This paper is focused on the real-time database and custom real-time engines (RTEs). However, some background will be provided on the other key processes in this environment. ## The kdb+tick environment -The real-time database (RDB) and all other real-time subscribers (RTS) do not exist in isolation. Instead they sit downstream of the feedhandler (FH) and tickerplant (TP) processes. The feedhandler feeds data into the tickerplant, which in turns publishes certain records to the real-time database and other real-time subscribers. Today’s data can be queried on the RDB. The historical data resides on disk and can be read into memory upon demand by the historical database process (HDB). +The real-time database (RDB) and all other real-time engines (RTE) do not exist in isolation. Instead they sit downstream of the feedhandler (FH) and tickerplant (TP) processes. The feedhandler feeds data into the tickerplant, which in turns publishes certain records to the real-time database and other RTEs. Today’s data can be queried on the RDB. The historical data resides on disk and can be read into memory upon demand by the historical database process (HDB). The incoming data feed could be from Reuters, Bloomberg, a particular exchange or some other internal data feed. The feedhandler receives this data and extracts the fields of interest. It will also perform some datatype casting and re-ordering of fields to normalize the data set with the corresponding table schemas present on the tickerplant. The feedhandler then pushes this massaged data to the tickerplant. @@ -113,7 +110,7 @@ hdb:.z.x 0 @[{system"l ",x};hdb;{show "Error message - ",x;exit 0}] ``` -Strictly speaking, an instance of the HDB is not required for this paper since all we really need is a tickerplant being fed data and then publishing this data downstream to the RDB and RTS. However, the RDB does communicate with the HDB at end of day once it has finished writing its records to the on-disk database. +Strictly speaking, an instance of the HDB is not required for this paper since all we really need is a tickerplant being fed data and then publishing this data downstream to the RDB and RTE. However, the RDB does communicate with the HDB at end of day once it has finished writing its records to the on-disk database. ## Real-time database (RDB) @@ -132,7 +129,7 @@ argument | semantics ### Real-time updates -Quite simply, the tickerplant provides the ability for a process (in this case the real-time database) to subscribe to certain tables, and for certain symbols (stock tickers, currency pairs etc.). Such a real-time subscriber will subsequently have relevant updates pushed to it by the tickerplant. The tickerplant asynchronously pushes the update as a 3-item list in the format `(upd;Table;data)`: +Quite simply, the tickerplant provides the ability for a process (in this case the real-time database) to subscribe to certain tables, and for certain symbols (stock tickers, currency pairs etc.). Such a RTE will subsequently have relevant updates pushed to it by the tickerplant. The tickerplant asynchronously pushes the update as a 3-item list in the format `(upd;Table;data)`: item | semantics ---------|---------- @@ -161,7 +158,7 @@ Some example updates: size:1000 2000)) ``` -Such a list is received by the real-time subscriber and is implicitly passed to the [`value`](../../ref/value.md) function. Here is a simple example of `value` in action: +Such a list is received by the RTE and is implicitly passed to the [`value`](../../ref/value.md) function. Here is a simple example of `value` in action: ```q q)upd:{:x-y} @@ -169,9 +166,9 @@ q)value (`upd;3;2) 1 ``` -In other words, the real-time subscriber passes two inputs to the function called `upd`. In the above examples, the inputs are the table name `` `trade`` and the table of new records. +In other words, the RTE passes two inputs to the function called `upd`. In the above examples, the inputs are the table name `` `trade`` and the table of new records. -The `upd` function should be defined on the real-time subscriber according to how the process is required to act in the event of an update. Often `upd` is defined as a binary (2-argument) function, but it could alternatively be defined as a dictionary which maps table names to unary function definitions. This duality works because of a fundamental and elegant feature of kdb+: [executing functions and indexing into data structures are equivalent](../../ref/apply.md). For example: +The `upd` function should be defined on the RTE according to how the process is required to act in the event of an update. Often `upd` is defined as a binary (2-argument) function, but it could alternatively be defined as a dictionary which maps table names to unary function definitions. This duality works because of a fundamental and elegant feature of kdb+: [executing functions and indexing into data structures are equivalent](../../ref/apply.md). For example: ```q /define map as a dictionary @@ -190,7 +187,7 @@ q)map[`bar;10] So the developer of the process needs to define `upd` according to their desired behavior. -Perhaps the simplest definition of `upd` is to be found in the vanilla RTS – the RDB. The script for this process is called `r.q` and within this script, we find the definition: +Perhaps the simplest definition of `upd` is to be found in the vanilla RTE – the RDB. The script for this process is called `r.q` and within this script, we find the definition: ```q upd:insert @@ -238,22 +235,22 @@ q)MC 4 ``` -The main challenge in developing a custom real-time subscriber is rewriting `upd` to achieve desired real-time behavior. +The main challenge in developing a custom RTE is rewriting `upd` to achieve desired real-time behavior. ### Tickerplant log replay -An important role of the tickerplant is to maintain a daily logfile on disk for replay purposes. When a real-time subscriber starts up, they could potentially replay this daily logfile, assuming they have read access to it. Such a feature could be useful if the subscriber crashes intraday and is restarted. In this scenario, the process would replay this logfile and then be fully up-to-date. Replaying this logfile, particularly late in the day when the tickerplant has written many messages to it, can take minutes. The exact duration will depend on three factors: +An important role of the tickerplant is to maintain a daily logfile on disk for replay purposes. When a RTE starts up, they could potentially replay this daily logfile, assuming they have read access to it. Such a feature could be useful if the subscriber crashes intraday and is restarted. In this scenario, the process would replay this logfile and then be fully up-to-date. Replaying this logfile, particularly late in the day when the tickerplant has written many messages to it, can take minutes. The exact duration will depend on three factors: 1. How many messages are in the logfile 2. The disk read speed 3. How quickly the process can replay a given message -The first and second factors are probably not controllable by the developer of the RTS. However the third factor is based on the efficiency and complexity of the particular replay function called `upd`. Defining this replay function efficiently is therefore of the upmost importance for quick intraday restarts. +The first and second factors are probably not controllable by the developer of the RTE. However the third factor is based on the efficiency and complexity of the particular replay function called `upd`. Defining this replay function efficiently is therefore of the upmost importance for quick intraday restarts. !!! note "One daily logfile" - The tickerplant maintains just one daily logfile. It does _not_ maintain separate logfiles split across different tables and symbols. This means that an RTS replaying such a logfile may only be interested in a fraction of the messages stored within. + The tickerplant maintains just one daily logfile. It does _not_ maintain separate logfiles split across different tables and symbols. This means that an RTE replaying such a logfile may only be interested in a fraction of the messages stored within. Ultimately the developer must decide if the process truly requires these records from earlier in the day. Changing the tickerplant’s code to allow subscriber specific logfiles should be technically possible, but is beyond the scope of this white paper. @@ -274,15 +271,15 @@ Focusing on the first message: item | semantics -----|----------- -1 | the symbol `` `upd`` is the name of the update/replay function on RTS +1 | the symbol `` `upd`` is the name of the update/replay function on RTE 2 | the symbol `` `trade`` is the table name of the update 3 | a column-oriented (columnar) list containing the new records -The format of the message in the tickerplant logfile is the same as the format of real-time updates sent to the RTS with one _critical_ difference – the data here is a list, _not_ a table. The RTS which wants to replay this logfile will need to define their `upd` to accommodate this list. This will mean in general that an RTS will have two different definitions of `upd` – one for tickerplant logfile replay and another for intraday updates via IPC (interprocess communication). +The format of the message in the tickerplant logfile is the same as the format of real-time updates sent to the RTE with one _critical_ difference – the data here is a list, _not_ a table. The RTE which wants to replay this logfile will need to define their `upd` to accommodate this list. This will mean in general that an RTE will have two different definitions of `upd` – one for tickerplant logfile replay and another for intraday updates via IPC (interprocess communication). For example, a q process with suitable definitions for the tables `trade` and `quote`, as well as the function `upd`, could replay `sym2014.08.23`. Again, a suitable definition for `upd` will depend on the desired behavior, but the function will need to deal with incoming lists as well as tables. -In the RDB (vanilla RTS), `upd` for both replay purposes and intraday update purposes is simply defined as: +In the RDB (vanilla RTE), `upd` for both replay purposes and intraday update purposes is simply defined as: ```q upd:insert @@ -350,8 +347,8 @@ For more information on tickerplant logfile replay, see [“Data Recovery for kd ### End of day -At end of day (EOD), the tickerplant sends messages to all its real-time subscribers, telling them to execute their unary end-of-day function called `.u.end`. The tickerplant supplies a date which is typically the previous day’s -date. When customizing your RTS, define `.u.end` to achieve whatever behavior you deem appropriate at EOD. On the RDB, `.u.end` is defined as follows: +At end of day (EOD), the tickerplant sends messages to all its RTEs, telling them to execute their unary end-of-day function called `.u.end`. The tickerplant supplies a date which is typically the previous day’s +date. When customizing your RTE, define `.u.end` to achieve whatever behavior you deem appropriate at EOD. On the RDB, `.u.end` is defined as follows: ```q / end of day: save, clear, hdb reload @@ -366,7 +363,7 @@ To summarize this behavior: the RDB persists its tables to disk in date-partitio ### Understanding the code in `r.q` -To help you modify `r.q` to create your own custom RTS, this section explains its inner workings. This script starts out with the following: +To help you modify `r.q` to create your own custom RTE, this section explains its inner workings. This script starts out with the following: ```q if[not "w"=first string .z.o;system "sleep 1"]; @@ -566,18 +563,18 @@ Reading this from the right, we obtain the location of the tickerplant process w -## Examples of custom real-time subscribers +## Examples of custom RTEs -Two quite different RTS instances are described below. +Two quite different RTE instances are described below. ### Real-time trade with as-of quotes One of the most popular and powerful joins in the q language is the [`aj`](../../ref/aj.md) function. This keyword was added to the language to solve a specific problem – how to join trade and quote tables together in such a way that for each trade, we grab the prevalent quote _as of_ the time of that trade. In other words, what is the last quote at or prior to the trade? -This function is relatively easy to use for one-off joins. However, what if you want to maintain trades with as-of quotes in real time? This section describes how to build an RTS with real-time trades and as-of quotes. This is a heavily modified version of `r.q`, written by the author and named `RealTimeTradeWithAsofQuotes.q`. +This function is relatively easy to use for one-off joins. However, what if you want to maintain trades with as-of quotes in real time? This section describes how to build an RTE with real-time trades and as-of quotes. This is a heavily modified version of `r.q`, written by the author and named `RealTimeTradeWithAsofQuotes.q`. -One additional feature this script demonstrates is the ability of any q process to write to and maintain its own kdb+ binary logfile for replay/recovery purposes. In this case, the RTS maintains its own daily logfile for trade records. This will be used for recovery in place of the standard tickerplant logfile as used by `r.q`. +One additional feature this script demonstrates is the ability of any q process to write to and maintain its own kdb+ binary logfile for replay/recovery purposes. In this case, the RTE maintains its own daily logfile for trade records. This will be used for recovery in place of the standard tickerplant logfile as used by `r.q`. This process should be started off as follows: @@ -592,7 +589,7 @@ The first section of the script simply parses the command-line arguments and use ```q / The purpose of this script is as follows: -1. Demonstrate how custom real-time subscribers can be created in q +1. Demonstrate how custom RTEs can be created in q 2. In this example, create an efficient engine for calculating the prevalent quotes as of trades in real-time. This removes the need for ad-hoc invocations of the aj function. @@ -624,10 +621,10 @@ The error flag above is set for purely testing purposes – when the developer r #### Initialize desired table schemas -The next section of code defines the behavior of this RTS upon connecting and subscribing to the tickerplant’s trade and quote tables. This function replaces `.u.rep` in `r.q`: +The next section of code defines the behavior of this RTE upon connecting and subscribing to the tickerplant’s trade and quote tables. This function replaces `.u.rep` in `r.q`: ```q -/initialize schemas for custom real-time subscriber +/initialize schemas for custom RTE InitializeSchemas:`trade`quote! ( {[x]`TradeWithQuote insert update bid:0n,bsize:0N,ask:0n,asize:0N from x}; @@ -635,7 +632,7 @@ InitializeSchemas:`trade`quote! ); ``` -The RTS’s trade table (named `TradeWithQuote`) maintains `bid`, `bsize`, `ask` and `asize` columns of appropriate type. For the quote table, we just maintain a keyed table called `LatestQuote`, keyed on `sym` which will maintain the most recent quote per symbol. This table will be used when joining prevalent quotes to incoming trades. +The RTE’s trade table (named `TradeWithQuote`) maintains `bid`, `bsize`, `ask` and `asize` columns of appropriate type. For the quote table, we just maintain a keyed table called `LatestQuote`, keyed on `sym` which will maintain the most recent quote per symbol. This table will be used when joining prevalent quotes to incoming trades. #### Intraday update behavior @@ -654,7 +651,7 @@ updTrade:{[d] LogfileHandle enlist (`replay;`TradeWithQuote;d); } ``` -Besides inserting the new trades with prevalent quote information into the trade table, the above function also appends the new records to its custom logfile. This logfile will be replayed upon recovery/startup of the RTS. Note that the replay function is named `replay`. This differs from the conventional TP logfile where the replay function was called `upd`. +Besides inserting the new trades with prevalent quote information into the trade table, the above function also appends the new records to its custom logfile. This logfile will be replayed upon recovery/startup of the RTE. Note that the replay function is named `replay`. This differs from the conventional TP logfile where the replay function was called `upd`. The next section defines the intraday behavior upon receiving new quotes: @@ -678,7 +675,7 @@ In `r.q`, `upd` is defined as a function, not a dictionary. However we can use t #### End of day -At end of day, the tickerplant sends a message to all real-time subscribers telling them to invoke their EOD function – `.u.end`: +At end of day, the tickerplant sends a message to all RTEs telling them to invoke their EOD function – `.u.end`: ```q /end of day function - triggered by tickerplant at EOD @@ -716,7 +713,7 @@ This function has been heavily modified from `r.q` to achieve the following desi #### Replay custom logfile -This section concerns the initialization and replay of the RTS’s custom logfile. +This section concerns the initialization and replay of the RTE’s custom logfile. ```q /Initialize name of custom logfile @@ -760,7 +757,7 @@ InitializeSchemas . h(".u.sub";`quote;args`syms) The output of a subscription to a given table (for example `trade`) from the tickerplant is a 2-list, as discussed previously. This pair is in turn passed to the function `InitializeSchemas`. -We can see this RTS in action by examining the five most recent trades for `GS.N`: +We can see this RTE in action by examining the five most recent trades for `GS.N`: ```q q)-5#select from TradeWithQuote where sym=`GS.N @@ -776,7 +773,7 @@ time sym price size bid bsize ask asize ### Real-time VWAP subscriber -This section describes how to build an RTS which enriches trade with VWAP (volume-weighted average price) information on a per-symbol basis. A VWAP can be defined as: +This section describes how to build an RTE which enriches trade with VWAP (volume-weighted average price) information on a per-symbol basis. A VWAP can be defined as: $$ VWAP = \frac{\sum_{i} (tradevolume_i)(tradeprice_i)}{\sum_{i} (trade price_i)}$$ @@ -843,7 +840,7 @@ IBM.N | 191.17041 MSFT.O | 45.146982 ``` -Just like the previous RTS example, this solution will comprise a heavily modified version of `r.q`, written by the author and named `real_time_vwap.q`. +Just like the previous RTE example, this solution will comprise a heavily modified version of `r.q`, written by the author and named `real_time_vwap.q`. This process should be started off as follows: @@ -860,7 +857,7 @@ code to the start of `RealTimeTradeWithAsofQuotes.q`. #### Initialize desired table schemas -The next section of code defines the behavior of this RTS upon connecting to the TP and subscribing to the `trade` table. This RTS will replay the TP’s logfile, much like the RDB. The following function replaces `.u.rep`. +The next section of code defines the behavior of this RTE upon connecting to the TP and subscribing to the `trade` table. This RTE will replay the TP’s logfile, much like the RDB. The following function replaces `.u.rep`. ```q /initialize schema function @@ -925,7 +922,7 @@ So whenever a trade update comes in, the VWAP for each affected symbol is update #### End of day -The EOD behavior on this RTS is very simple – clear out the tables: +The EOD behavior on this RTE is very simple – clear out the tables: ```q /end of day function - triggered by tickerplant at EOD @@ -936,7 +933,7 @@ The EOD behavior on this RTS is very simple – clear out the tables: #### Subscribe to TP -The RTS connects to the TP and subscribes to the `trade` table for user specified symbols. The RTS also requests TP logfile information (for replay purposes): +The RTE connects to the TP and subscribes to the `trade` table for user specified symbols. The RTE also requests TP logfile information (for replay purposes): ```q h:hopen args`tp /connect to tickerplant @@ -944,19 +941,19 @@ InitializeTrade . h "(.u.sub[`trade;",(.Q.s1 args`syms),"];`.u `i`L)" upd:updIntraDay /switch upd to intraday update mode ``` -The message returned from the TP is passed to the function `InitializeTrade`. Once the RTS has finished initializing or replaying the TP logfile, the definition of `upd` is then switched to `updIntraDay` so the RTS can deal with intraday updates appropriately. +The message returned from the TP is passed to the function `InitializeTrade`. Once the RTE has finished initializing or replaying the TP logfile, the definition of `upd` is then switched to `updIntraDay` so the RTE can deal with intraday updates appropriately. ## Performance considerations -The developer can build the RTS to achieve whatever real-time behavior is desired. However from a performance perspective, not all RTS instances are equal. The standard RDB is highly performant – meaning it should be able process updates at a very high frequency without maxing out CPU resources. In a real world environment, it is critical that the RTS can finish processing an incoming update before the next one arrives. The high level of RDB performance comes from the fact that its definition of `upd` is extremely simple: +The developer can build the RTE to achieve whatever real-time behavior is desired. However from a performance perspective, not all RTE instances are equal. The standard RDB is highly performant – meaning it should be able process updates at a very high frequency without maxing out CPU resources. In a real world environment, it is critical that the RTE can finish processing an incoming update before the next one arrives. The high level of RDB performance comes from the fact that its definition of `upd` is extremely simple: ```q upd:insert ``` -In other words, for both TP logfile replay and intraday updates, simply insert the records into the table. It doesn’t take much time to execute `insert` in kdb+. However, the two custom RTS instances discussed in this white paper have more complicated definitions of `upd` for intraday updates and will therefore be less performant. This section examines this relative performance. +In other words, for both TP logfile replay and intraday updates, simply insert the records into the table. It doesn’t take much time to execute `insert` in kdb+. However, the two custom RTE instances discussed in this white paper have more complicated definitions of `upd` for intraday updates and will therefore be less performant. This section examines this relative performance. For this test, the TP log will be used. This particular TP logfile has the following characteristics: @@ -1004,7 +1001,7 @@ upd:{[tblName;tblData] ``` This transformed logfile will now be used to test performance on the -RDB and two RTS instances. +RDB and two RTE instances. On the RDB, we obtained the following performance: @@ -1020,7 +1017,7 @@ q)\ts value each logs /execute each update It took 289 milliseconds to process over a quarter of a million updates, where each update had two records. Therefore, the average time taken to process a single two-row update is 1µs. -In the first example RTS (Real-time Trade With As-of Quotes), we obtained the following performance: +In the first example RTE (Real-time Trade With As-of Quotes), we obtained the following performance: ```q q)upd /custom real time update behavior @@ -1038,12 +1035,12 @@ q)\ts value each logs /execute each update It took 2185 milliseconds to process over a quarter of a million updates, where each update had two records. Therefore, the average time taken to process a single two-row update is 7.7 µs – over seven times slower than RDB. -In the second example RTS (Real-time VWAP), we obtained the following performance: +In the second example RTE (Real-time VWAP), we obtained the following performance: ```q / Because there are trades and quotes in the logfile -but this RTS is only designed to handle trades, +but this RTE is only designed to handle trades, a slight change to upd is necessary for the purpose of this performance experiment \ @@ -1059,16 +1056,16 @@ q)\ts value each logs /execute each update It took 9639 milliseconds to process over a quarter of a million updates, where each update had two records. Therefore, the average time taken to process a single two row update is 34 µs – over thirty times slower than RDB. -We can conclude that there was a significant difference in performance in processing updates across the various real-time subscribers. However even in the worst case, assuming the TP updates arrive no more frequently than once every 100 µs, the process should still function well. +We can conclude that there was a significant difference in performance in processing updates across the various RTEs. However even in the worst case, assuming the TP updates arrive no more frequently than once every 100 µs, the process should still function well. It should be noted that prior to this experiment being carried out on each process, all tables were emptied. ## Conclusions -This white paper explained the inner workings of the standard real-time database subscriber as well as an overview of the rest of the kdb+tick environment. The white paper then detailed examples of customizing the RDB to achieve useful real-time analytical behavior. +This white paper explained the inner workings of the standard RDB as well as an overview of the rest of the kdb+tick environment. The white paper then detailed examples of customizing the RDB to achieve useful real-time analytical behavior. -It’s important when building a custom RTS to consider the performance implications of adding complexity to the update logic. The more complex the definition of `upd`, the longer it will take to process intraday updates or replay the TP logfile. In the case of intraday updates, it is important to know the frequency of TP updates in order to know how much complexity you can afford to build into your `upd` function. +It’s important when building a custom RTE to consider the performance implications of adding complexity to the update logic. The more complex the definition of `upd`, the longer it will take to process intraday updates or replay the TP logfile. In the case of intraday updates, it is important to know the frequency of TP updates in order to know how much complexity you can afford to build into your `upd` function. It is the aim of the author that the reader will now have the understanding of how a kdb+tick subscriber can be built and customized fairly easily according to the requirements of the system. diff --git a/mkdocs.yml b/mkdocs.yml index a3a4675e6..668bc73ab 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -540,7 +540,7 @@ nav: - Order Book: wp/order-book.md - Publish and subscribe: kb/publish-subscribe.md - Query Routing: wp/query-routing/index.md - - Real-time tick subscribers: wp/rt-tick/index.md + - Real-time engines: wp/rt-tick/index.md - Advanced: - Distributed systems: wp/query-interface.md - Intraday writedown: wp/intraday-writedown/index.md From a2ea115f00d87721f4c1624303363f0a95762d22 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Fri, 26 Jul 2024 12:49:01 +0100 Subject: [PATCH 11/61] fix links --- docs/interfaces/index.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/docs/interfaces/index.md b/docs/interfaces/index.md index bced80459..2edde30a5 100644 --- a/docs/interfaces/index.md +++ b/docs/interfaces/index.md @@ -39,11 +39,10 @@ Our Fusion interfaces are [automl](https://github.com/KxSystems/automl)[Automate machine learning in kdb+](../ml.md) [cookbook](https://github.com/KxSystems/cookbook)Companion files to the Knowledge Base [help](https://github.com/KxSystems/help)Online **help** for q -[insights-assemblies](https://github.com/KxSystems/insights-assemblies) Deploy assemblies for **KX Insights** ==new== [jupyterq](https://github.com/KxSystems/jupyterq)**Jupyter** kernel for kdb+ [kdb](https://github.com/KxSystems/kdb)Companion files to **kdb+** [kdb-taq](https://github.com/KxSystems/kdb-taq)Processing **trade-and-quote** data -[kdb-tick](https://github.com/KxSystems/kdb-tick)[**Ticker**plant](../kb/kdb-tick.md) +[kdb-tick](https://github.com/KxSystems/kdb-tick)[Tickerplant](../architecture/index.md) [man](https://github.com/KxSystems/man)[man-style reference](../about/man.md) [ml](https://github.com/KxSystems/ml)[**Machine Learning** Toolkit](../ml.md) [mlnotebooks](https://github.com/KxSystems/mlnotebooks)Jupyter notebooks with ML examples From d9124a5ed9ccb650206c6e9096f7b3c746db94db Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Fri, 26 Jul 2024 13:54:28 +0100 Subject: [PATCH 12/61] publish-subscribe page now covered by u.q --- docs/architecture/index.md | 2 +- docs/architecture/uq.md | 36 ++++++++++++++++++++++++++++++++++- docs/kb/publish-subscribe.md | 37 ------------------------------------ docs/wp/capi/index.md | 13 +++++-------- mkdocs.yml | 1 - 5 files changed, 41 insertions(+), 48 deletions(-) delete mode 100644 docs/kb/publish-subscribe.md diff --git a/docs/architecture/index.md b/docs/architecture/index.md index 7426f63db..bf223221f 100644 --- a/docs/architecture/index.md +++ b/docs/architecture/index.md @@ -29,7 +29,7 @@ KX’s [Fusion interfaces](../interfaces/index.md#fusion-interfaces) connect kdb ### Tickerplant (TP) -A kdb+ processing acting as a TP (tickerplant) captures the initial data feed, writes it to the log file and [publishes](../kb/publish-subscribe.md) these messages to any registered subscribers. +A kdb+ processing acting as a TP (tickerplant) captures the initial data feed, writes it to the log file and publishes these messages to any registered subscribers. Aims for zero-latency. Includes ingesting data in batch mode. diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 7d5f42a5f..96e31a175 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -13,6 +13,24 @@ Contains functions to allow clients to subscribe to all or subsets of available This script is loaded by other processes, for example [a tickerplant](tickq.md). +## Usage + +To give the ability to publish data to any process, a few things need to be done: + +- load `u.q` +- declare the tables to be published in the top level namespace. Each table must contain a column called `sym`, which acts as the single key field to which subscribers subscribe +- initialize by calling [`.u.init[]`](#uinit) +- publish data by calling [`.u.pub[table name; table data]`](#upub) + +The list of tables that can be published and the processes currently subscribed are held in [`.u.w`](#variables). + +Subscriber processes must open a connection to the publisher and call [`.u.sub[tablename;list_of_symbols_to_subscribe_to]`](#usub). + +If a subscriber calls `.u.sub` again, the current subscription will be overwritten either for all tables (if a wildcard is used) or the specified table. +To add to a subscription (e.g. add more syms to a current subscription) the subscriber can call [`.u.add`](#uadd)). + +Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u.end`](rq.md#uend) function for end-of-day events + ## Variables | Name | Description | @@ -32,7 +50,7 @@ Initialise variables used to track registered clients. .u.init[] ``` -Initialises [variables](#variables) used to track client interest in data being published. +Initialises [variables](#variables) by retreiving all tables defined in the root namespace. Used to track client interest in data being published. ### .u.del @@ -142,3 +160,19 @@ Implementation of [`.z.pc`](../ref/dotz.md#zpc-close) callback for connection cl Called when a client disconnects. The client handle provided is used to call [`.u.del`](#udel) for all tables. This ensures all subscriptions are removed for that client. +## Example + +[`tick.q`](tickq.md) is an example of a tickerplant that uses `u.q` for pub/sub. + +In addition, the example scripts below demonstrate pub/sub in a standalone publisher and subscriber. +They can be downloaded from :fontawesome-brands-github:[KxSystems/cookbook/pubsub](https://github.com/KxSystems/cookbook/tree/master/pubsub). +Each script should be run from the OS command prompt e.g. + +```bash +$ q publisher.q +$ q subscriber.q +``` + +The publisher will generate some random data and publish it periodically on a timer. + +The subscriber will receive data from the publisher and output it to the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. diff --git a/docs/kb/publish-subscribe.md b/docs/kb/publish-subscribe.md deleted file mode 100644 index 7b5b37bc8..000000000 --- a/docs/kb/publish-subscribe.md +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: Publish and subscribe – Knowledge Base – kdb+ and q documentation -description: KxSystems/kdb-tick contains functionality to allow processes to publish data and subscribe to it. It is worth highlighting how the publish-and-subscribe code can be used by any process on a standalone basis. The pubsub functionality is supplied in the u.q script of kdb+tick. -keywords: kdb+, publish, q, subscribe ---- -# Publish and subscribe - -## Setup - -[`.u.q`](../architecture/uq.md) contains functionality to allow processes to publish data and subscribe to it. -Although commonly used by tickerplants, It is worth highlighting how the publish-and-subscribe code can be used by any process on a standalone basis. - -To give the ability to publish data to any process, a few things need to be done: - -- load `u.q` -- declare the tables to be published in the top level namespace. Each table must contain a column called `sym`, which acts as the single key field to which subscribers subscribe -- initialize by calling [`.u.init[]`](../architecture/uq.md#uinit) -- publish data by calling [`.u.pub[table name; table data]`](../architecture/uq.md#upub) - -The list of tables that can be published and the processes currently subscribed are held in [`.u.w`](../architecture/uq.md#variables). - -Subscriber processes must open a connection to the publisher and call [`.u.sub[tablename;list_of_symbols_to_subscribe_to]`](../architecture/uq.md#usub). - -If a subscriber calls `.u.sub` again, the current subscription will be overwritten either for all tables (if a wildcard is used) or the specified table. To add to a subscription (e.g. add more syms to a current subscription) the subscriber can call [`.u.add`](../architecture/uq.md#uadd)). - -## Example - -The example scripts below can be downloaded from :fontawesome-brands-github:[KxSystems/cookbook/pubsub](https://github.com/KxSystems/cookbook/tree/master/pubsub). Each script should be run from the OS command prompt e.g. - -```bash -$ q publisher.q -$ q subscriber.q -``` - -The publisher will generate some random data and publish it periodically on a timer. - -The subscriber will receive data from the publisher and output it to the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. diff --git a/docs/wp/capi/index.md b/docs/wp/capi/index.md index 856e567fd..611c9e0d8 100644 --- a/docs/wp/capi/index.md +++ b/docs/wp/capi/index.md @@ -1307,7 +1307,7 @@ q)csha256"kx tech" ### Subscribing to a kdb+ tickerplant -A kdb+ tickerplant is a kdb+ process specifically designed to handle incoming, high-frequency, data feeds from publishing processes. The primary responsibility of the tickerplant is to manage subscription requests and publish data quickly to its subscribers. In the vanilla kdb+ setup, illustrated below, the real-time database (RDB) and chained tickerplant kdb+ processes are the most common type of subscriber. However, C applications are also possible using the API. +A kdb+ [tickerplant](../../architecture/index.md) is a kdb+ process specifically designed to handle incoming, high-frequency, data feeds from publishing processes. The primary responsibility of the tickerplant is to manage subscription requests and publish data quickly to its subscribers. In the vanilla kdb+ setup, illustrated below, the real-time database (RDB) and chained tickerplant kdb+ processes are the most common type of subscriber. However, C applications are also possible using the API. ![Architecture](img/architecture.png) @@ -1341,7 +1341,7 @@ The tickerplant process is started from the command line as follows. $ q tick.q trade /logs/tickerplant/ -p 5010 ``` -Above, the first argument following `tick.q` is the name of the table schema file to use. The second argument is the location where the tickerplant log file will be created and the value following the `-p` option is the port the tickerplant will listen on. The C process will use this port number when initializing the connection. The final step in this setup is to create a kdb+ mock feedhandler process which will act as the trade data source for the tickerplant. Below is a simple publishing process which is sufficient for the demonstration. +Above, the first argument following [`tick.q`](../../architecture/tickq.md) is the name of the table schema file to use. The second argument is the location where the tickerplant log file will be created and the value following the `-p` option is the port the tickerplant will listen on. The C process will use this port number when initializing the connection. The final step in this setup is to create a kdb+ mock feedhandler process which will act as the trade data source for the tickerplant. Below is a simple publishing process which is sufficient for the demonstration. ```q /* File name: feed.q */ @@ -1368,10 +1368,7 @@ Once the tickerplant and feedhandler processes are up and running the C subscrib Subscriber processes are required to make an initial subscribe request to the tickerplant in order to receive data. -:fontawesome-regular-hand-point-right: -Knowledge Base: [Publish and subscribe](../../kb/publish-subscribe.md) - -This request involves calling the `.u.sub` function with two parameter arguments. The first argument is the table name, and the second is the list of symbols to subscribe to. +This request involves calling the [`.u.sub`](../../architecture/uq.md#usub) function with two parameter arguments. The first argument is the table name, and the second is the list of symbols to subscribe to. Specifying a backtick character for either parameter of `.u.sub` means _all_, as in, all tables and/or all symbols. If the `.u.sub` function is called synchronously the tickerplant will return the table schema. The following program demonstrates how the initial subscribe request can be made and the column names of table schema extracted. In the case below, a request is made for all trade records. @@ -1564,7 +1561,7 @@ Handle value : The integer value returned by the `khpu` function when a socket connection is established. -Update function name `.u.upd` +Update function name [`.u.upd`](../../architecture/tickq.md#uupd) : The function executed on the tickerplant which enables the data insertion. It may be called synchronously or asynchronously and takes two arguments, as follow. @@ -1588,7 +1585,7 @@ k(handle,".u.upd",r1(tableName),mixedList,(K)0) ### Publishing a single row using a mixed-list object -The next example shows how a `K` object, containing a mixed list corresponding to one row of data, can be passed to the `.u.upd` function. Below the `knk` function is used to create the mixed list containing a symbol, float and integer, constituting a single row. The function `.u.upd` is called in the `k` function with two parameters being passed, a symbol object corresponding to the table name and the singleRow object. +The next example shows how a `K` object, containing a mixed list corresponding to one row of data, can be passed to the [`.u.upd`](../../architecture/tickq.md#uupd) function. Below the `knk` function is used to create the mixed list containing a symbol, float and integer, constituting a single row. The function `.u.upd` is called in the `k` function with two parameters being passed, a symbol object corresponding to the table name and the singleRow object. ```c /* File name: singleRow.c */ diff --git a/mkdocs.yml b/mkdocs.yml index 668bc73ab..34f261cf0 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -538,7 +538,6 @@ nav: - Kubernetes: 'https://youtu.be/jqtkkCqBvr4' - Load balancing: kb/load-balancing.md - Order Book: wp/order-book.md - - Publish and subscribe: kb/publish-subscribe.md - Query Routing: wp/query-routing/index.md - Real-time engines: wp/rt-tick/index.md - Advanced: From c610947b0e51e4f047f460fbc0daac41a14fd500 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Sun, 28 Jul 2024 14:27:57 +0100 Subject: [PATCH 13/61] add details of multithreaded primatives --- docs/ref/abs.md | 6 ++++-- docs/ref/ceiling.md | 3 ++- docs/ref/exp.md | 1 + docs/ref/floor.md | 4 +++- docs/ref/log.md | 2 ++ docs/ref/mod.md | 2 ++ docs/ref/neg.md | 2 ++ docs/ref/not.md | 2 ++ docs/ref/null.md | 2 ++ docs/ref/reciprocal.md | 2 ++ docs/ref/within.md | 2 ++ docs/ref/xbar.md | 2 +- 12 files changed, 25 insertions(+), 5 deletions(-) diff --git a/docs/ref/abs.md b/docs/ref/abs.md index e65d70b67..b89ab8993 100644 --- a/docs/ref/abs.md +++ b/docs/ref/abs.md @@ -16,7 +16,7 @@ abs x abs[x] Where `x` is a numeric or temporal, returns the absolute value of `x`. -Null is returned if `x` is null. +Null is returned if `x` is null. ```q q)abs -1.0 @@ -25,6 +25,8 @@ q)abs 10 -43 0N 10 43 0N ``` +`abs` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration @@ -53,4 +55,4 @@ Range: `ihjefpmdznuvt` [`signum`](signum.md)
:fontawesome-solid-book-open: -[Mathematics](../basics/math.md) \ No newline at end of file +[Mathematics](../basics/math.md) diff --git a/docs/ref/ceiling.md b/docs/ref/ceiling.md index f62b4ccad..c4bd0815e 100644 --- a/docs/ref/ceiling.md +++ b/docs/ref/ceiling.md @@ -20,6 +20,7 @@ q)ceiling 01b 0 1i ``` +`ceiling` is a [multithreaded primitive](../kb/mt-primitives.md). ## :fontawesome-solid-sitemap: Implicit iteration @@ -74,4 +75,4 @@ Range: `hij` [`floor`](floor.md)
:fontawesome-solid-book-open: -[Mathematics](../basics/math.md) \ No newline at end of file +[Mathematics](../basics/math.md) diff --git a/docs/ref/exp.md b/docs/ref/exp.md index f0eb372ce..c1a77cfff 100644 --- a/docs/ref/exp.md +++ b/docs/ref/exp.md @@ -39,6 +39,7 @@ q)exp 00:00:00 00:00:12 12:00:00 1 162754.8 0w ``` +`exp` is a [multithreaded primitive](../kb/mt-primitives.md). ### :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/floor.md b/docs/ref/floor.md index bf7dd3618..6ec43e241 100644 --- a/docs/ref/floor.md +++ b/docs/ref/floor.md @@ -21,6 +21,8 @@ q)floor -2.1 0 2.1 -3 0 2 ``` +`floor` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration @@ -80,4 +82,4 @@ Range: `hijcs` [`ceiling`](ceiling.md)
:fontawesome-solid-book-open: -[Mathematics](../basics/math.md) \ No newline at end of file +[Mathematics](../basics/math.md) diff --git a/docs/ref/log.md b/docs/ref/log.md index 2d52f521e..af9a67311 100644 --- a/docs/ref/log.md +++ b/docs/ref/log.md @@ -35,6 +35,8 @@ q)log -2 0n 0 0.1 1 42 0n 0n -0w -2.302585 0 3.73767 ``` +`log` is a [multithreaded primitive](../kb/mt-primitives.md). + ### :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/mod.md b/docs/ref/mod.md index fdf85361d..643e747a3 100644 --- a/docs/ref/mod.md +++ b/docs/ref/mod.md @@ -28,6 +28,8 @@ q)-7 7 mod/:\:-2.5 -2 2 2.5 -0.5 -1 1 2 ``` +`mod` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/neg.md b/docs/ref/neg.md index bf3aa57aa..559bc11bb 100644 --- a/docs/ref/neg.md +++ b/docs/ref/neg.md @@ -34,6 +34,8 @@ q)neg 2000.01.01 2012.01.01 / negates the underlying data value An atomic function. +`neg` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Domain and range diff --git a/docs/ref/not.md b/docs/ref/not.md index d232b20f0..5285df4be 100644 --- a/docs/ref/not.md +++ b/docs/ref/not.md @@ -46,6 +46,8 @@ q)not (0W;-0w;0N) An atomic function. +`not` is a [multithreaded primitive](../kb/mt-primitives.md). + --- :fontawesome-solid-book: [`neg`](neg.md) diff --git a/docs/ref/null.md b/docs/ref/null.md index 44c0ab6f4..8a11d7b46 100644 --- a/docs/ref/null.md +++ b/docs/ref/null.md @@ -52,4 +52,6 @@ q)\ts `=v 66 268435648 ``` +`null` is a [multithreaded primitive](../kb/mt-primitives.md). + diff --git a/docs/ref/reciprocal.md b/docs/ref/reciprocal.md index 8949b311e..2af39afbf 100644 --- a/docs/ref/reciprocal.md +++ b/docs/ref/reciprocal.md @@ -25,6 +25,8 @@ q)reciprocal 1b 1f ``` +`reciprocal` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration `reciprocal` is an [atomic function](../basics/atomic.md). diff --git a/docs/ref/within.md b/docs/ref/within.md index 9479a9bcc..6a1073f0b 100644 --- a/docs/ref/within.md +++ b/docs/ref/within.md @@ -47,6 +47,8 @@ q)(1 3 10 6 4;"acyxmpu") within ((2;"b");(6;"r")) `within` uses [Find](find.md) to search for `x` in `y`. +`within` is a [multithreaded primitive](../kb/mt-primitives.md). + ---- :fontawesome-solid-book: diff --git a/docs/ref/xbar.md b/docs/ref/xbar.md index d81d9f350..5041b8b46 100644 --- a/docs/ref/xbar.md +++ b/docs/ref/xbar.md @@ -18,7 +18,7 @@ Where - `x` is a non-negative numeric atom - `y` is numeric or temporal -returns `y` rounded down to the nearest multiple of `x`. +returns `y` rounded down to the nearest multiple of `x`. `xbar` is a [multithreaded primitive](../kb/mt-primitives.md). ```q q)3 xbar til 16 From 772992b0fbe7df3465101374621f7f55d11c790c Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Sun, 28 Jul 2024 14:41:06 +0100 Subject: [PATCH 14/61] add details of multithreaded primative --- docs/ref/all-any.md | 6 +++++- docs/ref/and.md | 2 ++ docs/ref/avg.md | 4 ++++ docs/ref/exp.md | 1 + docs/ref/log.md | 2 ++ docs/ref/max.md | 1 + docs/ref/min.md | 2 ++ docs/ref/signum.md | 4 +++- docs/ref/sqrt.md | 4 +++- docs/ref/tan.md | 2 ++ 10 files changed, 25 insertions(+), 3 deletions(-) diff --git a/docs/ref/all-any.md b/docs/ref/all-any.md index 096d4140e..419a8f9af 100644 --- a/docs/ref/all-any.md +++ b/docs/ref/all-any.md @@ -60,6 +60,7 @@ domain: b g x h i j e f c s p m d z n u v t range: b . b b b b b b b . b b b b b b b b ``` +`all` is a [multithreaded primitive](../kb/mt-primitives.md). ## `any` @@ -99,6 +100,9 @@ q)if[any x in y;....] / use in control structure domain: b g x h i j e f c s p m d z n u v t range: b . b b b b b b b . b b b b b b b b ``` + +`any` is a [multithreaded primitive](../kb/mt-primitives.md). + ---- :fontawesome-solid-book: @@ -109,4 +113,4 @@ range: b . b b b b b b b . b b b b b b b b [`min`](min.md)
:fontawesome-solid-book-open: -[Logic](../basics/by-topic.md#logic) \ No newline at end of file +[Logic](../basics/by-topic.md#logic) diff --git a/docs/ref/and.md b/docs/ref/and.md index e8299ed1a..9840275fd 100644 --- a/docs/ref/and.md +++ b/docs/ref/and.md @@ -9,6 +9,8 @@ author: Stephen Taylor _Lesser of two values, logical AND_ +`and` is a [multithreaded primitive](../kb/mt-primitives.md). + :fontawesome-solid-book: [Lesser](lesser.md) diff --git a/docs/ref/avg.md b/docs/ref/avg.md index fc9ea8674..10d0a35bc 100644 --- a/docs/ref/avg.md +++ b/docs/ref/avg.md @@ -53,6 +53,8 @@ domain: b g x h i j e f c s p m d z n u v t range: f . f f f f f f f . f f f f f f f f ``` +`avg` is a [multithreaded primitive](../kb/mt-primitives.md). + ## `avgs` _Running averages_ @@ -198,6 +200,8 @@ t | f . f f f f f f f . f f f f f f f f Range: `f` +`wavg` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/exp.md b/docs/ref/exp.md index c1a77cfff..3da94cc5e 100644 --- a/docs/ref/exp.md +++ b/docs/ref/exp.md @@ -124,6 +124,7 @@ q)1.5 xexp -4.2 0 0.1 0n 0w 7.9999999999999982 ``` +`xexp` is a [multithreaded primitive](../kb/mt-primitives.md). ### :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/log.md b/docs/ref/log.md index af9a67311..c1bc384c2 100644 --- a/docs/ref/log.md +++ b/docs/ref/log.md @@ -112,6 +112,8 @@ q)"A"xlog"C" 1.00726 ``` +`xlog` is a [multithreaded primitive](../kb/mt-primitives.md). + ### :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/max.md b/docs/ref/max.md index 7c7be8d5c..c8a236173 100644 --- a/docs/ref/max.md +++ b/docs/ref/max.md @@ -41,6 +41,7 @@ domain: b g x h i j e f c s p m d z n u v t range: b . x h i j e f c . p m d z n u v t ``` +`max` is a [multithreaded primitive](../kb/mt-primitives.md). ## `maxs` diff --git a/docs/ref/min.md b/docs/ref/min.md index e0e932f4e..8a8a556cf 100644 --- a/docs/ref/min.md +++ b/docs/ref/min.md @@ -37,6 +37,8 @@ q)select min price by sym from t / use in a select statement `min` is an aggregate function, equivalent to `&/`. +`min` is a [multithreaded primitive](../kb/mt-primitives.md). + ## `mins` diff --git a/docs/ref/signum.md b/docs/ref/signum.md index cafe76299..59b5319f7 100644 --- a/docs/ref/signum.md +++ b/docs/ref/signum.md @@ -35,6 +35,8 @@ Find counts of price movements by direction: select count i by signum deltas price from trade ``` +`signum` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration @@ -81,4 +83,4 @@ Range: `i` [`abs`](abs.md)
:fontawesome-solid-book-open: -[Mathematics](../basics/math.md) \ No newline at end of file +[Mathematics](../basics/math.md) diff --git a/docs/ref/sqrt.md b/docs/ref/sqrt.md index d5524eb91..be813a433 100644 --- a/docs/ref/sqrt.md +++ b/docs/ref/sqrt.md @@ -35,6 +35,8 @@ q)sqrt 101b 1 0 1f ``` +`sqrt` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration @@ -83,4 +85,4 @@ Range: `fz` [`xlog`](log.md#xlog)
:fontawesome-solid-book-open: -[Mathematics](../basics/math.md) \ No newline at end of file +[Mathematics](../basics/math.md) diff --git a/docs/ref/tan.md b/docs/ref/tan.md index 8bd20441c..a23a50544 100644 --- a/docs/ref/tan.md +++ b/docs/ref/tan.md @@ -37,6 +37,8 @@ q)atan 42 1.546991 ``` +`tan` and `atan` are [multithreaded primitives](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration From f4a2bc80263c16f6bceeb07b56197c9e29444123 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Sun, 28 Jul 2024 19:28:41 +0100 Subject: [PATCH 15/61] add multithread primative details --- docs/ref/cor.md | 2 ++ docs/ref/cos.md | 2 ++ docs/ref/cov.md | 4 +++- docs/ref/dev.md | 3 +++ docs/ref/differ.md | 4 +++- docs/ref/distinct.md | 4 +++- docs/ref/div.md | 2 ++ docs/ref/or.md | 2 ++ docs/ref/sin.md | 2 ++ docs/ref/sum.md | 1 + docs/ref/take.md | 2 ++ docs/ref/var.md | 7 ++++--- 12 files changed, 29 insertions(+), 6 deletions(-) diff --git a/docs/ref/cor.md b/docs/ref/cor.md index 8378f53d1..66eb57330 100644 --- a/docs/ref/cor.md +++ b/docs/ref/cor.md @@ -32,6 +32,8 @@ q)1000101000b cor 0010011001b `cor` is an aggregate function, equivalent to `{cov[x;y]%dev[x]*dev y}`. +`cor` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Domain and range diff --git a/docs/ref/cos.md b/docs/ref/cos.md index 02c804df3..cd280a862 100644 --- a/docs/ref/cos.md +++ b/docs/ref/cos.md @@ -41,6 +41,8 @@ q)acos -0.4 / arccosine 1.982313 ``` +`cos` and `acos` are [multithreaded primitives](../kb/mt-primitives.md). + ## Domain and range diff --git a/docs/ref/cov.md b/docs/ref/cov.md index e5d22f531..38c1e68c6 100644 --- a/docs/ref/cov.md +++ b/docs/ref/cov.md @@ -56,7 +56,7 @@ t | f . f f f f f f f . f f f f f f f f Range: `f` - +`cov` is a [multithreaded primitive](../kb/mt-primitives.md). ## `scov` @@ -110,6 +110,8 @@ t | f . f f f f f f f . f f f f f f f f Range: `f` +`scov` is a [multithreaded primitive](../kb/mt-primitives.md). + ---- :fontawesome-solid-book: diff --git a/docs/ref/dev.md b/docs/ref/dev.md index fd3eccbd3..3467f77f9 100644 --- a/docs/ref/dev.md +++ b/docs/ref/dev.md @@ -50,6 +50,7 @@ a| 1.247219 b| 2 ``` +`dev` is a [multithreaded primitive](../kb/mt-primitives.md). ## `mdev` @@ -184,6 +185,8 @@ a| 1.527525 b| 2.828427 ``` +`sdev` is a [multithreaded primitive](../kb/mt-primitives.md). + ---- :fontawesome-solid-book: [`var`, `svar`](var.md) diff --git a/docs/ref/differ.md b/docs/ref/differ.md index 3f9bfcb54..67b7f1516 100644 --- a/docs/ref/differ.md +++ b/docs/ref/differ.md @@ -59,6 +59,8 @@ domain: b g x h i j e f c s p m d z n u v t range: b b b b b b b b b b b b b b b b b b ``` +`differ` is a [multithreaded primitive](../kb/mt-primitives.md). + ??? warning "Binary use deprecated" As of V3.6 the keyword is [variadic](../basics/variadic.md). @@ -69,4 +71,4 @@ range: b b b b b b b b b b b b b b b b b b --- :fontawesome-regular-hand-point-right: -Basics: [Comparison](../basics/comparison.md) \ No newline at end of file +Basics: [Comparison](../basics/comparison.md) diff --git a/docs/ref/distinct.md b/docs/ref/distinct.md index e1a2883d9..25e018ac8 100644 --- a/docs/ref/distinct.md +++ b/docs/ref/distinct.md @@ -40,6 +40,8 @@ q)distinct 2 + 0f,10 xexp -13 2 2.0000000000001 ``` +`distinct` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Errors @@ -55,4 +57,4 @@ type | `x` is an atom
:fontawesome-solid-book-open: [Precision](../basics/precision.md), -[Search](../basics/by-topic.md#search) \ No newline at end of file +[Search](../basics/by-topic.md#search) diff --git a/docs/ref/div.md b/docs/ref/div.md index 79aedbe32..2425e0d77 100644 --- a/docs/ref/div.md +++ b/docs/ref/div.md @@ -51,6 +51,8 @@ q)"\023" div 8 2i ``` +`div` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/or.md b/docs/ref/or.md index bb6b313bb..b7a4de491 100644 --- a/docs/ref/or.md +++ b/docs/ref/or.md @@ -8,6 +8,8 @@ keywords: and, greater, kdb+, logic, or, q _Greater of two values, logical OR_ +`or` is a [multithreaded primitive](../kb/mt-primitives.md). + :fontawesome-regular-hand-point-right: diff --git a/docs/ref/sin.md b/docs/ref/sin.md index 94dcc955e..170f1e0bd 100644 --- a/docs/ref/sin.md +++ b/docs/ref/sin.md @@ -35,6 +35,8 @@ q)asin 0.8 / arcsine 0.9272952 ``` +`sin` and `asin` are [multithreaded primitives](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/sum.md b/docs/ref/sum.md index 1ae364b9d..541b79460 100644 --- a/docs/ref/sum.md +++ b/docs/ref/sum.md @@ -80,6 +80,7 @@ q)sum each flip(0n 8;8 0n) /do this to fall back to vector case q)sum a 49999897.181933172 +`sum` is a [multithreaded primitive](../kb/mt-primitives.md). ## `sums` diff --git a/docs/ref/take.md b/docs/ref/take.md index 105c9dc0b..365ecb7f2 100644 --- a/docs/ref/take.md +++ b/docs/ref/take.md @@ -23,6 +23,8 @@ Where returns `y` as a list, dictionary or table described or selected by `x`. +`#` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Atom or list diff --git a/docs/ref/var.md b/docs/ref/var.md index 5bcc90103..d7ba9815d 100644 --- a/docs/ref/var.md +++ b/docs/ref/var.md @@ -8,9 +8,6 @@ author: Stephen Taylor _Variance, sample variance_ - - - ## `var` _Variance_ @@ -51,6 +48,8 @@ a| 1.555556 b| 4 ``` +`var` is a [multithreaded primitive](../kb/mt-primitives.md). + ## `svar` @@ -91,6 +90,8 @@ a| 2.333333 b| 8 ``` +`svar` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Domain and range From 377551a5cdd043d7989cb014878280d9feaa7f62 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Sun, 28 Jul 2024 19:35:05 +0100 Subject: [PATCH 16/61] add multithreaded primitive details --- docs/ref/aj.md | 2 ++ docs/ref/asof.md | 1 + docs/ref/bin.md | 2 ++ docs/ref/ij.md | 2 ++ docs/ref/in.md | 2 ++ docs/ref/lj.md | 2 ++ docs/ref/til.md | 2 ++ docs/ref/uj.md | 2 ++ 8 files changed, 15 insertions(+) diff --git a/docs/ref/aj.md b/docs/ref/aj.md index 445eefd80..ec195b597 100644 --- a/docs/ref/aj.md +++ b/docs/ref/aj.md @@ -57,6 +57,8 @@ time sym qty px 10:01:04 ge 150 ``` +`aj` is a [multithreaded primitive](../kb/mt-primitives.md). + !!! tip "There is no requirement for any of the join columns to be keys but the join will be faster on keys." diff --git a/docs/ref/asof.md b/docs/ref/asof.md index 2838fe913..259aca432 100644 --- a/docs/ref/asof.md +++ b/docs/ref/asof.md @@ -68,6 +68,7 @@ A 1996.05.23| 046298105 ASTRA AB CL-A ADS 1CL-ASEK2.50 0 N 100 A 2000.08.04| 00846U101 AGILENT TECHNOLOGIES INC 0 N 100 ``` +`asof` is a [multithreaded primitive](../kb/mt-primitives.md). ---- :fontawesome-solid-book: diff --git a/docs/ref/bin.md b/docs/ref/bin.md index 54f427be3..bf5bd680e 100644 --- a/docs/ref/bin.md +++ b/docs/ref/bin.md @@ -48,6 +48,8 @@ Essentially `bin` gives a half-open interval on the left. `bin` also operates on tuples and table columns and is the function used in [`aj`](aj.md) and [`lj`](lj.md). +`bin` and `binr` are [multithreaded primitives](../kb/mt-primitives.md). + !!! danger "If `x` is not sorted the result is undefined." diff --git a/docs/ref/ij.md b/docs/ref/ij.md index fe8d30ae4..5e01d7d04 100644 --- a/docs/ref/ij.md +++ b/docs/ref/ij.md @@ -59,6 +59,8 @@ k v s 4 400 c ``` +`ij` is a [multithreaded primitive](../kb/mt-primitives.md). + !!! detail "Changes in V3.0" Since V3.0, `ij` has changed behavior (similarly to `lj`): when there are nulls in `y`, `ij` uses the `y` null, where the earlier version left the corresponding value in `x` unchanged: diff --git a/docs/ref/in.md b/docs/ref/in.md index 7ad158587..36a332bec 100644 --- a/docs/ref/in.md +++ b/docs/ref/in.md @@ -62,6 +62,8 @@ q)(1 2;3 4) in ((1 2;3 4);9) / x is an item of y `in` uses [Find](find.md) to search for `x` in `y`. +`in` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Queries diff --git a/docs/ref/lj.md b/docs/ref/lj.md index 93643f7cb..69dbb0f9b 100644 --- a/docs/ref/lj.md +++ b/docs/ref/lj.md @@ -59,6 +59,8 @@ c d 2 20 ``` +`lj` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Changes in V4.0 diff --git a/docs/ref/til.md b/docs/ref/til.md index c44a82a73..464e09c97 100644 --- a/docs/ref/til.md +++ b/docs/ref/til.md @@ -31,6 +31,8 @@ q)til 5f `til` and [`key`](key.md) are synonyms, but the above usage is conventionally reserved to `til`. +`til` is a [multithreaded primitive](../kb/mt-primitives.md). + ---- :fontawesome-solid-book-open: [Mathematics](../basics/math.md) diff --git a/docs/ref/uj.md b/docs/ref/uj.md index 798fba633..5b4df10aa 100644 --- a/docs/ref/uj.md +++ b/docs/ref/uj.md @@ -58,6 +58,8 @@ a b| c d 3 7| 30 C ``` +`uj` is a [multithreaded primitive](../kb/mt-primitives.md). + !!! note "`uj` generalizes the [`,` Join](join.md) operator." From 22036009fcaf73765abdc9c711dcb3a50716a266 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Sun, 28 Jul 2024 19:47:27 +0100 Subject: [PATCH 17/61] tickerplant/rdb intro --- docs/architecture/rq.md | 4 ++++ docs/architecture/tickq.md | 4 ++++ 2 files changed, 8 insertions(+) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 852a85489..6e9cdf8cd 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -9,6 +9,10 @@ keywords: kdb+, q, rdb, streaming ## Overview +A kdb+ process acting as an RDB stores a current day’s data in-memory for client queries. +It can write its contents to disk at end-of-day, clearing out it in-memory data to prepare for the next day. +After writting data to disk, it will instruct a HDB to load the written data. + ### Customization `r.q` provides a starting point to most environments. The source code is freely avaialble and can be tailered to individual needs. For example: diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index a636c46d5..0f46d12e4 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -9,12 +9,16 @@ keywords: hdb, kdb+, q, tick, tickerplant, streaming ## Overview +All incoming streaming data is processed by a kdb+ process acting as a tickerplant. +A tickerplant writes all data to a tickerplant log (to permit data recovery) and publishes data to subscribed clients, for example a RDB. + ### Customization `tick.q` provides a starting point to most environments. The source code is freely avaialble and can be tailered to individual needs. ### Schema file +A tickerplant requires a schema file. A schema file should be created to describe the data you plan to capture, by specifing the tables that will be populated by the tickerplant environment. The [datatypes](../basics/datatypes.md) and [attributes](../ref/set-attribute.md) are denoted within the file as shown in this example: ```q From 73da1d39324f165afe7b4196088f795e8c474703 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Mon, 29 Jul 2024 09:55:34 +0100 Subject: [PATCH 18/61] add multithread primitive details --- docs/ref/add.md | 2 ++ docs/ref/cast.md | 1 + docs/ref/cut.md | 4 +++- docs/ref/divide.md | 2 ++ docs/ref/drop.md | 2 ++ docs/ref/find.md | 4 ++-- docs/ref/join.md | 2 ++ docs/ref/multiply.md | 2 ++ docs/ref/subtract.md | 2 ++ 9 files changed, 18 insertions(+), 3 deletions(-) diff --git a/docs/ref/add.md b/docs/ref/add.md index 81607aacd..f2ba32738 100644 --- a/docs/ref/add.md +++ b/docs/ref/add.md @@ -35,6 +35,8 @@ msoft| 3005 103 Add is generally faster than [Subtract](subtract.md). +`+` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/cast.md b/docs/ref/cast.md index 61bc6839c..996644dc2 100644 --- a/docs/ref/cast.md +++ b/docs/ref/cast.md @@ -45,6 +45,7 @@ Where `x` is: Casting does not change the underlying bit pattern of the data, only how it is represented. +`$`(cast) is a [multithreaded primitive](../kb/mt-primitives.md). ## :fontawesome-solid-sitemap: Iteration diff --git a/docs/ref/cut.md b/docs/ref/cut.md index e66af2ef9..43b1eccda 100644 --- a/docs/ref/cut.md +++ b/docs/ref/cut.md @@ -62,6 +62,8 @@ s4 p4 300 s1 p5 400 ``` +`_`(cut) is a [multithreaded primitive](../kb/mt-primitives.md). + !!! tip "Avoid confusion with underscores in names: separate the Cut operator with spaces." @@ -93,4 +95,4 @@ Otherwise `cut` behaves as `_` Cut. ---- :fontawesome-solid-book: -[Drop](drop.md) \ No newline at end of file +[Drop](drop.md) diff --git a/docs/ref/divide.md b/docs/ref/divide.md index c7812811a..eab6a2a4e 100644 --- a/docs/ref/divide.md +++ b/docs/ref/divide.md @@ -45,6 +45,8 @@ q)2010.01.01 % 2005.01.01 1.999453 ``` +`%` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/drop.md b/docs/ref/drop.md index f3b095801..a57711a41 100644 --- a/docs/ref/drop.md +++ b/docs/ref/drop.md @@ -15,6 +15,8 @@ _Drop items from a list, entries from a dictionary or columns from a table._ x _ y _[x;y] ``` +`_`(drop) is a [multithreaded primitive](../kb/mt-primitives.md). + ## Drop leading or trailing items diff --git a/docs/ref/find.md b/docs/ref/find.md index 7e7e70013..2bafe508d 100644 --- a/docs/ref/find.md +++ b/docs/ref/find.md @@ -9,8 +9,6 @@ keywords: find, kdb+, q, query, search _Find the first occurrence of an item in a list._ - - ```syntax x?y ?[x;y] ``` @@ -38,6 +36,8 @@ q)"abcde"?"d" 3 ``` +`?`(find) is a [multithreaded primitive](../kb/mt-primitives.md). + ## Type-specific diff --git a/docs/ref/join.md b/docs/ref/join.md index 6f9690ed3..fdda5f4d2 100644 --- a/docs/ref/join.md +++ b/docs/ref/join.md @@ -46,6 +46,8 @@ q)v,(type v)$0xab 1.00 2.34 -567.1 20.00 171e ``` +`,`(join) is a [multithreaded primitive](../kb/mt-primitives.md). + ## Dictionaries diff --git a/docs/ref/multiply.md b/docs/ref/multiply.md index 49fc6e955..c8f6a1933 100644 --- a/docs/ref/multiply.md +++ b/docs/ref/multiply.md @@ -40,6 +40,8 @@ price qty 34.5 17 ``` +`*` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration diff --git a/docs/ref/subtract.md b/docs/ref/subtract.md index c1ea857ea..24bf1c726 100644 --- a/docs/ref/subtract.md +++ b/docs/ref/subtract.md @@ -21,6 +21,8 @@ q)2000.11.22 - 03:44:55.666 2000.11.21D20:15:04.334000000 ``` +`-` is a [multithreaded primitive](../kb/mt-primitives.md). + ## :fontawesome-solid-sitemap: Implicit iteration From 01913f6bcbfff7767144068fce3f4292956a6712 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Mon, 29 Jul 2024 16:53:45 +0100 Subject: [PATCH 19/61] updated matlab for newer versions & fixed example --- docs/interfaces/img/matlab.png | Bin 11116 -> 0 bytes docs/interfaces/matlab-client-for-q.md | 262 +++++++++---------------- 2 files changed, 95 insertions(+), 167 deletions(-) delete mode 100644 docs/interfaces/img/matlab.png diff --git a/docs/interfaces/img/matlab.png b/docs/interfaces/img/matlab.png deleted file mode 100644 index 232263232c9d9949a242a63a3b0937a39a8e89c4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11116 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 HeF^$MMa@-R diff --git a/docs/interfaces/matlab-client-for-q.md b/docs/interfaces/matlab-client-for-q.md index 0f01587b9..f1eb2a8ac 100644 --- a/docs/interfaces/matlab-client-for-q.md +++ b/docs/interfaces/matlab-client-for-q.md @@ -1,24 +1,75 @@ --- -title: Working with Matlab | Interfaces | kdb+ and q documentation -description: How connect a Matlab client program to a kdb+ server process +title: Working with MATLAB | Interfaces | kdb+ and q documentation +description: How connect a MATLAB client program to a kdb+ server process --- -# ![Matlab](img/matlab.png) Working with Matlab +# Working with MATLAB - - -Support for Matlab is a part of [Datafeed Toolbox for Matlab](https://uk.mathworks.com/help/datafeed/kx-systems-inc-.html): since R2007a edition. - -MathWorks provides functions overview, usage instructions and some examples on the toolbox webpage. +## Installation !!! note "Versions" - As Matlab/datafeed toolbox evolves features or instruction below are subject to revisions. Please refer to toolbox documentation for latest version. + As MATLAB/datafeed toolbox evolves features or instruction below are subject to revisions. Please refer to toolbox documentation for latest version. Users have reported that this works with more recent versions (e.g. R2015b on RHEL 6.8/2016b and 2017a on macOS). See also community-supported native connector :fontawesome-brands-github: [dmarienko/kdbml](https://github.com/dmarienko/kdbml) -First, we start up a kdb+ process that we wish to communicate with from Matlab and load some sample data into it. +=== "MATLAB R2021a and later" + + Download and unzip [kx_kdbplus.zip](https://uk.mathworks.com/matlabcentral/answers/864350-trading-toolbox-functionality-for-kx-systems-inc-kdb-in-matlab-r2021a-and-later). + Add the resulting directory to your [MATLAB path](https://uk.mathworks.com/help/matlab/matlab_env/what-is-the-matlab-search-path.html), for example in MATLAB + ```matlab + >> addpath('/Users/Developer/matlabkx') + ``` + +=== "MATLAB prior to R2021a" + + Support for MATLAB is a part of [Datafeed Toolbox for MATLAB](https://uk.mathworks.com/help/releases/R2020b/datafeed/kx-systems-inc-.html): since R2007a edition. + +The MATLAB integration depends on the two Java files `c.jar` and `jdbc.jar`. + +:fontawesome-brands-github: +[KxSystems/kdb/c/c.jar](https://github.com/KxSystems/kdb/blob/master/c/c.jar) +:fontawesome-brands-github: +[KxSystems/kdb/c/jdbc.jar](https://github.com/KxSystems/kdb/blob/master/c/jdbc.jar) + +Add the JAR files to the classpath used by MATLAB. It can be added permanently by editing `classpath.txt` (type `edit classpath.txt` at the MATLAB prompt) or for the duration of a particular session using the `javaaddpath` function, for example + +```matlab +>> javaaddpath /home/myusername/jdbc.jar +>> javaaddpath /home/myusername/c.jar +``` + +!!! note "Installation directory" + + In these examples change `/home/myusername` to the directory where `jdbc.jar` and `c.jar` are installed. + +Alternatively, this can be achieved in a MATLAB source file (i.e., \*.m file) adding the following two functions before calling `kx` functions. + +```matlab +javaaddpath('/home/myusername/jdbc.jar') +javaaddpath('/home/myusername/c.jar') +``` + +Confirm they have been added successfully using the `javaclasspath` function. + +```matlab +>> javaclasspath + STATIC JAVA PATH +... + /opt/matlab/2015b/java/jar/toolbox/stats.jar + /opt/matlab/2015b/java/jar/toolbox/symbol.jar + + DYNAMIC JAVA PATH + + /home/myusername/jdbc.jar + /home/myusername/c.jar +>> +``` + +## Connecting to a q process + +First, we start up a kdb+ process that we wish to communicate with from MATLAB and load some sample data into it. Save following as `tradedata.q` file ```q @@ -48,7 +99,7 @@ totalvolume2:{[stock;minvolume] select sum(volume) from trade where sec = stock, Then -```powershell +```bash q tradedata.q -p 5001 ``` @@ -80,54 +131,7 @@ XYZ 13.64856 2900 6.435824 2005.04.03 q) ``` -The Matlab integration depends on the two Java files `c.jar` and `jdbc.jar`. -For the purposes of this recipe, we assume this is available on the machine Matlab is running on, at `C:\q\jdbc.jar`. - -:fontawesome-brands-github: -[KxSystems/kdb/c/c.jar](https://github.com/KxSystems/kdb/blob/master/c/c.jar) -:fontawesome-brands-github: -[KxSystems/kdb/c/jdbc.jar](https://github.com/KxSystems/kdb/blob/master/c/jdbc.jar) - -We then start a new Matlab session. From here on, `>>` represents the Matlab prompt. - - -## Connecting to a q process - -We assume a kdb+ process running on the local host on port 5001 and that the `jdbc.jar` is installed. - -First we need to add the JAR file to the classpath used by Matlab. We can either permanently add it by editing `classpath.txt` (type `edit classpath.txt` at the Matlab prompt) or for the duration of a particular session using the `javaaddpath` function. We’ll use the latter here. - -```matlab ->> javaaddpath /home/myusername/jdbc.jar ->> javaaddpath /home/myusername/c.jar -``` - -!!! note "Installation directory" - - In these examples change `/home/myusername` to the directory where `jdbc.jar` and `c.jar` are installed. - -Alternatively, this can be achieved in a Matlab source file (i.e., \*.m file) adding the following two functions before calling `kx` functions. - -```matlab -javaaddpath('/home/myusername/jdbc.jar') -javaaddpath('/home/myusername/c.jar') -``` - -We can confirm that we’ve added this successfully using the `javaclasspath` function. - -```matlab ->> javaclasspath - STATIC JAVA PATH -... - /opt/matlab/2015b/java/jar/toolbox/stats.jar - /opt/matlab/2015b/java/jar/toolbox/symbol.jar - - DYNAMIC JAVA PATH - - /home/myusername/jdbc.jar - /home/myusername/c.jar ->> -``` +We then start a new MATLAB session. From here on, `>>` represents the MATLAB prompt. We’re now ready to open a connection to the q process: @@ -145,7 +149,7 @@ q = We can also pass a username:password string as the third parameter to the `kx` function if it is required to log in to the q process. -The `q` value is a normal Matlab object and we can inspect the listed properties. We’ll use this value in all our communications with the q process. +The `q` value is a normal MATLAB object and we can inspect the listed properties. We’ll use this value in all our communications with the q process. We close a connection using the `close` function: @@ -171,13 +175,9 @@ We close a connection using the `close` function: java.net.SocketException: Socket closed at java.net.SocketOutputStream.socketWrite(Unknown Source) - at java.net.SocketOutputStream.write(Unknown Source) - at c.w(c.java:99) - at c.k(c.java:107) - at c.k(c.java:108) Error in ==> kx.fetch at 65 @@ -222,28 +222,29 @@ Then we can fetch it: hundreds = -java.lang.Object[]: - [100x1 int32] - [100x1 int32] + java.lang.Object[]: + + [100×1 int64] + [100×1 int64] ``` -We can use the `cell` function to strip the Java array wrapper away: +We can use the [`cell`](https://uk.mathworks.com/help/matlab/ref/cell.html) function to strip the Java array wrapper away: ```matlab >> hundreds_as_cell = cell(hundreds) hundreds_as_cell = - [100x1 int32] - [100x1 int32] + 2×1 cell array ->> + {100×1 int64} + {100×1 int64} ``` Tables are returned as an object with an array property for each column. Taking the first 10 rows of the `trade` table as an example: ```q -q) 10 # trade +q)10#trade sec price volume exchange date ---------------------------------------- ACME 89.5897 1300 6.58303 2005.04.26 @@ -258,18 +259,18 @@ ABC 58.26731 2100 5.220929 2004.09.10 XYZ 74.14568 2900 5.075229 2004.08.24 ``` -Will be returned in Matlab: +Will be returned in MATLAB: ```matlab ->> ten = fetch(q, '10 # trade') +>> ten = fetch(q, '10#trade') ten = - sec: {10x1 cell} - price: [10x1 double] - volume: [10x1 int32] - exchange: [10x1 double] - date: [10x1 double] + sec: {10×1 cell} + price: [10×1 double] + volume: [10×1 int64] + exchange: [10×1 double] + date: [10×1 double] ``` With suitable computation in q, we can return data suitable for immediate plotting. Here we compute a 10-item moving average over the `` `ACME `` prices: @@ -279,13 +280,13 @@ q)mavg[10;exec price from trade where sec=`ACME] 89.5897 65.9259 61.50204 53.32677 54.74408 57.39743 57.15958 62.33525 56.8732.. ``` ```matlab ->> plot (fetch(q,'mavg[10;exec price from trade where sec=`ACME]')) +>> acme = fetch(q,'mavg[10;exec price from trade where sec=`ACME]') ``` ## Metadata -The q integration in Matlab provides the `tables` meta function. +The q integration in MATLAB provides the `tables` meta function. ```matlab >> tables(q) @@ -297,7 +298,7 @@ ans = 'trade' ``` -The experienced q user can use the `\v` command to see all values in the directory: +The experienced q user can use the [`\v`](../basics/syscmds.md#v-variables) command to see all values in the directory: ```matlab >> fetch(q,'\v') @@ -322,7 +323,8 @@ We can use the `fetch` function to cause side effects in the kdb+ process, such Given a table `b`: ```q -q)show b +q)b:([] a:1 2; b:1 2) +q)b a b --- 1 1 @@ -371,7 +373,7 @@ a b A more complicated row shows the potential advantage to better effect: ```matlab ->> insert(q,'TRADE',{'`ACME',100.45,400,.0453,'2005.04.28'}) +>> insert(q,'trade',{'`ACME',100.45,400,.0453,'2005.04.28'}) ``` Be warned though, that errors will not be detected very well. For example the following expression silently fails! @@ -384,91 +386,17 @@ whereas the equivalent `fetch` call provokes an error: ```matlab >> fetch(q,'b,:(1;2;3)') -??? Java exception occurred: -c$KException: length - - at c.k(c.java:106) - - at c.k(c.java:107) - - at c.k(c.java:108) - -Error in ==> kx.fetch at 65 - t = c.handle.k(varargin{1}); -``` - - -## Moving data from one source to another - -As an example of moving data from one source to another, let us get a MSFT quote from Yahoo! and insert it into our q table of data. - -First we connect to Yahoo! and get the quote: - -```matlab ->> y = yahoo - -y = - - url: 'http://quote.yahoo.com' - ip: [] - port: [] - ->> msft = fetch(y,'MSFT') - -msft = - - Symbol: {'MSFT'} - Last: 30.7200 - Date: 733064 - Time: 0.6674 - Change: -0.3900 - Open: 31.0600 - High: 31.1200 - Low: 30.5100 - Volume: 50928424 -``` - -And then we insert it to a suitable table in q: - -```matlab ->> fetch(q,'yahoo_data:([symbol:`symbol$()];high:`float$();low:`float$())') ->> insert(q,'yahoo_data',{'`MSFT', msft.High, msft.Low}) +Error using fetch (line 64) +Java exception occurred: +kx.c$KException: length + at kx.c.k(c.java:110) + at kx.c.k(c.java:111) + at kx.c.k(c.java:112) ``` -And do the same for an IBM quote: - -```matlab ->> ibm = fetch(y,'IBM') - -ibm = - - Symbol: {'IBM'} - Last: 97.0900 - Date: 733064 - Time: 0.6660 - Change: 0.9200 - Open: 96.4200 - High: 97.2300 - Low: 96.1200 - Volume: 10474800 - ->> insert(q,'yahoo_data',{'`IBM', ibm.High, ibm.Low}) -``` - -Finally, let’s check the average high for the data we’re tracking: - -```matlab ->> fetch(q,'select avg high from yahoo_data') - -ans = - - high: 64.1750 -``` - - ## Async commands to q -The `exec` function is used for sending asynchronous commands to q; ones we do not expect a response to, and which may be performed in the background while we continue interacting with the Matlab process. +The `exec` function is used for sending asynchronous commands to q; ones we do not expect a response to, and which may be performed in the background while we continue interacting with the MATLAB process. Here we establish a large-ish data structure in the kdb+ process: @@ -492,6 +420,6 @@ ans = ## Getting more help -Start with `help kx` in your Matlab session and also see `help kx.fetch` and so on for further details of the integration. - +Start with `help kx` in your MATLAB session and also see `help kx.fetch` and so on for further details of the integration. +MathWorks provides functions overview, usage instructions and some examples on the [toolbox webpage](https://uk.mathworks.com/help/releases/R2020b/datafeed/kx-systems-inc-.html). From 08f3c347916ae43374fc01b73bec68a442815529 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Tue, 30 Jul 2024 17:58:47 +0100 Subject: [PATCH 20/61] added link to multithreaded primitives --- docs/basics/comparison.md | 2 ++ docs/ref/greater.md | 2 ++ docs/ref/lesser.md | 2 ++ 3 files changed, 6 insertions(+) diff --git a/docs/basics/comparison.md b/docs/basics/comparison.md index 25c04acbb..b8a28995a 100644 --- a/docs/basics/comparison.md +++ b/docs/basics/comparison.md @@ -66,6 +66,8 @@ q)(1 + 1e-13) = 1 1b ``` +`< > = >= <= <>` are [multithreaded primitives](../kb/mt-primitives.md). + !!! tip "For booleans, `<>` is the same as _exclusive or_ (XOR)." diff --git a/docs/ref/greater.md b/docs/ref/greater.md index daa99e5d2..3295c2b67 100644 --- a/docs/ref/greater.md +++ b/docs/ref/greater.md @@ -25,6 +25,8 @@ q)"sat"|"cow" "sow" ``` +`|` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Flags diff --git a/docs/ref/lesser.md b/docs/ref/lesser.md index 01ece948a..5e0c4520b 100644 --- a/docs/ref/lesser.md +++ b/docs/ref/lesser.md @@ -26,6 +26,8 @@ q)"sat"&"cow" "cat" ``` +`&` is a [multithreaded primitive](../kb/mt-primitives.md). + ## Flags From af07e6ac7ef293339bb5e91c0018bb2ecc7fc898 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 12:00:43 +0100 Subject: [PATCH 21/61] referencing old forum no longer used --- docs/learn/startingkdb/index.md | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/docs/learn/startingkdb/index.md b/docs/learn/startingkdb/index.md index 2e4c6b405..2281c7d31 100644 --- a/docs/learn/startingkdb/index.md +++ b/docs/learn/startingkdb/index.md @@ -5,15 +5,10 @@ keywords: kdb+, q, start, tutorial, --- # Starting kdb+ - - - - This is a quick-start guide to kdb+, aimed primarily at those learning independently. It covers system installation, the kdb+ environment, IPC, tables and typical databases, and where to find more material. After completing this you should be able to follow the Borror textbook [Q for Mortals](/q4m3/), and the [Reference](../../ref/index.md). One caution: you can learn kdb+ reasonably well by independent study, but for serious deployment of the product you need the help of a consultant. This is because kdb+ is typically used for very demanding applications that require experience to set up properly. Contact KX for help with such evaluations. - ## kdb+ The kdb+ system is both a database and a programming language. @@ -47,7 +42,7 @@ Several background articles and links can be found in the [Archive](../archive.m ### Discussion groups - The main discussion forum is the [k4 Topicbox](https://k4.topicbox.com/groups/k4). This is available only to licensed customers – please use a work email address to [apply for access](https://k4.topicbox.com/groups/k4?subscription_form=e1ca20f8-95f6-11e8-8090-9973fa3f0106). -- The [kdb+ Personal Developers](https://groups.google.com/forum/#!forum/personal-kdbplus) forum is an open Google discussion group for users of the free system. +- [KX Community discussion forum](https://learninghub.kx.com/forums/) is used to find answers, ask questions, and connect with our KX Community. ## :fontawesome-solid-download: Install free system From 298aee730c41af36fabf14adebb65d9db94d69ab Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 12:28:36 +0100 Subject: [PATCH 22/61] reference scripts used directly in learning --- docs/learn/startingkdb/index.md | 3 +-- docs/learn/startingkdb/language.md | 7 +++---- docs/learn/startingkdb/tables.md | 9 +++------ 3 files changed, 7 insertions(+), 12 deletions(-) diff --git a/docs/learn/startingkdb/index.md b/docs/learn/startingkdb/index.md index 2281c7d31..a428bd222 100644 --- a/docs/learn/startingkdb/index.md +++ b/docs/learn/startingkdb/index.md @@ -54,10 +54,9 @@ If you do not already have access to a licensed copy, go to [Get started](../ind Two sets of scripts are referenced in this guide: -1. The free system is distributed with the following example scripts in the main directory: +1. The free system is distributed with the following example script in the main directory: - `sp.q` – the Suppliers and Parts sample database - - `trade.q` – a stock trades sample database If you do not have these scripts, get them from :fontawesome-brands-github: [KxSystems/kdb](https://github.com/KxSystems/kdb) diff --git a/docs/learn/startingkdb/language.md b/docs/learn/startingkdb/language.md index 50716fe3c..00d946767 100644 --- a/docs/learn/startingkdb/language.md +++ b/docs/learn/startingkdb/language.md @@ -64,7 +64,6 @@ You can confirm that you are in the `QHOME` directory by calling a directory lis q)\ls *.q ... "sp.q" - "trade.q" ... ``` @@ -74,7 +73,6 @@ You can confirm that you are in the `QHOME` directory by calling a directory lis q)\dir *.q ... "sp.q" - "trade.q" ... ``` @@ -360,7 +358,7 @@ cog 3 20 60 A q script is a plain text file with extension `.q`, which contains q expressions that are executed when loaded. -For example, load the script `sp.q` and display the `s` table that it defines: +For example, load the script :fontawesome-brands-github:[`KxSystems/kdb/sp.q`](https://github.com/KxSystems/kdb/blob/master/sp.q) and display the `s` table that it defines: ```q q)\l sp.q / load script @@ -416,7 +414,8 @@ q)b ## Q queries -Q queries are similar to SQL, though often much simpler. +Q queries are similar to SQL, though often much simpler. +Loading the script :fontawesome-brands-github:[`KxSystems/kdb/sp.q`](https://github.com/KxSystems/kdb/blob/master/sp.q) to populate tables `s`,`p` and `sp` we can show some query examples: ```q \l sp.q diff --git a/docs/learn/startingkdb/tables.md b/docs/learn/startingkdb/tables.md index 2bba8e565..a0c659f5c 100644 --- a/docs/learn/startingkdb/tables.md +++ b/docs/learn/startingkdb/tables.md @@ -51,7 +51,7 @@ Here table `t` is defined with column names $c_{1-n}$, and corresponding values ## 4.3 Suppliers and parts -The script `sp.q` defines [C.J. Date’s Suppliers and Parts database](https://en.wikipedia.org/wiki/Suppliers_and_Parts_database). You can view this script in an editor to see the definitions. Load the script. +The script :fontawesome-brands-github:[`KxSystems/kdb/sp.q`](https://github.com/KxSystems/kdb/blob/master/sp.q) defines [C.J. Date’s Suppliers and Parts database](https://en.wikipedia.org/wiki/Suppliers_and_Parts_database). You can view this script in an editor to see the definitions. Load the script. ```q q)\l sp.q @@ -163,15 +163,12 @@ s4 p5 100 ## Stock data -The following is a typical layout populated with random data. Load the `trades.q` script. +The following is a typical layout populated with random data. Load the :fontawesome-brands-github:[`KxSystems/cookbook/start/trades.q`](https://github.com/KxSystems/cookbook/blob/master/start/trades.q) script. ```q -q)\l start/trades.q +q)\l trades.q ``` -:fontawesome-brands-github: -[KxSystems/cookbook/start/trades.q](https://github.com/KxSystems/cookbook/blob/master/start/trades.q) - A trade table might include: date, time, symbol, price, size, condition code. ```q From 71f091ee6fce27603c1f77b8a18ea029cc65503d Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 12:29:21 +0100 Subject: [PATCH 23/61] add links, format code --- docs/architecture/rq.md | 2 +- docs/wp/rt-tick/index.md | 5 ++--- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 6e9cdf8cd..9abdc5c41 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -80,7 +80,7 @@ upd[x;y] Where * `x` is a symbol atom naming a table -* `y` is table data to add to table `x`. +* `y` is table data to add to table `x`, which can contain one or more rows. ### .u.end diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index fd36c1504..857df38de 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -48,8 +48,7 @@ Although the inner workings of the tickerplant process are beyond the scope of t : refers to the schema file (in this case called `sym.q`), assumed to reside in the subdirectory called `tick` (relative to `tick.q`). This schema file simply defines the tables that exist in the TP – here we define two tables, `trade` and `quote`, as follows. ```q -quote:([]time:`timespan$();sym:`symbol$();bid:`float$();ask:`float$(); - bsize:`int$();asize:`int$()) +quote:([]time:`timespan$();sym:`symbol$();bid:`float$();ask:`float$();bsize:`int$();asize:`int$()) trade:([]time:`timespan$();sym:`symbol$();price:`float$();size:`int$()) ``` @@ -93,7 +92,7 @@ getask:{[s] prices[s]+getmovement[s]} /generate ask price Points to note from the above: 1. The data sent to the tickerplant is in columnar (column-oriented) list format. In other words, the tickerplant expects data as lists, not tables. This point will be relevant later when the RDB wishes to replay the tickerplant logfile. -1. The function triggered on the tickerplant upon receipt of these updates is `.u.upd`. +1. The function triggered on the tickerplant upon receipt of these updates is [`.u.upd`](../../architecture/tickq.md#uupd). 1. If you wish to increase the frequency of updates sent to the tickerplant for testing purposes, simply change the timer value at the end of this script accordingly. From 59592b726d67d012e9d67dc153cf6070b6365faa Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 15:05:42 +0100 Subject: [PATCH 24/61] script ref were used, rather than on seperate page --- docs/learn/startingkdb/hdb.md | 31 +++++++++++++++---------------- docs/learn/startingkdb/index.md | 32 -------------------------------- 2 files changed, 15 insertions(+), 48 deletions(-) diff --git a/docs/learn/startingkdb/hdb.md b/docs/learn/startingkdb/hdb.md index d4b697848..03f5f83ef 100644 --- a/docs/learn/startingkdb/hdb.md +++ b/docs/learn/startingkdb/hdb.md @@ -36,17 +36,15 @@ These storage strategies give best efficiency for searching and retrieval. For e For example, a simple partitioning scheme on a single disk might be as shown right. Here, the daily and master tables are small enough to be written to single files, while the trade and quote tables are splayed and partitioned by date. -## Sample database +## Sample partitioned database -The script `buildhdb.q` will build a sample HDB. It builds a month’s random data in directory `start/db`, and takes a few seconds to run. - -:fontawesome-brands-github: -[KxSystems/cookbook/start/buildhdb.q](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) +The script :fontawesome-brands-github:[`KxSystems/cookbook/start/buildhdb.q`](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) will build a sample HDB. +It builds a month’s random data in directory `start/db`, and takes a few seconds to run. Load q, then: ```q -q)\l start/buildhdb.q +q)\l buildhdb.q ``` To load the database, enter: @@ -86,8 +84,7 @@ date | x 2013.05.06| 14182 ... -q)select cnt:count i,sum size,last price, wprice:size wavg price - by 5 xbar time.minute from t +q)select cnt:count i,sum size,last price, wprice:size wavg price by 5 xbar time.minute from t minute| cnt size price wprice ------| ----------------------- 09:30 | 44 2456 47.83 47.60555 @@ -115,11 +112,11 @@ time price bid ask ## Sample segmented database -The `buildhdb.q` script can be customized to build a segmented database. In practice, database segments should be on separate drives, but for illustration, the segments are here written to a single drive. Both the database root, and the location of the database segments need to be specified. - -For example, edit the first few lines of the script as below. +The `buildhdb.q` script can be customized to build a segmented database. +In practice, database segments should be on separate drives, but for illustration, the segments are here written to a single drive. +Both the database root, and the location of the database segments need to be specified. -Ensure that the directory given in `dsp` is the full pathname, and that it is created, writeable and empty. +For example, edit the first few lines of the script :fontawesome-brands-github:[`KxSystems/cookbook/start/buildhdb.q`](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) as below. ```q dst:`:start/dbs / new database root @@ -131,9 +128,12 @@ end:2013.12.31 ... ``` -For Windows, `dsp` might be: ``dsp:`:c:/dbss``. +Ensure that the directory given in `dsp` is the full pathname, and that it is created, writeable and empty. For Windows, `dsp` might be: ``dsp:`:c:/dbss``. + +!!! warning "This example writes approximately 7GB of created data to disk." -Load the modified script, which should now take a minute or so. This should write the partioned data to subdirectories of `dsp`, and create a `par.txt` file like: +Load the modified script, which should now take a minute or so. This should write the partioned data to subdirectories of the directory specified by `dsp` +`par.txt` can be found within the `dsp` directory, which lists the disks/directories containing the data of the segmented database. ```txt /dbss/d0 @@ -151,8 +151,7 @@ q)\l start/dbs q)(count quote), count trade 61752871 12356516 -q)select cnt:count i,sum size,size wavg price from trade - where date in 2012.09.17+til 5, sym=`IBM +q)select cnt:count i,sum size,size wavg price from trade where date in 2012.09.17+til 5, sym=`IBM cnt size price -------------------- 4033 217537 37.35015 diff --git a/docs/learn/startingkdb/index.md b/docs/learn/startingkdb/index.md index a428bd222..a68c6803d 100644 --- a/docs/learn/startingkdb/index.md +++ b/docs/learn/startingkdb/index.md @@ -50,38 +50,6 @@ Several background articles and links can be found in the [Archive](../archive.m If you do not already have access to a licensed copy, go to [Get started](../index.md) to download and install q. -## :fontawesome-solid-file: Example files - -Two sets of scripts are referenced in this guide: - -1. The free system is distributed with the following example script in the main directory: - - - `sp.q` – the Suppliers and Parts sample database - - If you do not have these scripts, get them from - :fontawesome-brands-github: [KxSystems/kdb](https://github.com/KxSystems/kdb) - and save them in your `QHOME` directory. - -2. Other example files are in the :fontawesome-brands-github: [KxSystems/cookbook/start](https://github.com/KxSystems/cookbook/tree/master/start) directory. - - Move the `start` directory under your `QHOME` directory, e.g. `q/start`. For example, there should be a file: - - === ":fontawesome-brands-linux: Linux :fontawesome-brands-apple: macOS" - - ~/q/start/buildhdb.q - - === ":fontawesome-brands-windows: Windows" - - c:\q\start\buildhdb.q - - -!!! tip "Text editor for :fontawesome-brands-windows: Windows" - - Since q source is in plain text files, it is worth installing a good text editor such as Notepad++ or Notepad2. - - Some text editors have extensions to provide e.g. syntax highlighting for q. See the [list of editor integrations](../../interfaces/index.md#editor-integrations) - - ## Graphical user interface When q is run, it displays a console where you can enter commands and see the results. This is all you need to follow the tutorial, and if you just want to learn a little about q, then it is easiest to work in the console. From 2631977a88872d4a34ce522492b31a99c43b1408 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 17:28:08 +0100 Subject: [PATCH 25/61] vwap example didnt work - fixed --- docs/wp/rt-tick/index.md | 129 +++++++++++++++++++++------------------ 1 file changed, 68 insertions(+), 61 deletions(-) diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index 857df38de..adf3d330d 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -286,7 +286,7 @@ upd:insert In other words, when the RDB replays a given message, it simply inserts the record/s into the corresponding table. This is the same definition of `upd` used for intraday updates via IPC. These updates succeed because the second argument to `insert` can be either a columnar list or a table. -A q process replays a tickerplant logfile using the operator `-11!`. Although this operator can be used in different ways, the simplest syntax is: +A q process replays a tickerplant logfile using the operator [`-11!`](../../basics/internal.md#-11-streaming-execute). Although this operator can be used in different ways, the simplest syntax is: ```q -11! `:TPDailyLogfile @@ -772,7 +772,12 @@ time sym price size bid bsize ask asize ### Real-time VWAP subscriber -This section describes how to build an RTE which enriches trade with VWAP (volume-weighted average price) information on a per-symbol basis. A VWAP can be defined as: +#### Overview + +This section describes how to build an RTE which enriches trade with VWAP (volume-weighted average price) information on a per-symbol basis. +If a situation occurs were this RTE is restarted, it will recalculate VWAP from the TP log file. Upon an end-of-day event it will clear current records. + +A VWAP can be defined as: $$ VWAP = \frac{\sum_{i} (tradevolume_i)(tradeprice_i)}{\sum_{i} (trade price_i)}$$ @@ -839,100 +844,102 @@ IBM.N | 191.17041 MSFT.O | 45.146982 ``` -Just like the previous RTE example, this solution will comprise a heavily modified version of `r.q`, written by the author and named `real_time_vwap.q`. - -This process should be started off as follows: - -```bash -q tick/real_time_vwap.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N -p 5004 -``` +#### Example script -This process will subscribe to only the `trade` table for symbols `MSFT.O`, `IBM.N` and `GS.N` and will listen on port 5004. The structure and design philosophy behind `real_time_vwap.q` is very similar to `RealTimeTradeWithAsofQuotes.q`. - -The first section of the script simply parses the command-line -arguments and uses these to update some default values – identical -code to the start of `RealTimeTradeWithAsofQuotes.q`. - - -#### Initialize desired table schemas - -The next section of code defines the behavior of this RTE upon connecting to the TP and subscribing to the `trade` table. This RTE will replay the TP’s logfile, much like the RDB. The following function replaces `.u.rep`. +Just like the previous RTE example, this solution will comprise a heavily modified version of [`r.q`](../../architecture/rq.md), written by the author and named `real_time_vwap.q`. ```q /initialize schema function -InitializeTrade:{[TradeInfo;logfile] +InitializeTrade:{[TradeInfo;logfile] `trade set TradeInfo 1; - if[null first logfile;update v:0n,s:0Ni,rvwap:0n from `trade;:()]; + if[null first logfile;update v:0n,s:0Ni,rvwap:0n from `trade;:()]; -11!logfile; update v:sums (size*price),s:sums size by sym from `trade; - update rvwap:v%s from `trade; } -``` + update rvwap:v%s from `trade; + `vwap upsert select last rvwap by sym from trade;} -This binary function `InitializeTrade` will be executed upon startup. It is passed two arguments, just like `.u.rep`: - -argument | semantics -------------|---------- -`TradeInfo` | pair: table name (`` `trade``); empty table definition -`Logfile` | pair: TP logfile record count and location - -The `vwap` table is then simply defined as: - -```q /this keyed table maps a symbol to its current vwap vwap:([sym:`$()] rvwap:`float$()) -``` - -When `InitializeTrade` is executed, the TP logfile will be replayed using `-11!`. For the purpose of this replay, the function `upd` is simply defined as: -```q /For TP logfile replay, upd is a simple insert for trades upd:{if[not `trade=x;:()];`trade insert y} -``` - -In other words, insert `trade` records into the `trade` table and ignore `quote` records. - - -#### Intraday update behavior - -The next code section defines the intraday behavior upon receiving -new trades: -```q / -This intraday function is triggered upon incoming updates from TP. +This intraday function is triggered upon incoming updates from TP. Its behavior is as follows: 1. Add s and v columns to incoming trade records -2. Increment incoming records with the last previous s and v values +2. Increment incoming records with the last previous s and v values (on per sym basis) 3. Add rvwap column to incoming records (rvwap is v divided by s) 4. Insert these enriched incoming records to the trade table 5. Update vwap table \ updIntraDay:{[t;d] - d:update s:sums size,v:sums size*price by sym from d; - d:d pj select last v,last s by sym from trade; + d:update s:sums size,v:sums size*price by sym from d; + d:d pj select last v,last s by sym from trade; d:update rvwap:v%s from d; `trade insert d; `vwap upsert select last rvwap by sym from trade; } + +/end of day function - triggered by tickerplant at EOD +/Empty tables +.u.end:{{delete from x}each tables `. } /clear out trade and vwap tables + +args:.Q.opt .z.x +args:`$args +h:hopen hsym first args`tp /connect to tickerplant +InitializeTrade . h "(.u.sub[`trade;",(.Q.s1 args`syms),"];`.u `i`L)" +upd:updIntraDay /switch upd to intraday update mode ``` -So whenever a trade update comes in, the VWAP for each affected symbol is updated and the new trades are enriched with this information. +This process should be started off as follows: +```bash +q tick/real_time_vwap.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N -p 5004 +``` -#### End of day +This process will subscribe to only the `trade` table for symbols `MSFT.O`, `IBM.N` and `GS.N` and will listen on port 5004. The structure and design philosophy behind `real_time_vwap.q` is very similar to `RealTimeTradeWithAsofQuotes.q`. -The EOD behavior on this RTE is very simple – clear out the tables: +The first section of the script simply parses the command-line +arguments and uses these to update some default values – identical +code to the start of `RealTimeTradeWithAsofQuotes.q`. + + +#### InitializeTrade function + +`InitializeTrade` defines the behavior of this RTE after connecting to the TP and subscribing to the `trade` table. +This RTE will replay the TP’s logfile, much like the RDB. The `InitializeTrade` function replaces [`.u.rep`](../../architecture/rq.md#urep). + +This binary function `InitializeTrade` will be executed upon startup. It is passed two arguments, just like `.u.rep`: + +argument | semantics +------------|---------- +`TradeInfo` | pair: table name (`` `trade``); empty table definition +`Logfile` | pair: TP logfile record count and location + +The `vwap` table is then simply defined as: ```q -/end of day function - triggered by tickerplant at EOD -/Empty tables -.u.end:{{delete from x}each tables `. } /clear out trade and vwap tables +/this keyed table maps a symbol to its current vwap +vwap:([sym:`$()] rvwap:`float$()) ``` +When `InitializeTrade` is executed, the TP logfile will be replayed and its contents executed using [`-11!`](../../basics/internal.md#-11-streaming-execute). -#### Subscribe to TP +Replaying will cause the `upd` function to be executed, which in this script is defined as insert `trade` records into the `trade` table and ignoring `quote` records. + +#### updIntraDay function + +`updIntraDay` is called whenever a trade update comes in, the VWAP for each affected symbol is updated and the new trades are enriched with this information. -The RTE connects to the TP and subscribes to the `trade` table for user specified symbols. The RTE also requests TP logfile information (for replay purposes): +#### .u.end function + +The EOD behavior on this RTE is very simple – clear out the tables. + +#### TP subscription + +The RTE connects to the TP and subscribes to the `trade` table for user specified symbols. +The RTE also requests TP logfile information (for replay purposes): ```q h:hopen args`tp /connect to tickerplant @@ -940,8 +947,8 @@ InitializeTrade . h "(.u.sub[`trade;",(.Q.s1 args`syms),"];`.u `i`L)" upd:updIntraDay /switch upd to intraday update mode ``` -The message returned from the TP is passed to the function `InitializeTrade`. Once the RTE has finished initializing or replaying the TP logfile, the definition of `upd` is then switched to `updIntraDay` so the RTE can deal with intraday updates appropriately. - +The message returned from the TP is passed to the function `InitializeTrade`. +Once the RTE has finished initializing or replaying the TP logfile, the definition of `upd` is then switched to `updIntraDay` so the RTE can deal with intraday updates appropriately. ## Performance considerations From 8ba1c07cef19ed4d42e9b3469ed6ef74cdbb1611 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 17:47:13 +0100 Subject: [PATCH 26/61] bug in as-of-quote example --- docs/wp/rt-tick/index.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index adf3d330d..e6cab4ad2 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -684,7 +684,8 @@ At end of day, the tickerplant sends a message to all RTEs telling them to invok logfile::hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D; .[logfile;();:;()]; /Initialize the new log file LogfileHandle::hopen logfile; - {delete from x}each tables `. /clear out tables } + {delete from x}each tables `. /clear out tables + } ``` This function has been heavily modified from `r.q` to achieve the following desired behavior: From 29358b2b4100db99eec3c79849554dff044be318 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Wed, 31 Jul 2024 18:02:16 +0100 Subject: [PATCH 27/61] prevent some erroneous output for example script --- docs/wp/rt-tick/index.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index e6cab4ad2..f7dd9cd27 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -750,9 +750,9 @@ The next part of the script is probably the most critical – the process connec ```q / connect to tickerplant and subscribe to trade and quote for portfolio -h:hopen args`tp /connect to tickerplant -InitializeSchemas . h(".u.sub";`trade;args`syms) -InitializeSchemas . h(".u.sub";`quote;args`syms) +h:hopen args`tp; /connect to tickerplant +InitializeSchemas . h(".u.sub";`trade;args`syms); +InitializeSchemas . h(".u.sub";`quote;args`syms); ``` The output of a subscription to a given table (for example `trade`) from the tickerplant is a 2-list, as discussed previously. This pair is in turn passed to the function `InitializeSchemas`. From 722253fbcf538960aecd179c2b096a5089e581ea Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Thu, 1 Aug 2024 12:18:25 +0100 Subject: [PATCH 28/61] add matlab null details --- docs/interfaces/matlab-client-for-q.md | 27 ++++++++++++++++++++++++++ 1 file changed, 27 insertions(+) diff --git a/docs/interfaces/matlab-client-for-q.md b/docs/interfaces/matlab-client-for-q.md index f1eb2a8ac..60475b023 100644 --- a/docs/interfaces/matlab-client-for-q.md +++ b/docs/interfaces/matlab-client-for-q.md @@ -417,6 +417,33 @@ ans = >> close(q) ``` +## Handling null + +kdb+ has the ability to set values to null. MATLAB doesnt have a corresponding null type, so if your data contains nulls you may wish to filter or detect them. + +MATLAB has the ability to call static methods within Java. The `NULL` method can provide the null values for the different [data types](../basics/datatypes.md). For example + +```matlab +NullInt=kx.c.NULL('i') +NullLong=kx.c.NULL('j') +NullDouble=kx.c.NULL('f') +NullDate=kx.c.NULL('d') +``` + +With this, you can test values for null. The following shows that the comparison will return true when requesting null values from a kdb+ connection named conn: + +```matlab +fetch(conn,'0Ni')== NullInt +fetch(conn,'0N')== NullLong +fetch(conn,'0Nd')== NullDate +isequaln(fetch(conn,'0Ni'),NullInt) +isequaln(fetch(conn,'0N'), NullLong) +isequaln(fetch(conn,'0Nd'), NullDate) +isequaln(fetch(conn,'0Nf'), NullDouble) +``` + +An alternative is to have your query include a filter for nulls (if they are populated), so they arent retrieved by MATLAB. + ## Getting more help From 76042ed99c43df85201e980779cd70a2064f7f98 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Thu, 1 Aug 2024 13:32:49 +0100 Subject: [PATCH 29/61] put example code into 1 section rather than split --- docs/wp/rt-tick/index.md | 435 ++++++++++++++++++++------------------- 1 file changed, 225 insertions(+), 210 deletions(-) diff --git a/docs/wp/rt-tick/index.md b/docs/wp/rt-tick/index.md index f7dd9cd27..9289ff4ed 100644 --- a/docs/wp/rt-tick/index.md +++ b/docs/wp/rt-tick/index.md @@ -566,211 +566,6 @@ Reading this from the right, we obtain the location of the tickerplant process w Two quite different RTE instances are described below. - -### Real-time trade with as-of quotes - -One of the most popular and powerful joins in the q language is the [`aj`](../../ref/aj.md) function. This keyword was added to the language to solve a specific problem – how to join trade and quote tables together in such a way that for each trade, we grab the prevalent quote _as of_ the time of that trade. In other words, what is the last quote at or prior to the trade? - -This function is relatively easy to use for one-off joins. However, what if you want to maintain trades with as-of quotes in real time? This section describes how to build an RTE with real-time trades and as-of quotes. This is a heavily modified version of `r.q`, written by the author and named `RealTimeTradeWithAsofQuotes.q`. - -One additional feature this script demonstrates is the ability of any q process to write to and maintain its own kdb+ binary logfile for replay/recovery purposes. In this case, the RTE maintains its own daily logfile for trade records. This will be used for recovery in place of the standard tickerplant logfile as used by `r.q`. - -This process should be started off as follows: - -```bash -q tick/RealTimeTradeWithAsofQuotes.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N -p 5003 -``` - -This process will subscribe to both trade and quote tables for symbols `MSFT.O`, `IBM.N` and `GS.N` and will listen on port 5003. The author has deliberately made some of the q syntax more easily understandable compared to `r.q`. - -The first section of the script simply parses the command-line arguments and uses these to update some default values: - -```q -/ -The purpose of this script is as follows: -1. Demonstrate how custom RTEs can be created in q -2. In this example, create an efficient engine for calculating - the prevalent quotes as of trades in real-time. - This removes the need for ad-hoc invocations of the aj function. -3. In this example, this subscriber also maintains its own binary - log file for replay purposes. - This replaces the standard tickerplant log file replay functionality. -\ -show "RealTimeTradeWithAsofQuotes.q" -/sample usage -/q tick/RealTimeTradeWithAsofQuotes.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N - -/default command line arguments - tp is location of tickerplant. -/syms are the symbols we wish to subscribe to -default:`tp`syms!("::5000";"") - -args:.Q.opt .z.x /transform incoming cmd line arguments into a dictionary -args:`$default,args /upsert args into default -args[`tp] : hsym first args[`tp] - -/drop into debug mode if running in foreground AND -/errors occur (for debugging purposes) -\e 1 - -if[not "w"=first string .z.o;system "sleep 1"] -``` - -The error flag above is set for purely testing purposes – when the developer runs this script in the foreground, if errors occur at runtime as a result of incoming IPC messages, the process will drop into debug mode. For example, if there is a problem with the definition of `upd`, then when an update is received from the tickerplant we will drop into debug mode and (hopefully) identify the issue. - - -#### Initialize desired table schemas - -The next section of code defines the behavior of this RTE upon connecting and subscribing to the tickerplant’s trade and quote tables. This function replaces `.u.rep` in `r.q`: - -```q -/initialize schemas for custom RTE -InitializeSchemas:`trade`quote! - ( - {[x]`TradeWithQuote insert update bid:0n,bsize:0N,ask:0n,asize:0N from x}; - {[x]`LatestQuote upsert select by sym from x} - ); -``` - -The RTE’s trade table (named `TradeWithQuote`) maintains `bid`, `bsize`, `ask` and `asize` columns of appropriate type. For the quote table, we just maintain a keyed table called `LatestQuote`, keyed on `sym` which will maintain the most recent quote per symbol. This table will be used when joining prevalent quotes to incoming trades. - - -#### Intraday update behavior - -The next code section defines the intraday behavior upon receiving new trades: - -```q -/intraday update functions -/Trade Update -/1. Update incoming data with latest quotes -/2. Insert updated data to TradeWithQuote table -/3. Append message to custom logfile -updTrade:{[d] - d:d lj LatestQuote; - `TradeWithQuote insert d; - LogfileHandle enlist (`replay;`TradeWithQuote;d); } -``` - -Besides inserting the new trades with prevalent quote information into the trade table, the above function also appends the new records to its custom logfile. This logfile will be replayed upon recovery/startup of the RTE. Note that the replay function is named `replay`. This differs from the conventional TP logfile where the replay function was called `upd`. - -The next section defines the intraday behavior upon receiving new quotes: - -```q -/Quote Update -/1. Calculate latest quote per sym for incoming data -/2. Update LatestQuote table -updQuote:{[d] - `LatestQuote upsert select by sym from d; } -``` - -The following dictionary `upd` acts as a case statement – when an update for the trade table is received, `updTrade` will be triggered with the message as argument. Likewise, when an update for the quote table is received, `updQuote` will be triggered. - -```q -/upd dictionary will be triggered upon incoming update from tickerplant -upd:`trade`quote!(updTrade;updQuote) -``` - -In `r.q`, `upd` is defined as a function, not a dictionary. However we can use this dictionary definition for reasons discussed previously. - - -#### End of day - -At end of day, the tickerplant sends a message to all RTEs telling them to invoke their EOD function – `.u.end`: - -```q -/end of day function - triggered by tickerplant at EOD -.u.end:{ - hclose LogfileHandle; /close the connection to the old log file - /create the new logfile - logfile::hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D; - .[logfile;();:;()]; /Initialize the new log file - LogfileHandle::hopen logfile; - {delete from x}each tables `. /clear out tables - } -``` - -This function has been heavily modified from `r.q` to achieve the following desired behavior: - -`hclose LogfileHandle` - -: Close connection to the custom logfile. - -``logfile::hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D`` - -: Create the name of the new custom logfile. This logfile is a daily logfile – meaning it only contains one day’s trade records and it has today’s date in its name, just like the tickerplant’s logfile. - -`.[logfile;();:;()]` - -: Initialize this logfile with an empty list. - -`LogfileHandle::hopen logfile` - -: Establish a connection (handle) to this logfile for streaming writes. - -``{delete from x}each tables `.`` - -: Empty out the tables. - - -#### Replay custom logfile - -This section concerns the initialization and replay of the RTE’s custom logfile. - -```q -/Initialize name of custom logfile -logfile:hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D - -replay:{[t;d]t insert d} /custom log file replay function -``` - -At this point, the name of today’s logfile and the definition of the logfile replay function have been established. The replay function will be invoked when replaying the process’s custom daily logfile. It is defined to simply insert the on-disk records into the in memory (`TradeWithQuote`) table. This will be a fast operation ensuring recovery is achieved quickly and efficiently. - -Upon startup, the process uses a try-catch to replay its custom daily logfile. If it fails for any reason (possibly because the logfile does not yet exist), it will send an appropriate message to standard out and will initialize this logfile. Replay of the logfile is achieved with the standard operator `-11!` as discussed previously. - -```q -/attempt to replay custom log file -@[{-11!x;show"successfully replayed custom log file"}; logfile; - {[e] - m:"failed to replay custom log file"; - show m," - assume it does not exist. Creating it now"; - .[logfile;();:;()]; /Initialize the log file - } ] -``` - -Once the logfile has been successfully replayed/initialized, a handle (connection) is established to it for subsequent streaming appends (upon new incoming trades from tickerplant): - -```q - /open a connection to log file for writing -LogfileHandle:hopen logfile -``` - - -#### Subscribe to TP - -The next part of the script is probably the most critical – the process connects to the tickerplant and subscribes to the trade and quote table for user-specified symbols. - -```q -/ connect to tickerplant and subscribe to trade and quote for portfolio -h:hopen args`tp; /connect to tickerplant -InitializeSchemas . h(".u.sub";`trade;args`syms); -InitializeSchemas . h(".u.sub";`quote;args`syms); -``` - -The output of a subscription to a given table (for example `trade`) from the tickerplant is a 2-list, as discussed previously. This pair is in turn passed to the function `InitializeSchemas`. - -We can see this RTE in action by examining the five most recent trades for `GS.N`: - -```q -q)-5#select from TradeWithQuote where sym=`GS.N -time sym price size bid bsize ask asize ---------------------------------------------------------------------- -0D21:50:58.857411000 GS.N 178.83 790 178.8148 25   178.8408 98 -0D21:51:00.158357000 GS.N 178.8315 312 178.8126 12   178.831 664 -0D21:51:01.157842000 GS.N 178.8463 307 178.8193 767 178.8383 697  -0D21:51:03.258055000 GS.N 178.8296 221 178.83 370 178.8627 358 -0D21:51:03.317152000 GS.N 178.8314 198 178.8296 915  178.8587 480 -``` - - ### Real-time VWAP subscriber #### Overview @@ -906,7 +701,7 @@ arguments and uses these to update some default values – identical code to the start of `RealTimeTradeWithAsofQuotes.q`. -#### InitializeTrade function +#### Initialization `InitializeTrade` defines the behavior of this RTE after connecting to the TP and subscribing to the `trade` table. This RTE will replay the TP’s logfile, much like the RDB. The `InitializeTrade` function replaces [`.u.rep`](../../architecture/rq.md#urep). @@ -929,15 +724,15 @@ When `InitializeTrade` is executed, the TP logfile will be replayed and its cont Replaying will cause the `upd` function to be executed, which in this script is defined as insert `trade` records into the `trade` table and ignoring `quote` records. -#### updIntraDay function +#### Intraday update behavior `updIntraDay` is called whenever a trade update comes in, the VWAP for each affected symbol is updated and the new trades are enriched with this information. -#### .u.end function +#### End of day -The EOD behavior on this RTE is very simple – clear out the tables. +At end of day, the tickerplant sends a message to all RTEs telling them to invoke their EOD function (`.u.end`). The function will clear tables used on this RTE. -#### TP subscription +#### Subscribe to TP The RTE connects to the TP and subscribes to the `trade` table for user specified symbols. The RTE also requests TP logfile information (for replay purposes): @@ -951,6 +746,226 @@ upd:updIntraDay /switch upd to intraday update mode The message returned from the TP is passed to the function `InitializeTrade`. Once the RTE has finished initializing or replaying the TP logfile, the definition of `upd` is then switched to `updIntraDay` so the RTE can deal with intraday updates appropriately. +### Real-time trade with as-of quotes + +#### Overview + +One of the most popular and powerful joins in the q language is the [`aj`](../../ref/aj.md) function. This keyword was added to the language to solve a specific problem – how to join trade and quote tables together in such a way that for each trade, we grab the prevalent quote _as of_ the time of that trade. In other words, what is the last quote at or prior to the trade? + +This function is relatively easy to use for one-off joins. However, what if you want to maintain trades with as-of quotes in real time? This section describes how to build an RTE with real-time trades and as-of quotes. + +One additional feature this script demonstrates is the ability of any q process to write to and maintain its own kdb+ binary logfile for replay/recovery purposes. In this case, the RTE maintains its own daily logfile for trade records. This will be used for recovery in place of the standard tickerplant logfile. + +#### Example script + +This is a heavily modified version of an RDB ([`r.q`](../../architecture/rq.md)), written by the author and named `RealTimeTradeWithAsofQuotes.q`. + +```q +/ +The purpose of this script is as follows: +1. Demonstrate how custom RTEs can be created in q +2. In this example, create an efficient engine for calculating + the prevalent quotes as of trades in real-time. + This removes the need for ad-hoc invocations of the aj function. +3. In this example, this subscriber also maintains its own binary + log file for replay purposes. + This replaces the standard tickerplant log file replay functionality. +\ +show "RealTimeTradeWithAsofQuotes.q" +/sample usage +/q tick/RealTimeTradeWithAsofQuotes.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N + +/default command line arguments - tp is location of tickerplant. +/syms are the symbols we wish to subscribe to +default:`tp`syms!("::5000";"") + +args:.Q.opt .z.x /transform incoming cmd line arguments into a dictionary +args:`$default,args /upsert args into default +args[`tp] : hsym first args[`tp] + +/drop into debug mode if running in foreground AND +/errors occur (for debugging purposes) +\e 1 + +if[not "w"=first string .z.o;system "sleep 1"] + +/initialize schemas for custom RTE +InitializeSchemas:`trade`quote! + ( + {[x]`TradeWithQuote insert update bid:0n,bsize:0N,ask:0n,asize:0N from x}; + {[x]`LatestQuote upsert select by sym from x} + ); + +/intraday update functions +/Trade Update +/1. Update incoming data with latest quotes +/2. Insert updated data to TradeWithQuote table +/3. Append message to custom logfile +updTrade:{[d] + d:d lj LatestQuote; + `TradeWithQuote insert d; + LogfileHandle enlist (`replay;`TradeWithQuote;d); } + +/Quote Update +/1. Calculate latest quote per sym for incoming data +/2. Update LatestQuote table +updQuote:{[d] + `LatestQuote upsert select by sym from d; } + +/upd dictionary will be triggered upon incoming update from tickerplant +upd:`trade`quote!(updTrade;updQuote) + +/end of day function - triggered by tickerplant at EOD +.u.end:{ + hclose LogfileHandle; /close the connection to the old log file + /create the new logfile + logfile::hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D; + .[logfile;();:;()]; /Initialize the new log file + LogfileHandle::hopen logfile; + {delete from x}each tables `. /clear out tables + } + +/Initialize name of custom logfile +logfile:hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D; + +replay:{[t;d]t insert d} /custom log file replay function + +/attempt to replay custom log file +@[{-11!x;show"successfully replayed custom log file"}; logfile; + {[e] + m:"failed to replay custom log file"; + show m," - assume it does not exist. Creating it now"; + .[logfile;();:;()]; /Initialize the log file + } ] + +/open a connection to log file for writing +LogfileHandle:hopen logfile + +/ connect to tickerplant and subscribe to trade and quote for portfolio +h:hopen args`tp; /connect to tickerplant +InitializeSchemas . h(".u.sub";`trade;args`syms); +InitializeSchemas . h(".u.sub";`quote;args`syms); +``` + +This process should be started off as follows: + +```bash +q tick/RealTimeTradeWithAsofQuotes.q -tp localhost:5000 -syms MSFT.O IBM.N GS.N -p 5003 +``` + +This process will subscribe to both trade and quote tables for symbols `MSFT.O`, `IBM.N` and `GS.N` and will listen on port 5003. The author has deliberately made some of the q syntax more easily understandable compared to `r.q`. + +The first section of the script simply parses the command-line arguments and uses these to update some default values. + +The error flag [`\e`](../../basics/syscmds.md#e-error-trap-clients) is set for purely testing purposes. +When the developer runs this script in the foreground, +if errors occur at runtime as a result of incoming IPC messages, the process will drop into debug mode. +For example, if there is a problem with the definition of `upd`, then when an update is received from the tickerplant +we will drop into debug mode and (hopefully) identify the issue. + +We can see this RTE in action by examining the five most recent trades for `GS.N`: + +```q +q)-5#select from TradeWithQuote where sym=`GS.N +time sym price size bid bsize ask asize +--------------------------------------------------------------------- +0D21:50:58.857411000 GS.N 178.83 790 178.8148 25   178.8408 98 +0D21:51:00.158357000 GS.N 178.8315 312 178.8126 12   178.831 664 +0D21:51:01.157842000 GS.N 178.8463 307 178.8193 767 178.8383 697  +0D21:51:03.258055000 GS.N 178.8296 221 178.83 370 178.8627 358 +0D21:51:03.317152000 GS.N 178.8314 198 178.8296 915  178.8587 480 +``` + +#### Initialize desired table schemas + +`InitializeSchemas` defines the behavior of this RTE upon connecting and subscribing to the tickerplant’s trade and quote tables. +`InitializeSchemas` (defined as a dictionary which maps table names to unary function definitions) replaces [`.u.rep`](../../architecture/rq.md#urep) in `r.q`: + +The RTE’s trade table (named `TradeWithQuote`) maintains `bid`, `bsize`, `ask` and `asize` columns of appropriate type. +For the quote table, we just maintain a keyed table called `LatestQuote`, keyed on `sym` which will maintain the most recent quote per symbol. +This table will be used when joining prevalent quotes to incoming trades. + + +#### Intraday update behavior + +`updTrade` defines the intraday behavior upon receiving new trades. + +Besides inserting the new trades with prevalent quote information into the trade table, `updTrade` +also appends the new records to its custom logfile. This logfile will be replayed upon recovery/startup of the RTE. +Note that the replay function is named `replay`. This differs from the conventional TP logfile where the replay function was called `upd`. + +`updQuote` defines the intraday behavior upon receiving new quotes. + +The `upd` dictionary acts as a case statement – when an update for the trade table is received, +`updTrade` will be triggered with the message as argument. +Likewise, when an update for the quote table is received, `updQuote` will be triggered. + +In `r.q`, `upd` is defined as a function, not a dictionary. However we can use this dictionary definition for reasons discussed previously. + + +#### End of day + +At end of day, the tickerplant sends a message to all RTEs telling them to invoke their EOD function (`.u.end`): + +This function has been heavily modified from `r.q` to achieve the following desired behavior: + +* `hclose LogfileHandle` + * Close connection to the custom logfile. +* ``logfile::hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D`` + * Create the name of the new custom logfile. This logfile is a daily logfile – meaning it only contains one day’s trade records and it has today’s date in its name, just like the tickerplant’s logfile. +* `.[logfile;();:;()]` + * Initialize this logfile with an empty list. +* `LogfileHandle::hopen logfile` + * Establish a connection (handle) to this logfile for streaming writes. +* ``{delete from x}each tables `.`` + * Empty out the tables. + + +#### Replay custom logfile + +This section concerns the initialization and replay of the RTE’s custom logfile. + +```q +/Initialize name of custom logfile +logfile:hsym `$"RealTimeTradeWithAsofQuotes_",string .z.D + +replay:{[t;d]t insert d} /custom log file replay function +``` + +At this point, the name of today’s logfile and the definition of the logfile replay function have been established. The replay function will be invoked when replaying the process’s custom daily logfile. It is defined to simply insert the on-disk records into the in memory (`TradeWithQuote`) table. This will be a fast operation ensuring recovery is achieved quickly and efficiently. + +Upon startup, the process uses a try-catch to replay its custom daily logfile. If it fails for any reason (possibly because the logfile does not yet exist), it will send an appropriate message to standard out and will initialize this logfile. Replay of the logfile is achieved with the standard operator `-11!` as discussed previously. + +```q +/attempt to replay custom log file +@[{-11!x;show"successfully replayed custom log file"}; logfile; + {[e] + m:"failed to replay custom log file"; + show m," - assume it does not exist. Creating it now"; + .[logfile;();:;()]; /Initialize the log file + } ] +``` + +Once the logfile has been successfully replayed/initialized, a handle (connection) is established to it for subsequent streaming appends (upon new incoming trades from tickerplant): + +```q + /open a connection to log file for writing +LogfileHandle:hopen logfile +``` + +#### Subscribe to TP + +The next part of the script is probably the most critical – the process connects to the tickerplant and subscribes to the trade and quote table for user-specified symbols. + +```q +/ connect to tickerplant and subscribe to trade and quote for portfolio +h:hopen args`tp; /connect to tickerplant +InitializeSchemas . h(".u.sub";`trade;args`syms); +InitializeSchemas . h(".u.sub";`quote;args`syms); +``` + +The output of a subscription to a given table (for example `trade`) from the tickerplant is a 2-list, as discussed previously. This pair is in turn passed to the function `InitializeSchemas`. + ## Performance considerations From f765ce248f2a9df8d38fa42edcb621bd1d5ab3ff Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Thu, 1 Aug 2024 21:29:18 +0100 Subject: [PATCH 30/61] fix typo in ssl --- docs/kb/ssl.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/kb/ssl.md b/docs/kb/ssl.md index 824179708..016353648 100644 --- a/docs/kb/ssl.md +++ b/docs/kb/ssl.md @@ -202,8 +202,8 @@ $ export KX_SSL_CA_CERT_FILE=$HOME/certs/ca-cert.pem with the client as: ```bash $ export SSL_CERT_FILE=$HOME/certs/client-cert.pem -$ export SSL_KEY_FILE=$HOME/certs/client-private-key.pep -$ export KX_SSL_CA_CERT_FILE=/tmp/new/ca-cert.pem +$ export SSL_KEY_FILE=$HOME/certs/client-private-key.pem +$ export KX_SSL_CA_CERT_FILE=$HOME/certs/ca-cert.pem ``` :fontawesome-brands-github: From 266aef75528fa4930292cb1b1b2fa555f5164184 Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Fri, 2 Aug 2024 09:39:57 +0100 Subject: [PATCH 31/61] fixed incorrect calc for num blocks --- docs/kb/file-compression.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/kb/file-compression.md b/docs/kb/file-compression.md index bc9feb7f1..bcdab33cf 100644 --- a/docs/kb/file-compression.md +++ b/docs/kb/file-compression.md @@ -207,7 +207,7 @@ and on macOS, the OS command `purge` can be used. ### Compression parameters -The `logicalBlockSize` represents how much data is taken as a compression unit, and consequently the minimum size of a block to decompress. E.g. using a `logicalBlockSize` of 128kB, a file of size 128000kB would be cut into 100 blocks, and each block compressed independently of the others. Later, if a single byte is requested from that compressed file, a minimum of 128kB would be decompressed to access that byte. Fortunately those types of access patterns are rare, and typically you would be extracting clumps of data that make a logical block size of 128kB quite reasonable. +The `logicalBlockSize` represents how much data is taken as a compression unit, and consequently the minimum size of a block to decompress. E.g. using a `logicalBlockSize` of 128kB, a file of size 128000kB would be cut into 1000 blocks, and each block compressed independently of the others. Later, if a single byte is requested from that compressed file, a minimum of 128kB would be decompressed to access that byte. Fortunately those types of access patterns are rare, and typically you would be extracting clumps of data that make a logical block size of 128kB quite reasonable. Experiment to discover what suits your data, hardware and access patterns best. A good balance for TAQ data and typical TAQ queries is to use algorithm 1 (the same algorithm as used for IPC compression) with 128kB `logicalBlockSize`. To trade performance for better compression, choose gzip with compression level 6. From 3226674abcfcfae28a441bf5d8288d9ec0df96ce Mon Sep 17 00:00:00 2001 From: Simon Shanks Date: Fri, 2 Aug 2024 11:23:12 +0100 Subject: [PATCH 32/61] fix details on ssl thread for c api --- docs/interfaces/c-client-for-q.md | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/docs/interfaces/c-client-for-q.md b/docs/interfaces/c-client-for-q.md index 5259d8364..cd340b48b 100644 --- a/docs/interfaces/c-client-for-q.md +++ b/docs/interfaces/c-client-for-q.md @@ -584,9 +584,8 @@ if(handle==-3){ } ``` -Prior to 4.1t 2023.11.10, SSL/TLS connections can be used from the initialization thread only, i.e. the thread which first calls any `khp` function since the start of the application. It can now be used for one-shot synchronous requests. - -The lib is sensitive to the same environment variables as kdb+, noted at [Knowledge Base: SSL/TLS](../kb/ssl.md) +The lib is sensitive to the same environment variables as kdb+, noted at [Knowledge Base: SSL/TLS](../kb/ssl.md). +Using `khpunc` for SSL/TLS connections can be used from the initialization thread only, see [SSL/TLS thread](../kb/ssl.md#thread-support) support for more details. The OpenSSL libs are loaded dynamically, the first time a TLS connection is requested. It may be forced on startup with From c98d30c26fae1cf2fafa57b1802ecc26badedba9 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:01:51 +0100 Subject: [PATCH 33/61] Update docs/architecture/rq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/rq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 9abdc5c41..62e878227 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -11,7 +11,7 @@ keywords: kdb+, q, rdb, streaming A kdb+ process acting as an RDB stores a current day’s data in-memory for client queries. It can write its contents to disk at end-of-day, clearing out it in-memory data to prepare for the next day. -After writting data to disk, it will instruct a HDB to load the written data. +After writing data to disk, it instructs a HDB to load the written data. ### Customization From a8cdf53095af646d360c948671c92f5bea3574c0 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:02:05 +0100 Subject: [PATCH 34/61] Update docs/architecture/rq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/rq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 62e878227..7c48a1872 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -19,7 +19,7 @@ After writing data to disk, it instructs a HDB to load the written data. #### Memory use -The default RDB will store all of a days data in memory before end-of-day writting to disk. The host machines should be configured that all required resources +The default RDB stores all of a days data in memory before end-of-day writing to disk. The host machines should be configured that all required resources can handle the demands that may be made of them (both for today and the future). Depending on when there may be periods of low/no activity, [garbage collection](../ref/dotq.md#gc-garbage-collect) could be deployed after clearing tables at end-of-day, or a system for intra-day writedowns. From fadf500190979646839c2fd749d09956652cd40f Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:02:15 +0100 Subject: [PATCH 35/61] Update docs/architecture/rq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/rq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 7c48a1872..8c1a1679f 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -25,7 +25,7 @@ Depending on when there may be periods of low/no activity, [garbage collection]( #### User queries -A gateway process should control user queries and authorisation/authentication, using RDBs/RTEs/HDBs to retrieve the required information. +A gateway process should control user queries and authorization/authentication, using RDBs/RTEs/HDBs to retrieve the required information. If known/common queries can be designed, the RDB can load additional scripts to pre-define functions a gateway can call. ### End-of-day From db30bce2caf3814b5e3570e552d7cfccdb3c7d56 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:02:27 +0100 Subject: [PATCH 36/61] Update docs/architecture/rq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/rq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 8c1a1679f..990218c2b 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -30,7 +30,7 @@ If known/common queries can be designed, the RDB can load additional scripts to ### End-of-day -The end-of-day event is governed by the tickerplant process. The tickerplant calls the RDB [`.u.end`](#uend) function when this event occur. +The end-of-day event is governed by the tickerplant process. The tickerplant calls the RDB [`.u.end`](#uend) function when this event occurs. The main end-of-day event for an RDB is to save todays data from memory to disk, clear its tables and cause the HDB to be aware of a new days dataset for it to access. !!! Note "[.u.rep](#urep) sets the HDB directory be the same as the tickerplant log file directory. This can be edited to use a different directory if required" From 778290abc3876b9be01cd87e8516afaf60f75326 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:02:42 +0100 Subject: [PATCH 37/61] Update docs/wp/data-recovery.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/wp/data-recovery.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/wp/data-recovery.md b/docs/wp/data-recovery.md index d083cef7c..66d3100c4 100644 --- a/docs/wp/data-recovery.md +++ b/docs/wp/data-recovery.md @@ -63,7 +63,7 @@ Here, `functionname` and `tablename` are symbols, and `tabledata` is a row of da `upd `trade (0D14:56:01.122310000;`SGDUSD;"B";5000;98.14) ``` -In a standard tick system, each kdb+ message will call a function named `upd`. Each process may have a different definition of this function. A TP will publish each time the `upd` function is called (if its timer is not set to batch the data), and an RDB will simply insert the data into the relevant table. +In a standard tick system, each kdb+ message calls a function named `upd`. Each process may have a different definition of this function. A TP publishes each time the `upd` function is called (if its timer is not set to batch the data), and an RDB inserts the data into the relevant table. ## Recovery From 26b62156929f3a36837486f908ae6a6252ca7a22 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:03:02 +0100 Subject: [PATCH 38/61] Update docs/wp/data-recovery.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/wp/data-recovery.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/wp/data-recovery.md b/docs/wp/data-recovery.md index 66d3100c4..3aee5c1d5 100644 --- a/docs/wp/data-recovery.md +++ b/docs/wp/data-recovery.md @@ -116,7 +116,7 @@ kdb+ messages were described above in [_kdb+ messages and upd function_](#kdb-me [`-11!`](../basics/internal.md#-11-streaming-execute) is an internal function that will read each message in a tplog in turn, running the function `functioname` on the `(tablename;tabledata)` parameters. -In this section, we will focus on normal usage of the `-11!` function where the tplog is uncorrupted, and move on to discuss how to use it to recover from a corrupted tplog below in [_Replay errors_](#replay-errors). +This section focuses on normal usage of the `-11!` function where the tplog is uncorrupted, and then discusses how to use it to recover from a corrupted tplog below in [_Replay errors_](#replay-errors). There are three distinct usages of `-11!` when passed a list of two parameters, depending on the value of the first parameter. From 84eaa91b9dcc8808dcdb8419f01454f231977a4e Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:03:23 +0100 Subject: [PATCH 39/61] Update docs/wp/data-recovery.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/wp/data-recovery.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/wp/data-recovery.md b/docs/wp/data-recovery.md index 3aee5c1d5..ff4c6157c 100644 --- a/docs/wp/data-recovery.md +++ b/docs/wp/data-recovery.md @@ -114,7 +114,7 @@ kdb+ messages were described above in [_kdb+ messages and upd function_](#kdb-me ### `-11!` functionality -[`-11!`](../basics/internal.md#-11-streaming-execute) is an internal function that will read each message in a tplog in turn, running the function `functioname` on the `(tablename;tabledata)` parameters. +[`-11!`](../basics/internal.md#-11-streaming-execute) is an internal function that reads each message in a tplog in turn, running the function `functioname` on the `(tablename;tabledata)` parameters. This section focuses on normal usage of the `-11!` function where the tplog is uncorrupted, and then discusses how to use it to recover from a corrupted tplog below in [_Replay errors_](#replay-errors). From a0e79c5baa26e9415156cc8e2b41a8a924889b3e Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:15:51 +0100 Subject: [PATCH 40/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index 0f46d12e4..e6a729ca3 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -14,7 +14,7 @@ A tickerplant writes all data to a tickerplant log (to permit data recovery) and ### Customization -`tick.q` provides a starting point to most environments. The source code is freely avaialble and can be tailered to individual needs. +`tick.q` provides a starting point to most environments. The source code is freely available and can be tailored to individual needs. ### Schema file From 48ae81868634f992d4334f2580cfaea4887cb6fa Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:16:11 +0100 Subject: [PATCH 41/61] Update docs/architecture/rq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/rq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/rq.md b/docs/architecture/rq.md index 990218c2b..7bd3164d0 100644 --- a/docs/architecture/rq.md +++ b/docs/architecture/rq.md @@ -40,7 +40,7 @@ The main end-of-day event for an RDB is to save todays data from memory to disk, Using IPC, the RDB process can retrieve the current tickerplant log location and use via the [variables](tickq.md#variables) the tickerplant maintains. The function [`.u.rep`](#urep) is then used to populate any tables from the log. -!!! Note "The RDB should have tickerplant log available from a directory on the same machine. The RDB/tickerplant can be changed to reside on different hosts but this will entail the increased resource utilisation of transmitting the log file contents over the network." +!!! Note "The RDB should be able to access the tickerplant log from a directory on the same machine. The RDB/tickerplant can be changed to reside on different hosts but this increases the resources needed to transmit the log file contents over the network." ## Usage From 45fcb4c41e69c16a37d8a0497deefd28ac03f3b8 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:17:09 +0100 Subject: [PATCH 42/61] Update docs/wp/capi/index.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/wp/capi/index.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/wp/capi/index.md b/docs/wp/capi/index.md index 611c9e0d8..548659976 100644 --- a/docs/wp/capi/index.md +++ b/docs/wp/capi/index.md @@ -1341,7 +1341,7 @@ The tickerplant process is started from the command line as follows. $ q tick.q trade /logs/tickerplant/ -p 5010 ``` -Above, the first argument following [`tick.q`](../../architecture/tickq.md) is the name of the table schema file to use. The second argument is the location where the tickerplant log file will be created and the value following the `-p` option is the port the tickerplant will listen on. The C process will use this port number when initializing the connection. The final step in this setup is to create a kdb+ mock feedhandler process which will act as the trade data source for the tickerplant. Below is a simple publishing process which is sufficient for the demonstration. +Above, the first argument following [`tick.q`](../../architecture/tickq.md) is the name of the table schema file to use. The second argument is the location where the tickerplant log file is created and the value following the `-p` option is the port the tickerplant listens on. The C process uses this port number when initializing the connection. The final step in this setup is to create a kdb+ mock feedhandler process, which acts as the trade data source for the tickerplant. Below is a simple publishing process. ```q /* File name: feed.q */ From fba5958c4a00f43bf5bcb421a74c5ece895c5ea0 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:17:29 +0100 Subject: [PATCH 43/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index e6a729ca3..f2e6ed10c 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -19,7 +19,7 @@ A tickerplant writes all data to a tickerplant log (to permit data recovery) and ### Schema file A tickerplant requires a schema file. -A schema file should be created to describe the data you plan to capture, by specifing the tables that will be populated by the tickerplant environment. +A schema file describes the data you plan to capture, by specifying the tables to be populated by the tickerplant environment. The [datatypes](../basics/datatypes.md) and [attributes](../ref/set-attribute.md) are denoted within the file as shown in this example: ```q quote:([]time:`timespan$(); sym:`g#`symbol$(); bid:`float$(); ask:`float$(); bsize:`long$(); asize:`long$(); mode:`char$(); ex:`char$()) From 9d7abb66ca1d5358e8c9a86303ee758924d854e4 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:17:47 +0100 Subject: [PATCH 44/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index f2e6ed10c..e7fed33c3 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -34,7 +34,7 @@ _The mode is controlled via the [`-t`](#usage) command line parameter._ Batch mode can alleviate CPU use on both the tickerplant and its subscribers by grouping together multiple ticks within the timer interval prior to sending/writing. This will be at the expense of tickerplant memory (required memory to hold several ticks) and increased latency that may occur between adding to the batch and sending. There is no ideal setting for all deployments as it depends on the frequency of the ticks received. -Real-time mode will process every tick as soon as they occur. +Real-time mode processes every tick as soon as they occur. !!! note "A feedhandler can be written to send messages comprising of multiple ticks to a tickerplant. In this situation real-time mode will already be processing batches of messages." From da357a6f0e6da7e240d758d9b5a73ffa9bfbcc06 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:18:24 +0100 Subject: [PATCH 45/61] Update docs/wp/data-recovery.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/wp/data-recovery.md | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/docs/wp/data-recovery.md b/docs/wp/data-recovery.md index ff4c6157c..6513ec095 100644 --- a/docs/wp/data-recovery.md +++ b/docs/wp/data-recovery.md @@ -98,9 +98,11 @@ if[l; l enlist(`upd;t;x); j+:1] ### Replaying a tplog -Recovery of an RDB typically involves simply restarting the process. On startup, an RDB will subscribe to a TP and will receive information about the message count ([`.u.i`](../architecture/tickq.md#variables)) and location of the tplog ([`.u.L`](../architecture/tickq.md#variables)) in return. +Recovery of an RDB involves restarting the process. On startup, an RDB subscribes to a TP and receives the following information: +- message count ([`.u.i`](../architecture/tickq.md#variables)) +- location of the tplog ([`.u.L`](../architecture/tickq.md#variables)). -It will then replay this tplog to recover all the data that has passed through the TP up to that point in the day. The replay is achieved using [`-11!`](../basics/internal.md#-11-streaming-execute), the streaming replay function. +It then replays this tplog to recover all the data that has passed through the TP up to that point in the day. The replay is achieved using [`-11!`](../basics/internal.md#-11-streaming-execute), the streaming replay function. This is called within `.u.rep`, which is executed when the RDB connects to the TP. From 97c6c3ff244f648a5105a1c6946b24d2965f491f Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:18:41 +0100 Subject: [PATCH 46/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index e7fed33c3..5b9459535 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -32,7 +32,7 @@ The default setup requires the first two columns to be `time` and `sym`. _The mode is controlled via the [`-t`](#usage) command line parameter._ Batch mode can alleviate CPU use on both the tickerplant and its subscribers by grouping together multiple ticks within the timer interval prior to sending/writing. -This will be at the expense of tickerplant memory (required memory to hold several ticks) and increased latency that may occur between adding to the batch and sending. +This comes at the expense of tickerplant memory (required memory to hold several ticks) and increased latency that may occur between adding to the batch and sending. There is no ideal setting for all deployments as it depends on the frequency of the ticks received. Real-time mode processes every tick as soon as they occur. From 2448c7e7fd5b85fe12960c9ed42ae9d6249744c3 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:18:58 +0100 Subject: [PATCH 47/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index 5b9459535..1f06b91d4 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -167,7 +167,7 @@ Actions performed: * using [`.u.L`](#variables), change last 10 chars to provided date and create log file if it doesnt yet exist * set [`.u.i`](#variables) and [`.u.j`](#variables) to count of valid messages currently in log file -* if log file is found to be corrupt (size bigger than size of number of valid messages) an error will be returned +* if log file is found to be corrupt (size bigger than size of number of valid messages) an error is returned * open new/existing log file ### .u.ts From 8b23bc5e03ba2fc6eaf5c461b15e126a5db4d9e1 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:21:05 +0100 Subject: [PATCH 48/61] Update docs/architecture/tickq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/tickq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/tickq.md b/docs/architecture/tickq.md index 1f06b91d4..8e5d5da64 100644 --- a/docs/architecture/tickq.md +++ b/docs/architecture/tickq.md @@ -197,7 +197,7 @@ Where #### Batch Mode -Add each recieved message to batch and record message to tickerplant log. Batch will be published on running timer. +Add each received message to the batch and record message to the tickerplant log. Batch is published on running timer. Actions performed: * If the first element of `y` is not a timespan (or list of timespan) From e599d1fea24f1972aab31ab81b1a2e25df0c668b Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:21:26 +0100 Subject: [PATCH 49/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 96e31a175..f7c5d91c8 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -9,7 +9,7 @@ keywords: kdb+, q, tick, tickerplant, streaming ## Overview -Contains functions to allow clients to subscribe to all or subsets of available data, publishing to interested clients & alerting clients to events (i.e. end-of-day). Tracks client subscription interest and removes a client subscription details on their disconnection. +Contains functions to allow clients to subscribe to all or subsets of available data, publishing to interested clients and alerting clients to events, for example, end-of-day. Tracks client subscription interest and removes client subscription details on their disconnection. This script is loaded by other processes, for example [a tickerplant](tickq.md). From 82175a65d0a7b281958e82e1470ce450d5e26618 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:21:52 +0100 Subject: [PATCH 50/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index f7c5d91c8..2a46294fc 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -15,7 +15,7 @@ This script is loaded by other processes, for example [a tickerplant](tickq.md). ## Usage -To give the ability to publish data to any process, a few things need to be done: +To allow the ability to publish data to any process, do the following: - load `u.q` - declare the tables to be published in the top level namespace. Each table must contain a column called `sym`, which acts as the single key field to which subscribers subscribe From 21cd945d707d480c774d6abcc795b9bdcdfd7021 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:22:17 +0100 Subject: [PATCH 51/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 2a46294fc..3230d8be3 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -26,8 +26,8 @@ The list of tables that can be published and the processes currently subscribed Subscriber processes must open a connection to the publisher and call [`.u.sub[tablename;list_of_symbols_to_subscribe_to]`](#usub). -If a subscriber calls `.u.sub` again, the current subscription will be overwritten either for all tables (if a wildcard is used) or the specified table. -To add to a subscription (e.g. add more syms to a current subscription) the subscriber can call [`.u.add`](#uadd)). +If a subscriber calls `.u.sub` again, the current subscription is overwritten either for all tables (if a wildcard is used) or the specified table. +To add to a subscription, for example, add more `syms` to a current subscription, the subscriber can call [`.u.add`](#uadd)). Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u.end`](rq.md#uend) function for end-of-day events From 3b5f60d18673f9e0b964ca0da79abbf0c7cf93ec Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:22:38 +0100 Subject: [PATCH 52/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 3230d8be3..73dea04e4 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -29,7 +29,7 @@ Subscriber processes must open a connection to the publisher and call [`.u.sub[t If a subscriber calls `.u.sub` again, the current subscription is overwritten either for all tables (if a wildcard is used) or the specified table. To add to a subscription, for example, add more `syms` to a current subscription, the subscriber can call [`.u.add`](#uadd)). -Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u.end`](rq.md#uend) function for end-of-day events +Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u.end`](rq.md#uend) function for end-of-day events. ## Variables From 9653dba6f91a3e3574053fb21e18048832bb0bbe Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:22:54 +0100 Subject: [PATCH 53/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 73dea04e4..6fd4f9738 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -35,7 +35,7 @@ Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u | Name | Description | | ---- | ---- | -| .u.w | Dictionary of registered client interest in data being processed i.e. (tables->(handle;syms) | +| .u.w | Dictionary of registered client interest in data being processed (for example, tables->(handle;syms) | | .u.t | Table names | ## Functions From a5fef08255c55ab134ed444a575eebbfc1826da7 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:23:06 +0100 Subject: [PATCH 54/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 6fd4f9738..27fbd89c8 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -40,7 +40,7 @@ Clients should define a [`upd`](rq.md#upd) function to receive updates, and [`.u ## Functions -Functions are open source & open to customisation. +Functions are open source and open to customisation. ### .u.init From 996aef3867edc01d698be77bf2c6d994b12534c3 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:23:24 +0100 Subject: [PATCH 55/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 27fbd89c8..d4b9e49c8 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -119,7 +119,7 @@ Actions performed: Returns 2 element list. The first element is the table name. The second element depends on whether `x` refers to a keyed table. * If `x` is a keyed table, [`.u.sel`](#usel) is used to select from the keyed table the required syms -* otherwise returns an empty table `x` (i.e. schema definition of table), with the [grouped attribute](../ref/set-attribute.md#grouped-and-parted) applied to the sym column. +* otherwise returns an empty table `x` (schema definition of table), with the [grouped attribute](../ref/set-attribute.md#grouped-and-parted) applied to the sym column. ### .u.sub From 6c2d359b35f33930d0f0a7589ba616b5a9154f2d Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:23:45 +0100 Subject: [PATCH 56/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index d4b9e49c8..9180d5bbf 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -166,7 +166,7 @@ Called when a client disconnects. The client handle provided is used to call [`. In addition, the example scripts below demonstrate pub/sub in a standalone publisher and subscriber. They can be downloaded from :fontawesome-brands-github:[KxSystems/cookbook/pubsub](https://github.com/KxSystems/cookbook/tree/master/pubsub). -Each script should be run from the OS command prompt e.g. +Each script should be run from the OS command prompt as shown in the following example. ```bash $ q publisher.q From fa3f3fceee808a262a00fc93c3fcec29793eed1d Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:24:07 +0100 Subject: [PATCH 57/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 9180d5bbf..08f38d9ef 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -173,6 +173,6 @@ $ q publisher.q $ q subscriber.q ``` -The publisher will generate some random data and publish it periodically on a timer. +The publisher generates some random data and publishes it periodically on a timer. The subscriber will receive data from the publisher and output it to the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. From 9586ea739aa81d1896b86332287e1beec085eb99 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:24:46 +0100 Subject: [PATCH 58/61] Update docs/architecture/uq.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/architecture/uq.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/architecture/uq.md b/docs/architecture/uq.md index 08f38d9ef..2499fc5a6 100644 --- a/docs/architecture/uq.md +++ b/docs/architecture/uq.md @@ -175,4 +175,4 @@ $ q subscriber.q The publisher generates some random data and publishes it periodically on a timer. -The subscriber will receive data from the publisher and output it to the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. +The subscriber receives data from the publisher and is displayed on the screen. You can modify the subscription request and the `upd` function of the subscriber as required. You can run multiple subscribers at once. From a156a59d48a55c921640025f2f88e292d4a27ca2 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:25:02 +0100 Subject: [PATCH 59/61] Update docs/learn/startingkdb/hdb.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/learn/startingkdb/hdb.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/learn/startingkdb/hdb.md b/docs/learn/startingkdb/hdb.md index 03f5f83ef..51c82d018 100644 --- a/docs/learn/startingkdb/hdb.md +++ b/docs/learn/startingkdb/hdb.md @@ -38,7 +38,7 @@ For example, a simple partitioning scheme on a single disk might be as shown rig ## Sample partitioned database -The script :fontawesome-brands-github:[`KxSystems/cookbook/start/buildhdb.q`](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) will build a sample HDB. +The script :fontawesome-brands-github:[`KxSystems/cookbook/start/buildhdb.q`](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) builds a sample HDB. It builds a month’s random data in directory `start/db`, and takes a few seconds to run. Load q, then: From b71372e07929887f4c7fc131b2c99ff635cc6889 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:25:45 +0100 Subject: [PATCH 60/61] Update docs/learn/startingkdb/hdb.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/learn/startingkdb/hdb.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/learn/startingkdb/hdb.md b/docs/learn/startingkdb/hdb.md index 51c82d018..27cbfeb56 100644 --- a/docs/learn/startingkdb/hdb.md +++ b/docs/learn/startingkdb/hdb.md @@ -39,7 +39,7 @@ For example, a simple partitioning scheme on a single disk might be as shown rig ## Sample partitioned database The script :fontawesome-brands-github:[`KxSystems/cookbook/start/buildhdb.q`](https://github.com/KxSystems/cookbook/blob/master/start/buildhdb.q) builds a sample HDB. -It builds a month’s random data in directory `start/db`, and takes a few seconds to run. +It builds a month’s random data in directory `start/db`. Load q, then: From 4d81c71ccc6c73339f68a585df0f9a069d79d974 Mon Sep 17 00:00:00 2001 From: Simon Shanks <59612559+sshanks-kx@users.noreply.github.com> Date: Fri, 2 Aug 2024 19:26:03 +0100 Subject: [PATCH 61/61] Update docs/ref/aj.md Co-authored-by: natalietanner <142605282+natalietanner@users.noreply.github.com> --- docs/ref/aj.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/ref/aj.md b/docs/ref/aj.md index ec195b597..a9a9c0bc9 100644 --- a/docs/ref/aj.md +++ b/docs/ref/aj.md @@ -59,7 +59,7 @@ time sym qty px `aj` is a [multithreaded primitive](../kb/mt-primitives.md). -!!! tip "There is no requirement for any of the join columns to be keys but the join will be faster on keys." +!!! tip "There is no requirement for any of the join columns to be keys but the join is faster on keys." ## `aj`, `aj0`