Hello, and welcome to the DevStream blog!

In this very first “hello world” post, I will:

  • give you a high-level picture of how DevStream functions;
  • walk you through our codebase;
  • explain briefly how to create your own plugin.

If you haven’t heard of DevStream yet, please have a quick glance over our README.

Without further adieu, let’s get started.


TL;DR

Too bad, there isn’t one. But don’t close the tab right away; let me tell you a story.

Most of the time, I’m a “normal” end-user. And by “normal”, I mean I don’t bother myself with creating GitHub issues and pull requests and follow the updates for days (if not weeks). I love and use open-source projects, but that’s about it. Perhaps when I’m in an extremely good (or bad) mood, I will open up a discussion to give some feedback (or WTFs) and ramble about it. I don’t really contribute on a daily basis, though. Don’t get me wrong: I’d love it if I had more time; but I work a 9-to-5 job to live, just like most of the readers do.

If you are like me, should you read this post anyway?

Well, yes. Because although this article is about DevStream, it isn’t. We will cover quite some topics that are non-specific to DevStream. We will discuss generic topics about life, the universe, and everything. Hang tight. Here we go.


Criticism Wanted

We released early (Mar 2nd, 2022.) Maybe a little bit too early, one would say. We weren’t in stealth mode for long; 8 weeks, tops.

Of course, that wasn’t a mistake. We didn’t accidentally click the “publish” button on Product Hunt and leak it out. We did this pretty much on purpose. In fact, even before our version 0.1 was released, we had already decided to aim for an early release. It’s pretty much by design. Since it’s still early and immature, there might be bugs. I know you probably want to swear already, but bear with me:

I mean, technically, we could work on it for a much longer period, polish it up, add more features, support more tools, and what have you. But in that case, we might end up with something we thought is perfect but utterly useless to you, which is what really matters.

All we had was an idea. We didn’t know “how”, or even “what”. All we knew is, we wanted to build something that is useful for the end-users. So why not release it to the users and ask them to tell us what to build and how to build?

The idea isn’t mine, though. It would be arrogant if I said that I invented that. In fact, I learned it from lean manufacturing and agile development. To be honest, if I learned only one thing from lean manufacturing and agile, it’s that we should reduce the product-to-customer cycle. It’s The Toyota Way.

OK, enough rambling. If you are interested in DevStream, simply go to our GitHub repo, follow our README and quickstart, and have a go. I’m sure you will have some “WTF” moments during your first try, in which case, don’t hesitate to head to our GitHub Issues page and hit the “New Issue” button hard. We have prepared a few templates there to help you quickly get your dissatisfaction out of your system. Heck, you can even create an issue about the issue templates themselves. For helpful contributions, we will give you a “good first issue” label and who knows, probably the marketing team will reach out and award you with a little physical prize as well!


Intrigued?

Then let’s dive a little deeper into it.

Architecture

A picture is worth a thousand words:

architecture-overview

We use the Go plugin. I know this is kind of a big topic so here I’m only going to talk about it briefly:

  • First of all, we want to support many DevOps tools, and one tool corresponding to one plugin sounds natural.
  • Second of all, although there are other ways to do the “core-plugin” architecture, we reviewed Go’s plugin and it met our needs.
  • Oops, I thought there was a “third” or “last but not least”, but there isn’t. Sorry.

State

We consider DevStream (CLI tool dtm, don’t ask me why; it’s another whole story. Read this if you enjoy stories) as a “state machine”.

Simply put, given input and the current state, DevStream will calculate what to do, so that “what you want” (what you described in the input) is “what you get”.

If you are interested, read more here.

Directory structure

We follow Golang’s standard project layout. But don’t click that link, it’s too long and too verbose and contains probably many things you don’t immediately need right now. Continue to read the TL;DR version:

  • /cmd: main applications for this project. Don’t put a lot of code in the application directory. If you think the code can be imported and used in other projects, then it should live in the /pkg directory. If the code is not reusable or if you don’t want others to reuse it, put that code in the /internal directory.
  • /pkg: library code that’s ok to use by external apps. Other projects will import these libs expecting them to work, so think twice before you put something here.
  • /internal: private application and library code; things you don’t want others to import in their own apps.
  • /hack: contains many scripts that ensure continuous development of DevStream. We didn’t invent this; we simply stole Kubernetes’ idea.
  • /build: packaging and CI-related stuff.
  • /docs: pretty much self-explanatory. Need I say more?
  • /examples: examples for your apps or public libs. Here we use it to store some sample configurations.

Of course, this isn’t the only way to set up a repo structure for a Golang app. If you are a beginner, though, this can save you some time. For example, after reading this, you should know that if you want to find the entrance of the app, you shall look into the cmd folder. That’s where the main app lives.

Core / Plugin

How Exactly Does dtm apply Work? Is it “Automatic”, or “Automagic”?

  1. It all starts with /cmd/devstream/main.go.
  2. It executes the apply command in /cmd/devstream/apply.go.
  3. The Engine /internal/pkg/pluginengine runs the logic for apply at /internal/pkg/pluginengine/cmd_apply.go.

Then, happy digging around in the “Apply” function!


You Release Waaaaay Too Slow! I WANT THE LATEST STUFF!! NOW!!!

Alright, alright, here you go.

First, install Go. If you haven’t done so, please refer to the official doc here.

git clone https://github.com/devstream-io/devstream.git && cd stream
go install golang.org/x/tools/cmd/goimports@latest

# make sure your GOPATH/bin is in your PATH
# for example, run this:
export PATH=`go env GOPATH`/bin:$PATH

make build

# after you run "make build", you DO NOT need to run "dtm init" anymore
# run "dtm apply", "dtm delete" directly
# because "init" will try to download plugins from the internet,
# instead of using what you just built
dtm apply -f path_to_your_config.yaml

Want More?

Congratulations if you have followed so far. Why not go the extra mile by creating a new plugin of your own and play with it?

Each plugin needs to implement four interfaces, defined as follows:

Refer to an existing plugin for more detail. For example, this one.

I know, I know, before you complain, let me tell you this: due to the nature that each plugin is in fact a separate thing from others, there will be some duplicated scaffolding, copying, and pasting when you try to create a new plugin.

EXCEPT I WON’T ALLOW IT TO HAPPEN.

Good news! We created a tool to make it a whole lot easier for you. NO COPY-PASTE. Run:

dtm develop create-plugin --name=my-pluing-name-here

and we will bootstrap the required code, files, and directory structure for you. How about that? Happy hacking, buddy!

If you enjoy reading this post, please like, comment, and subscribe! I will see you next time.