What if you want to print the whole book? You could say

$ pr ch1.1 ch1.2 ch1.3 ...

But you would soon get bored typing filenames and start to make mistakes. This is where filename shorthand comes in. If you say

$ pr ch*

The shell takes the * to mean “any string of characters,” so ch* is a pattern that matches all filenames in the current directory that begin with ch. The shell creates the list, in alphabetical order, and passes the list to pr. The pr command never sees the *; the pattern match that the shell does in the current directory generates a list of strings that are passed to pr.

The crucial point is that filename shorthand is not a property of the pr command, but a service of the shell. Thus you can use it to generate a sequence of filenames for any command.

— Brian W. Kernighan and Rob Pike, The Unix Programming Environment, 1984

In the last article, we were concerned with a method’s ability to verify the sanity of the arguments passed to it.

Consider the following:

@class Example {
	@public @method run @Object* object {
		echo "@object.member"
	}
}

How can the method verify that the pointer passed to it is valid, and points to an Object type?

In C++, some code like this:

class Object;

void run(Object object) {
	std::cout << object.member << std::endl;
}

Will throw a compiler error if you try to call the run method with an argument of the wrong type.

Imagine a hypothetical language, like C++, but which permits untyped parameters in function headers (in fact, C++20 allows this with the auto keyword). You could imagine that what was really going on under the hood was something like an implicit static cast to the expected type:

// Pseudo-code
void run(untyped object) {
	Object casted_object = static_cast<Object>(object);
	std::cout << casted_object.member << std::endl;
}

The static cast would verify at compile time that the object passed to the method is of the correct type. When this function is called later, the compiler would take a good long look at the argument passed to it, and throw an error if it’s not of the correct type.

Great. So why can’t we do this in Bash++?

Well, unfortunately Bash++ is unlike any other language that I know about – it’s an object-oriented language built overtop of the shell. The shell is amazing because it’s so flexible, but it’s also a bit of a pain because it’s so flexible. The shell isn’t better or worse than “proper” programming languages – it’s just different (and for different things).

Let’s imagine trying to follow through on type-checking an argument in Bash++.

@class Example {
	@public @method run arg1 arg2 @Object* arg3 arg4 arg5 {
		echo "Argument 1: $arg1"
		echo "Argument 2: $arg2"
		echo "Argument 3's data member: @arg3.member"
		echo "Argument 4: $arg4"
		echo "Argument 5: $arg5"
	}
}

OK. So the compiler knows that the run method expects its third argument to be of the type Object. Now let’s take a look at a place where this method is called:

@Example example
@Object object

var1="Argument1 Argument2"

@example.run $var1 &@object "Argument4" "Argument5"

Which argument is the third one?

How about this case?

var1=$(ls)
@example.run $var1 &@object "Argument4" "Argument5"

How many arguments are being passed to the run method?

As you can tell by now – and with the help of the book by Kernighan and Pike – it is absolutely impossible to verify the type of the nth argument passed to a method in Bash++ at compile time, because it’s impossible to tell which argument is the nth argument. It’s impossible for the compiler to even figure how many arguments are being passed to a method. The output of ls could be different every single time you run it, and the shell will happily pass that output to the run method, as a series of an indeterminate number of arguments.

Aside
Kernighan and Pike offer a small exercise for the reader on this point:

If you type

$ rm *

why can’t rm warn you that you’re about to delete all your files?

I offer a similar one here:

  1. Why can’t we have method overloading in Bash++? (That is, two methods in the same class which share the same name, distinguished by how many or what kind of parameters they take).
  2. Why can methods only accept pointers as parameters—why not objects proper?

Nevertheless, we still have to make sure that the method is given the tools it needs to do its job. We can’t just leave the programmer high and dry, and say “good luck, hope that third argument really was an Object pointer, and not some random garbage!”

So, as of Bash++ v0.3.2, a method which accepts a non-primitive pointer as an argument will perform an implicit dynamic cast of that argument to the expected type. In other words, this:

@class Example {
	@public @method run @Object* object {
		echo "@object.member"
	}
}

Is now equivalent to this:

@class Example {
	@public @method run {
		@Object* object=@dynamic_cast<Object> "$1"
		echo "@object.member"
	}
}

The @dynamic_cast will attempt to cast the first argument passed to the method to the expected type at runtime. If the cast fails, the pointer will be set to @nullptr.

This may or may not be the right call – in fact I’ve never heard of any other programming language doing implicit dynamic casts – but I don’t see any alternative.

The safe way to write that method would be to check that the cast was successful:

@class Example {
	@public @method run @Object* object {
		if [[ @object == @nullptr ]]; then
			echo "Error."
			return 1
		fi
		echo "@object.member"
	}
}

This is fortunately idiomatic in Bash as it stands, and very much like how functions are almost always written in the shell – verifying (at runtime) that the arguments passed to them are correct before using them.

Of course, if you want to forego the dynamic cast for whatever reason, you can still do a “dirty” cast as you like:

@class Example {
	@public @method run {
		@Object* object="$1"
		echo "@object.member"
	}
}