Compare these bog standard GlideRecord and GlideQuery examples and you’ll see something is different between them:

let gr = new GlideRecord(someTable);
gr.addQuery(someField, someValue);
gr.addNotNullQuery(someOtherField);
gr.query();
while (gr.next()) {
  // do the thing...
}
new GlideQuery(someTable)
  .where(someField, someValue)
  .whereNotNullQuery(someOtherField)
  .select(someFields)
  .forEach((record) => {
    // do the thing...
  });

The difference might be subtle or it might jump out at you right away. They’re the same number of lines and they seem to be doing all the same steps in the same order (indeed they are). What is it about GlideRecord and GlideQuery that makes the syntax appear so different, and, if this is important, why is it important?

GlideRecord is founded on object mutation

In the GlideRecord example, we first instantiate an object and assign it to a variable. From that point on we call methods on the object that cause the object to change. We don’t have to assign the output of those method calls to any new variable, because each time we add a new query the object itself changes. When we execute the query the object changes again, and each time we iterate to the next record the object changes yet again. And the whole time it remains assigned to the same variable name.

These changes in the object are sometimes called mutations or side effects, and some consider it best practice to avoid them. But aren’t variables meant to vary? After all, it’s right there in the name “variable”. What’s really wrong with allowing the GlideRecord object to mutate like this? We saw one extreme (and thankfully uncommon) example in Part 0 and Part 2 where mutation due to the next method caused undesirable behavior, but are there other drawbacks?

GlideQuery objects are immutable

In functional programming there’s a design pattern called a pure function. Pure functions always produce the same output given the same input and have no observable side effects. All GlideQuery objects are immutable, meaning there’s no way to change them after they’re created. All the methods of the GlideQuery API are pure functions, so, when you call any of them, instead of mutating the object, the method leaves the original object unchanged and returns a new object that’s like the old one but with the new step applied.

This is cumbersome, but it helps to see what’s really going on if we rewrite the GlideQuery example above like this:

let query1 = new GlideQuery(someTable)
let query2 = query1.where(someField, someValue)
let query3 = query2.whereNotNullQuery(someOtherField)
let stream = query3.select(someFields)
stream.forEach((record) => {
  // do the thing...
});

Here you can see clearly that with each new step we get back a new GlideQuery or Stream object (we’ll talk more about Stream and Optional objects in Part 4). In this example we’re assigning each object to a new variable, then using that variable to call the method for the next step.

Fluent method chaining

One immediate benefit of this pure function architecture is it enables GlideQuery’s fluent method chaining style. We usually don’t have any need to keep all those intermediate objects so we don’t bother assigning them to variables. Each chained method call simply acts on the new object returned from the preceding method call. This generally leads to shorter and cleaner-looking lines of code and fewer overall keystrokes.

When you chain methods like this, even though the method calls span multiple lines, you’re actually creating one long single expression. This means in some cases, like our first GlideQuery example above, you don’t need to assign anything to a variable. If you do start with declaring a variable, keep in mind the value assigned to the variable will be the result of the entire expression.

let n = new GlideQuery(table)
  .where(someField, someValue)
  .count()

The GlideQuery object instantiated on the first line never gets assigned to n. Instead, the entire method chain is evaluated and, ultimately, the variable will be assigned the return value from the count method.

And notice the indentation convention used here. Indenting each of the chained method call lines with a single tab is deliberate and recommended since it reminds us the whole thing is one long expression, not a series of individual expressions or statements.

Object re-usability

A less obvious but very powerful feature of this pure functional architecture is object re-use. Since the intermediate objects are never mutated, if we do decide to save one by assigning it to a variable, we can go back and re-use it later.

Let’s say we want to send an e-mail to everyone in our department who got an above average score on the most recent customer satisfaction survey. In order to do that, we first need to know what the average is. For simplicity, let’s assume we’ve already calculated each team member’s score and stored it in a custom field on the User [sys_user] table.

Without GlideQuery, the best way to do this is with GlideAggregate to get the average and then GlideRecord to loop through the users and send the e-mails.

let ga = new GlideAggregate('sys_user');
ga.addQuery('department', someDepartment);
ga.addQuery('active', true);
ga.addNotNullQuery('u_csat_score');
ga.addAggregate('AVG', 'u_csat_score');
ga.query();
ga.next();
let average = ga.getAggregate('AVG', 'u_csat_score');

let gr = new GlideRecord('sys_user');
gr.addQuery('department', someDepartment);
gr.addQuery('active', true);
gr.addNotNullQuery('u_csat_score');
gr.addQuery('u_csat_score', '>', average);
gr.query();
while (gr.next()) {
  // send e-mail...
}

Notice how the first four lines are nearly identical in the two different blocks of our script? This is a simple example, but imagine if our query were even more complicated. Wouldn’t it be great if we could somehow only write those lines once? We can’t do that with GlideAggregate and GlideRecord, but we can with GlideQuery.

let query = new GlideQuery('sys_user')
  .where('department', someDepartment)
  .where('active', true)
  .whereNotNull('u_csat_score');

let average = query  // using query once
  .avg('u_csat_score')
  .get();

query  // using query a second time
  .where('u_csat_score', '>', average)
  .select(someFields)
  .forEach((user) => {
    // send e-mail...
  });

After calling the avg and get methods you might think the query object is all used up, but since it doesn’t mutate like a GlideRecord object we can re-use it—even adding an additional where clause—to send the e-mails.1

I’ve used this in Script Includes where many different methods start by executing nearly the same query. The query can be assigned to a property of the object and then re-used across all the methods. They’re also great for enabling more dynamic function behavior since immutable GlideQuery objects can safely be passed around as function arguments. Once you know what’s possible you’ll find all sorts of uses for this feature.

Conclusion

Hopefully I’ve convinced you GlideQuery’s stateless, immutable design is safer than GlideRecord overall and enables both a more fluid syntax and some handy new use cases. Stay tuned for Part 4 where we’ll cover the myriad ways you can process Streams and Optionals. 


  1. This is possible in the first place due to GlideQuery being one API, not split into separate APIs for aggregate and normal queries. However, even if our example didn’t involve GlideAggregate and instead used GlideRecord twice, the first GlideRecord object couldn’t be re-used the way a GlideQuery object can.