ENCODING ======== Let's say you have the following tab-separated data format in a file called "data.tsv": foo bar 1 2 1.230000 three abacus 0 -1 -0.250000 foo baz 1 0 -1.000000 Call the columns a, b, x, y, and z. Suppose that a and b are string values, x and y are integer values, and z is a floating point value. The first thing you need to do is tell odb about all the possible string values found in your data since it stores strings as indexes into a string index file. The easiest way to do this is to specify the data schema and use the -x option to extract all the string values: $ odb encode -fa:string,b:string,x:int,y:int,z:float data.tsv -x | sort -u | odb strings By default, this creates a file called "strings.idx" that allows easy and efficient encoding and decoding of string data, representing every string as a 64-bit value. Next, we can encode the TSV data into ODB format: $ odb encode -fa:string,b:string,x:int,y:int,z:float data.tsv >data Now you have a working, usable ODB data file. To see what's in it, use the odb cat command: $ odb cat data a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 three abacus 0 -1 -0.250000 foo baz 1 0 -1.000000 If you want to get tab-separated values back, you can use the invese decode operation: $ odb decode data foo bar 1 2 1.230000 three abacus 0 -1 -0.250000 foo baz 1 0 -1.000000 Odb also supports timestamp and date field types, which can be input and output in various formats, specified using the -T option for timestamps and -D option for dates, according to the strftime and strptime standard C library functions (see man strftime for details). SORTING ======= Now let's say you want to sort the data by various columns. Just use the sort command: $ odb sort data -f a,b a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 $ odb sort data -f b,a a b x y z ------------------------------------------------------------------------------ three abacus 0 -1 -0.250000 foo bar 1 2 1.230000 foo baz 1 0 -1.000000 $ odb sort data -f x,y a b x y z ------------------------------------------------------------------------------ three abacus 0 -1 -0.250000 foo baz 1 0 -1.000000 foo bar 1 2 1.230000 $ odb sort data -f -z a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 The names of fields are separated by commas, and prefixing one with a minus sign sorts by descending order instead of ascending order. Any set and order of fields can be used for sorting. It should be noted that the data is sorted in place: the actual data file is altered when the sort occurs, so that if you cat it after a sort operation, it will have the order of the most recent sort: $ odb cat data a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 To avoid sorting data in place, you can cat the file data to the sort command, which will sort the data stream but not affect the original file: $ odb cat data | odb sort -f -a,b a b x y z ------------------------------------------------------------------------------ three abacus 0 -1 -0.250000 foo bar 1 2 1.230000 foo baz 1 0 -1.000000 $ odb cat data a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 SLICING ======= Like the UNIX cat command, you can use odb cat to concatenate files with like schemas. This includes concatenating a file with itself, thereby effectively repeating every record twice: $ odb cat data data a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 Unlike the UNIX cat command, however, the odb cat command can do a lot more than just concatenate file contents. It also subsumes the functionality of the cut, tac, head and tail commands, and much more. Using the -f option lets you select which fields to output, as well as their order: $ odb cat data -f z,b,a z b a ------------------------------------ 1.230000 bar foo -1.000000 baz foo -0.250000 abacus three The command always uses the order given and respects repeated fields: $ odb cat data -f b,a,b b a b ---------------------- bar foo bar baz foo baz abacus three abacus It also allows you to rename fields: $ odb cat data -f b,a,b=c b a b ---------------------- bar foo bar baz foo baz abacus three abacus It also allows you to reinterpret fields as being of a different type. In particular, strings are internally encoded as 64-bit integer indices: $ odb cat data -f a,a=i:int,b,b=j:int a i b j --------------------------------------------------------- foo 3 bar 1 foo 3 baz 2 three 4 abacus 0 You can also specify ranges of rows to output using the -r option, using a Matlab-like range notation start:step:stop. The start specifies the first row to output, step specifies how many rows to advance between rows to output, and stop specifies the largest row to be output. The start and stop offsets are 1-based with positive values counting from the first record and negative values counting from the last records. Abbreviated versions of the range notation include all of the following with defaults as indicated: notation start step stop ------------------------------------ : 1 1 -1 :stop 1 1 start: 1 -1 start:stop 1 :+step: 1 -1 start:+step: -1 :+step:stop 1 :-step: -1 1 start:-step: 1 :-step:stop -1 ------------------------------------ If you want to output the first 2 records, for example, you can do this: $ odb cat data -r :2 a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo baz 1 0 -1.000000 To output the last two rows, do this: $ odb cat data -r -2: a b x y z ------------------------------------------------------------------------------ foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 Odd rows: $ odb cat data -r 1:2: a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 three abacus 0 -1 -0.250000 Even rows: $ odb cat data -r 2:2: a b x y z ------------------------------------------------------------------------------ foo baz 1 0 -1.000000 To output data inreversed order, like the tac command, use the range -1:-1:1 like so: $ odb cat date -r -1:-1:1 a b x y z ------------------------------------------------------------------------------ three abacus 0 -1 -0.250000 foo baz 1 0 -1.000000 foo bar 1 2 1.230000 You can slice streamed data as well: $ odb sort data data data | odb cat -r :2: a b x y z ------------------------------------------------------------------------------ foo bar 1 2 1.230000 foo bar 1 2 1.230000 foo baz 1 0 -1.000000 three abacus 0 -1 -0.250000 three abacus 0 -1 -0.250000 You cannot, however, use negative offsets other than -1, or negitive strides when slicing streamed data: $ odb sort data data data | odb cat -r -2:2: negative range offsets cannot be used with streamed inputs $ odb sort data data data | odb cat -r -1:-1:1 negative range strides cannot be used with streamed inputs OTHER ===== The paste command horizontally concatenates its argument data just like the UNIX paste command does. It's arguments do not have to have compatible schemas, but they should have the same number of rows. The join command (not yet implemented) does an inner join on multiple inputs by the fields given with the -f option.