Skip to content

Latest commit

 

History

History
143 lines (114 loc) · 5.88 KB

CONTRIBUTING.md

File metadata and controls

143 lines (114 loc) · 5.88 KB

Dynatrace Operator

How to Contribute

You are welcome to contribute to Dynatrace Operator. Use issues for discussing proposals or to raise a question. If you have improvements to Dynatrace Operator, please submit your pull request. For those just getting started, consult this guide.

Coding style guide

General

  • Use descriptive names (namespace is better than ns, dynakube is better than dk, etc.)
  • Avoid using client.Client for 'getting' resources, use client.Reader (also known as apiReader) instead.
    • client.Client uses a cache (or tries to) that requires more permissions than normally, and can also give you outdated results.
  • Do not create methods with more than two parameters (in extremely rare occasions maybe three) except constructors and factory functions. Structs and interfaces exist for a reason.
  • Avoid returning responses (e.g., reconcile.Result, admission.Patched) in anything but Reconcile or Handle functions.

Cuddling of statements

Statements must be cuddled, i.e., written as a single block, if an if-statement directly follows a single assignment and the condition is directly related to the assignment. This commonly occurs with error handling, but is not restricted to it.
Example:

err := assignment1()
if err != nil {
  do()
}

value1 := assignment2()
if value1 {
  do()
}

Statements must not be cuddled with each other if multiple of the same type of statements follow each other. A statement must be cuddled with following statements, if they are of the same type.
Example:

value1 := assignment1()
value2 := assignment2()
value3, err := assignment3()

if err != nil {
  do()
}
if value1 == "something" {
  do()
}
if value2 {
  do()
}

Reconciler vs Controller

A Controller is a struct that DIRECTLY handles the reconcile Requests.

Important characteristics:

  • We pass it to ctrl.NewControllerManagedBy(mgr)
  • Has a Reconcile(ctx context.Context, request reconcile.Request) (reconcile.Result, error) function.
  • Calls other Reconcilers when needed
  • Examples: DynakubeController, WebhookCertController, NodesController, OneAgentProvisioner, CSIGarbageCollector

A Reconciler is a struct that INDIRECTLY handles the reconcile Requests.

Important characteristics:

  • Has a Reconcile(<whatever is necessary>) function
  • Is called/used BY a Controller
  • Examples: OneAgentReconciler, IstioReconciler...

Errors

Do's

  • If an error is returned by an external function from an external package, it must be wrapped with errors.WithStack().
  • If an error is instantiated by internal code, it must be instantiated with errors.New()
  • If an error is returned by an internal function and propagated, it must be propagated as is and must not be wrapped with errors.WithStack().
    • The stacktrace is already complete when errors.New is called, wrapping it with errors.WithStack() convolutes it.

Don'ts

  • Errors must not be logged with log.Errorf
    • Errors are propagated to the controller or reconciler and then automatically logged by the Operator-SDK framework

Logging

Do's

  • Use a package global logger (1 per package), should be named log and be declared in the config.go of the package. (temporary solution)
    • In case of larger packages it's advisable to introduce separate loggers for different parts of the package, these sub-loggers should still be derived from the main logger of the package and given a name.
      • Example: in webhook/mutation var nsLog = log.WithName("namespace") (the name of this logger is mutation-webhook.namespace)
  • Use the logger defined in the dynatrace-operator/src/logger and always give it a name.
    • The name of the logger (given via .WithName("...")) should use - to divide longer names.
    • Example: var log = logger.Factory.GetLogger("mutation-webhook")

Don'ts

  • Don't use fmt.Sprintf for creating log messages, the values you wish to replace via Sprintf should be provided to the logger as key-value pairs.
    • Example: log.Info("deleted volume info", "ID", volume.VolumeID, "PodUID", volume.PodName, "Version", volume.Version, "TenantUUID", volume.TenantUUID)
  • Don't start a log message with uppercase characters, unless it's a name of an exact proper noun.

Testing

Do's

  • Write unit-tests ;)
  • Use assert and require. (github.com/stretchr/testify/assert, github.com/stretchr/testify/require)
  • require is your friend, use it for checking errors (require.NoError(t, err)) or anywhere where executing the rest of the asserts in case of the check failing would just be noise in the output.
  • Abstract the setup/assert phase as much as possible so it can be reused in other tests in the package.
  • Use t.Run and give a title that describes what you are testing in that run.
  • Use this structure:
func TestMyFunction(t *testing.T) {
    	// Common setup, used for multiple cases
    	testString := "test"

    	// Each test case of a function gets a t.Run
    	t.Run(`useful title`, func(t *testing.T) {
        	// Arrange/Setup
		testInt := 1

        	// Act
        	out, err := MyFunction(testString, testInt)

        	// Assert
		require.Nil(t, err)
		assert.Equal(t, out, testString)
	})
    	t.Run(`other useful title`, func(t *testing.T) {
        	// Arrange
		testInt := 4

        	// Act
        	out, err := MyFunction(testString, testInt)

        	// Assert
		require.Error(t, err)
		assert.Empty(t, out)
	})
}

Don'ts

  • Don't name the testing helper functions/variables/constants in a way that could be confused with actual functions. (e.g. add test to the beginning)
  • Avoid magic ints/strings/... where possible, give names to your values and try to reuse them where possible
  • Don't name test functions like TestMyFunctionFirstCase, instead use single TestMyFunction test function with multiple t.Run cases in it that have descriptive names.