Skip to main content
Version: 2.0.2

Counters

Counters are a bucket-level Riak data type that can be used by themselves, associated with a bucket/key pair, or used within a map. A counter's value can only be a positive integer, negative integer, or zero.

The examples in this section will show you how to use counters on their own.

Set Up a Bucket Type

If you've already created and activated a bucket type with the datatype parameter set to counter, skip to the next section.

Start by creating a bucket type with the datatype parameter set to counter:

riak-admin bucket-type create counters '{"props":{"datatype":"counter"}}'
note

The counters bucket type name provided above is an example and is not required to be counters. You are free to name bucket types whatever you like, with the exception of default.

After creating a bucket with a Riak data type, confirm the bucket property configuration associated with that type is correct:

riak-admin bucket-type status counters

This returns a list of bucket properties and their values in the form of property: value.

If our counters bucket type has been set properly we should see the following pair in our console output:

datatype: counter

Once we have confirmed the bucket type is properly configured, we can activate the bucket type to be used in Riak KV:

riak-admin bucket-type activate counters

We can check if activation has been successful by using the same bucket-type status command shown above:

riak-admin bucket-type status counters

After creating and activating our new counters bucket type, we can setup our client to start using the bucket type as detailed in the next section.

Client Setup

First, we need to direct our client to the bucket type/bucket/key location that contains our counter.

For this example we'll use the counters bucket type created and activated above and a bucket called counters:

// In the Java client, a bucket/bucket type combination is specified
// using a Namespace object. To specify bucket, bucket type, and key,
// use a Location object that incorporates the Namespace object, as is
// done below.
Namespace countersBucket = new Namespace("counters", "counters");
Location location = new Location(countersBucket, "<insert_key_here>");

Create a Counter

To create a counter, you need to specify a bucket/key pair to hold that counter. Here is the general syntax for doing so:

// Here, we'll use the Namespace object that we created above and
// incorporate it into a Location object that includes the key (as yet
// unspecified) for our counter

// Using the countersBucket Namespace object from above:
Location counter = new Location(countersBucket, "<key>");

// Or we can specify the Location all at once:
Location counter = new Location(new Namespace("counters", "counters"), "<key>");

Let's say that we want to create a counter called traffic_tickets in our counters bucket to keep track of our legal misbehavior. We can create this counter and ensure that the counters bucket will use our counters bucket type like this:

// Using the countersBucket Namespace object from above:

Location trafficTickets = new Location(countersBucket, "traffic_tickets");

Increment a Counter

Now that our client knows which bucket/key pairing to use for our counter, traffic_tickets will start out at 0 by default. If we happen to get a ticket that afternoon, we can increment the counter:

// Using the "trafficTickets" Location from above:

CounterUpdate cu = new CounterUpdate(1);
UpdateCounter update = new UpdateCounter.Builder(trafficTickets, cu)
.build();
client.execute(update);

Increment a Counter by More Than 1

The default value of an increment operation is 1, but you can increment by more than 1 (but always by an integer).

Continuing with our traffic_tickets example, let's say we receive 5 tickets in a single day:

// Using the "trafficTickets" Location from above:
CounterUpdate cu = new CounterUpdate(5);
UpdateCounter update = new UpdateCounter.Builder(trafficTickets, cu)
.build();
client.execute(update);

Retrieve Counter Value

We can retrieve the value of the counter and view how many tickets have accumulated:

// Using the "trafficTickets" Location from above:
FetchCounter fetch = new FetchCounter.Builder(trafficTickets)
.build();
FetchCounter.Response response = client.execute(fetch);
RiakCounter counter = response.getDatatype();
Long ticketsCount = counter.view();

Decrement a Counter

Counters enable you to decrement values in addition to incrementing them as seen above.

For example, let's say we hire an expert lawyer who gets one of the traffic tickets stricken from our record:

// Using the "trafficTickets" Location from above:
CounterUpdate cu = new CounterUpdate(-1);
UpdateCounter update = new UpdateCounter.Builder(trafficTickets, cu)
.build();
client.execute(update);