Skip to content

Commit 0c12482

Browse files
committed
Added readme, added tests, and fixed up some more comments.
1 parent c5f7710 commit 0c12482

File tree

4 files changed

+180
-15
lines changed

4 files changed

+180
-15
lines changed

README.rst

+22-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
.. raw:: html
2-
2+
33
<img src="https://cdn.rawgit.com/michelvocks/ef3894f63c3bb004bca1a2fd5f7eb644/raw/c36d614db8afe229b466b38de1636a82ad809f64/gaia-logo-text.png" width="650px">
44

55
|build-status| |go-report| |go-doc| |apache2| |chat| |codecov|
@@ -13,8 +13,8 @@ Motivation
1313

1414
.. begin-motivation
1515
16-
*Automation Engineer*, *DevOps*, *SRE*, *Cloud Engineer*,
17-
*Platform Engineer* - they all have one in common:
16+
*Automation Engineer*, *DevOps*, *SRE*, *Cloud Engineer*,
17+
*Platform Engineer* - they all have one in common:
1818
The majority of tech people are not motivated to take up this work and they are hard to recruit.
1919

2020
One of the main reasons for this is the abstraction and poor execution of many automation tools. They come with their own configuration (`YAML`_ syntax) specification or limit the user to one specific programming language. Testing is nearly impossible because most automation tools lack the ability to mock services and subsystems. Even tiny things, for example parsing a JSON file, are sometimes really painful because external, outdated libraries were used and not included in the standard framework.
@@ -26,11 +26,11 @@ How does it work?
2626

2727
.. begin-architecture
2828
29-
Gaia is based on `HashiCorp's go-plugin`_. It's a plugin system that uses `gRPC`_ to communicate over `HTTP/2`_. HashiCorp developed this tool initially for `Packer`_ but it's now heavily used by `Terraform`_, `Nomad`_, and `Vault`_ too.
29+
Gaia is based on `HashiCorp's go-plugin`_. It's a plugin system that uses `gRPC`_ to communicate over `HTTP/2`_. HashiCorp developed this tool initially for `Packer`_ but it's now heavily used by `Terraform`_, `Nomad`_, and `Vault`_ too.
3030

3131
Plugins, which we named pipelines, are applications which can be written in any programming language as long as `gRPC`_ is supported. All functions, which we call Jobs, are exposed to Gaia and can form up a dependency graph which describes the order of execution.
3232

33-
Pipelines can be compiled locally or simply over the build system. Gaia clones the git repository and automatically builds the included pipeline. If a change (`git push`_) happened, Gaia will automatically rebuild the pipeline for you.
33+
Pipelines can be compiled locally or simply over the build system. Gaia clones the git repository and automatically builds the included pipeline. If a change (`git push`_) happened, Gaia will automatically rebuild the pipeline for you.
3434

3535
After a pipeline has been started, all log output are returned back to Gaia and displayed in a detailed overview with their final result status.
3636

