Skip to main content
Version: 2.1.3

Maps

Maps are the most versatile of the Riak data types because all other data types can be embedded within them, including maps themselves. This enables the creation of complex, custom data types from a few basic building blocks.

Using counters, sets, and maps within maps are similar to working with those types at the bucket level.

Set Up a Bucket Type

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

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

riak-admin bucket-type create maps '{"props":{"datatype":"map"}}'
note

The maps bucket type name provided above is an example and is not required to be maps. 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 maps

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

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

datatype: map

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 maps

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

riak-admin bucket-type status maps

After creating and activating our new maps 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 map.

The syntax for creating a map is analogous to the syntax for creating other data types:

// 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.

Location map = new Location(new Namespace("<bucket_type>", "<bucket>"), "<key>");

Create a Map

For this example, say we want to use Riak KV to store information about our company's customers. We'll use the maps bucket type created and activated previously and a bucket called customers. Each customer's data will be contained in its own key in the customers bucket.

We can create a map for the user Ahmed (ahmed_info) using the maps bucket type:

// In the Java client, you specify the location of data types
// before you perform operations on them:

Location ahmedMap = new Location(new Namespace("maps", "customers"), "ahmed_info");

Registers

Registers are essentially named binaries (like strings). Any binary value can act as the value of a register. Like flags, registers cannot be used on their own and must be embedded in Riak maps.

Registers Within Maps

Continuing with our previous customers example, let's store some information in our map.

The first piece of information we want to store in our map is Ahmed's name and phone number, both of which are best stored as registers:

// Using our "ahmedMap" location from above:

RegisterUpdate ru1 = new RegisterUpdate("Ahmed");
RegisterUpdate ru2 = new RegisterUpdate("5551234567");
MapUpdate mu = new MapUpdate()
.update("first_name", ru1)
.update("phone_number", ru2);
UpdateMap update = new UpdateMap.Builder(ahmedMap, mu)
.build();
client.execute(update);

If a register did not previously exist, Riak KV will create that register for you.

Flags

Flags behave much like Boolean values, except that instead of true or false flags have the values enable or disable.

Flags cannot be used on their own, i.e. a flag cannot be stored in a bucket/key by itself. Instead, flags can only be stored within maps.

To disable an existing flag, you have to read it or provide a context.

Flags Within Maps

Now let's say that we add an Enterprise plan to our pricing model. We'll create an enterprise_customer flag to track whether Ahmed has signed up for the new plan. He hasn't yet, so we'll set it to false:

// Using our "ahmedMap" location from above:

MapUpdate mu = new MapUpdate()
.update("enterprise_customer", new FlagUpdate(false));
UpdateMap update = new UpdateMap.Builder(ahmedMap, mu)
.build();
client.execute(update);

We can retrieve the value of that flag at any time:

// Using our "ahmedMap" location from above:

FetchMap fetch = new FetchMap.Builder(ahmedMap).build();
FetchMap.Response response = client.execute(fetch);
RiakMap map = response.getDatatype();
System.out.println(map.getFlag("enterprise_customer").view());

Counters Within Maps

We also want to know how many times Ahmed has visited our website. We'll use a page_visits counter for that and run the following operation when Ahmed visits our page for the first time:

// Using our "ahmedMap" location from above:

MapUpdate mu = new MapUpdate()
.update("page_visits", cu);
UpdateMap update = new UpdateMap.Builder(ahmedMap, new CounterUpdate(1))
.build();
client.execute(update);

Even though the page_visits counter did not exist previously, the above operation will create it (with a default starting point of 0) and the increment operation will bump the counter up to 1.

Sets Within Maps

We'd also like to know what Ahmed's interests are so that we can better design a user experience for him. Through his purchasing decisions, we find out that Ahmed likes robots, opera, and motorcycles. We'll store that information in a set inside of our map:

// Using our "ahmedMap" location from above:

SetUpdate su = new SetUpdate()
.add("robots")
.add("opera")
.add("motorcycles");
MapUpdate mu = new MapUpdate()
.update("interests", su);
UpdateMap update = new UpdateMap.Builder(ahmedMap, mu)
.build();
client.execute(update);

We can then verify that the interests set includes these three interests:

// Using our "ahmedMap" location from above:

FetchMap fetch = new FetchMap.Builder(ahmedMap)
.build();
FetchMap.Response response = client.execute(fetch);
RiakMap map = response.getDatatype();
RiakSet interestSet = map.getSet("interests");
Set<BinaryValue> interests = interestSet.view();
System.out.println(interests.contains(BinaryValue.create("robots")));

// Checking for "opera" and "motorcycles" works the same way

