8000 google-cloud-python/docs/bigtable-data-api.rst at master · enterstudio/google-cloud-python · GitHub
[go: up one dir, main page]

7FFF
Skip to content

Latest commit

 

History

History
344 lines (241 loc) · 11.8 KB

File metadata and controls

344 lines (241 loc) · 11.8 KB

Data API

After creating a :class:`Table <google.cloud.bigtable.table.Table>` and some column families, you are ready to store and retrieve data.

Cells vs. Columns vs. Column Families

  • As explained in the :doc:`table overview <bigtable-table-api>`, tables can have many column families.
  • As described below, a table can also have many rows which are specified by row keys.
  • Within a row, data is stored in a cell. A cell simply has a value (as bytes) and a timestamp. The number of cells in each row can be different, depending on what was stored in each row.
  • Each cell lies in a column (not a column family). A column is really just a more specific modifier within a column family. A column can be present in every column family, in only one or anywhere in between.
  • Within a column family there can be many columns. For example, within the column family foo we could have columns bar and baz. These would typically be represented as foo:bar and foo:baz.

Modifying Data

Since data is stored in cells, which are stored in rows, we use the metaphor of a row in classes that are used to modify (write, update, delete) data in a :class:`Table <google.cloud.bigtable.table.Table>`.

Direct vs. Conditional vs. Append

There are three ways to modify data in a table, described by the MutateRow, CheckAndMutateRow and ReadModifyWriteRow API methods.

Row Factory

A single factory can be used to create any of the three row types. To create a :class:`DirectRow <google.cloud.bigtable.row.DirectRow>`:

row = table.row(row_key)

Unlike the previous string values we've used before, the row key must be bytes.

To create a :class:`ConditionalRow <google.cloud.bigtable.row.ConditionalRow>`, first create a :class:`RowFilter <google.cloud.bigtable.row.RowFilter>` and then

cond_row = table.row(row_key, filter_=filter_)

To create an :class:`AppendRow <google.cloud.bigtable.row.AppendRow>`

append_row = table.row(row_key, append=True)

Building Up Mutations

In all three cases, a set of mutations (or two sets) are built up on a row before they are sent of in a batch via

row.commit()

Direct Mutations

Direct mutations can be added via one of four methods

Conditional Mutations

Making conditional modifications is essentially identical to direct modifications: it uses the exact same methods to accumulate mutations.

However, each mutation added must specify a state: will the mutation be applied if the filter matches or if it fails to match.

For example:

cond_row.set_cell(column_family_id, column, value,
                  timestamp=timestamp, state=True)

will add to the set of true mutations.

Append Mutations

Append mutations can be added via one of two methods

Notice that no timestamp was specified. This is because append mutations operate on the latest value of the specified column.

If there are no cells in the specified column, then the empty string (bytes case) or zero (integer case) are the assumed values.

Starting Fresh

If accumulated mutations need to be dropped, use

row.clear()

Reading Data

Read Single Row from a Table

To make a ReadRows API request for a single row key, use :meth:`Table.read_row() <google.cloud.bigtable.table.Table.read_row>`:

>>> row_data = table.read_row(row_key)
>>> row_data.cells
{
    u'fam1': {
        b'col1': [
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
        ],
        b'col2': [
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
        ],
    },
    u'fam2': {
        b'col3': [
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
            <google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>,
        ],
    },
}
>>> cell = row_data.cells[u'fam1'][b'col1'][0]
>>> cell
<google.cloud.bigtable.row_data.Cell at 0x7f80d150ef10>
>>> cell.value
b'val1'
>>> cell.timestamp
datetime.datetime(2016, 2, 27, 3, 41, 18, 122823, tzinfo=<UTC>)

Rather than returning a :class:`DirectRow <google.cloud.bigtable.row.DirectRow>` or similar class, this method returns a :class:`PartialRowData <google.cloud.bigtable.row_data.PartialRowData>` instance. This class is used for reading and parsing data rather than for modifying data (as :class:`DirectRow <google.cloud.bigtable.row.DirectRow>` is).

A filter can also be applied to the results:

row_data = table.read_row(row_key, filter_=filter_val)

The allowable filter_ values are the same as those used for a :class:`ConditionalRow <google.cloud.bigtable.row.ConditionalRow>`. For more information, see the :meth:`Table.read_row() <google.cloud.bigtable.table.Table.read_row>` documentation.

Stream Many Rows from a Table

To make a ReadRows API request for a stream of rows, use :meth:`Table.read_rows() <google.cloud.bigtable.table.Table.read_rows>`:

row_data = table.read_rows()

Using gRPC over HTTP/2, a continual stream of responses will be delivered. In particular

See the :class:`PartialRowsData <google.cloud.bigtable.row_data.PartialRowsData>` documentation for more information.

As with :meth:`Table.read_row() <google.cloud.bigtable.table.Table.read_row>`, an optional filter_ can be applied. In addition a start_key and / or end_key can be supplied for the stream, a limit can be set and a boolean allow_row_interleaving can be specified to allow faster streamed results at the potential cost of non-sequential reads.

See the :meth:`Table.read_rows() <google.cloud.bigtable.table.Table.read_rows>` documentation for more information on the optional arguments.

Sample Keys in a Table

Make a SampleRowKeys API request with :meth:`Table.sample_row_keys() <google.cloud.bigtable.table.Table.sample_row_keys>`:

keys_iterator = table.sample_row_keys()

The returned row keys will delimit contiguous sections of the table of approximately equal size, which can be used to break up the data for distributed tasks like mapreduces.

As with :meth:`Table.read_rows() <google.cloud.bigtable.table.Table.read_rows>`, the returned keys_iterator is connected to a cancellable HTTP/2 stream.

The next key in the result can be accessed via

next_key = keys_iterator.next()

or all keys can be iterated over via

for curr_key in keys_iterator:
    do_something(curr_key)

Just as with reading, the stream can be canceled:

keys_iterator.cancel()
0