Epic refactoring to improve testability.
It all started when I wanted to write ListServers(), but doing that
meant acquiring the test context, so that I could intercept server-side
communications. However, that couldn't be easily acquired with the old
software configuration. In fact, it was patently impossible to do,
without breaking a lot of encapsulation and conflating concerns.
Additionally, I knew I didn't want to make ComputeApi() a method of an
AccessProvider interface; considering how similar the OpenStack APIs
are, I was banking on that design decision causing a lot of duplicate
code, even if said code was simple. Not only that, but it conflated
concerns; again, something I wanted to avoid.
So, I needed to do a couple of things.
1) Realize that module-global functions are delegators to the global
context. My original implementation of ComputeApi() wasn't, which meant
that I had zero access to any contexts created in unit testing.
2) Realize that the Context interface is the true Gophercloud global
API. This meant I had to make a ComputeApi() method on the Context
interface, and implement it. This proved very convenient -- it granted
me access automatically to all test contexts.
As a checklist bullet point, whenever adding a new global-level function
to gophercloud, do it in at least these steps: a) add the function as a
method on Context. Seriously -- this is the only real way to make it
testable. b) Add a very dumb delegator function in global_context.go
which dispatches to its eponymously-named method on globalContext.
3) Making this simple change was sufficient to start to test an
implementation of ListServers(). However, invoking "c := TestContext();
c.foo(); c.bar();" was becoming repetitive and error-prone. So, I
refactored that into a Java-style DSL. These things aren't terribly Go
idiomatic, but for my needs here, they work nicely.
4) I refactored the two different implementations of custom transports
into a single "transport-double" type. This type will supports both
canned responses and some basic request validation. It's expandable by
simply adding more configuration fields and DSL methods of its own.
5) api.go is no more -- it previously served two separate purposes, each
of which has its own source file now. interfaces.go holds the
definition of all publicly visible APIs, while global_context.go
provides the default global Context, its initialization, and the
module-global entry points that delegate to the global context.
With these changes having been made, *now* I'm ready to actually start
testing ListServers() development! It only took 24 hours and 4
refreshes of the feature branch just to get this far. :-)
The nice thing is, though, that going forward, these changes should
contribute to making future endpoint binding implementations
significantly easier than what I had to do before.
diff --git a/global_context.go b/global_context.go
new file mode 100644
index 0000000..b071c09
--- /dev/null
+++ b/global_context.go
@@ -0,0 +1,48 @@
+package gophercloud
+
+// globalContext is the, well, "global context."
+// Most of this SDK is written in a manner to facilitate easier testing,
+// which doesn't require all the configuration a real-world application would require.
+// However, for real-world deployments, applications should be able to rely on a consistent configuration of providers, etc.
+var globalContext *Context
+
+// providers is the set of supported providers.
+var providers = map[string]Provider{
+ "rackspace-us": Provider{
+ AuthEndpoint: "https://identity.api.rackspacecloud.com/v2.0/tokens",
+ },
+ "rackspace-uk": Provider{
+ AuthEndpoint: "https://lon.identity.api.rackspacecloud.com/v2.0/tokens",
+ },
+}
+
+// Initialize the global context to sane configuration.
+// The Go runtime ensures this function is called before main(),
+// thus guaranteeing proper configuration before your application ever runs.
+func init() {
+ globalContext = TestContext()
+ for name, descriptor := range providers {
+ globalContext.RegisterProvider(name, descriptor)
+ }
+}
+
+// Authenticate() grants access to the OpenStack-compatible provider API.
+//
+// Providers are identified through a unique key string.
+// Specifying an unsupported provider will result in an ErrProvider error.
+//
+// The supplied AuthOptions instance allows the client to specify only those credentials
+// relevant for the authentication request. At present, support exists for OpenStack
+// Identity V2 API only; support for V3 will become available as soon as documentation for it
+// becomes readily available.
+//
+// For Identity V2 API requirements, you must provide at least the Username and Password
+// options. The TenantId field is optional, and defaults to "".
+func Authenticate(provider string, options AuthOptions) (*Access, error) {
+ return globalContext.Authenticate(provider, options)
+}
+
+// Instantiates a Cloud Servers object for the provider given.
+func ComputeApi(acc AccessProvider, criteria ApiCriteria) (ComputeProvider, error) {
+ return globalContext.ComputeApi(acc, criteria)
+}