We learn from a recent purchasing decision that Ahmed actually doesn't seem to like opera. He's much more keen on indie pop. Let's change the interests set to reflect that:

// Using our "ahmedMap" location from above:

SetUpdate su = new SetUpdate()
.remove("opera")
.add("indie pop");
MapUpdate mu = new MapUpdate()
.update("interests", su);
UpdateMap update = new UpdateMap.Builder(ahmedMap, mu)
.build();
client.execute(update);

Maps Within Maps

We've stored a wide of variety of information---of a wide variety of types---within the ahmed_info map thus far, but we have yet to explore recursively storing maps within maps (which can be nested as deeply as you wish).

Our company is doing well and we have lots of useful information about Ahmed, but now we want to store information about Ahmed's contacts as well. We'll start with storing some information about Ahmed's colleague Annika inside of a map called annika_info.

First, we'll store Annika's first name, last name, and phone number in registers:

// Using our "ahmedMap" location from above:

RegisterUpdate ru1 = new RegisterUpdate("Annika");
RegisterUpdate ru2 = new RegisterUpdate("Weiss");
RegisterUpdate ru3 = new RegisterUpdate("5559876543");

MapUpdate annikaUpdate = new MapUpdate()
.update("first_name", ru1)
.update("last_name", ru2)
.update("phone_number", ru3);
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.build();
client.execute(update);

The value of a register in a map can be obtained without a special method:

// Using our "ahmedMap" location from above:

FetchMap fetch = new FetchMap.Builder(ahmedMap).build();
FetchMap.Response response = client.execute(fetch);
String annikaFirstName = response.getDatatype()
.getMap("annika_info")
.getRegister("first_name")
.view()
.toString();

Registers can also be removed:

// This example uses our "ahmedMap" location from above. Operations that
// remove fields from maps require that you first fetch the opaque context
// attached to the map and then include the context in the update operation:

FetchMap fetch = new FetchMap.Builder(ahmedMap)
.build();
FetchMap.Response response = client.execute(fetch);
Context ctx = response.getContext();
MapUpdate annikaUpdate = new MapUpdate()
.removeRegister("first_name");
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.withContext(ctx)
.build();
client.execute(update);

Now, we'll store whether Annika is subscribed to a variety of plans within the company as well:

// Using our "ahmedMap" location from above:

FetchMap fetch = new FetchMap.Builder(ahmedMap).build();
FetchMap.Response response = client.execute(fetch);
Context ctx = response.getContext();
MapUpdate annikaUpdate = new MapUpdate()
.update("enterprise_plan", new FlagUpdate((false))
.update("family_plan", new FlagUpdate(false))
.update("free_plan", new FlagUpdate(true));
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.withContext(ctx)
.build();
client.execute(update);

The value of a flag can be retrieved at any time:

// Using our "ahmedMap" location from above:

FetchMap fetch = new FetchMap.Builder(ahmedMap).build();
FetchMap.Response response = client.execute(fetch);
boolean enterprisePlan = response.getDatatype()
.getMap("annika_info")
.getFlag("enterprise_plan")
.view();

It's also important to track the number of purchases that Annika has made with our company. Annika just made her first widget purchase:

// Using our "ahmedMap" location from above:

MapUpdate annikaUpdate = new MapUpdate()
.update("widget_purchases", new CounterUpdate(1));
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.build();
client.execute(update);

Now let's store Annika's interests in a set:

// Using our "ahmedMap" location from above:

SetUpdate su = new SetUpdate().add("tango dancing");
MapUpdate annikaUpdate = new MapUpdate()
.update("interests", su);
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.build();
client.execute(update);

We can remove that interest in just the way that we would expect:

// Using our "ahmedMap" location from above:

SetUpdate su = new SetUpdate().remove("tango dancing");
MapUpdate annikaUpdate = new MapUpdate()
.update("interests", su);
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.withUpdate(ahmedUpdate)
.build();
client.execute(update);

If we wanted to add store information about one of Annika's specific purchases, we could do so within a map:

// Using our "ahmedMap" location from above:

MapUpdate purchaseUpdate = new MapUpdate()
.update("first_purchase", new FlagUpdate(true)
.update("amount", new RegisterUpdate("1271"))
.update("items", new SetUpdate().add("large widget"));
MapUpdate annikaUpdate = new MapUpdate()
.update("purchase", purchaseUpdate);
MapUpdate ahmedUpdate = new MapUpdate()
.update("annika_info", annikaUpdate);
UpdateMap update = new UpdateMap.Builder(ahmedMap, ahmedUpdate)
.withUpdate(ahmedUpdate)
.build();
client.execute(update);