@@ -107,7 +107,7 @@ Here is an example:
107107
jobs := sdk.Jobs{
108108
sdk.Job{
109109
Handler: DoSomethingAwesome,
110-
Title: "DoSomethingAwesome",
110+
Title: "DoSomethingAwesome",
111111
Description: "This job does something awesome.",
112112
113113
// Increase the priority if this job should be executed later than other jobs.
@@ -123,7 +123,7 @@ Here is an example:
123123
124124
Like you can see, pipelines are defined by jobs. Usually, a function represents a job. You can define as many jobs in your pipeline as you want.
125125

126-
At the end, we define a jobs array that populates all jobs to gaia. We also add some information like a title, a description and the priority.
126+
At the end, we define a jobs array that populates all jobs to gaia. We also add some information like a title, a description and the priority.
127127

128128
The priority is really important and should always be used. If, for example, job A has a higher priority (decimal number) as job B, job A will be executed **after** job B. Priority defines therefore the order of execution. If two or more jobs have the same priority, those will be executed simultanously. You can compare it with the `Unix nice level`_.
129129

@@ -132,6 +132,11 @@ Gaia will compile it and add it to it's store for later execution.
132132

133133
Please find a bit more sophisticated example in our `go-example repo`_.
134134

135+
Security
136+
--------
137+
138+
See the Documentation located here: |security-docs|.
139+
135140
Documentation and more
136141
======================
137142

@@ -144,11 +149,11 @@ What problem solves **Gaia**?
144149
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
145150
Literally every tool which were designed for automation, continuous integration (CI), and continuous deployment (CD) like Spinnaker, Jenkins, Gitlab CI/CD, TravisCI, CircleCI, Codeship, Bamboo and many more, introduced their own configuration format. Some of them don't even support *configuration/automation as code*. This works well for simple tasks like running a ``go install`` or ``mvn clean install`` but in the real world there is more to do.
146151

147-
Gaia is the first platform which does not limit the user and provides full support for almost all common programming languages without losing the features offered by todays CI/CD tools.
152+
Gaia is the first platform which does not limit the user and provides full support for almost all common programming languages without losing the features offered by todays CI/CD tools.
148153

149154
What is a **pipeline**?
150155
~~~~~~~~~~~~~~~~~~~~~~~
151-
A pipeline is a real application with at least one function (we call it Job). Every programming language can be used as long as gRPC is supported. We offer SDKs (currently only Go but others are already in development) to support the development.
156+
A pipeline is a real application with at least one function (we call it Job). Every programming language can be used as long as gRPC is supported. We offer SDKs (currently only Go but others are already in development) to support the development.
152157

153158
What is a **job**?
154159
~~~~~~~~~~~~~~~~~~
@@ -169,12 +174,12 @@ Gaia is currently in alpha version available. We extremely recommend to not use
169174

170175
One of the main issues currently is the lack of unit- and integration tests. This is on our to-do list and we are working on this topic with high priority.
171176

172-
It is planned that other programming languages should be supported in the next few month. It is up to the community which languages will be supported next.
177+
It is planned that other programming languages should be supported in the next few month. It is up to the community which languages will be supported next.
173178

174179
Contributing
175180
============
176181

177-
Gaia can only evolve and become a great product with the help of contributors. If you like to contribute, please have a look at our `issues section`_. We do our best to mark issues for new contributors with the label *good first issue*.
182+
Gaia can only evolve and become a great product with the help of contributors. If you like to contribute, please have a look at our `issues section`_. We do our best to mark issues for new contributors with the label *good first issue*.
178183

179184
If you think you found a good first issue, please consider this list as a short guide:
180185

@@ -225,7 +230,7 @@ If you have any questions feel free to contact us on `gitter`_.
225230
:alt: Apache licensed
226231
:target: https://github.com/gaia-pipeline/gaia/blob/master/LICENSE
227232

228-
.. |chat| image:: https://badges.gitter.im/Join%20Chat.svg
233+
.. |chat| image:: https://badges.gitter.im/Join%20Chat.svg
229234
:alt: Gitter
230235
:target: https://gitter.im/gaia-pipeline
231236

@@ -247,15 +252,18 @@ If you have any questions feel free to contact us on `gitter`_.
247252
.. |sh-create-pipeline-history| image:: https://cdn.rawgit.com/michelvocks/6868118d0da06a422e69e453497eb30d/raw/142a2969c4d27d4135ef8f96213bb166009fda1e/create_pipeline_history.png
248253
:alt: gaia create pipeline history screenshot
249254
:width: 650px
250-
255+
251256
.. |sh-pipeline-detailed| image:: https://cdn.rawgit.com/michelvocks/6868118d0da06a422e69e453497eb30d/raw/51b4d6cbc3d86b1fe9531250db5456595423d9ec/pipeline_detailed.png
252257
:alt: gaia pipeline detailed screenshot
253258
:width: 650px
254-
259+
255260
.. |sh-pipeline-logs| image:: https://cdn.rawgit.com/michelvocks/6868118d0da06a422e69e453497eb30d/raw/51b4d6cbc3d86b1fe9531250db5456595423d9ec/pipeline_logs.png
256261
:alt: gaia pipeline logs screenshot
257262
:width: 650px
258263

259264
.. |sh-settings| image:: https://cdn.rawgit.com/michelvocks/6868118d0da06a422e69e453497eb30d/raw/142a2969c4d27d4135ef8f96213bb166009fda1e/settings.png
260265
:alt: gaia settings screenshot
261266
:width: 650px
267+
.. |security-docs|
268+
:alt: Security in Gaia
269+
:target: https://github.com/gaia-pipeline/gaia/blob/master/security/README.md

handlers/vault.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ func SetSecret(c echo.Context) error {
5555
if err != nil {
5656
return c.String(http.StatusInternalServerError, err.Error())
5757
}
58-
return c.String(http.StatusOK, "secret successfully set")
58+
return c.String(http.StatusCreated, "secret successfully set")
5959
}
6060

6161
// ListSecrets retrieves all secrets from the vault.

handlers/vault_test.go

+125
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
package handlers
2+
3+
import (
4+
"bytes"
5+
"encoding/json"
6+
"io/ioutil"
7+
"net/http"
8+
"net/http/httptest"
9+
"os"
10+
"testing"
11+
12+
"github.com/gaia-pipeline/gaia"
13+
"github.com/gaia-pipeline/gaia/store"
14+
hclog "github.com/hashicorp/go-hclog"
15+
"github.com/labstack/echo"
16+
)
17+
18+
func TestVaultWorkflowAddListDelete(t *testing.T) {
19+
dataDir, err := ioutil.TempDir("", "temp")
20+
if err != nil {
21+
t.Fatalf("error creating data dir %v", err.Error())
22+
}
23+
24+
defer func() {
25+
gaia.Cfg = nil
26+
os.RemoveAll(dataDir)
27+
}()
28+
29+
gaia.Cfg = &gaia.Config{
30+
Logger: hclog.NewNullLogger(),
31+
DataPath: dataDir,
32+
CAPath: dataDir,
33+
VaultPath: dataDir,
34+
}
35+
36+
dataStore := store.NewStore()
37+
err = dataStore.Init()
38+
if err != nil {
39+
t.Fatalf("cannot initialize store: %v", err.Error())
40+
}
41+
42+
e := echo.New()
43+
InitHandlers(e, dataStore, nil)
44+
t.Run("can add secret", func(t *testing.T) {
45+
body := map[string]string{
46+
"Key": "Key",
47+
"Value": "Value",
48+
}
49+
bodyBytes, _ := json.Marshal(body)
50+
req := httptest.NewRequest(echo.POST, "/api/"+apiVersion+"/secret", bytes.NewBuffer(bodyBytes))
51+
req.Header.Set("Content-Type", "application/json")
52+
rec := httptest.NewRecorder()
53+
c := e.NewContext(req, rec)
54+
55+
SetSecret(c)
56+
57+
if rec.Code != http.StatusCreated {
58+
t.Fatalf("expected response code %v got %v", http.StatusCreated, rec.Code)
59+
}
60+
})
61+
62+
t.Run("can update secret", func(t *testing.T) {
63+
body := map[string]string{
64+
"Key": "Key",
65+
"Value": "Value",
66+
}
67+
bodyBytes, _ := json.Marshal(body)
68+
req := httptest.NewRequest(echo.PUT, "/api/"+apiVersion+"/secret", bytes.NewBuffer(bodyBytes))
69+
req.Header.Set("Content-Type", "application/json")
70+
rec := httptest.NewRecorder()
71+
c := e.NewContext(req, rec)
72+
73+
SetSecret(c)
74+
75+
if rec.Code != http.StatusCreated {
76+
t.Fatalf("expected response code %v got %v", http.StatusCreated, rec.Code)
77+
}
78+
})
79+
80+
t.Run("can list secrets", func(t *testing.T) {
81+
req := httptest.NewRequest(echo.GET, "/api/"+apiVersion+"/secrets", nil)
82+
req.Header.Set("Content-Type", "application/json")
83+
rec := httptest.NewRecorder()
84+
c := e.NewContext(req, rec)
85+
86+
ListSecrets(c)
87+
88+
if rec.Code != http.StatusOK {
89+
t.Fatalf("expected response code %v got %v", http.StatusCreated, rec.Code)
90+
}
91+
body, _ := ioutil.ReadAll(rec.Body)
92+
expectedBody := "[{\"key\":\"Key\",\"value\":\"**********\"}]"
93+
if string(body) != expectedBody {
94+
t.Fatalf("response body did not equal expected body: expected: %s, actual: %s", expectedBody, string(body))
95+
}
96+
})
97+
98+
t.Run("can delete secrets", func(t *testing.T) {
99+
req := httptest.NewRequest(echo.DELETE, "/api/"+apiVersion+"/secret/:key", nil)
100+
req.Header.Set("Content-Type", "application/json")
101+
rec := httptest.NewRecorder()
102+
c := e.NewContext(req, rec)
103+
c.SetParamNames("key")
104+
c.SetParamValues("Key")
105+
106+
RemoveSecret(c)
107+
108+
if rec.Code != http.StatusOK {
109+
t.Fatalf("expected response code %v got %v", http.StatusCreated, rec.Code)
110+
}
111+
})
112+
113+
t.Run("can delete fails if no secret is provided", func(t *testing.T) {
114+
req := httptest.NewRequest(echo.DELETE, "/api/"+apiVersion+"/secret/:key", nil)
115+
req.Header.Set("Content-Type", "application/json")
116+
rec := httptest.NewRecorder()
117+
c := e.NewContext(req, rec)
118+
119+
RemoveSecret(c)
120+
121+
if rec.Code != http.StatusBadRequest {
122+
t.Fatalf("expected response code %v got %v", http.StatusCreated, rec.Code)
123+
}
124+
})
125+
}

security/README.md

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
# Security
2+
3+
## Certificates
4+
5+
Gaia, when first started will create a signed certificate in a location
6+
defined by the user under `gaia.Cfg.CAPath` which can be set by the runtime flag
7+
`-capath=/etc/gaia/cert` for example. It is recommended that the certificate
8+
is kept separate from the main Gaia work folder and in a secure location.
9+
10+
This certificate is used in two places. First, in the communication between the
11+
admin portal and the back-end. Second, by the Vault.
12+
13+
## The Vault
14+
15+
The Vault is a secure storage for secret values like, password, tokens and other
16+
things that the user would like to pass securly into a Pipeline. The Vault is
17+
encrypted using AES cipher technology where the key is derived from the above
18+
certificate and the IV is included in the encrypted content.
19+
20+
The Vault file's location can be configured through the runtime variable called
21+
`VaultPath`. For maximum security it is recommended that this file is kept on an
22+
encrypted, mounted drive. In case there is a breach the drive can be quickly removed
23+
and the file deleted, thus rotating all of the secrets at once, under Gaia.
24+
25+
To create an encrypted MacOSX image follow this guide: [Encrypted Secure Disk Image on Mac](https://www.howtogeek.com/183826/how-to-create-an-encrypted-file-container-disk-image-on-a-mac/).
26+
27+
To create an encrypted disk on Linux follow this guide: [Encrypted Disk Image on Linux](http://freesoftwaremagazine.com/articles/create_encrypted_disk_image_gnulinux/).
28+
29+
The admin will never see the secure values, not when editing, not when adding and not
30+
when looking at the list of secrets. Only the Key names are displayed at all times.
31+
32+
It's possible to Add, Delete, Update and List secrets in the system.

0 commit comments

Comments
 (0)