How to create a native macOS app on Go (Golang) and React.js with full code protection (without magic)

Vic Shóstak
6 min readDec 4, 2019

Introduction

Welcome to the next article in the “How to” series! This time, we will analyze the creation of a native desktop application for Apple macOS, write a little code on Golang and React.js, which will have copy protection.

This time, we will try to do it without magic! 😉

Objectives of the article

  1. Show one of the easiest ways to create a native desktop application for macOS on Golang;
  2. Show the option to protect the code of your application from being modified by third parties (for example, during commercial distribution);

Work Environment

  • Go 1.12.5;
  • Node.js 12.3.1;

Operating system

  • Apple macOS 10.14.5 Mojave (darwin/amd64);

Package and Dependency Manager

  • dep 0.5.3 (Go);
  • npm 6.9.0 (Node.js);

Used Golang Packages

  • net/http - standard package for creating a web server (godoc);
  • gobuffalo/packr - package for packaging all the necessary sources into one executable binary file (GitHub);
  • zserge/webview - cross-platform package for creating a native operating system window with a built-in browser (GitHub);

Used Node.js libraries

  • facebook/create-react-app - front-end for macOS-application (GitHub);
  • axios/axios - for easier writing of AJAX requests (GitHub);

Theoretical base

To better understand what is happening, I suggest you examine the work of some of the packages on which we will rely and use.

net/http

A package that provides an implementation of the HTTP client and server. Included in the standard Go delivery and does not require separate installation and configuration.

It is interesting to us, as it is very easy to understand, has good documentation and has the function http.FileServer().

For more details, see official documentation.

http.FileServer()

This function is the key and gives the web server full access to the specified folder and all its files. That is, the http.FileServer() function allows you to mount a folder to any specified address (route) of the web server.

For example, mount the root folder ./static/images/photos so that it is available at http://localhost:8000/photos :

http.Handle("/photos", http.FileServer("./static/images/photos"))

gobuffalo/packr

Package with a talking title. It is he who will allow us to pack all the necessary files into one binary file.

Please note, we are working with the packr v1 branch.

Suppose we have the following project directory structure:

tree .

├── main.go
└── templates
├── admin
│ └── index.html
└── index.html

The file ./main.go contains:

Now let’s compile the project into an executable binary file. At the same time, the packr package will pack the entire contents of the./templates folder into it:

packr build ./main.go

If you want to create a binary file for an OS or architecture other than the one you are working with now, then call packr like this:

# Example for GNU/Linux, x64 bit
GOOS=linux GOARCH=amd64 packr build ./main.go

zserge/webview

A tiny cross-platform web-browsing package used to create modern graphical interfaces.

Please note that the article describes how to work with the version 0.1.0.

The file ./main.go contains:

The Project structure

tree .├── vendor
├── ui
│ ├── build
│ ├── node_modules
│ ├── public
│ ├── src
│ ├── package-lock.json
│ └── package.json
├── helloworld.app
├── Gopkg.lock
├── Gopkg.lock
├── Makefile
└── main.go

Description of main files and folders

  • vendor — all packages installed using dep will be stored here;
  • ui — folder with React.js application (frontend);
  • ui/build — folder with production-version of React-app after the build;
  • ui/src — folder with the source code of the React-app;
  • ui/package.json — dependency file npm;
  • helloworld.app — macOS application (specially prepared folder);
  • Gopkg.toml — dependency file dep;
  • Makefile — make script for easy way to build app;
  • main.go — Golang application source code (backend);

Write the code

Enough theory. As he said, without exaggeration, one of the great programmers of our time, Linus Torvalds:

Talk is cheap. Show me the code.

— Linus Torvalds

Let’s follow this advice and write some code.

I will not analyze each line of code separately, as I consider it redundant and counter-productive. All code listings are provided with detailed comments.

Looking for full code example?

No problem! 👌 I created repository on my GitHub especially for you:

https://github.com/koddr/example-go-react-macos-app-1

Just git clone and make.

Memo for beginners/copy-paste developers

Great, when there is a full code listing at the end of the article, right? You can immediately, without reading the text, copy all the program code and see its execution …

At this point, I would like to appeal to all readers who do not want to spend time on theory:

Do not mindlessly copy code from the Internet! This will not help you (in understanding the code and subject of the article), nor the author (in explaining/helping in the comments).

App frontend

React.js is a powerful, but at the same time, easy-to-learn JavaScript-library for creating user interfaces, which is perfect for us to implement the frontend part of the application.

For this article, we will not use anything but the standard React.js page.

Like everything in modern frontend, we start with the installation of React.js and all the necessary auxiliary libraries.

# Create a folder for app and go into it# According to the structure of finished app,
# install React.js in ./ui directory
npx create-react-app ui# Go to folder and check that everything workscd ui && npm start && open http://localhost:3000# Next, stop dev server (press Ctrl+C) and install axios librarynpm i --save axios
Code listing for ./ui/src/App.js file
Code listing for ./ui/src/App.js file

App backend

Install the necessary Go packages:

dep ensure -add github.com/gobuffalo/packr
dep ensure -add github.com/zserge/webview

Also, we need the packr utility, which should be available for calling from the console in $GOPATH/bin/packr:

go get -u github.com/gobuffalo/packr/packr
Code listing for ./main.go file
Code listing for ./main.go file

Build a native macOS App 🏗

# Creating the directory structure of macOS app
mkdir -p helloworld.app/Contents/MacOS
# Compile ./main.go to app folder
go build -o helloworld.app/Contents/MacOS/helloworld
# Run application
open helloworld.app

Demo time 🎯

Full native macOS application about 5 min of code, , included React.js and code protection by Golang.
Full native macOS application about 5 min of code, included React.js and code protection by Golang.

Cross compilation for Windows and GNU/Linux

The theoretical block and the code given in the article are relevant for developing similar applications for other operating systems. In this case, the code remains unchanged.

Write the code once — run everywhere!

This is made possible by the cross-system nature. You can compile the executable file.

  • GNU/Linux — executable binary file;
  • Microsoft Windows — executable file .exe;
  • Apple macOS — a binary file located inside the .app structure;

We will look at this in the following articles.

Stay tuned, comment and write only good code!

Securing material

You are at the end of the article. Now you know a lot more than 15 minutes ago. Take my congratulations! 🎉

Separate 10–15 minutes and the read text restored in memory and the studied code from articles. Next, try to answer the questions and do the exercises in order to better consolidate the material.

Yes, you can pry, but only if you could not remember.

Questions

  1. What is the function of the standard go package net/http used to mount folders to the specified address (route)?
  2. What does the Marshal function do from the standard Go package encoding/json?
  3. What parameters need to be changed in the source code of the Full HD application?
  4. If you want to start a web server without goroutine?
  5. What is the command packr build ./main.go?

Exercises

  • Rewrite the code of the AJAX request (in the frontend application) without using the axios library. Hint: use the features Fetch API.
  • Add more JSON data to the frontend output in the showMessage function. Example: add a new attribute Emoji to theMessage structure and output it (with your favorite smiley) after the Text attribute.
  • Try to improve the appearance of your application, for example, using the Material UI visual component library (GitHub).

--

--

Vic Shóstak

Founder of True web artisans 🎯 Full stack Web Developer (Golang, Python, modern JavaScript, Docker), UX evangelist, DX philosopher, PWA adept.