Documentation ¶
Overview ¶
Package frameless - The Hitchhiker's Guide to the Grumpytecture.
Introduction ¶
Everything you read here handle with a grain of salt because it is just my personal opinion on this subject. This research project primary was created for myself, to refresh and sort out my experience about design for my self.
The reason ¶
There is a nice trade-off between using certain conventions or a framework that try to collect experience and build safety guards for those who may not experienced the same pitfalls yet. Using a framework on hand allows new recruits to able to create value for the company, on the other hand, hides a lot of internals from them and prevents understanding certain problems from simplicity point of view. As a side effect, frameworks often leak into the design of the application. It is nice to have something that able to provide ability to avoid common pitfalls, but I believe, that it is important to keep in front of our eye, that if we have a hammer, we should not think about every problem as nails. There is nothing wrong using frameworks, but we have to make sure, they are only used for a certain problem only, and not for gluing together everything. For example, if a framework provide help handling HTTP requests, we should not over using it, to external resources (like DB), business logic (use-cases) and business entities. As long we discipline ourselves to keep everything for what it meant to, we will receive maintainability, observability and minimal mental model needs in our project. Trough this, we can make sure, that our application use certain knowledge from the framework, and not an "xy framework app" that is heavily vendor locked to that framework. Having to deep connection with a framework can easily cause our application to be volatile against breaking changes in the framework.
This conventions that I collected in this project are serve me one purpose, which is to overcome the fact that as a human, we are in generally bad at programming. We have limited mental model capacity, and it takes time to build it up, and if one mistake made during that, the wrongly build mental model will cause bugs, and it needs minimum rubber duck debugging to fix the model. And to fight this and similar problems all the techniques here aim to minimise during programming the required mental model, help practice roles alone such as the "driver" and the "navigator" and in general to design the application in a way, where existing code base more or less protected from changes. The later is especially useful, because modern programming relies on scientific approach to test the system, and providing full edge case coverage can be exponentially hard. Don't mix it together with the test coverage % that only check whether the code path is being used once from a test case or not. Therefore when a code being used by the masses (users), it is likely used in a way that we didn't explicitly specified, and in order to not break expected system behaviors, minimising the need to change existing code base can help in general. I heavily sympathise on TDD/BDD but even with a full % coverage, the edge cases between components are in general harder to test than in simple unit tests with contracts.
The quality of the software in this project therefore defined by factors as how likely you have to change existing code base, how fast you receive back feedback regarding your change, at what quality level it able to provide feedback about system behavior changes, and how big mental model you need to build in order to understand the application on high level.
The Pressure ¶
Because different stakeholders expect results to be delivered, and if possible as soon as possible, It's rare to have a moment to think every decision through in a life of a project. The most often challenging ones are decisions made to avoid some boilerplate in the name of minimalism, but on the long run results in interface violations because rewrites.
I like to think through things, play in mind that what could be the future outcome of a given decision... How will it affect maintainability? How easy will the code be consumed if someone joins the team of that project and want to read it alone? How much effort will be required to build the mental model for the code? These are the questions I like to sort out beforehand by building principles that help me keep my self disciplined.
You will not find anything regarding complete out of the box solutions, just some idea and practice I like to follow while working on projects. Please don't expect in this repo examples in a way, that you can easily wire into your project. You can check projects that use idioms from here if you interested in such examples.
Why Golang ¶
Most of these experience sourced from working in other languages, and programming in golang's minimalist environment is kinda like a chill out place for my mind, but the knowledge I try to form it into the code here is language independent and could be used in any other languages as well. In languages where there are no interfaces, I highly recommend creating the specification that ensure this simple contract.
Principles that I liked and try to follow when I design ¶
Rule 1. You can't tell where a program is going to spend its time. Bottlenecks occur in surprising places, so don't try to guess and build in a speed hack until you've proven that's where the bottleneck really is.
Rule 2. Measure. Don't tune for speed until you've measured, and even then, still don't tune unless one part of the code overwhelms the rest.
Rule 3. Fancy algorithms are slow when n is small, and n is usually small. Fancy algorithms have big constants. Until you know that n is frequently going to be big, don't get fancy. (Even if n does get big, use Rule 2 first.)
Rule 4. Fancy algorithms are buggier than simple ones, and they're much harder to implement. Use simple algorithms as well as simple data structures.
Rule 5. Data dominates. If you've chosen the right data structures and organized things well, the algorithms will almost always be self-evident. Data structures, not algorithms, are central to programming.
Rule 6. Don't create production code that is not proven to be required. Prove it with user story and tests. If it is "hard to test", take a break and think over.
Rule 7. Use contracts wherever it makes sense instead of concrete type declarations to enforce dependency inversion.
Rule 8. Try creating code parts that a stranger could understand within 15m, else try to reduce the code in a way that it requires a lesser mind model.
Most of the rules originated from one of my favorite programmers, who was the initial reason why I started to read about golang, Rob Pike. Pike's rules 1 and 2 restate Tony Hoare's famous maxim "Premature optimization is the root of all evil." Ken Thompson rephrased Pike's rules 3 and 4 as "When in doubt, use brute force.". Rules 3 and 4 are instances of the design philosophy KISS. Rule 5 was previously stated by Fred Brooks in The Mythical Man-Month.
Resources ¶
https://12factor.net/ https://en.wikipedia.org/wiki/Law_of_Demeter https://golang.org/pkg/encoding/json/#Decoder https://en.wikipedia.org/wiki/Iterator_pattern https://en.wikipedia.org/wiki/Adapter_pattern https://en.wikipedia.org/wiki/You_aren%27t_gonna_need_it https://en.wikipedia.org/wiki/Single_responsibility_principle https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html
Domain Entity ¶
Entity encapsulate the most general and high-level rules of the application.
TL;DR: These structures are representing purely data related to some kind of real world entity. It may have high level functions that use it's own data. It knows about nothing else but it self only.
This interface here is only for documentation purpose
"An entity can be an object with methods, or it can be a set of data structures and functions" Robert Martin
In enterprise environment, this or the specification of this object can be shared between applications. If you don’t have an enterprise, and are just writing a single application, then these entities are the business objects of the application. They encapsulate the most general and high-level rules. Entity scope must be free from anything related to other software layers implementation knowledge such as SQL or HTTP request objects.
They are the least likely to change when something external changes. For example, you would not expect these objects to be affected by a change to page navigation, or security. No operational change to any particular application should affect the entity layer. By convention these structures should be placed on the top folder level of the project.
Interactor ¶
Interactor or also often referenced as Domain Logic, Business Logic or Consumer. Interactor implement a business rule to a specific audience of the software. This interface here is only for documentation purpose.
TL;DR: Using Interactor imposes discipline upon focusing on the audience who's business rule you works on.
This can be a function or a struct of function, it's up to the implementation. It has to be implemented in a framework independent way. The function arguments should be explicit Entity structures or primitives. When stream of data required, use case should declare framework and technology independent data providers.
In my future examples this data source will be fulfilled by Iterator pattern implementations.
As an easy to follow practice that you start build your application by implementing domain use cases, until you have all your business logic / use case implemented. Your test should work only with Entity structures and primitives exclusively.
If you cannot avoid to depend on external resources, use interface to represent they need. During my research, I played around multiple solutions, and the one I liked the most is the following: You describe in a shared specification, in a test that is Exposed and importable by the external interface specification, and in that you describe what is your expectation from the use-case point of view from the provided external resource, when you call it with a specific data structure. for more about this, read the "Query" and "Supplier" type.
Here is a definition from Robert Martin:
The software in this layer contains application specific business rules. It encapsulates and implements all of the use cases of the system. These use cases orchestrate the flow of data to and from the entities, and direct those entities to use their enterprise wide business rules to achieve the goals of the use case. We do not expect changes in this layer to affect the entities. We also do not expect this layer to be affected by changes to externalises such as the database, the UI, or any of the common frameworks. This layer is isolated from such concerns. We do, however, expect that changes to the operation of the application will affect the use-cases and therefore the software in this layer. If the details of a use-case change, then some code in this layer will certainly be affected.
When your application has all the use-case, then you decide the right external interface to expose them.
TIP: If you don't know how to start, imagine that every audience category your system defines has a dedicated engineer. For example in case of a web-shop: Buyer, Seller, Content Manager, Application Manager, DataBaseAdministrator just to name a few. Each engineer work on one user story for one of the audience category. You are one of the engineers. Almost every other engineer on the other user stories push code really frequently (for example 1 push / min). How would you structure and create your code in a way that you are safe from merge conflicts ? How would you design your code dependency in a way that other engineers activity unlikely to affect your code ?
Supplier ¶
Supplier is a specific implementation of an external resource that required for an Interactor.
TL;DR: Supplier (External) imposes discipline upon dependency inversion. You encapsulate all the technology specific implementations, as a separate structure that adapt to a predefined stable but easily extendable interface, so you can try anything out while frameworks and ORMs evolve, yet your domain rules will be keep safe from this changes.
A common example for a Supplier is a Entity Storage implementation, also known as Entity Repositories.
Index ¶
- func FinishOnePhaseCommit(errp *error, cm OnePhaseCommitProtocol, tx context.Context)
- func FinishTx(errp *error, commit, rollback func() error)
- func Recover(errp *error)
- type Creator
- type Decoder
- type DecoderFunc
- type Deleter
- type Error
- type EventCreate
- type EventDeleteAll
- type EventDeleteByID
- type EventUpdate
- type Finder
- type FixtureFactory
- type ID
- type Iterator
- type MetaAccessor
- type OnePhaseCommitProtocol
- type Publisher
- type Subscriber
- type Subscription
- type T
- type TwoPhaseCommitProtocol
- type Updater
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func FinishOnePhaseCommit ¶ added in v0.48.0
func FinishOnePhaseCommit(errp *error, cm OnePhaseCommitProtocol, tx context.Context)
Example ¶
package main import ( "context" "github.com/adamluzsi/frameless" ) func main() { var cm frameless.OnePhaseCommitProtocol myMethod := func(ctx context.Context) (returnError error) { tx, err := cm.BeginTx(ctx) if err != nil { return err } defer frameless.FinishOnePhaseCommit(&returnError, cm, tx) // do something with in tx return nil } _ = myMethod }
Output:
func FinishTx ¶ added in v0.48.0
Example ¶
package main import ( "context" "database/sql" "github.com/adamluzsi/frameless" ) func main() { db, err := sql.Open(`fake`, `DSN`) if err != nil { panic(err) } myMethod := func(ctx context.Context) (returnError error) { tx, err := db.Begin() if err != nil { return err } defer frameless.FinishTx(&returnError, tx.Commit, tx.Rollback) // do something with in tx return nil } _ = myMethod }
Output:
Types ¶
type Creator ¶ added in v0.31.0
type Creator interface { // Create takes a ptr to a entity<T> and store it into the resource. // It also updates the entity<T> ext:"ID" field with the associated uniq resource id. // The reason behind this links the id and not returning the id is that, // in most case the Create error value is the only thing that is checked for errors, // and introducing an extra value also introduce boiler plates in the handling. Create(ctx context.Context, ptr interface{}) error }
type Decoder ¶
type Decoder interface { // Decode will populate/replace/configure the value of the received pointer type // and in case of failure, returns an error. Decode(ptr interface{}) error }
Decoder is the interface to represent value decoding into a passed pointer type. Most commonly this happens with value decoding that was received from some sort of external resource. Decoder in other words the interface for populating/replacing a public struct with values that retried from an external resource.
type DecoderFunc ¶ added in v0.39.0
type DecoderFunc func(interface{}) error
DecoderFunc enables to use anonymous functions to be a valid DecoderFunc
func (DecoderFunc) Decode ¶ added in v0.39.0
func (lambda DecoderFunc) Decode(i interface{}) error
Decode proxy the call to the wrapped Decoder function
type Deleter ¶ added in v0.31.0
type Deleter interface { // DeleteByID will remove a <T> type entity from the storage by a given ID DeleteByID(ctx context.Context, id interface{}) error // DeleteAll will erase all entity from the resource that has <T> type DeleteAll(context.Context) error }
Deleter request to destroy a business entity in the Resource that implement it's test.
type Error ¶
type Error string
Error is an implementation for the error interface that allow you to declare exported globals with the `const` keyword.
TL;DR: const ErrSomething errs.Error = "something is an error"
type EventCreate ¶ added in v0.41.0
type EventCreate struct{ Entity T }
type EventDeleteAll ¶ added in v0.41.0
type EventDeleteAll struct{}
type EventDeleteByID ¶ added in v0.41.0
type EventDeleteByID struct{ ID ID }
type EventUpdate ¶ added in v0.41.0
type EventUpdate struct{ Entity T }
type Finder ¶ added in v0.31.0
type Finder interface { // FindByID will link an entity that is found in the resource to the received ptr, // and report back if it succeeded finding the entity in the resource. // It also reports if there was an unexpected exception during the execution. // It was an intentional decision to not use error to represent "not found" case, // but tell explicitly this information in the form of return bool value. FindByID(ctx context.Context, ptr, id interface{}) (found bool, err error) // FindAll will return all entity that has <T> type FindAll(context.Context) Iterator }
type FixtureFactory ¶ added in v0.49.0
type FixtureFactory interface { // Fixture will create a fixture data for the provided T type. // The created fixture data expected to have random data in its fields. // It is expected that the created fixture will have no content for extID field. Fixture(T interface{}, ctx context.Context) (_T interface{}) }
type Iterator ¶
type Iterator interface { // Closer is required to make it able to cancel iterators where resource being used behind the scene // for all other case where the underling io is handled on higher level, it should simply return nil io.Closer // Next will ensure that Decode return the next item when it is executed Next() bool // Err return the cause if for some reason by default the More return false all the time Err() error // Decoder will populate an object with values and/or return error // this is required to retrieve the current value from the iterator Decoder }
Iterator define a separate object that encapsulates accessing and traversing an aggregate object. Clients use an iterator to access and traverse an aggregate without knowing its representation (data structures). Interface design inspirited by https://golang.org/pkg/encoding/json/#Decoder https://en.wikipedia.org/wiki/Iterator_pattern
type MetaAccessor ¶ added in v0.43.1
type OnePhaseCommitProtocol ¶ added in v0.31.0
type OnePhaseCommitProtocol interface { // BeginTx creates a context with a transaction. // All statements that receive this context should be executed within the given transaction in the context. // After a BeginTx command will be executed in a single transaction until an explicit COMMIT or ROLLBACK is given. // // In case the resource support some form of isolation level, // or other ACID related property of the transaction, // then it is advised to prepare this information in the context before calling BeginTx. // e.g.: // ... // var err error // ctx = r.ContextWithIsolationLevel(ctx, sql.LevelSerializable) // ctx, err = r.BeginTx(ctx) // BeginTx(context.Context) (context.Context, error) // CommitTx Commit commits the current transaction. // All changes made by the transaction become visible to others and are guaranteed to be durable if a crash occurs. CommitTx(context.Context) error // RollbackTx rolls back the current transaction and causes all the updates made by the transaction to be discarded. RollbackTx(context.Context) error }
type Publisher ¶ added in v0.41.0
type Publisher interface { // Subscribe create a subscription that will consume an event feed. // If event stream repeatability from a certain point is a requirement, // it needs to be further specified with a resource contract. Subscribe(context.Context, Subscriber) (Subscription, error) }
type Subscriber ¶ added in v0.31.0
type Subscriber interface { // Handle handles the the subscribed event. // Context may or may not have meta information about the received event. // To ensure expectations, define a resource specification <contract> about what must be included in the context. Handle(ctx context.Context, event interface{}) error // Error allow the subscription implementation to be notified about unexpected situations // that needs to be handled by the subscriber. // For e.g. the connection is lost and the subscriber might have cached values // that must be invalidated on the next successful Handle call Error(ctx context.Context, err error) error }
type Subscription ¶ added in v0.31.0
type TwoPhaseCommitProtocol ¶ added in v0.31.0
type TwoPhaseCommitProtocol interface { OnePhaseCommitProtocol // PrepareTx communicate with the resource that the current transaction is done and should be prepared for commit later. // // Prepare transaction is not intended for use in applications or interactive sessions. // Its purpose is to allow an external transaction manager to perform atomic global transactions across multiple databases or other transactional resources. // Unless you're writing a transaction manager, you probably shouldn't be using PrepareTx. // // This command must be used on a context made with BeginTx. // Calling CommitTx or RollbackTx with the received context // must be interpreted as Two Phase Commit Protocol's Commit or Rollback action. PrepareTx(context.Context) (context.Context, error) }
Source Files ¶
Directories ¶
Path | Synopsis |
---|---|
adapters
|
|
postgresql
Module
|
|
Package specs Summary This package implements generic CRUD operation related testing specifications that commonly appears upon interacting with external resources.
|
Package specs Summary This package implements generic CRUD operation related testing specifications that commonly appears upon interacting with external resources. |
Package iterators provide iterator implementations.
|
Package iterators provide iterator implementations. |
postgresql
module
|
|