Function Graphs

This section should be useful to new users as well as experienced developers. Here, we explain how and why we use functions for building data mapping recipes. If you are planning to do so, this section is essential reading.

Mapping Functions

A function represents a mapping between one set of values and another. Consider a basic example in which the function N(i) maps an input number to text form.

#zoom:1.0
#direction:right
#.value: fill=#D0FFD0 visual=frame
#.function: fill=#E0FFE0 visual=sender
[example 1|
[<value> input: 1] ->[<function> N(1)]
[<function> N(1)] -> [<value>output: "one"]
[<value> input: 2] ->[<function> N(2)]
[<function> N(2)] -> [<value>output: "two"]
[<value> input: 3] ->[<function> N(3)]
[<function> N(3)] -> [<value>output: "three"]
]

In this example, as will often be the case, the type of input is different than that of the output. The input could be a long, and the output type a String, for example. This example also shows data flowing through the function. This is a common representation in flow-based programming.

Value Semantics

If we consider the words above to be the names of users in a population, it is easy to see how we could think of function N as simply enumerating their names. In this case, the word values are semantically properties – the names of some arbitrary members of a population. We say that it is arbitrary in this case simply because a property does not uniquely identify a member of a population. We are not yet speaking about identity semantics. So far we have chosen only to assign the meaning of property to the output values.

If we take it a step further and say that the words are unique identifiers for users, then we can say that we are indeed identifying specific instances of a population. This would be the case, for example, if you had a system in which each unique number name represented a unique user. The most important point here is that it is up to you how uniqueness in your particular system is represented and thus how you should model it. So long as the data mapping functions mirror the identity and property semantics of your application, the results will make sense.

Both properties and instances are important in a dataset. How do we have both together in a way that is meaningful? How do we make it so that we can deal with instances of things as well as the properties so that we actually get stable properties for specific instances?

For this, we need to combine functions. To model the relationship between the identity of an instance at one level and one of its associated properties, we simply mirror the relationship in terms of functions. We combine functions. In formal terms, we create a composed function.

As a starting point, we must have a basic model of how the identities and properties relate to each other:

#direction:down
#zoom:1.0
[identity: user_id] --> [property: first name]
[identity: user_id] --> [property: last name]

For the first time, we see a graph structure that represents the association between instances and properties of those instances in a data set. It is not unlike an entity-relationship diagram. We will call this type of diagram an entity-property graph.

Although basic, we see three values and how they are related. first_name depends on user_id, as does last_name. This is the same as saying that you must have the value for user_id before you can calculate the values for first and last name. It also shows why we draw the lines in this direction when modeling data flow.

It illustrates cardinal relationships but does not describe how the data corresponds from one value to another. For this, we need to fill in the blanks. First, let’s talk about where everything begins in our data flow.

Input Coordinates

When mapping data via functions we have to have an original input value. For the sake of simplicity, we will stick to whole numbers in sequence. Knowing that our original input data is simply the counting numbers allows us to do interesting things later with cardinality. For now, the most important thing to remember is that we have a primary input that we call the coordinate – an integer from a sequence.

Composition and Dataflow

Assume that you have two functions:

one that maps numbers to user ids:

#zoom:0.75
#direction:right
#.input: fill=#FFFFFF visual=frame
#.function: fill=#FFFFFF visual=sender
#.userid: fill=#E2D58B visual=frame
[<input> input: number] ->[<function>U]
[<function>U] -> [<userid>output: user_id]

and another that maps user ids to first names:

#zoom:0.75
#direction:right
#.userid: fill=#E2D58B visual=frame
#.function: fill=#FFFFFF visual=sender
#.firstname: fill=#44BBA4 visual=frame
[<userid> input: user_id] ->[<function>F]
[<function>F] -> [<firstname>output: first_name] 

If you combine them together in data-flow form, the results looks like this:

