Skip to Content
Technical Articles
Author's profile photo DJ Adams

Getting BTP resource GUIDs with the btp CLI – part 1

Learn how to use the btp CLI to determine resource GUIDs in your global account. This post is part 1, covering the bgu mechanism.

Update: the second blog post in this series is now also available: Getting BTP resource GUIDs with the btp CLI – part 2 – JSON and jq.

In the Developer Keynote during SAP TechEd, specifically in the Command Line Magic section, I used a little mechanism I called bgu (for “btp GUID”), to determine the GUIDs for various resources (a subaccount, and later, a directory) in my global account. Here you can see bgu in action:

bgu%20used%20to%20show%20GUID%20of%20trial%20subaccount

In case you’re interested in re-watching this section of the Developer Keynote, there’s a link in the Further reading and viewing section later.

In this post I explain what this bgu mechanism is and how it works, because it may be useful to you too. Moreover, the background information and context should provide you with some extra knowledge about BTP and the command line interface tool, btp.

Starting at a high level

The bgu mechanism is actually a function in my shell environment, which calls a script called btpguid. This in turn uses the btp CLI to examine the global account’s resource hierarchy and pick information out of it – specifically the GUID for a given resource name. Let’s break that down.

We can ask what bgu is with the Bash shell’s type builtin. This itself is an opportunity for us to enjoy a little bit of meta before we start, by asking what the type of type is:

; type type
type is a shell builtin

Anyway, enough of that, let’s ask what bgu is:

