Chapter 3 - The zproject Tool
In the last chapter I explained a lot of rules and conventions for writing a Scalable C project. It looks like a lot to remember. The good news is that if we are consistent, it pays off. For example if we always put our sources into src and our headers into include, it is easier to reuse build scripts between projects.
Speaking of build scripts...
Problem: Infinite Sucking Pits of Darkness
I'm speaking of Makefiles. Wikipedia tells us Make was invented by Bell Labs in 1976. Wikipedia lies! The real truth is that Makefiles are digital demon devisements from the darkest depths of Dis. Some say Bell Labs was the portal through which they clawed their way into our innocent world. We still don't know their true purpose. All we know is, they are eternal and cannot be killed. And we know the dying sound our soul makes as it leaves our body.
Makefiles spawned an entire legion of descendant demons called the autotools. There are said to be ancient scrolls that provide the incantations to tame autotools demons. A piece of one landed on my desk. This is what it said:
Makefiles and build systems that laughing call themselves "makefile generators" as if that made things simpler are inevitable. There is no escape to an alternate universe, if you are writing C code. Oh, please someone tell me how "you need make to reduce build times." I need a good laugh while Travis CI trundles through a fifteen-minute build, every time I push a commit to GitHub.
That being said...
Solution: make it someone else's problem.
Happily, this solution actually worked. It comes as close to killing makefiles as possible, after about 25 years of research. As often, the brilliance and genius comes from the collective mind.
What my team, at iMatix did, many years ago, was to build a way to generate code from high level models. We used this a lot and got good at it. Our GSL language makes it possible to develop DSLs (domain specific languages) quickly, and then build backends that turn these DSLs into code. Not to be confused with the GNU Scientific Library (also known as GSL).
What the ZeroMQ community did, over about two years, was build a DSL for packaging, and write dozens of backends for it. This tool is called zproject, and it is what I'll explain in this chapter.
Remember this lesson:
Never give up. If you wait long enough, the ZeroMQ community may solve your problem for you.
Problem: I don't got zproject
Solution: get it from GitHub.
Remember this lesson:
Once you go master, you never go back.
Problem: we need an example
The fastest way to learn any new tool is by example. Let's make a minimal project by hand, then apply magic. Our project is called "Global Domination." Right now version 0.1 is small and modest. It is a skeleton project that fits the rules of Chapter 2, and does nothing more.
Solution: get the code from GitHub.
The minimal project contains one empty class, and supporting files:
LICENSE -- MPLv2 license text
README.md -- this file
include/globaldom.h -- project public header
include/gdom_server.h -- Global Domination server API
src/gdom_server.c -- Global Domination server
src/gdom_classes.h -- project private header
src/gdom_selftest.c -- project selftest tool
build.sh -- build and test Global Domination
.gitignore -- tell git what files to ignore
To build and test GlobDom 0.1, run build.sh.
Remember this lesson:
Learn the basics of Bash, it will save your life many times.
Problem: people expect "./configure && make"
It is possible, and I've done this on real projects, to work without Makefiles. You compile stuff, chuck it into libraries, and link your executables. Yet it bounces off the wall of expectations. Also, the endless weirdness of the real world. Any real build process gets complex. And so people turn to Makefile generators, much like the victim of a street mugging turning to Somali warlords for help.
Let me be frank, for a change. I do not like the GNU build system, even after mastering it. It uses a flat yet vast macro language to generate Makefiles by sheer brute force. It may be powerful, yet so are the technicals those Somali warlords like to drive to work. The only good thing about autotools is that if (and this is a large if) you can master it, or find someone who's done this, then it is solid.
Happily we figured out how to use autotools' considerable power from a position of blissful ignorance. Let me show you how.
Solution: tell zproject what your project looks like.
First, create a file project.xml in the globdom root directory, like this:
This should be self-explanatory. If you've heard bad things about XML, or been traumatized by it in the past, my sympathies. Give some people a set of hammers, and they think they're rock star drummers. XML is not a programming language. It is however great for writing models to generate code from. You'll learn the fun and profit in this.
Second, run the gsl command to build the project model:
And now, the "configure/make" thing works. Run "./autogen.sh" first, as that produces the configure script:
Remember this lesson:
If everyone expects cake, give them cake.
Problem: sorry, I meant "cmake..."
No problems. As you saw, zproject supports both CMake and the GNU build system. So:
Solution: zproject targets both autotools and CMake.
If autotools are the Somali warlords of build systems, then CMake is the Texas politician who promises wealth and power. CMake still wants your soul, yet it is has much more charisma. "I can take care of Visual Studio for you!" it says, smiling, playing off our inner fears.
My main gripe with CMake is that it is just a better build scripting system. It doesn't change the basic fact that I don't want to write build scripts because they're always doing the same bloody work!
Solution: don't script when you can model.
With zproject we don't write scripts. Instead we document our projects as abstract XML models. Then, we can run arbitrary backends on this model, each doing what it must. It is a profound and valuable shift.
Just to finish my complaints about CMake, it lacks a "clean" command. And since it leaves trash lying all over the place, and since that trash really gets in the way sometimes, this is bad. The solution people use is to build in a temporary directory. It isn't great. Thus, zproject generates a make distclean
target for cmake similar to the autotools one which cleans up all cmake generated files nicely. It isn't great either but better than no cleaning at all.
Still, since we get CMake support for free, why complain.
Remember this lesson:
Even if you hate a particular build system, someone out there is addicted to it.
Problem: what do I add to git?
Ah, our directory is now a mess of different files. We have some made by hand, some produced by autotools, some by CMake, and some by the compiler and linker.
We cannot add everything to git, because many of these files change every time we compile, and do not belong in the repository. Yet we need the basic build scripts in git, otherwise no-one will know how to use our code.
Solution: put the output of zproject in git.
Let's rewind. First, save project.xml. Then reset the clock using git clean. Then run zproject again:
Let's look at what zproject actually generated for us. Run git status to see all new and changed files:
Two of our hand-written files got smashed by generated versions. That's intentional. We won't modify these by hand ever again, as they track the project. Then we got a lot of new files, for autotools and for CMake. And then we got a project header called include/global_domination.h. Cute! But useless! Let's tolerate that for now, and fix it later.
Add these files to git and commit:
Now run ./autogen.sh && ./configure && make check again. You should see lots of output, ending like this:
Remember this lesson:
When using git clean, save any hand-written files first.
Problem: git status shows lots of junk
Building your project will produce lots of files and directories scattered around. Running git clean too often is a bad idea, as it will wipe any new files you've written.
Solution: use a more complete .gitignore file.
It gets tedious to write a complete .gitignore file. Happily we have a tool whose intention is precisely to do the tedious things involved in building Scalable C projects. Here is how we get a complete .gitignore file:
You'll see this in the output:
Now type git status and the image comes in focus:
Let's add this and commit:
Remember this lesson:
zproject generates .gitignore for us, if we don't have it.
Problem: I need to make a new class
Global domination is well on the way! Now we'd like to add a client class. We'll offer two APIs. One is for those who wish to run the GlobDom server in their code. The second is for those who want to access it over the network. So let's make a client class. Like our server class, it'll do nothing, yet. First draw the outline, then fill it in.
Solution: add new classes to project.xml.
Here is how we define the client class in project.xml:
Then we run gsl project.xml again and see what git status gives us. There's a few changes to build scripts, then two new files:
Take a quick look at these generated files. The include/gdom_client.h header defines a bare minimum for a typical class:
And src/gdom_client.c implements these three methods with little more than air and used chewing gum. Let's do the usual sanity check:
Remember this lesson:
zproject generates classes for you if you need them.
Problem: generated sources don't have a license
These two new files need a license blurb at their start. Now, we could add the license blurb by hand. zproject does not touch the class header or source once those files exist. Plus, there are other files that zproject gives us, like gdom_selftest.c, which also need a license blurb.
Solution: specify a blurb in our project model.
We'll put the blurb into a separate XML file so it doesn't clutter our project model. Create a file license.xml with this content:
And now include that into your project.xml thus:
Now, let's check that this works:
If gsl complains, fix the XML syntax and try again. Now look at the src/gdom_client.c file. It should start like this:
Take a peek at src/gdom_selftest.c and you will see the same blurb.
Remember this lesson:
Stop deleting src/gdom_client.c. From now you, you edit this by hand.
Problem: the client wants to run a server
As luck would have it, Global Domination's sales team already has a potential customer, who's offering good money for a server. You just need a demo, and the sooner the better.
Solution: write a server program.
What the server program does is create a gdom_server instance, print an optimistic message, wait for Ctrl-C, and then exit. How hard can it be?
Let's start by adding a 'main' element to project.xml:
Now regenerate the project with the usual gsl project.xml. You'll notice that zproject tells you:
The skeleton src/gdomd.c doesn't do anything useful, or even pretend to. Let's open it up in an editor and add some body to it:
The zsys_info method is one of a bunch that do system logging. The others are:
zsys_error - log error condition - highest priority.
zsys_warning - log warning condition - high priority.
zsys_notice - log normal condition - normal priority.
zsys_info - log information message - low priority.
Run man zsys to see some of the other methods in the CZMQ zsys class.
Let's build and test our new server:
It should say:
Remember this lesson:
Better a hundred small steps than one giant leap.
Problem: update the version number
After such a lot of work we should release a new version. We defined the version number in our public header file, include/globaldom.h. It should be simple to bump it:
Edit include/globaldom.h to say #define GLOBDOM_VERSION_MINOR 2
Print the version number from the header:
Run make and... we get compile errors like this:
The reason is that our project header files are confused. We have a mix of hand-written files (the original include/globaldom.h) and generated files doing the same thing. Take a look at include/gdom_library.h and you'll see it does the same (and more) as we did by hand.
Solution: define the version and header in our project model.
First, let's fix the header mismatch. Add this header attribute to the project item:
Now delete these two files:
The first is junk and has to go. The second is our hand-written project header. When we delete it, and then build the project again, zproject gives us a new skeleton header. It does nothing except pull in gdom_library.h, which has all of our public API. This split lets us add hand-written code (to globaldom.h) while also generating the public API (in gdom_library.h).
It sounds tricky yet once things are working, you can more or less forget about it. Rebuild the project with gsl project.xml as usual. You should see this output:
Now look at the version macros in include/gdom_library.h:
To define a version in the project, we add a 'version' item like this:
Run gsl project.xml again and see how include/gdom_library.h changes. Now we can fix our main program, and make the project:
Remember this lesson:
Simple is hard.
Problem: someone's patch broke my code
Congratulations, you got a contributor! It is a precious thing, someone joining your project. To criticize their work is clumsy, and foolish. Yet most open source projects treat patches like lollipops offered by strangers in a park. I've already discussed why optimistic merging is sane and effective.
Yet, it's nice to give people rapid feedback on their work. It might take us a day to see someone's patch and spot a mistake. We have computers to do this kind of stuff.
Solution: enable Travis CI on your project.
"CI" is short for "build everything and run make check, and see if it works." I'm sure there are other CI systems yet Travis wins for being simple, fast, and reliable.
To enable Travis, add this line to your project.xml:
Run gsl project.xml again and you'll see this output:
In all cases, zproject builds the autotools and cmake targets. We've now added one more target, travis. The .travis.yml script is where we tell Travis what to do.
Do git status and you'll see two new files:
The ci_build.sh file is always generated, whereas we will modify .travis.yml by hand, as we decide to expand our test cover. Here is what the skeleton Travis script looks like:
You can, if this is your kind of soup, go and learn Travis' scripting language.
Go to travis-ci.org (not .com!).
Click "Log in with GitHub" at the top right.
Authorize Travis for your organizations.
When you return to Travis, click your user icon at the top right.
Click "Sync account" so Travis knows what repositories you have.
Find your Global Domination project.
Click the toggle to enable Travis.
Nothing happens until you push the .travis.yml file to your repository. So:
Now Travis should start building and testing your project. Every time you push commits, it will run the .travis.yml script and report any errors.
Remember this lesson:
Travis is there to help you, not add more stress.
Problem: Travis isn't building my project :(
The first time you try to use Travis it's not obvious. Just enabling builds on your repo does not start a build. So if you carefully pushed your new .travis.yml file, and then enabled builds, nothing happened.
You need to push a commit before Travis wakes up. You don't need to change anything. Git lets you create empty commits.
Solution: push an empty commit.
Run these commands:
Remember this lesson:
With enough empty commits you can draw pictures on your GitHub profile.
Problem: my client is using Windows
Consider this an opportunity, not a problem. "We need support for Visual Studio 2010," says your client. "Hmm, that could take a week or more. Is that OK?" you reply. The client doesn't blink. "We could make it work with VS2015 too, that way you're compatible with the next Windows 10 service pack," you continue. The client still doesn't blink. "We put our best people on it... it'll cost more but you'll get it faster." The client finally blinks. "OK, make it happen," they say.
Solution: zproject does Visual Studio.
Here's a way to see all the targets that zproject supports. This is a growing list:
Here is the kind of thing that zproject will report:
So to support VS2010 and VS2015, we add these two targets to our project.xml:
And then we regenerate the project packaging with gsl project.xml. Here is what zproject reports now:
You can generate a single target using the -target command line switch. E.g.:
When you run git status you will now see a new subdirectory called builds. Add this to your repository, and commit it:
Your repository now holds project files for two versions of Visual Studio.
Remember this lesson:
zproject supports a lot of targets.
Problem: my client wants a stable release
Hold on to your horses, cowboy! One thing at a time. What the client wants and what the client needs are often two different things. Don't throw the word "stable" around lightly. It means different things to different people:
It means stuff that is firm, robust, and won't crash.
It means stuff that is resistant to change.
It means stuff that isn't screamingly insane.
It means stuff you put your horses in.
Our code is definitely one of these, and definitely not one of these. The other two, meh. As a compromise, let's "tag" the release. This is somewhat like shooting an orange dart, labeled "Little Buffy", into a wild buffalo. The tag is a short way of saying "that raging mountain of anger heading straight for you," or as we say in the trade, commit #bbc2c1.
Solution: tag this version.
Here is how we tag the current commit and send that off to GitHub:
Here's a subtle yet important detail. We change the version number, then we do work, and then we tag it. And then we change the version number again. That means after making our "stable release", we right away update the version number, then regenerate everything, then push it to GitHub again.
In project.xml:
And then:
Remember this lesson:
The version in git master is always the next version you intend to release.
Problem: how does my client get a tagged release?
Solution: use GitHub's release page.
GitHub offers one-click downloads of the tarballs and zip files for any given release of a project. There really isn't a better way of distributing tarballs.The ZeroMQ project uses a download server, yet that's a historical artifact, not a preference over GitHub.
If you want a specific tagged release from git, there are two ways. Either way you first clone the git repository. Then you can do either of these:
I tend to use the second form, as it leaves me with a clean state. It is exactly as if I rewound history. The first form leaves the repository in a "detached head" state. Unless you are a masochist, forget it. The hard reset is clean and effective. You can make commits, and they'll be in the right place. Just do git pull origin master before sending your commits back to GitHub.
Remember this lesson:
When it comes to git, ignorance is bliss.
Problem: how do I build my project on Windows?
If you use Linux in your work, then Windows can be a bit of a mystery. I especially hate the Visual Studio user interface. It greeted me by asking me to register a new user account. It then tried to show me "cool" stuff about my project. The only cool stuff in a C project is the code, which seemed impossible to find. Windows 1.0 introduced flat frames without overlap, and Visual Studio has been trying to get back there ever since.
Happily you can entirely ignore Visual Studio's aspirations to be an "environment", and treat it just like any other C/C++ compiler.
Solution: Windows has a command line.
I'll summarize what you need to have, know, and do in order to build your project via the command line:
You need a PC running Windows 8 or 10 with Internet connection.
You need a Visual Studio compiler. You can use the Community Edition for free.
Ensure the programfiles environment variable is set properly. Use the command line (or PowerShell! if you want a Kung Fury-style retro experience).
You need to install git and/or unzip for the command line. I'd recommend to build from git.
Using git, clone Global Domination plus all dependent projects:
Build the projects in order:
Laugh with glee as you realize you just accomplished months of work in a few hours.
Remember this lesson:
It is already amazing the horse can dance. Don't expect it to also keep a rhythm.
Problem: Java
Ah, thither yon sweet Java! Thy perfumed voice lulls my dreams with paths untold! Let me count the ways I love thee... OK, done.
The main problem with Java is that people use the language. And not just a few people either. It is the COBOL of the 21st Century, built by Big Software and sold to managers. If C++ suffers from "hey, how abstract can we make this stuff before our brains strangle themselves?" syndrome, then Java suffers from "just one more path, I promise!" pain.
Inevitably, a quiet and yet determined character will approach you on a dark street corner and offer you money. "Just give me Global Domination in Java," they'll say. "Just once," they'll insist, "How bad can it be?"
The answer is, pretty bad. Java took the traditional way of integrating native C libraries into "higher-level" languages, which is to say "pain and grief." The official name for this torture chamber is JNI, which stands for "Just Nasty, Innit?" Like many parts of the thankless job of programming, it seems OK once you're used to it.
Yet writing JNI code is a mind-numbing exercise that belies the delicacy of it. You need all the grace of an elephant hopping from rock to rock in a molten lava pool. Get it wrong and badness happens. Here's a slice of JNI:
I'm not going to explain this. You can find enough JNI tutorials on-line to decipher it. What I'm going to do is explain how to generate everything you need to make it Someone Else's Problem.
Solution: use the 'java' target.
Add this to project.xml:
And then gsl project.xml as usual. You'll see this output from zproject:
And when you type git status, you will only project.xml changed, and nothing else. There is a new directory bindings/jni and it is empty.
Problem: bindings/jni is empty
To generate a binding we need a little more than just the class name. We need a description of the methods in the class. We call this the "API model" and it sits in a subdirectory called api and has the file extension .api. We write this file as XML, as we do all our models.
Solution: write an API model for gdom_client.
Create a new file api/gdom_client.api with this content:
And then run gsl project.xml as usual. It looks just like before, yet take a look at bindings/jni now. It's full of life. Let's quickly add the newly generated files to our repository:
You should see a bunch of files in bindings/jni. Let's add these to git and push our changes:
Remember this lesson:
Small steps will lead you on a great journey.
Problem: how do I build the Java binding?
This is a part of zproject that will still change. Right now, every target in zproject has its own build process. It's easier than working entirely by hand. Yet, if you want to build several targets, it's inconsistent. If this bothers you, send a patch. For instance I've thought of writing a Bash front end script that would build any given target.
Having said that...
Solution: use gradle.
Install Gradle if you don't already have it:
Then, install the Java Development Kit if you don't already have it installed. Check if you already have it:
And then if you need it, go install a JDK. I'm not going to explain how, because documenting Java is like blowing on molten rock to cool it down. It is useless and you easily get burnt, and one of those things other people do better.
In the bindings/jni directory, this command does the work:
This calls javah to build the JNI headers in src/native/include, and then compiles the C and Java pieces to create a jar file a sharable library (.so). It then runs the library test suite.
The result is a pretty jar file: build/libs/gdom-jni-0.3.0.jar. I've no idea what people do with jars, to be honest. I assume one installs them somewhere in the seventh circle of Hell so that one's Java "apps" (the more horrid the tech, the cuter the euphemisms) can find them. If you ever figure this out, send me a pull request.
Problem: the JNI layer does not build on Windows
Developing on Windows means using Microsoft Visual Studio, as we already know to our regret. To build the JNI layer on Windows, we ask for the java-msvc target. This generates all the 'solutions' (note again the cute euphemism) and project files for various versions of MSVS (aka MSVC for "Microsoft Visual C/C++").
Solution: use the 'java-msvc' target.
Add this to project.xml:
And then gsl project.xml as usual. You'll see this output from zproject:
To actually build:
You need MS Visual Studio 2010 or later.
You need the Java SDK. Set the JAVA_HOME environment to the installation location, e.g. C:\Program Files\Javajdk1.8.0_66.
Check out dependent projects (libzmq, czmq) from github, at the same level as this project.
In each project, open a console in builds/msvc/vs2010 and run the build.bat batch file.
In this project, open a console in bindings/jni/msvc/vs2010 and run the build.bat batch file.
The resulting libraries (gdomjni.dll and gdomjni.lib) should land in bindings/jni/msvc/bin, if my calculations are correct.
Remember this lesson:
Writing or creating these project files by hand would be painful. Code generation rocks!
Problem: sorry, I meant "Android" not "Windows"
We get it. You deliver magic to a client, charge way too little, and instead of gratitude, the client swings around and demands more. "Windows is out. Can you make it run on Android? We need it on Android, we're going to demo in three days' time."
To compile C/C++ and JNI for Android is like tattooing your own back. "Difficult and painful" starts to describe it. You can make it work, if you are patient and willing to learn. The basic steps are:
Install a funky version of gcc called "the Android Native Development Kit" or NDK, which can compile your C/C++ code to the various different CPU architectures that Android runs on.
Set up a whole load of paths and environment variables so that the NDK can compile your code.
Compile it, and hope you've not any non-portable dependencies to trip you up.
Much of the hard work is already done in zproject's jni target. You should probably read bindings/jni/android/build.sh and try to understand what it's doing.
Solution: gratitude to whomever wrote the jni target in zproject.
Install the latest Android Native Development Kit (NDK).
Set these environment variables, like this (to build for ARM, which is the most common Android CPU):
Then in the bindings/jni/android directory, run ./build.sh.
This does the following:
It compiles the Global Domination C sources for Android, into a native library libgdom.so in builds/android/.
It compiles the JNI Java classes into a jar file gdom-jni-0.3.0.jar in bindings/jni/build/libs.
It compiles the JNI C sources for Android, into a native library libgdomjni.so.
It combines all these into gdom-android.jar (basically a zip file with a specific layout), that you can use in your Android projects.
Problem: project name is not consistent
Sometimes you get it wrong.
Problem: contributors aren't reading the C4
CONTRIBUTING.md
Problem: the server blocks my main thread
-> generate
Problem: self tests are slow
selftest command line tool
Problem: someone asked me for packages
debian, redhat, nuget, docker packaging
Problem: my stuff is getting popular
bindings for ruby, python, QML
Problem: can I Globally Dominate Android?
yes you can! In two ways!
Problem: zproject doesn't support X
draft/stable/legacy
Last updated