#zoom:0.75
#direction:right
#.input: fill=#FFFFFF visual=frame
#.function: fill=#FFFFFF visual=sender
#.userid: fill=#E2D58B visual=frame
#.firstname: fill=#44BBA4 visual=frame
[<input> input: number] ->[<function>U]
[<function>U] -> [<userid> user_id]
[<userid> user_id] ->[<function>F]
[<function>F] -> [<firstname>output: first_name]

The value of user id is the output of U and the input of F in this configuration. If we compose these functions U(i) and F(i) together, they become a single function that maps a number to an instance to a first name. Somewhat formally, we would call this F(U(i)). In other words, we would apply F to the result of applying U to the original input. This wording can get out of hand for larger compositions, so we will emphasize flow-based descriptions moving forward.

Function Graphs

Given an entity-property graph, we can build a new kind of picture with more detail: the function graph. You can think of the function graph as a symbolic encoding of the value relationships in a data set. We simply add the coordinate input value at the front, and then drop in a place-holder for each required mapping function:

#zoom:1.0
#direction:down
#.function: fill=#FFFFFF visual=sender
#.userid: visual=frame
#.firstname: visual=frame
#.lastname: visual=frame
[<input>coordinate] ->[<function>U]
[<function>U] -> [<userid>identity: user_id]
[<userid>identity: user_id] ->[<function>F]
[<function>F] -> [<firstname>property: first_name]
[<userid>identity: user_id] ->[<function>L]
[<function>L] -> [<lastname>property: last_name]

This function graph is merely a template. The functions U, F, and L are still symbolic. We haven’t picked anything concrete to put in their places. However, we can see the function semantics and signatures that must be fulfilled.

The key insight here is that each complete path of this graph represents a composed function that can render self-consistent data. This is the basic building block for modeling virtual datasets with interesting data relationships.

Type Signatures

No matter what type system is used, the input and output types determine what kind of functions may be plugged in. This is called the type signature of the function. In terms of data flow between pure functions, we are really only concerned about the types of inputs and outputs. Our examples above glossed over data types at the concrete level.

There are concrete data types at every stage of function composition. It is also true that more than one data type can fulfill the need for a counting number, etc. At some point we have to start being particular about the data types needed in our data set. Here, we remove the explanatory “identity and property” markers, and go to concrete data types and labels:

#zoom:1.0
#direction:down
#.function: fill=#FFFFFF visual=sender
#.userid: visual=frame
#.firstname: visual=frame
#.lastname: visual=frame
[<input>coordinate: long] ->[<function>U]
[<function>U] -> [<userid>user_id: long]
[<userid>user_id: long] ->[<function>F]
[<function>F] -> [<firstname>first_name: String]
[<userid>user_id: long] ->[<function>L]
[<function>L] -> [<lastname>last_name: String]

Even though the function graph above is not realizable yet, it is useful as is. Function graphs of this type can be referred to as function graph template.

From this function graph template, we have two distinct paths – two different composed functions to be created. If we traverse each path from the coordinate, we see two composed function templates. They are, in pseudo-code form:

  • long->U->long->F->String
  • long->U->long->L->String

When decomposed further, these composed function templates yield a more specific function template for each function in the flow which must be individually assignable from the available functions in the runtime.

  • long->U->long
  • long->F->String
  • long->L->String

So long as the runtime library has a matching function available for the function name and type signature, it is possible to realize a usable mapping function at runtime.

Binding a Template

The explanations above avoided talking about function naming directly in order to focus on the data flow and type semantics of function graphs. However, you may have functions which are actually named U, F, and L in your library which have the same type signatures. In Java, these might look like

long U(long input) ...
String F(long input) ...
String L(long input) ...

In this case, it would be possible to bind the example function graph template above using simple function name resolution to method (or static functions if you prefer) names. If the call to bind the function graph template was successful, the result would be called a Lambda, which could be used just like any other function at runtime.

The function placeholders in the examples above can be satisfied by any function instance that fits the type signatures. Further, how these functions are realized at runtime are ultimately left to the individual data mapping libraries. Each library supports a set of loadable functions that can be asked for by name. How this name is resolved to an actual function is not limited to simple name-based lookup as described above.