; type bgu
bgu is a function
bgu ()
{
    btpguid "$@";
    if [[ $# -gt 1 ]]; then
        btpctx > "$HOME/.status";
    fi
}

So there we are, bgu is a function that I’ve defined and made available in my shell. All it does is call btpguid with all of the arguments that were passed:

btpguid "$@"

Then, depending on circumstances, it calls another script (btpctx) to write some info to a status file. This is not relevant here (it’s related to my tmux-based status line in my terminal) so let’s just focus on the call to btpguid.

So what is btpguid? Let’s find out:

; type btpguid
btpguid is /home/user/.dotfiles/scripts/btp/btpguid

So btpguid is a script. Great, let’s have a look at it!

The btpguid script

#!/usr/bin/env bash

# btpguid - return BTP subaccount/directory GUIDs

# Usage: btpguid [-t|--target] displayname

# Returns the GUID for the given subaccount or directory, which is
# specified by name. If the option -t or --target is specified, it
# will also set that subaccount or directory as the target.

# Requires the btp CLI. Will direct you to log in first if you're
# not already logged in.

# It uses the detail from the output of this command:
# btp get accounts/global-account --show-hierarchy

# The output looks like this:

# Showing details for global account af39080b-1527-40a1-b78a-3b605af7e811...
#
# ├─ c35b11e4trial (af39080b-1527-40a1-b78a-3b605af7e811 - global account)
# │  ├─ trial (b6501bff-e0ac-4fdf-8898-81f305d25335 - subaccount)
# │  ├─ techedhouse (e57c5b13-9480-4a68-9c04-a603d7a017a9 - directory)
#
# type:            id:                                    display name:   parent id:
# global account   af39080b-1527-40a1-b78a-3b605af7e811   c35b11e4trial
# subaccount       b6501bff-e0ac-4fdf-8898-81f305d25335   trial           af39080b-...
# directory        e57c5b13-9480-4a68-9c04-a603d7a017a9   techedhouse     af39080b-...

# It's the second part of the output (the table) that is used.

# Uses the "${2:-$1}" technique seen in fff - see
# https://qmacro.org/autodidactics/2021/09/03/exploring-fff-part-1-main/
# for details.

gethier() {
  btp get accounts/global-account --show-hierarchy 2> /dev/null
}

main() {

  local hierarchy subtype guid displayname rc=0

  displayname="${2:-$1}"

  [[ -z $displayname ]] && {
    echo "No display name specified"
    exit 1
  }

  hierarchy="$(gethier)" || { btp login && hierarchy="$(gethier)"; }
  read -r subtype guid <<< "$(grep -P -o "^(subaccount|directory)\s+(\S+)(?=\s+$displayname)" <<< "$hierarchy")"

  # Set the subtype as target if requested
  [[ $1 == -t ]] || [[ $1 == --target ]] && {
    btp target "--${subtype}" "$guid" &> /dev/null
    rc=$?
  }

  echo "$guid"
  return $rc

}

main "$@"

Hopefully the comments provide the general idea – given the name of a resource, say “trial” or “techedhouse”, this script will find and return that resource’s GUID. In addition, if the --target option is specified, it will also set the default command context, but we’ll leave that for another time.

An overview of structure in BTP

Before we dig in to some of the details of the script, let’s just spend a few moments thinking about subaccounts and the hierarchies that can be built using directories.

Basically, within a global account, you can create subaccounts, and you can set up those subaccounts using directories. This is a flexible and simple way to organise resources, assignments, adminstrative access and more using a well understood paradigm.

You can read more about this on the SAP Help Portal in Account Models With Directories and Subaccounts [Feature Set B].

Let’s look at an example. The structure that existed at the end of the Command Line Magic section of the Developer Keynote looked like this:

├─ 1a99110dtrial (c63c501e-e589-467d-8875-1821927ea713 - global account)
│  ├─ trial (00516298-b174-418e-9824-8824de04bfa3 - subaccount)
│  ├─ techedhouse (2558794c-f8cd-4422-b071-3b21c2922a02 - directory)
│  │  ├─ messaging (3ea88c9c-010b-4bf0-9fdb-5c29c9087660 - subaccount)

We saw, very briefly, a representation of this structure in the BTP cockpit too, right at the end of this section of the keynote:

Staring at this structure for a few seconds, we see that it’s made up of directories and subaccounts; the “messaging” subaccount sits within the “techedhouse” directory, which itself sits alongside (at the same level as) the default “trial” subaccount that was set up automatically for me when my global account was created.

Resources, GUIDs and command substitution

When managing these subaccount and directory resources, GUIDs are used. We saw multiple examples where GUIDs are required – here are three of them. However, note that each time, instead of finding and specifying a GUID manually, a command substitution (they look like this: $(...)) is used, to make things easier:

In each case, instead of manually looking up the GUID for a resource, and then copy-pasting that in for the value to use with --to-subaccount, --to-directory and --directory above, the bgu mechanism was used to do that for us. As the manual section on command substitution says:

command substitution allows the output of a command to replace the command itself

In other words, when you see something like this (taken from the first example above):

--to-subaccount $(bgu trial)

then what happens is that the command bgu trial is executed, and the output is then substituted as the value for the --to-subaccount parameter.

(In case you’re wondering, command substitution comes in two forms: `...` and $(...); the former is now deprecated.)

Determining the GUIDs

Now that we understand the structure of resources in BTP accounts, we can turn our attention to the heart of the btpguid script. This incarnation of the script, which was used in the Developer Keynote, invokes a btp CLI command (which we’ll see shortly), and parses some of its output:

hierarchy="$(gethier)" || { btp login && hierarchy="$(gethier)"; }
read -r subtype guid <<< "$(grep -P -o "^(subaccount|directory)\s+(\S+)(?=\s+$displayname)" <<< "$hierarchy")"

Let’s break that down so we understand what’s going on.

The first line executes the gethier function which is defined earlier in the script.

In case you’re wondering about the rest of the first line, this is to deal with the situation where you’re not yet (or no longer) logged in with btp, and if that’s the case, you’re guided to log in first, and then the call is re-tried.

The gethier function just runs the following btp command:

btp get accounts/global-account --show-hierarchy 2> /dev/null

I’m redirecting standard error (with 2>) to /dev/null, to get rid of anything printed there. Currently the btp CLI outputs an “OK” to standard error, and I don’t want to see that anywhere.

The --show-hierarchy parameter is responsible for the lovely detail that you saw earlier. Let’s look at an example of that detail before diving into the second line, so we know what we’re dealing with. This example is from the script’s comments above:

Showing details for global account af39080b-1527-40a1-b78a-3b605af7e811...

├─ c35b11e4trial (af39080b-1527-40a1-b78a-3b605af7e811 - global account)
│  ├─ trial (b6501bff-e0ac-4fdf-8898-81f305d25335 - subaccount)
│  ├─ techedhouse (e57c5b13-9480-4a68-9c04-a603d7a017a9 - directory)

type:            id:                                    display name:   parent id:
global account   af39080b-1527-40a1-b78a-3b605af7e811   c35b11e4trial
subaccount       b6501bff-e0ac-4fdf-8898-81f305d25335   trial           af39080b-...
directory        e57c5b13-9480-4a68-9c04-a603d7a017a9   techedhouse     af39080b-...

Right, so what is the second line doing? From a high level, it’s looking for lines in this output starting with either “subaccount” or “directory”, grabbing the GUID and resource type on the line that’s found, and assigning them to two variables, guid and subtype respectively:

read -r subtype guid <<< "$(grep -P -o "^(subaccount|directory)\s+(\S+)(?=\s+$displayname)" <<< "$hierarchy")"

It looks a little complex, but if you stare at it for a few minutes, this pattern emerges:

read [into two variables] from the results of searching and extracting values from the hierarchy data

A couple of “here strings” in the form of the <<< construct are used (see section 3.6.7 of the Bash manual section on redirections). Such “here strings” allow us to supply the value of a variable as input data to a command or builtin that would normally expect to read from standard input (STDIN). If you’re interested in understanding here strings and how they fit in, have a look at Input/output redirection, here documents and here strings.

Knowing this, we can break the line down into parts. The first part is this, inside the command substitution construct $(...):

grep -P -o "^(subaccount|directory)\s+(\S+)(?=\s+$displayname)" <<< "$hierarchy"

Here are some notes to help you interpret this:

Part Description
grep this is the command to search for patterns in data
-P this tells grep that we’re going to use a Perl Compatible Regular Expression (PCRE)
-o this tells grep to output not the entire matched line, but only the parts that are matched and captured (via parentheses)
"$hierarchy" this is the value of the hierarchy variable that holds the output from btp get accounts/global-account --show-hierarchy
^ this anchors the pattern to the beginning of the line (in other words, either “subaccount” or “directory” needs to be right at the start of the line for the match to be successful)
\s and \S these are very common metacharacters used in regular expressions, and represent “a whitespace character” and “anything but a whitespace character” respectively
+ this is a modifier which represents “at least one, possibly more” and is different from \* which is “zero or more” and ? which means “optional (i.e. either no occurrence or just one occurrence)”
$displayname because the entire pattern is enclosed in double quotes ("...") the shell will substitute the value of this variable into the pattern; the variable holds the value specified when btpguid is invoked, i.e. the name of the resource we’re looking for
(...) these are matching parentheses, called “capturing groups”, to identify and grab what we want from the match
(?=...) this is a positive lookahead assertion which allows us to say things like “must be followed by” without consuming anything in the match; note also that despite there being parentheses, this is not itself a capturing group and therefore what’s being asserted is not grabbed
<<< this is a here string construct that provides the input to grep from the value of the hierarchy variable instead of from standard input

With that in mind, let’s look again at the pattern, in quotes:

"^(subaccount|directory)\s+(\S+)(?=\s+$displayname)"

If the displayname variable contains “techedhouse”, then, after parameter substitution within these double quotes (i.e. at the shell level), we have this as the actual pattern:

^(subaccount|directory)\s+(\S+)(?=\s+techedhouse)

Spoken out loud we might say: the line must start with either ‘subaccount’ or ‘directory’ right at the beginning, and whichever it is, we want to capture it; that must be directly followed by at least one whitespace character (\s+), followed by at least one non-whitespace character (\S+), and we want to capture those non-whitespace characters*; oh, but also this must be followed ((?=) by at least one whitespace character (\s+) and then ‘techedhouse’.

* those non-whitespace characters will be the GUID

Let’s test this out manually, to see what happens. Let’s assume that we’re looking for the GUID of a resource with a display name of “techedhouse”:

; btp get accounts/global-account --show-hierarchy 2>/dev/null \
  | grep -P -o '^(subaccount|directory)\s+(\S+)(?=\s+techedhouse)'
directory        2558794c-f8cd-4422-b071-3b21c2922a02

Note that the whitespace and “techedhouse” inside the positive lookahead assertion (i.e. (?=\s+techedhouse)) is not captured and therefore we don’t see it as a third value in the output.

With the two values output like this:

directory        2558794c-f8cd-4422-b071-3b21c2922a02

we can better understand what’s happening with the read builtin; let’s substitute the output to see it in action:

read -r subtype guid <<< "directory        2558794c-f8cd-4422-b071-3b21c2922a02"

And guess what – after this, the subtype variable will contain “directory” and the guid variable will contain “2558794c-f8cd-4422-b071-3b21c2922a02”. And then with the echo a little bit further on in the script, this GUID is emitted to standard output before the script ends. Nice!

In case you’re wondering about the -r option to the read builtin, it’s to stop any backslashes in the input being interpreted inappropriately. I wrote about this in a recent post on my Autodidactics blog – see The read command section of Exploring fff part 2 – get_ls_colors.

Wrapping up this part

We can see how the power of the Unix Philosophy helps us prepare, run, and handle output of executables on the command line. Here we put together just a few lines to help us level up and be even more efficient, by allowing us to determine our SAP Business Technology Platform account’s resource GUIDs with zero effort, and use those determined GUIDs in the context of other commands.

In part 2 we’ll learn a little bit more about the Unix Philosophy and then examine alternative output formats for complex data structures and relationships; formats that are more predictable and – with the right tools – more reliably parseable.

Further reading and viewing

Here’s a quick list of resources that you may wish to consume, relating to what you’ve read in this post:

The second blog post in this series is now also available: Getting BTP resource GUIDs with the btp CLI – part 2 – JSON and jq.

Assigned Tags

      5 Comments
      You must be Logged on to comment or reply to a post.
      Author's profile photo Helmut Tammen
      Helmut Tammen

      Thanks DJ for this wonderful explanation of your bash script. I learn so much from you 🙂

      Author's profile photo DJ Adams
      DJ Adams
      Blog Post Author

      You're welcome, Helmut! It makes me happy that folks like you are discovering things here.

      Author's profile photo Morten Wittrock
      Morten Wittrock

      Out of habit, I would normally also comment out the blank lines between sections in a long multi-line comment, but looking at your btpguid script, I actually quite like the look of not doing it.

      Author's profile photo DJ Adams
      DJ Adams
      Blog Post Author

      Thanks Morten, I love the attention to detail in this comment! I've always wondered about whether I should or not, but I agree with you, not doing it makes the sections stand out a little better.

      Author's profile photo Bodhisattwa Pal
      Bodhisattwa Pal

      Hello DJ Adams

      I was reading how to extend business processes in SAP S/4 with

      We have a requirement where in  when the Business Partner  is getting created we need to default some fields to some constant value .

      In earlier days we used to implement Badis and User Exits .

      Now most of the blogs speak how to add extension logic on BTP  , that consume the event  after the Business Partner has been created or changed .

      But if we want to write the logic in BTP  which would be execute during  the Business partner is getting created (not after rthe Buisness Partner is  created .

      Can we achieve this kind of functionality using BTP .