For new contributors, we've put together a detailed tutorial here!
There are three main ways you can get involved in our open-source project, and each is described briefly below.
If you want to start fixing open bugs, we'd really appreciate that! Bug fixing is central to any project. The best way to get started is by heading to our bug tracker and finding open bugs that you think nobody is working on. It might be useful to comment on the thread to see the current state of the issue and if anybody has made any breakthroughs on it so far.
Gophercloud's documentation is automatically generated from the source code and can be read online at godoc.org.
If you feel that a certain section could be improved - whether it's to clarify ambiguity, correct a technical mistake, or to fix a grammatical error - please feel entitled to do so! We welcome doc pull requests with the same childlike enthusiasm as any other contribution!
If you've found something we've left out, we'd love for you to add it! Please first open an issue to indicate your interest to a core contributor - this enables quick/early feedback and can help steer you in the right direction by avoiding known issues. It might also help you avoid losing time implementing something that might not ever work or is outside the scope of the project.
While you're implementing the feature, one tip is to prefix your Pull Request
title with [wip]
- then people know it's a work in progress. Once the PR is
ready for review, you can remove the [wip]
tag and request a review.
We ask that you do not submit a feature that you have not spent time researching and testing first-hand in an actual OpenStack environment. While we appreciate the contribution, submitting code which you are unfamiliar with is a risk to the users who will ultimately use it. See our acceptance tests readme for information about how you can create a local development environment to better understand the feature you're working on.
Please do not hesitate to ask questions or request clarification. Your contribution is very much appreciated and we are happy to work with you to get it merged.
As a contributor you will need to setup your workspace in a slightly different way than just downloading it. Here are the basic instructions:
-
Configure your
$GOPATH
and rungo get
as described in the main README but add-tags "fixtures acceptance"
to get dependencies for unit and acceptance tests.go get -tags "fixtures acceptance" github.com/gophercloud/gophercloud
-
Move into the directory that houses your local repository:
cd ${GOPATH}/src/github.com/gophercloud/gophercloud
-
Fork the
gophercloud/gophercloud
repository and update your remote refs. You will need to rename theorigin
remote branch toupstream
, and add your fork asorigin
instead:git remote rename origin upstream git remote add origin git@github.com:<my_username>/gophercloud.git
-
Checkout the latest development branch:
git checkout master
-
If you're working on something (discussed more in detail below), you will need to checkout a new feature branch:
git checkout -b my-new-feature
-
Use a standard text editor or IDE of your choice to make your changes to the code or documentation. Once finished, commit them.
git status git add path/to/changed/file.go git commit
-
Submit your branch as a Pull Request. When submitting a Pull Request, please follow our Style Guide.
Further information about using Git can be found here.
Happy Hacking!
When working on a new or existing feature, testing will be the backbone of your work since it helps uncover and prevent regressions in the codebase. There are two types of test we use in Gophercloud: unit tests and acceptance tests, which are both described below.
Unit tests are the fine-grained tests that establish and ensure the behavior of individual units of functionality. We usually test on an operation-by-operation basis (an operation typically being an API action) with the use of mocking to set up explicit expectations. Each operation will set up its HTTP response expectation, and then test how the system responds when fed this controlled, pre-determined input.
To make life easier, we've introduced a bunch of test helpers to simplify the process of testing expectations with assertions:
import (
"testing"
"github.com/gophercloud/gophercloud/testhelper"
)
func TestSomething(t *testing.T) {
result, err := Operation()
testhelper.AssertEquals(t, "foo", result.Bar)
testhelper.AssertNoErr(t, err)
}
func TestSomethingElse(t *testing.T) {
testhelper.CheckEquals(t, "expected", "actual")
}
AssertEquals
and AssertNoErr
will throw a fatal error if a value does not
match an expected value or if an error has been declared, respectively. You can
also use CheckEquals
and CheckNoErr
for the same purpose; the only difference
being that t.Errorf
is raised rather than t.Fatalf
.
Here is a truncated example of mocked HTTP responses:
import (
"testing"
th "github.com/gophercloud/gophercloud/testhelper"
fake "github.com/gophercloud/gophercloud/testhelper/client"
"github.com/gophercloud/gophercloud/openstack/networking/v2/networks"
)
func TestGet(t *testing.T) {
// Setup the HTTP request multiplexer and server
th.SetupHTTP()
defer th.TeardownHTTP()
th.Mux.HandleFunc("/networks/d32019d3-bc6e-4319-9c1d-6722fc136a22", func(w http.ResponseWriter, r *http.Request) {
// Test we're using the correct HTTP method
th.TestMethod(t, r, "GET")
// Test we're setting the auth token
th.TestHeader(t, r, "X-Auth-Token", fake.TokenID)
// Set the appropriate headers for our mocked response
w.Header().Add("Content-Type", "application/json")
w.WriteHeader(http.StatusOK)
// Set the HTTP body
fmt.Fprintf(w, `
{
"network": {
"status": "ACTIVE",
"name": "private-network",
"admin_state_up": true,
"tenant_id": "4fd44f30292945e481c7b8a0c8908869",
"shared": true,
"id": "d32019d3-bc6e-4319-9c1d-6722fc136a22"
}
}
`)
})
// Call our API operation
network, err := networks.Get(fake.ServiceClient(), "d32019d3-bc6e-4319-9c1d-6722fc136a22").Extract()
// Assert no errors and equality
th.AssertNoErr(t, err)
th.AssertEquals(t, n.Status, "ACTIVE")
}
As we've already mentioned, unit tests have a very narrow and confined focus - they test small units of behavior. Acceptance tests on the other hand have a far larger scope: they are fully functional tests that test the entire API of a service in one fell swoop. They don't care about unit isolation or mocking expectations, they instead do a full run-through and consequently test how the entire system integrates together. When an API satisfies expectations, it proves by default that the requirements for a contract have been met.
Please be aware that acceptance tests will hit a live API - and may incur service charges from your provider. Although most tests handle their own teardown procedures, it is always worth manually checking that resources are deleted after the test suite finishes.
We provide detailed information about how to set up local acceptance test environments in our acceptance tests readme.
To run all tests:
go test -tags fixtures ./...
To run all tests with verbose output:
go test -v -tags fixtures ./...
To run tests that match certain build tags:
go test -tags "fixtures foo bar" ./...
To run tests for a particular sub-package:
cd ./path/to/package && go test -tags fixtures ./...
See here