-
-
Save thomaspoignant/5b72d579bd5f311904d973652180c705 to your computer and use it in GitHub Desktop.
GOCMD=go | |
GOTEST=$(GOCMD) test | |
GOVET=$(GOCMD) vet | |
BINARY_NAME=example | |
VERSION?=0.0.0 | |
SERVICE_PORT?=3000 | |
DOCKER_REGISTRY?= #if set it should finished by / | |
EXPORT_RESULT?=false # for CI please set EXPORT_RESULT to true | |
GREEN := $(shell tput -Txterm setaf 2) | |
YELLOW := $(shell tput -Txterm setaf 3) | |
WHITE := $(shell tput -Txterm setaf 7) | |
CYAN := $(shell tput -Txterm setaf 6) | |
RESET := $(shell tput -Txterm sgr0) | |
.PHONY: all test build vendor | |
all: help | |
## Build: | |
build: ## Build your project and put the output binary in out/bin/ | |
mkdir -p out/bin | |
GO111MODULE=on $(GOCMD) build -mod vendor -o out/bin/$(BINARY_NAME) . | |
clean: ## Remove build related file | |
rm -fr ./bin | |
rm -fr ./out | |
rm -f ./junit-report.xml checkstyle-report.xml ./coverage.xml ./profile.cov yamllint-checkstyle.xml | |
vendor: ## Copy of all packages needed to support builds and tests in the vendor directory | |
$(GOCMD) mod vendor | |
watch: ## Run the code with cosmtrek/air to have automatic reload on changes | |
$(eval PACKAGE_NAME=$(shell head -n 1 go.mod | cut -d ' ' -f2)) | |
docker run -it --rm -w /go/src/$(PACKAGE_NAME) -v $(shell pwd):/go/src/$(PACKAGE_NAME) -p $(SERVICE_PORT):$(SERVICE_PORT) cosmtrek/air | |
## Test: | |
test: ## Run the tests of the project | |
ifeq ($(EXPORT_RESULT), true) | |
GO111MODULE=off go get -u github.com/jstemmer/go-junit-report | |
$(eval OUTPUT_OPTIONS = | tee /dev/tty | go-junit-report -set-exit-code > junit-report.xml) | |
endif | |
$(GOTEST) -v -race ./... $(OUTPUT_OPTIONS) | |
coverage: ## Run the tests of the project and export the coverage | |
$(GOTEST) -cover -covermode=count -coverprofile=profile.cov ./... | |
$(GOCMD) tool cover -func profile.cov | |
ifeq ($(EXPORT_RESULT), true) | |
GO111MODULE=off go get -u github.com/AlekSi/gocov-xml | |
GO111MODULE=off go get -u github.com/axw/gocov/gocov | |
gocov convert profile.cov | gocov-xml > coverage.xml | |
endif | |
## Lint: | |
lint: lint-go lint-dockerfile lint-yaml ## Run all available linters | |
lint-dockerfile: ## Lint your Dockerfile | |
# If dockerfile is present we lint it. | |
ifeq ($(shell test -e ./Dockerfile && echo -n yes),yes) | |
$(eval CONFIG_OPTION = $(shell [ -e $(shell pwd)/.hadolint.yaml ] && echo "-v $(shell pwd)/.hadolint.yaml:/root/.config/hadolint.yaml" || echo "" )) | |
$(eval OUTPUT_OPTIONS = $(shell [ "${EXPORT_RESULT}" == "true" ] && echo "--format checkstyle" || echo "" )) | |
$(eval OUTPUT_FILE = $(shell [ "${EXPORT_RESULT}" == "true" ] && echo "| tee /dev/tty > checkstyle-report.xml" || echo "" )) | |
docker run --rm -i $(CONFIG_OPTION) hadolint/hadolint hadolint $(OUTPUT_OPTIONS) - < ./Dockerfile $(OUTPUT_FILE) | |
endif | |
lint-go: ## Use golintci-lint on your project | |
$(eval OUTPUT_OPTIONS = $(shell [ "${EXPORT_RESULT}" == "true" ] && echo "--out-format checkstyle ./... | tee /dev/tty > checkstyle-report.xml" || echo "" )) | |
docker run --rm -v $(shell pwd):/app -w /app golangci/golangci-lint:latest-alpine golangci-lint run --deadline=65s $(OUTPUT_OPTIONS) | |
lint-yaml: ## Use yamllint on the yaml file of your projects | |
ifeq ($(EXPORT_RESULT), true) | |
GO111MODULE=off go get -u github.com/thomaspoignant/yamllint-checkstyle | |
$(eval OUTPUT_OPTIONS = | tee /dev/tty | yamllint-checkstyle > yamllint-checkstyle.xml) | |
endif | |
docker run --rm -it -v $(shell pwd):/data cytopia/yamllint -f parsable $(shell git ls-files '*.yml' '*.yaml') $(OUTPUT_OPTIONS) | |
## Docker: | |
docker-build: ## Use the dockerfile to build the container | |
docker build --rm --tag $(BINARY_NAME) . | |
docker-release: ## Release the container with tag latest and version | |
docker tag $(BINARY_NAME) $(DOCKER_REGISTRY)$(BINARY_NAME):latest | |
docker tag $(BINARY_NAME) $(DOCKER_REGISTRY)$(BINARY_NAME):$(VERSION) | |
# Push the docker images | |
docker push $(DOCKER_REGISTRY)$(BINARY_NAME):latest | |
docker push $(DOCKER_REGISTRY)$(BINARY_NAME):$(VERSION) | |
## Help: | |
help: ## Show this help. | |
@echo '' | |
@echo 'Usage:' | |
@echo ' ${YELLOW}make${RESET} ${GREEN}<target>${RESET}' | |
@echo '' | |
@echo 'Targets:' | |
@awk 'BEGIN {FS = ":.*?## "} { \ | |
if (/^[a-zA-Z_-]+:.*?##.*$$/) {printf " ${YELLOW}%-20s${GREEN}%s${RESET}\n", $$1, $$2} \ | |
else if (/^## .*$$/) {printf " ${CYAN}%s${RESET}\n", substr($$1,4)} \ | |
}' $(MAKEFILE_LIST) |
Associate medium article to explain the content of the makefile is available here: https://link.medium.com/JF1bkIuVUdb
I suggest this line to make dynamic help:
help: ## Show this help.
@awk 'BEGIN {FS = ":.*?## "} /^[a-zA-Z_-]+:.*?## / {printf "${YELLOW}%-16s${GREEN}%s${RESET}\n", $$1, $$2}' $(MAKEFILE_LIST)
I suggest this line to make dynamic help:
help: ## Show this help. @awk 'BEGIN {FS = ":.*?## "} /^[a-zA-Z_-]+:.*?## / {printf "${YELLOW}%-16s${GREEN}%s${RESET}\n", $$1, $$2}' $(MAKEFILE_LIST)
Good catch @nstrappazzonc, I've updated the gist with your recommandation.
Revision 6 rework the help function to support categories if needed.
The new output of the help will looks like:
❯ make help
Usage:
make <target>
Targets:
Build:
build Build your project and put the output binary in out/bin/
clean Remove build related file
vendor Copy of all packages needed to support builds and tests in the vendor directory
watch Run the code with cosmtrek/air to have automatic reload on changes
Test:
test Run the tests of the project
coverage Run the tests of the project and export the coverage
Lint:
lint Run all available linters
lint-dockerfile Lint your Dockerfile
lint-go Use golintci-lint on your project
lint-yaml Use yamllint on the yaml file of your projects
Docker:
docker-build Use the dockerfile to build the container
docker-release Release the container with tag latest and version
Help:
help Show this help.
Hey @thomaspoignant. Where does ${CYAN}
is coming from?
@tsologub I guess it's missing :-), just add that line after the line #11:
CYAN := $(shell tput -Txterm setaf 6)
@ mowtschan good catch, I've forgotten to add the variable.
@tsologub I've updated the gist.
you can make the version dynamic with:
VERSION ?= $(shell git describe --tags --always --dirty --match=v* 2> /dev/null || cat $(PWD)/.version 2> /dev/null || echo v0)
So why is my project asking me to manually crate go mod init?
Do you manually create this and check it into your source control? Why not have a target in the Makefile to build this?
So why is my project asking me to manually crate go mod init?
Do you manually create this and check it into your source control? Why not have a target in the Makefile to build this?
@snoby I did not put it in the makefile because this is a one time thing but you can consider to add it yes.
Before using this makefile please edit the
BINARY_NAME
variable