//////////////////////////////////////////////////////////////////////////////// // THE HIGH-LEVEL VIEW // //////////////////////////////////////////////////////////////////////////////// // Process: When you first open a connection, send the magic number // for the version of the protobuf you're targeting (in the [Version] // enum). This should **NOT** be sent as a protobuf; just send the // little-endian 32-bit integer over the wire raw. This number should // only be sent once per connection. // The magic number shall be followed by an authorization key. The // first 4 bytes are the length of the key to be sent as a little-endian // 32-bit integer, followed by the key string. Even if there is no key, // an empty string should be sent (length 0 and no data). // Following the authorization key, the client shall send a magic number // for the communication protocol they want to use (in the [Protocol] // enum). This shall be a little-endian 32-bit integer. // The server will then respond with a NULL-terminated string response. // "SUCCESS" indicates that the connection has been accepted. Any other // response indicates an error, and the response string should describe // the error. // Next, for each query you want to send, construct a [Query] protobuf // and serialize it to a binary blob. Send the blob's size to the // server encoded as a little-endian 32-bit integer, followed by the // blob itself. You will recieve a [Response] protobuf back preceded // by its own size, once again encoded as a little-endian 32-bit // integer. You can see an example exchange below in **EXAMPLE**. // A query consists of a [Term] to evaluate and a unique-per-connection // [token]. // Tokens are used for two things: // * Keeping track of which responses correspond to which queries. // * Batched queries. Some queries return lots of results, so we send back // batches of <1000, and you need to send a [CONTINUE] query with the same // token to get more results from the original query. //////////////////////////////////////////////////////////////////////////////// message VersionDummy { // We need to wrap it like this for some // non-conforming protobuf libraries // This enum contains the magic numbers for your version. See **THE HIGH-LEVEL // VIEW** for what to do with it. enum Version { V0_1 = 0x3f61ba36; V0_2 = 0x723081e1; // Authorization key during handshake V0_3 = 0x5f75e83e; // Authorization key and protocol during handshake V0_4 = 0x400c2d20; // Queries execute in parallel V1_0 = 0x34c2bdc3; // Users and permissions } // The protocol to use after the handshake, specified in V0_3 enum Protocol { PROTOBUF = 0x271ffc41; JSON = 0x7e6970c7; } } // You send one of: // * A [START] query with a [Term] to evaluate and a unique-per-connection token. // * A [CONTINUE] query with the same token as a [START] query that returned // [SUCCESS_PARTIAL] in its [Response]. // * A [STOP] query with the same token as a [START] query that you want to stop. // * A [NOREPLY_WAIT] query with a unique per-connection token. The server answers // with a [WAIT_COMPLETE] [Response]. // * A [SERVER_INFO] query. The server answers with a [SERVER_INFO] [Response]. message Query { enum QueryType { START = 1; // Start a new query. CONTINUE = 2; // Continue a query that returned [SUCCESS_PARTIAL] // (see [Response]). STOP = 3; // Stop a query partway through executing. NOREPLY_WAIT = 4; // Wait for noreply operations to finish. SERVER_INFO = 5; // Get server information. } optional QueryType type = 1; // A [Term] is how we represent the operations we want a query to perform. optional Term query = 2; // only present when [type] = [START] optional int64 token = 3; // This flag is ignored on the server. `noreply` should be added // to `global_optargs` instead (the key "noreply" should map to // either true or false). optional bool OBSOLETE_noreply = 4 [default = false]; // If this is set to [true], then [Datum] values will sometimes be // of [DatumType] [R_JSON] (see below). This can provide enormous // speedups in languages with poor protobuf libraries. optional bool accepts_r_json = 5 [default = false]; message AssocPair { optional string key = 1; optional Term val = 2; } repeated AssocPair global_optargs = 6; } // A backtrace frame (see `backtrace` in Response below) message Frame { enum FrameType { POS = 1; // Error occurred in a positional argument. OPT = 2; // Error occurred in an optional argument. } optional FrameType type = 1; optional int64 pos = 2; // The index of the positional argument. optional string opt = 3; // The name of the optional argument. } message Backtrace { repeated Frame frames = 1; } // You get back a response with the same [token] as your query. message Response { enum ResponseType { // These response types indicate success. SUCCESS_ATOM = 1; // Query returned a single RQL datatype. SUCCESS_SEQUENCE = 2; // Query returned a sequence of RQL datatypes. SUCCESS_PARTIAL = 3; // Query returned a partial sequence of RQL // datatypes. If you send a [CONTINUE] query with // the same token as this response, you will get // more of the sequence. Keep sending [CONTINUE] // queries until you get back [SUCCESS_SEQUENCE]. WAIT_COMPLETE = 4; // A [NOREPLY_WAIT] query completed. SERVER_INFO = 5; // The data for a [SERVER_INFO] request. This is // the same as `SUCCESS_ATOM` except that there will // never be profiling data. // These response types indicate failure. CLIENT_ERROR = 16; // Means the client is buggy. An example is if the // client sends a malformed protobuf, or tries to // send [CONTINUE] for an unknown token. COMPILE_ERROR = 17; // Means the query failed during parsing or type // checking. For example, if you pass too many // arguments to a function. RUNTIME_ERROR = 18; // Means the query failed at runtime. An example is // if you add together two values from a table, but // they turn out at runtime to be booleans rather // than numbers. } optional ResponseType type = 1; // If `ResponseType` is `RUNTIME_ERROR`, this may be filled in with more // information about the error. enum ErrorType { INTERNAL = 1000000; RESOURCE_LIMIT = 2000000; QUERY_LOGIC = 3000000; NON_EXISTENCE = 3100000; OP_FAILED = 4100000; OP_INDETERMINATE = 4200000; USER = 5000000; PERMISSION_ERROR = 6000000; } optional ErrorType error_type = 7; // ResponseNotes are used to provide information about the query // response that may be useful for people writing drivers or ORMs. // Currently all the notes we send indicate that a stream has certain // special properties. enum ResponseNote { // The stream is a changefeed stream (e.g. `r.table('test').changes()`). SEQUENCE_FEED = 1; // The stream is a point changefeed stream // (e.g. `r.table('test').get(0).changes()`). ATOM_FEED = 2; // The stream is an order_by_limit changefeed stream // (e.g. `r.table('test').order_by(index: 'id').limit(5).changes()`). ORDER_BY_LIMIT_FEED = 3; // The stream is a union of multiple changefeed types that can't be // collapsed to a single type // (e.g. `r.table('test').changes().union(r.table('test').get(0).changes())`). UNIONED_FEED = 4; // The stream is a changefeed stream and includes notes on what state // the changefeed stream is in (e.g. objects of the form `{state: // 'initializing'}`). INCLUDES_STATES = 5; } repeated ResponseNote notes = 6; optional int64 token = 2; // Indicates what [Query] this response corresponds to. // [response] contains 1 RQL datum if [type] is [SUCCESS_ATOM] or // [SERVER_INFO]. [response] contains many RQL data if [type] is // [SUCCESS_SEQUENCE] or [SUCCESS_PARTIAL]. [response] contains 1 // error message (of type [R_STR]) in all other cases. repeated Datum response = 3; // If [type] is [CLIENT_ERROR], [TYPE_ERROR], or [RUNTIME_ERROR], then a // backtrace will be provided. The backtrace says where in the query the // error occurred. Ideally this information will be presented to the user as // a pretty-printed version of their query with the erroneous section // underlined. A backtrace is a series of 0 or more [Frame]s, each of which // specifies either the index of a positional argument or the name of an // optional argument. (Those words will make more sense if you look at the // [Term] message below.) optional Backtrace backtrace = 4; // Contains n [Frame]s when you get back an error. // If the [global_optargs] in the [Query] that this [Response] is a // response to contains a key "profile" which maps to a static value of // true then [profile] will contain a [Datum] which provides profiling // information about the execution of the query. This field should be // returned to the user along with the result that would normally be // returned (a datum or a cursor). In official drivers this is accomplished // by putting them inside of an object with "value" mapping to the return // value and "profile" mapping to the profile object. optional Datum profile = 5; } // A [Datum] is a chunk of data that can be serialized to disk or returned to // the user in a Response. Currently we only support JSON types, but we may // support other types in the future (e.g., a date type or an integer type). message Datum { enum DatumType { R_NULL = 1; R_BOOL = 2; R_NUM = 3; // a double R_STR = 4; R_ARRAY = 5; R_OBJECT = 6; // This [DatumType] will only be used if [accepts_r_json] is // set to [true] in [Query]. [r_str] will be filled with a // JSON encoding of the [Datum]. R_JSON = 7; // uses r_str } optional DatumType type = 1; optional bool r_bool = 2; optional double r_num = 3; optional string r_str = 4; repeated Datum r_array = 5; message AssocPair { optional string key = 1; optional Datum val = 2; } repeated AssocPair r_object = 6; } // A [Term] is either a piece of data (see **Datum** above), or an operator and // its operands. If you have a [Datum], it's stored in the member [datum]. If // you have an operator, its positional arguments are stored in [args] and its // optional arguments are stored in [optargs]. // // A note about type signatures: // We use the following notation to denote types: // arg1_type, arg2_type, argrest_type... -> result_type // So, for example, if we have a function `avg` that takes any number of // arguments and averages them, we might write: // NUMBER... -> NUMBER // Or if we had a function that took one number modulo another: // NUMBER, NUMBER -> NUMBER // Or a function that takes a table and a primary key of any Datum type, then // retrieves the entry with that primary key: // Table, DATUM -> OBJECT // Some arguments must be provided as literal values (and not the results of sub // terms). These are marked with a `!`. // Optional arguments are specified within curly braces as argname `:` value // type (e.x `{noreply:BOOL}`) // Many RQL operations are polymorphic. For these, alterantive type signatures // are separated by `|`. // // The RQL type hierarchy is as follows: // Top // DATUM // NULL // BOOL // NUMBER // STRING // OBJECT // SingleSelection // ARRAY // Sequence // ARRAY // Stream // StreamSelection // Table // Database // Function // Ordering - used only by ORDER_BY // Pathspec -- an object, string, or array that specifies a path // Error message Term { enum TermType { // A RQL datum, stored in `datum` below. DATUM = 1; MAKE_ARRAY = 2; // DATUM... -> ARRAY // Evaluate the terms in [optargs] and make an object MAKE_OBJ = 3; // {...} -> OBJECT // * Compound types // Takes an integer representing a variable and returns the value stored // in that variable. It's the responsibility of the client to translate // from their local representation of a variable to a unique _non-negative_ // integer for that variable. (We do it this way instead of letting // clients provide variable names as strings to discourage // variable-capturing client libraries, and because it's more efficient // on the wire.) VAR = 10; // !NUMBER -> DATUM // Takes some javascript code and executes it. JAVASCRIPT = 11; // STRING {timeout: !NUMBER} -> DATUM | // STRING {timeout: !NUMBER} -> Function(*) UUID = 169; // () -> DATUM // Takes an HTTP URL and gets it. If the get succeeds and // returns valid JSON, it is converted into a DATUM HTTP = 153; // STRING {data: OBJECT | STRING, // timeout: !NUMBER, // method: STRING, // params: OBJECT, // header: OBJECT | ARRAY, // attempts: NUMBER, // redirects: NUMBER, // verify: BOOL, // page: FUNC | STRING, // page_limit: NUMBER, // auth: OBJECT, // result_format: STRING, // } -> STRING | STREAM // Takes a string and throws an error with that message. // Inside of a `default` block, you can omit the first // argument to rethrow whatever error you catch (this is most // useful as an argument to the `default` filter optarg). ERROR = 12; // STRING -> Error | -> Error // Takes nothing and returns a reference to the implicit variable. IMPLICIT_VAR = 13; // -> DATUM // * Data Operators // Returns a reference to a database. DB = 14; // STRING -> Database // Returns a reference to a table. TABLE = 15; // Database, STRING, {read_mode:STRING, identifier_format:STRING} -> Table // STRING, {read_mode:STRING, identifier_format:STRING} -> Table // Gets a single element from a table by its primary or a secondary key. GET = 16; // Table, STRING -> SingleSelection | Table, NUMBER -> SingleSelection | // Table, STRING -> NULL | Table, NUMBER -> NULL | GET_ALL = 78; // Table, DATUM..., {index:!STRING} => ARRAY // Simple DATUM Ops EQ = 17; // DATUM... -> BOOL NE = 18; // DATUM... -> BOOL LT = 19; // DATUM... -> BOOL LE = 20; // DATUM... -> BOOL GT = 21; // DATUM... -> BOOL GE = 22; // DATUM... -> BOOL NOT = 23; // BOOL -> BOOL // ADD can either add two numbers or concatenate two arrays. ADD = 24; // NUMBER... -> NUMBER | STRING... -> STRING SUB = 25; // NUMBER... -> NUMBER MUL = 26; // NUMBER... -> NUMBER DIV = 27; // NUMBER... -> NUMBER MOD = 28; // NUMBER, NUMBER -> NUMBER FLOOR = 183; // NUMBER -> NUMBER CEIL = 184; // NUMBER -> NUMBER ROUND = 185; // NUMBER -> NUMBER // DATUM Array Ops // Append a single element to the end of an array (like `snoc`). APPEND = 29; // ARRAY, DATUM -> ARRAY // Prepend a single element to the end of an array (like `cons`). PREPEND = 80; // ARRAY, DATUM -> ARRAY //Remove the elements of one array from another array. DIFFERENCE = 95; // ARRAY, ARRAY -> ARRAY // DATUM Set Ops // Set ops work on arrays. They don't use actual sets and thus have // performance characteristics you would expect from arrays rather than // from sets. All set operations have the post condition that they // array they return contains no duplicate values. SET_INSERT = 88; // ARRAY, DATUM -> ARRAY SET_INTERSECTION = 89; // ARRAY, ARRAY -> ARRAY SET_UNION = 90; // ARRAY, ARRAY -> ARRAY SET_DIFFERENCE = 91; // ARRAY, ARRAY -> ARRAY SLICE = 30; // Sequence, NUMBER, NUMBER -> Sequence SKIP = 70; // Sequence, NUMBER -> Sequence LIMIT = 71; // Sequence, NUMBER -> Sequence OFFSETS_OF = 87; // Sequence, DATUM -> Sequence | Sequence, Function(1) -> Sequence CONTAINS = 93; // Sequence, (DATUM | Function(1))... -> BOOL // Stream/Object Ops // Get a particular field from an object, or map that over a // sequence. GET_FIELD = 31; // OBJECT, STRING -> DATUM // | Sequence, STRING -> Sequence // Return an array containing the keys of the object. KEYS = 94; // OBJECT -> ARRAY // Return an array containing the values of the object. VALUES = 186; // OBJECT -> ARRAY // Creates an object OBJECT = 143; // STRING, DATUM, ... -> OBJECT // Check whether an object contains all the specified fields, // or filters a sequence so that all objects inside of it // contain all the specified fields. HAS_FIELDS = 32; // OBJECT, Pathspec... -> BOOL // x.with_fields(...) <=> x.has_fields(...).pluck(...) WITH_FIELDS = 96; // Sequence, Pathspec... -> Sequence // Get a subset of an object by selecting some attributes to preserve, // or map that over a sequence. (Both pick and pluck, polymorphic.) PLUCK = 33; // Sequence, Pathspec... -> Sequence | OBJECT, Pathspec... -> OBJECT // Get a subset of an object by selecting some attributes to discard, or // map that over a sequence. (Both unpick and without, polymorphic.) WITHOUT = 34; // Sequence, Pathspec... -> Sequence | OBJECT, Pathspec... -> OBJECT // Merge objects (right-preferential) MERGE = 35; // OBJECT... -> OBJECT | Sequence -> Sequence // Sequence Ops // Get all elements of a sequence between two values. // Half-open by default, but the openness of either side can be // changed by passing 'closed' or 'open for `right_bound` or // `left_bound`. BETWEEN_DEPRECATED = 36; // Deprecated version of between, which allows `null` to specify unboundedness // With the newer version, clients should use `r.minval` and `r.maxval` for unboundedness BETWEEN = 182; // StreamSelection, DATUM, DATUM, {index:!STRING, right_bound:STRING, left_bound:STRING} -> StreamSelection REDUCE = 37; // Sequence, Function(2) -> DATUM MAP = 38; // Sequence, Function(1) -> Sequence // The arity of the function should be // Sequence..., Function(sizeof...(Sequence)) -> Sequence FOLD = 187; // Sequence, Datum, Function(2), {Function(3), Function(1) // Filter a sequence with either a function or a shortcut // object (see API docs for details). The body of FILTER is // wrapped in an implicit `.default(false)`, and you can // change the default value by specifying the `default` // optarg. If you make the default `r.error`, all errors // caught by `default` will be rethrown as if the `default` // did not exist. FILTER = 39; // Sequence, Function(1), {default:DATUM} -> Sequence | // Sequence, OBJECT, {default:DATUM} -> Sequence // Map a function over a sequence and then concatenate the results together. CONCAT_MAP = 40; // Sequence, Function(1) -> Sequence // Order a sequence based on one or more attributes. ORDER_BY = 41; // Sequence, (!STRING | Ordering)..., {index: (!STRING | Ordering)} -> Sequence // Get all distinct elements of a sequence (like `uniq`). DISTINCT = 42; // Sequence -> Sequence // Count the number of elements in a sequence, or only the elements that match // a given filter. COUNT = 43; // Sequence -> NUMBER | Sequence, DATUM -> NUMBER | Sequence, Function(1) -> NUMBER IS_EMPTY = 86; // Sequence -> BOOL // Take the union of multiple sequences (preserves duplicate elements! (use distinct)). UNION = 44; // Sequence... -> Sequence // Get the Nth element of a sequence. NTH = 45; // Sequence, NUMBER -> DATUM // do NTH or GET_FIELD depending on target object BRACKET = 170; // Sequence | OBJECT, NUMBER | STRING -> DATUM // OBSOLETE_GROUPED_MAPREDUCE = 46; // OBSOLETE_GROUPBY = 47; INNER_JOIN = 48; // Sequence, Sequence, Function(2) -> Sequence OUTER_JOIN = 49; // Sequence, Sequence, Function(2) -> Sequence // An inner-join that does an equality comparison on two attributes. EQ_JOIN = 50; // Sequence, !STRING, Sequence, {index:!STRING} -> Sequence ZIP = 72; // Sequence -> Sequence RANGE = 173; // -> Sequence [0, +inf) // NUMBER -> Sequence [0, a) // NUMBER, NUMBER -> Sequence [a, b) // Array Ops // Insert an element in to an array at a given index. INSERT_AT = 82; // ARRAY, NUMBER, DATUM -> ARRAY // Remove an element at a given index from an array. DELETE_AT = 83; // ARRAY, NUMBER -> ARRAY | // ARRAY, NUMBER, NUMBER -> ARRAY // Change the element at a given index of an array. CHANGE_AT = 84; // ARRAY, NUMBER, DATUM -> ARRAY // Splice one array in to another array. SPLICE_AT = 85; // ARRAY, NUMBER, ARRAY -> ARRAY // * Type Ops // Coerces a datum to a named type (e.g. "bool"). // If you previously used `stream_to_array`, you should use this instead // with the type "array". COERCE_TO = 51; // Top, STRING -> Top // Returns the named type of a datum (e.g. TYPE_OF(true) = "BOOL") TYPE_OF = 52; // Top -> STRING // * Write Ops (the OBJECTs contain data about number of errors etc.) // Updates all the rows in a selection. Calls its Function with the row // to be updated, and then merges the result of that call. UPDATE = 53; // StreamSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | // SingleSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | // StreamSelection, OBJECT, {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | // SingleSelection, OBJECT, {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT // Deletes all the rows in a selection. DELETE = 54; // StreamSelection, {durability:STRING, return_changes:BOOL} -> OBJECT | SingleSelection -> OBJECT // Replaces all the rows in a selection. Calls its Function with the row // to be replaced, and then discards it and stores the result of that // call. REPLACE = 55; // StreamSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | SingleSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT // Inserts into a table. If `conflict` is replace, overwrites // entries with the same primary key. If `conflict` is // update, does an update on the entry. If `conflict` is // error, or is omitted, conflicts will trigger an error. INSERT = 56; // Table, OBJECT, {conflict:STRING, durability:STRING, return_changes:BOOL} -> OBJECT | Table, Sequence, {conflict:STRING, durability:STRING, return_changes:BOOL} -> OBJECT // * Administrative OPs // Creates a database with a particular name. DB_CREATE = 57; // STRING -> OBJECT // Drops a database with a particular name. DB_DROP = 58; // STRING -> OBJECT // Lists all the databases by name. (Takes no arguments) DB_LIST = 59; // -> ARRAY // Creates a table with a particular name in a particular // database. (You may omit the first argument to use the // default database.) TABLE_CREATE = 60; // Database, STRING, {primary_key:STRING, shards:NUMBER, replicas:NUMBER, primary_replica_tag:STRING} -> OBJECT // Database, STRING, {primary_key:STRING, shards:NUMBER, replicas:OBJECT, primary_replica_tag:STRING} -> OBJECT // STRING, {primary_key:STRING, shards:NUMBER, replicas:NUMBER, primary_replica_tag:STRING} -> OBJECT // STRING, {primary_key:STRING, shards:NUMBER, replicas:OBJECT, primary_replica_tag:STRING} -> OBJECT // Drops a table with a particular name from a particular // database. (You may omit the first argument to use the // default database.) TABLE_DROP = 61; // Database, STRING -> OBJECT // STRING -> OBJECT // Lists all the tables in a particular database. (You may // omit the first argument to use the default database.) TABLE_LIST = 62; // Database -> ARRAY // -> ARRAY // Returns the row in the `rethinkdb.table_config` or `rethinkdb.db_config` table // that corresponds to the given database or table. CONFIG = 174; // Database -> SingleSelection // Table -> SingleSelection // Returns the row in the `rethinkdb.table_status` table that corresponds to the // given table. STATUS = 175; // Table -> SingleSelection // Called on a table, waits for that table to be ready for read/write operations. // Called on a database, waits for all of the tables in the database to be ready. // Returns the corresponding row or rows from the `rethinkdb.table_status` table. WAIT = 177; // Table -> OBJECT // Database -> OBJECT // Generates a new config for the given table, or all tables in the given database // The `shards` and `replicas` arguments are required. If `emergency_repair` is // specified, it will enter a completely different mode of repairing a table // which has lost half or more of its replicas. RECONFIGURE = 176; // Database|Table, {shards:NUMBER, replicas:NUMBER [, // dry_run:BOOLEAN] // } -> OBJECT // Database|Table, {shards:NUMBER, replicas:OBJECT [, // primary_replica_tag:STRING, // nonvoting_replica_tags:ARRAY, // dry_run:BOOLEAN] // } -> OBJECT // Table, {emergency_repair:STRING, dry_run:BOOLEAN} -> OBJECT // Balances the table's shards but leaves everything else the same. Can also be // applied to an entire database at once. REBALANCE = 179; // Table -> OBJECT // Database -> OBJECT // Ensures that previously issued soft-durability writes are complete and // written to disk. SYNC = 138; // Table -> OBJECT // Set global, database, or table-specific permissions GRANT = 188; // -> OBJECT // Database -> OBJECT // Table -> OBJECT // * Secondary indexes OPs // Creates a new secondary index with a particular name and definition. INDEX_CREATE = 75; // Table, STRING, Function(1), {multi:BOOL} -> OBJECT // Drops a secondary index with a particular name from the specified table. INDEX_DROP = 76; // Table, STRING -> OBJECT // Lists all secondary indexes on a particular table. INDEX_LIST = 77; // Table -> ARRAY // Gets information about whether or not a set of indexes are ready to // be accessed. Returns a list of objects that look like this: // {index:STRING, ready:BOOL[, progress:NUMBER]} INDEX_STATUS = 139; // Table, STRING... -> ARRAY // Blocks until a set of indexes are ready to be accessed. Returns the // same values INDEX_STATUS. INDEX_WAIT = 140; // Table, STRING... -> ARRAY // Renames the given index to a new name INDEX_RENAME = 156; // Table, STRING, STRING, {overwrite:BOOL} -> OBJECT // * Control Operators // Calls a function on data FUNCALL = 64; // Function(*), DATUM... -> DATUM // Executes its first argument, and returns its second argument if it // got [true] or its third argument if it got [false] (like an `if` // statement). BRANCH = 65; // BOOL, Top, Top -> Top // Returns true if any of its arguments returns true (short-circuits). OR = 66; // BOOL... -> BOOL // Returns true if all of its arguments return true (short-circuits). AND = 67; // BOOL... -> BOOL // Calls its Function with each entry in the sequence // and executes the array of terms that Function returns. FOR_EACH = 68; // Sequence, Function(1) -> OBJECT //////////////////////////////////////////////////////////////////////////////// ////////// Special Terms //////////////////////////////////////////////////////////////////////////////// // An anonymous function. Takes an array of numbers representing // variables (see [VAR] above), and a [Term] to execute with those in // scope. Returns a function that may be passed an array of arguments, // then executes the Term with those bound to the variable names. The // user will never construct this directly. We use it internally for // things like `map` which take a function. The "arity" of a [Function] is // the number of arguments it takes. // For example, here's what `_X_.map{|x| x+2}` turns into: // Term { // type = MAP; // args = [_X_, // Term { // type = Function; // args = [Term { // type = DATUM; // datum = Datum { // type = R_ARRAY; // r_array = [Datum { type = R_NUM; r_num = 1; }]; // }; // }, // Term { // type = ADD; // args = [Term { // type = VAR; // args = [Term { // type = DATUM; // datum = Datum { type = R_NUM; // r_num = 1}; // }]; // }, // Term { // type = DATUM; // datum = Datum { type = R_NUM; r_num = 2; }; // }]; // }]; // }]; FUNC = 69; // ARRAY, Top -> ARRAY -> Top // Indicates to ORDER_BY that this attribute is to be sorted in ascending order. ASC = 73; // !STRING -> Ordering // Indicates to ORDER_BY that this attribute is to be sorted in descending order. DESC = 74; // !STRING -> Ordering // Gets info about anything. INFO is most commonly called on tables. INFO = 79; // Top -> OBJECT // `a.match(b)` returns a match object if the string `a` // matches the regular expression `b`. MATCH = 97; // STRING, STRING -> DATUM // Change the case of a string. UPCASE = 141; // STRING -> STRING DOWNCASE = 142; // STRING -> STRING // Select a number of elements from sequence with uniform distribution. SAMPLE = 81; // Sequence, NUMBER -> Sequence // Evaluates its first argument. If that argument returns // NULL or throws an error related to the absence of an // expected value (for instance, accessing a non-existent // field or adding NULL to an integer), DEFAULT will either // return its second argument or execute it if it's a // function. If the second argument is a function, it will be // passed either the text of the error or NULL as its // argument. DEFAULT = 92; // Top, Top -> Top // Parses its first argument as a json string and returns it as a // datum. JSON = 98; // STRING -> DATUM // Returns the datum as a JSON string. // N.B.: we would really prefer this be named TO_JSON and that exists as // an alias in Python and JavaScript drivers; however it conflicts with the // standard `to_json` method defined by Ruby's standard json library. TO_JSON_STRING = 172; // DATUM -> STRING // Parses its first arguments as an ISO 8601 time and returns it as a // datum. ISO8601 = 99; // STRING -> PSEUDOTYPE(TIME) // Prints a time as an ISO 8601 time. TO_ISO8601 = 100; // PSEUDOTYPE(TIME) -> STRING // Returns a time given seconds since epoch in UTC. EPOCH_TIME = 101; // NUMBER -> PSEUDOTYPE(TIME) // Returns seconds since epoch in UTC given a time. TO_EPOCH_TIME = 102; // PSEUDOTYPE(TIME) -> NUMBER // The time the query was received by the server. NOW = 103; // -> PSEUDOTYPE(TIME) // Puts a time into an ISO 8601 timezone. IN_TIMEZONE = 104; // PSEUDOTYPE(TIME), STRING -> PSEUDOTYPE(TIME) // a.during(b, c) returns whether a is in the range [b, c) DURING = 105; // PSEUDOTYPE(TIME), PSEUDOTYPE(TIME), PSEUDOTYPE(TIME) -> BOOL // Retrieves the date portion of a time. DATE = 106; // PSEUDOTYPE(TIME) -> PSEUDOTYPE(TIME) // x.time_of_day == x.date - x TIME_OF_DAY = 126; // PSEUDOTYPE(TIME) -> NUMBER // Returns the timezone of a time. TIMEZONE = 127; // PSEUDOTYPE(TIME) -> STRING // These access the various components of a time. YEAR = 128; // PSEUDOTYPE(TIME) -> NUMBER MONTH = 129; // PSEUDOTYPE(TIME) -> NUMBER DAY = 130; // PSEUDOTYPE(TIME) -> NUMBER DAY_OF_WEEK = 131; // PSEUDOTYPE(TIME) -> NUMBER DAY_OF_YEAR = 132; // PSEUDOTYPE(TIME) -> NUMBER HOURS = 133; // PSEUDOTYPE(TIME) -> NUMBER MINUTES = 134; // PSEUDOTYPE(TIME) -> NUMBER SECONDS = 135; // PSEUDOTYPE(TIME) -> NUMBER // Construct a time from a date and optional timezone or a // date+time and optional timezone. TIME = 136; // NUMBER, NUMBER, NUMBER, STRING -> PSEUDOTYPE(TIME) | // NUMBER, NUMBER, NUMBER, NUMBER, NUMBER, NUMBER, STRING -> PSEUDOTYPE(TIME) | // Constants for ISO 8601 days of the week. MONDAY = 107; // -> 1 TUESDAY = 108; // -> 2 WEDNESDAY = 109; // -> 3 THURSDAY = 110; // -> 4 FRIDAY = 111; // -> 5 SATURDAY = 112; // -> 6 SUNDAY = 113; // -> 7 // Constants for ISO 8601 months. JANUARY = 114; // -> 1 FEBRUARY = 115; // -> 2 MARCH = 116; // -> 3 APRIL = 117; // -> 4 MAY = 118; // -> 5 JUNE = 119; // -> 6 JULY = 120; // -> 7 AUGUST = 121; // -> 8 SEPTEMBER = 122; // -> 9 OCTOBER = 123; // -> 10 NOVEMBER = 124; // -> 11 DECEMBER = 125; // -> 12 // Indicates to MERGE to replace, or remove in case of an empty literal, the // other object rather than merge it. LITERAL = 137; // -> Merging // JSON -> Merging // SEQUENCE, STRING -> GROUPED_SEQUENCE | SEQUENCE, FUNCTION -> GROUPED_SEQUENCE GROUP = 144; SUM = 145; AVG = 146; MIN = 147; MAX = 148; // `str.split()` splits on whitespace // `str.split(" ")` splits on spaces only // `str.split(" ", 5)` splits on spaces with at most 5 results // `str.split(nil, 5)` splits on whitespace with at most 5 results SPLIT = 149; // STRING -> ARRAY | STRING, STRING -> ARRAY | STRING, STRING, NUMBER -> ARRAY | STRING, NULL, NUMBER -> ARRAY UNGROUP = 150; // GROUPED_DATA -> ARRAY // Takes a range of numbers and returns a random number within the range RANDOM = 151; // NUMBER, NUMBER {float:BOOL} -> DATUM CHANGES = 152; // TABLE -> STREAM ARGS = 154; // ARRAY -> SPECIAL (used to splice arguments) // BINARY is client-only at the moment, it is not supported on the server BINARY = 155; // STRING -> PSEUDOTYPE(BINARY) GEOJSON = 157; // OBJECT -> PSEUDOTYPE(GEOMETRY) TO_GEOJSON = 158; // PSEUDOTYPE(GEOMETRY) -> OBJECT POINT = 159; // NUMBER, NUMBER -> PSEUDOTYPE(GEOMETRY) LINE = 160; // (ARRAY | PSEUDOTYPE(GEOMETRY))... -> PSEUDOTYPE(GEOMETRY) POLYGON = 161; // (ARRAY | PSEUDOTYPE(GEOMETRY))... -> PSEUDOTYPE(GEOMETRY) DISTANCE = 162; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) {geo_system:STRING, unit:STRING} -> NUMBER INTERSECTS = 163; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> BOOL INCLUDES = 164; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> BOOL CIRCLE = 165; // PSEUDOTYPE(GEOMETRY), NUMBER {num_vertices:NUMBER, geo_system:STRING, unit:STRING, fill:BOOL} -> PSEUDOTYPE(GEOMETRY) GET_INTERSECTING = 166; // TABLE, PSEUDOTYPE(GEOMETRY) {index:!STRING} -> StreamSelection FILL = 167; // PSEUDOTYPE(GEOMETRY) -> PSEUDOTYPE(GEOMETRY) GET_NEAREST = 168; // TABLE, PSEUDOTYPE(GEOMETRY) {index:!STRING, max_results:NUM, max_dist:NUM, geo_system:STRING, unit:STRING} -> ARRAY POLYGON_SUB = 171; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> PSEUDOTYPE(GEOMETRY) // Constants for specifying key ranges MINVAL = 180; MAXVAL = 181; } optional TermType type = 1; // This is only used when type is DATUM. optional Datum datum = 2; repeated Term args = 3; // Holds the positional arguments of the query. message AssocPair { optional string key = 1; optional Term val = 2; } repeated AssocPair optargs = 4; // Holds the optional arguments of the query. // (Note that the order of the optional arguments doesn't matter; think of a // Hash.) } //////////////////////////////////////////////////////////////////////////////// // EXAMPLE // //////////////////////////////////////////////////////////////////////////////// // ```ruby // r.table('tbl', {:read_mode => 'outdated'}).insert([{:id => 0}, {:id => 1}]) // ``` // Would turn into: // Term { // type = INSERT; // args = [Term { // type = TABLE; // args = [Term { // type = DATUM; // datum = Datum { type = R_STR; r_str = "tbl"; }; // }]; // optargs = [["read_mode", // Term { // type = DATUM; // datum = Datum { type = R_STR; r_bool = "outdated"; }; // }]]; // }, // Term { // type = MAKE_ARRAY; // args = [Term { // type = DATUM; // datum = Datum { type = R_OBJECT; r_object = [["id", 0]]; }; // }, // Term { // type = DATUM; // datum = Datum { type = R_OBJECT; r_object = [["id", 1]]; }; // }]; // }] // } // And the server would reply: // Response { // type = SUCCESS_ATOM; // token = 1; // response = [Datum { type = R_OBJECT; r_object = [["inserted", 2]]; }]; // } // Or, if there were an error: // Response { // type = RUNTIME_ERROR; // token = 1; // response = [Datum { type = R_STR; r_str = "The table `tbl` doesn't exist!"; }]; // backtrace = [Frame { type = POS; pos = 0; }, Frame { type = POS; pos = 0; }]; // }