Sign Up
Log In
Log In
or
Sign Up
Places
All Projects
Status Monitor
Collapse sidebar
SUSE:SLE-12-SP4:GA
golang-github-prometheus-promu.35598
promu-0.16.0.obscpio
Overview
Repositories
Revisions
Requests
Users
Attributes
Meta
File promu-0.16.0.obscpio of Package golang-github-prometheus-promu.35598
07070100000000000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001700000000promu-0.16.0/.circleci07070100000001000081A4000000000000000000000001660D2C8E0000075D000000000000000000000000000000000000002200000000promu-0.16.0/.circleci/config.yml--- version: 2.1 executors: # Whenever the Go version is updated here, .promu.yml should also be updated. golang: docker: - image: cimg/go:1.22 jobs: test: executor: golang steps: - setup_remote_docker - checkout - run: go build -o promu-intermediate - run: make check_license style unused lint test build PROMU=./promu-intermediate - store_artifacts: path: promu destination: /build/promu - run: rm -v promu promu-intermediate - run: git diff --exit-code build: executor: golang environment: PROMU: ./promu-intermediate working_directory: /home/circleci/.go_workspace/src/github.com/prometheus/promu parallelism: 5 steps: - setup_remote_docker - checkout - run: go build -o promu-intermediate - run: ./promu-intermediate crossbuild -v --parallelism $CIRCLE_NODE_TOTAL --parallelism-thread $CIRCLE_NODE_INDEX - persist_to_workspace: root: . paths: - .build - store_artifacts: path: .build destination: /build release_tags: executor: golang environment: PROMU: ./promu-intermediate steps: - checkout - attach_workspace: at: . - run: go build -o promu-intermediate - run: ./promu-intermediate crossbuild tarballs - run: ./promu-intermediate checksum .tarballs - run: ./promu-intermediate release .tarballs - store_artifacts: path: .tarballs destination: releases workflows: version: 2 promu: jobs: - test: filters: tags: only: /.*/ - build: filters: tags: only: /.*/ - release_tags: requires: - test - build filters: tags: only: /^v[0-9]+(\.[0-9]+){2}(-.+|[^-.]*)$/ branches: ignore: /.*/ 07070100000002000041ED000000000000000000000003660D2C8E00000000000000000000000000000000000000000000001500000000promu-0.16.0/.github07070100000003000081A4000000000000000000000001660D2C8E0000006E000000000000000000000000000000000000002400000000promu-0.16.0/.github/dependabot.ymlversion: 2 updates: - package-ecosystem: "gomod" directory: "/" schedule: interval: "monthly" 07070100000004000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001F00000000promu-0.16.0/.github/workflows07070100000005000081A4000000000000000000000001660D2C8E000004CB000000000000000000000000000000000000003100000000promu-0.16.0/.github/workflows/golangci-lint.yml--- # This action is synced from https://github.com/prometheus/prometheus name: golangci-lint on: push: paths: - "go.sum" - "go.mod" - "**.go" - "scripts/errcheck_excludes.txt" - ".github/workflows/golangci-lint.yml" - ".golangci.yml" pull_request: permissions: # added using https://github.com/step-security/secure-repo contents: read jobs: golangci: permissions: contents: read # for actions/checkout to fetch code pull-requests: read # for golangci/golangci-lint-action to fetch pull requests name: lint runs-on: ubuntu-latest steps: - name: Checkout repository uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 - name: install Go uses: actions/setup-go@0c52d547c9bc32b1aa3301fd7a9cb496313a4491 # v5.0.0 with: go-version: 1.22.x - name: Install snmp_exporter/generator dependencies run: sudo apt-get update && sudo apt-get -y install libsnmp-dev if: github.repository == 'prometheus/snmp_exporter' - name: Lint uses: golangci/golangci-lint-action@3cfe3a4abbb849e10058ce4af15d205b6da42804 # v4.0.0 with: version: v1.56.2 07070100000006000081A4000000000000000000000001660D2C8E0000002B000000000000000000000000000000000000001800000000promu-0.16.0/.gitignore/promu .build .tarballs testoutput /vendor 07070100000007000081A4000000000000000000000001660D2C8E000001ED000000000000000000000000000000000000001B00000000promu-0.16.0/.golangci.ymlissues: max-issues-per-linter: 0 max-same-issues: 0 linters: disable: - errcheck - unused enable: - errorlint - gocritic - gofumpt - goimports - govet - revive - staticcheck linters-settings: goimports: local-prefixes: github.com/prometheus/promu revive: rules: # https://github.com/mgechev/revive/blob/master/RULES_DESCRIPTIONS.md#unused-parameter - name: unused-parameter severity: warning disabled: true 07070100000008000081A4000000000000000000000001660D2C8E000002BC000000000000000000000000000000000000001800000000promu-0.16.0/.promu.ymlgo: # Whenever the Go version is updated here, # .circle/config.yml should also be updated. version: 1.22 repository: path: github.com/prometheus/promu build: tags: all: [netgo, static_build] windows: [static_build] ldflags: | -s -X github.com/prometheus/common/version.Version={{.Version}} -X github.com/prometheus/common/version.Revision={{.Revision}} -X github.com/prometheus/common/version.Branch={{.Branch}} -X github.com/prometheus/common/version.BuildUser={{user}}@{{host}} -X github.com/prometheus/common/version.BuildDate={{date "20060102-15:04:05"}} tarball: files: - LICENSE - NOTICE 07070100000009000081A4000000000000000000000001660D2C8E00000BCF000000000000000000000000000000000000001A00000000promu-0.16.0/CHANGELOG.md## 0.16.0 / 2024-04-02 * [FEATURE] Do not discover user/host for reproducible builds #267 * [BUGFIX] Fix example/prometheus build error #274 ## 0.15.0 / 2023-07-04 * [CHANGE] Use unmarshalstrict #240 * [CHANGE] Add linux/riscv64 to default platforms #254 ## 0.14.0 / 2022-12-09 * [FEATURE] Add the ability to override tags per GOOS ## 0.13.0 / 2021-11-06 * [ENHANCEMENT] Add windows/arm64 platform #225 ## 0.12.0 / 2021-04-12 * [CHANGE] Unified CGO crossbuild image #219 ## 0.11.1 / 2021-03-20 * [BUGFIX] Deduplicate platforms when two regexes match the same platform #214 * [BUGFIX] Regexes are evaluated against all archs, and don't stop at the first match #214 ## 0.11.0 / 2021-03-20 * [FEATURE] Add the ability to run parallel build threads independently #212 ## 0.10.0 / 2021-03-17 * [FEATURE] Add parallel crossbuilds #208 ## 0.9.0 / 2021-03-16 Note: promu crossbuild --platform flag is now a regexp. To use multiple options, the flag can be repeated. * [CHANGE] Use regexp for crossbuild platforms #204 ## 0.8.1 / 2021-03-12 This release is cut to publish `darwin/arm64` and `illumos/amd64` binaries of promu. * [ENHANCEMENT] Promu is now built from its default branch instead of a released binary #205 ## 0.8.0 / 2021-03-11 Note: The default build now requires Go 1.16. * [FEATURE] Add `darwin/arm64` and `illumos/amd64` to the default build #201 ## 0.7.0 / 2020-11-03 * [FEATURE] Produce ZIP archives for Windows releases #195 ## 0.6.1 / 2020-09-06 * [BUGFIX] Fix cgo builds on illumos by avoiding static linking #192 ## 0.6.0 / 2020-09-02 * [CHANGE] Remove default build of darwin/386 #187 * [FEATURE] Add 'check changelog' command. #168 * [FEATURE] Support remotes other than "origin". #174 * [ENHANCEMNT] Improved error handling when parsing CHANGELOG. #161 * [ENANCEMENT] Support arm64 on BSDs. #186 ## 0.5.0 / 2019-06-21 * [CHANGE] Remove --broken from git describe. #145 * [FEATURE] Add support for aix/ppc64. #151 * [ENHANCEMENT] cmd/release: add --timeout option. #142 * [ENHANCEMENT] cmd/release: create release in GitHub if none exists. #148 ## 0.4.0 / 2019-05-03 * [FEATURE] Fallback to `git describe` output if no VERSION. #130 * [BUGFIX] cmd/tarball: restore --prefix flag. #133 * [BUGFIX] cmd/release: don't leak credentials in case of error. #136 ## 0.3.0 / 2019-02-18 * [FEATURE] Make extldflags extensible by configuration. #125 * [ENHANCEMENT] Avoid bind-mounting to allow building with a remote docker engine #95 ## 0.2.0 / 2018-11-07 * [FEATURE] Adding changes to support s390x * [FEATURE] Add option to disable static linking * [FEATURE] Add support for 32bit MIPS. * [FEATURE] Added check_licenses Command to Promu * [ENHANCEMENT] Allow to customize nested options via env variables * [ENHANCEMENT] Bump Go version to 1.11 * [ENHANCEMENT] Add warning if promu info is unable to determine repo info * [BUGFIX] Fix build on SmartOS by not setting gcc's -static flag * [BUGFIX] Fix git repository url parsing ## 0.1.0 / 2017-09-22 Initial release 0707010000000A000081A4000000000000000000000001660D2C8E00000098000000000000000000000000000000000000002000000000promu-0.16.0/CODE_OF_CONDUCT.md# Prometheus Community Code of Conduct Prometheus follows the [CNCF Code of Conduct](https://github.com/cncf/foundation/blob/main/code-of-conduct.md). 0707010000000B000081A4000000000000000000000001660D2C8E00000230000000000000000000000000000000000000001D00000000promu-0.16.0/CONTRIBUTING.md# Contributing Prometheus uses GitHub to manage reviews of pull requests. * If you have a trivial fix or improvement, go ahead and create a pull request, addressing (with `@...`) the maintainer of this repository (see [MAINTAINERS.md](MAINTAINERS.md)) in the description of the pull request. * If you plan to do something more involved, first discuss your ideas on our [mailing list](https://groups.google.com/forum/?fromgroups#!forum/prometheus-developers). This will avoid unnecessary work and surely give you and us a good deal of inspiration. 0707010000000C000081A4000000000000000000000001660D2C8E00002C5D000000000000000000000000000000000000001500000000promu-0.16.0/LICENSE Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright [yyyy] [name of copyright owner] Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. 0707010000000D000081A4000000000000000000000001660D2C8E0000002C000000000000000000000000000000000000001C00000000promu-0.16.0/MAINTAINERS.md* Steve Durrheimer <s.durrheimer@gmail.com> 0707010000000E000081A4000000000000000000000001660D2C8E0000034E000000000000000000000000000000000000001600000000promu-0.16.0/Makefile# Copyright 2018 The Prometheus Authors # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. include Makefile.common .PHONY: build build: @echo ">> installing promu" GO111MODULE=$(GO111MODULE) GOOS= GOARCH= $(GO) install github.com/prometheus/promu @echo ">> rebuilding binaries using promu" GO111MODULE=$(GO111MODULE) $(PROMU) build --prefix $(PREFIX) 0707010000000F000081A4000000000000000000000001660D2C8E000023CD000000000000000000000000000000000000001D00000000promu-0.16.0/Makefile.common# Copyright 2018 The Prometheus Authors # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # A common Makefile that includes rules to be reused in different prometheus projects. # !!! Open PRs only against the prometheus/prometheus/Makefile.common repository! # Example usage : # Create the main Makefile in the root project directory. # include Makefile.common # customTarget: # @echo ">> Running customTarget" # # Ensure GOBIN is not set during build so that promu is installed to the correct path unexport GOBIN GO ?= go GOFMT ?= $(GO)fmt FIRST_GOPATH := $(firstword $(subst :, ,$(shell $(GO) env GOPATH))) GOOPTS ?= GOHOSTOS ?= $(shell $(GO) env GOHOSTOS) GOHOSTARCH ?= $(shell $(GO) env GOHOSTARCH) GO_VERSION ?= $(shell $(GO) version) GO_VERSION_NUMBER ?= $(word 3, $(GO_VERSION)) PRE_GO_111 ?= $(shell echo $(GO_VERSION_NUMBER) | grep -E 'go1\.(10|[0-9])\.') PROMU := $(FIRST_GOPATH)/bin/promu pkgs = ./... ifeq (arm, $(GOHOSTARCH)) GOHOSTARM ?= $(shell GOARM= $(GO) env GOARM) GO_BUILD_PLATFORM ?= $(GOHOSTOS)-$(GOHOSTARCH)v$(GOHOSTARM) else GO_BUILD_PLATFORM ?= $(GOHOSTOS)-$(GOHOSTARCH) endif GOTEST := $(GO) test GOTEST_DIR := ifneq ($(CIRCLE_JOB),) ifneq ($(shell command -v gotestsum 2> /dev/null),) GOTEST_DIR := test-results GOTEST := gotestsum --junitfile $(GOTEST_DIR)/unit-tests.xml -- endif endif PROMU_VERSION ?= 0.15.0 PROMU_URL := https://github.com/prometheus/promu/releases/download/v$(PROMU_VERSION)/promu-$(PROMU_VERSION).$(GO_BUILD_PLATFORM).tar.gz SKIP_GOLANGCI_LINT := GOLANGCI_LINT := GOLANGCI_LINT_OPTS ?= GOLANGCI_LINT_VERSION ?= v1.56.2 # golangci-lint only supports linux, darwin and windows platforms on i386/amd64/arm64. # windows isn't included here because of the path separator being different. ifeq ($(GOHOSTOS),$(filter $(GOHOSTOS),linux darwin)) ifeq ($(GOHOSTARCH),$(filter $(GOHOSTARCH),amd64 i386 arm64)) # If we're in CI and there is an Actions file, that means the linter # is being run in Actions, so we don't need to run it here. ifneq (,$(SKIP_GOLANGCI_LINT)) GOLANGCI_LINT := else ifeq (,$(CIRCLE_JOB)) GOLANGCI_LINT := $(FIRST_GOPATH)/bin/golangci-lint else ifeq (,$(wildcard .github/workflows/golangci-lint.yml)) GOLANGCI_LINT := $(FIRST_GOPATH)/bin/golangci-lint endif endif endif PREFIX ?= $(shell pwd) BIN_DIR ?= $(shell pwd) DOCKER_IMAGE_TAG ?= $(subst /,-,$(shell git rev-parse --abbrev-ref HEAD)) DOCKERFILE_PATH ?= ./Dockerfile DOCKERBUILD_CONTEXT ?= ./ DOCKER_REPO ?= prom DOCKER_ARCHS ?= amd64 BUILD_DOCKER_ARCHS = $(addprefix common-docker-,$(DOCKER_ARCHS)) PUBLISH_DOCKER_ARCHS = $(addprefix common-docker-publish-,$(DOCKER_ARCHS)) TAG_DOCKER_ARCHS = $(addprefix common-docker-tag-latest-,$(DOCKER_ARCHS)) SANITIZED_DOCKER_IMAGE_TAG := $(subst +,-,$(DOCKER_IMAGE_TAG)) ifeq ($(GOHOSTARCH),amd64) ifeq ($(GOHOSTOS),$(filter $(GOHOSTOS),linux freebsd darwin windows)) # Only supported on amd64 test-flags := -race endif endif # This rule is used to forward a target like "build" to "common-build". This # allows a new "build" target to be defined in a Makefile which includes this # one and override "common-build" without override warnings. %: common-% ; .PHONY: common-all common-all: precheck style check_license lint yamllint unused build test .PHONY: common-style common-style: @echo ">> checking code style" @fmtRes=$$($(GOFMT) -d $$(find . -path ./vendor -prune -o -name '*.go' -print)); \ if [ -n "$${fmtRes}" ]; then \ echo "gofmt checking failed!"; echo "$${fmtRes}"; echo; \ echo "Please ensure you are using $$($(GO) version) for formatting code."; \ exit 1; \ fi .PHONY: common-check_license common-check_license: @echo ">> checking license header" @licRes=$$(for file in $$(find . -type f -iname '*.go' ! -path './vendor/*') ; do \ awk 'NR<=3' $$file | grep -Eq "(Copyright|generated|GENERATED)" || echo $$file; \ done); \ if [ -n "$${licRes}" ]; then \ echo "license header checking failed:"; echo "$${licRes}"; \ exit 1; \ fi .PHONY: common-deps common-deps: @echo ">> getting dependencies" $(GO) mod download .PHONY: update-go-deps update-go-deps: @echo ">> updating Go dependencies" @for m in $$($(GO) list -mod=readonly -m -f '{{ if and (not .Indirect) (not .Main)}}{{.Path}}{{end}}' all); do \ $(GO) get -d $$m; \ done $(GO) mod tidy .PHONY: common-test-short common-test-short: $(GOTEST_DIR) @echo ">> running short tests" $(GOTEST) -short $(GOOPTS) $(pkgs) .PHONY: common-test common-test: $(GOTEST_DIR) @echo ">> running all tests" $(GOTEST) $(test-flags) $(GOOPTS) $(pkgs) $(GOTEST_DIR): @mkdir -p $@ .PHONY: common-format common-format: @echo ">> formatting code" $(GO) fmt $(pkgs) .PHONY: common-vet common-vet: @echo ">> vetting code" $(GO) vet $(GOOPTS) $(pkgs) .PHONY: common-lint common-lint: $(GOLANGCI_LINT) ifdef GOLANGCI_LINT @echo ">> running golangci-lint" $(GOLANGCI_LINT) run $(GOLANGCI_LINT_OPTS) $(pkgs) endif .PHONY: common-lint-fix common-lint-fix: $(GOLANGCI_LINT) ifdef GOLANGCI_LINT @echo ">> running golangci-lint fix" $(GOLANGCI_LINT) run --fix $(GOLANGCI_LINT_OPTS) $(pkgs) endif .PHONY: common-yamllint common-yamllint: @echo ">> running yamllint on all YAML files in the repository" ifeq (, $(shell command -v yamllint 2> /dev/null)) @echo "yamllint not installed so skipping" else yamllint . endif # For backward-compatibility. .PHONY: common-staticcheck common-staticcheck: lint .PHONY: common-unused common-unused: @echo ">> running check for unused/missing packages in go.mod" $(GO) mod tidy @git diff --exit-code -- go.sum go.mod .PHONY: common-build common-build: promu @echo ">> building binaries" $(PROMU) build --prefix $(PREFIX) $(PROMU_BINARIES) .PHONY: common-tarball common-tarball: promu @echo ">> building release tarball" $(PROMU) tarball --prefix $(PREFIX) $(BIN_DIR) .PHONY: common-docker-repo-name common-docker-repo-name: @echo "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)" .PHONY: common-docker $(BUILD_DOCKER_ARCHS) common-docker: $(BUILD_DOCKER_ARCHS) $(BUILD_DOCKER_ARCHS): common-docker-%: docker build -t "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:$(SANITIZED_DOCKER_IMAGE_TAG)" \ -f $(DOCKERFILE_PATH) \ --build-arg ARCH="$*" \ --build-arg OS="linux" \ $(DOCKERBUILD_CONTEXT) .PHONY: common-docker-publish $(PUBLISH_DOCKER_ARCHS) common-docker-publish: $(PUBLISH_DOCKER_ARCHS) $(PUBLISH_DOCKER_ARCHS): common-docker-publish-%: docker push "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:$(SANITIZED_DOCKER_IMAGE_TAG)" DOCKER_MAJOR_VERSION_TAG = $(firstword $(subst ., ,$(shell cat VERSION))) .PHONY: common-docker-tag-latest $(TAG_DOCKER_ARCHS) common-docker-tag-latest: $(TAG_DOCKER_ARCHS) $(TAG_DOCKER_ARCHS): common-docker-tag-latest-%: docker tag "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:$(SANITIZED_DOCKER_IMAGE_TAG)" "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:latest" docker tag "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:$(SANITIZED_DOCKER_IMAGE_TAG)" "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$*:v$(DOCKER_MAJOR_VERSION_TAG)" .PHONY: common-docker-manifest common-docker-manifest: DOCKER_CLI_EXPERIMENTAL=enabled docker manifest create -a "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME):$(SANITIZED_DOCKER_IMAGE_TAG)" $(foreach ARCH,$(DOCKER_ARCHS),$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME)-linux-$(ARCH):$(SANITIZED_DOCKER_IMAGE_TAG)) DOCKER_CLI_EXPERIMENTAL=enabled docker manifest push "$(DOCKER_REPO)/$(DOCKER_IMAGE_NAME):$(SANITIZED_DOCKER_IMAGE_TAG)" .PHONY: promu promu: $(PROMU) $(PROMU): $(eval PROMU_TMP := $(shell mktemp -d)) curl -s -L $(PROMU_URL) | tar -xvzf - -C $(PROMU_TMP) mkdir -p $(FIRST_GOPATH)/bin cp $(PROMU_TMP)/promu-$(PROMU_VERSION).$(GO_BUILD_PLATFORM)/promu $(FIRST_GOPATH)/bin/promu rm -r $(PROMU_TMP) .PHONY: proto proto: @echo ">> generating code from proto files" @./scripts/genproto.sh ifdef GOLANGCI_LINT $(GOLANGCI_LINT): mkdir -p $(FIRST_GOPATH)/bin curl -sfL https://raw.githubusercontent.com/golangci/golangci-lint/$(GOLANGCI_LINT_VERSION)/install.sh \ | sed -e '/install -d/d' \ | sh -s -- -b $(FIRST_GOPATH)/bin $(GOLANGCI_LINT_VERSION) endif .PHONY: precheck precheck:: define PRECHECK_COMMAND_template = precheck:: $(1)_precheck PRECHECK_COMMAND_$(1) ?= $(1) $$(strip $$(PRECHECK_OPTIONS_$(1))) .PHONY: $(1)_precheck $(1)_precheck: @if ! $$(PRECHECK_COMMAND_$(1)) 1>/dev/null 2>&1; then \ echo "Execution of '$$(PRECHECK_COMMAND_$(1))' command failed. Is $(1) installed?"; \ exit 1; \ fi endef 07070100000010000081A4000000000000000000000001660D2C8E000000E9000000000000000000000000000000000000001400000000promu-0.16.0/NOTICEPrometheus Utility Tool Copyright 2013-2016 The Prometheus Authors The following components were used in this product as inspiration: Glu https://github.com/gliderlabs/glu Copyright 2015 Glider Labs Licensed under the MIT License 07070100000011000081A4000000000000000000000001660D2C8E00000785000000000000000000000000000000000000001700000000promu-0.16.0/README.md# Prometheus Utility Tool [![CircleCI](https://circleci.com/gh/prometheus/promu/tree/master.svg?style=shield)][circleci] ## Usage ```help usage: promu [<flags>] <command> [<args> ...] promu is the utility tool for building and releasing Prometheus projects Flags: -h, --help Show context-sensitive help (also try --help-long and --help-man). -c, --config=".promu.yml" Path to config file -v, --verbose Verbose output Commands: help [<command>...] Show help. build [<flags>] [<binary-names>...] Build a Go project check licenses [<flags>] [<location>...] Inspect source files for each file in a given directory check changelog [<flags>] Check that CHANGELOG.md follows the guidelines checksum [<location>...] Calculate the SHA256 checksum for each file in the given location crossbuild [<flags>] [<tarballs>] Crossbuild a Go project using Golang builder Docker images info Print info about current project and exit release [<flags>] [<location>...] Upload all release files to the Github release tarball [<flags>] [<location>...] Create a tarball from the built Go project version [<flags>] Print the version and exit ``` ## `.promu.yml` config file See documentation example [here](doc/examples/prometheus/.promu.yml) ## Compatibility * Go 1.14+ ## More information * This tool is part of our reflection about [Prometheus component Builds](https://docs.google.com/document/d/1Ql-f_aThl-2eB5v3QdKV_zgBdetLLbdxxChpy-TnWSE) * All of the core developers are accessible via the [Prometheus Developers Mailinglist](https://groups.google.com/forum/?fromgroups#!forum/prometheus-developers) and the `#prometheus` channel on `irc.freenode.net`. ## Contributing Refer to [CONTRIBUTING.md](CONTRIBUTING.md) ## License Apache License 2.0, see [LICENSE](LICENSE). [circleci]: https://circleci.com/gh/prometheus/promu 07070100000012000081A4000000000000000000000001660D2C8E000000AC000000000000000000000000000000000000001900000000promu-0.16.0/SECURITY.md# Reporting a security issue The Prometheus security policy, including how to report vulnerabilities, can be found here: <https://prometheus.io/docs/operating/security/> 07070100000013000081A4000000000000000000000001660D2C8E00000007000000000000000000000000000000000000001500000000promu-0.16.0/VERSION0.16.0 07070100000014000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001100000000promu-0.16.0/cmd07070100000015000081A4000000000000000000000001660D2C8E0000192F000000000000000000000000000000000000001A00000000promu-0.16.0/cmd/build.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "bytes" "fmt" "log" "os" "path" "strconv" "strings" "text/template" "time" kingpin "github.com/alecthomas/kingpin/v2" "github.com/prometheus/promu/pkg/repository" "github.com/prometheus/promu/util/sh" ) const ( sourceDateEpoch = "SOURCE_DATE_EPOCH" ) var ( buildcmd = app.Command("build", "Build a Go project") buildCgoFlagSet bool buildCgoFlag = buildcmd.Flag("cgo", "Enable CGO"). PreAction(func(c *kingpin.ParseContext) error { buildCgoFlagSet = true return nil }).Bool() prefixFlagSet bool prefixFlag = buildcmd.Flag("prefix", "Specific dir to store binaries (default is .)"). PreAction(func(c *kingpin.ParseContext) error { prefixFlagSet = true return nil }).String() binariesArg = buildcmd.Arg("binary-names", "Comma separated list of binaries to build").Default("all").Strings() ) // Check if binary names passed to build command are in the config. // Returns an array of Binary to build, or error. func validateBinaryNames(binaryNames []string, cfgBinaries []Binary) ([]Binary, error) { var binaries []Binary OUTER: for _, binaryName := range binaryNames { for _, binary := range cfgBinaries { if binaryName == binary.Name { binaries = append(binaries, binary) continue OUTER } } return nil, fmt.Errorf("binary %s not found in config", binaryName) } return binaries, nil } func buildBinary(ext string, prefix string, ldflags string, tags []string, binary Binary) { info("Building binary: " + binary.Name) binaryName := fmt.Sprintf("%s%s", binary.Name, ext) fmt.Printf(" > %s\n", binaryName) repoPath := config.Repository.Path flags := config.Build.Flags params := []string{ "build", "-o", path.Join(prefix, binaryName), "-ldflags", ldflags, } params = append(params, sh.SplitParameters(flags)...) if len(tags) > 0 { params = append(params, "-tags", strings.Join(tags, ",")) } params = append(params, path.Join(repoPath, binary.Path)) info("Building binary: " + "go " + strings.Join(params, " ")) if err := sh.RunCommand("go", params...); err != nil { fatal(fmt.Errorf("command failed: %s: %w", strings.Join(params, " "), err)) } } func buildAll(ext string, prefix string, ldflags string, tags []string, binaries []Binary) { for _, binary := range binaries { buildBinary(ext, prefix, ldflags, tags, binary) } } func runBuild(binariesString string) { // Check required configuration if len(strings.TrimSpace(config.Repository.Path)) == 0 { log.Fatalf("missing required '%s' configuration", "repository.path") } if buildCgoFlagSet { config.Go.CGo = *buildCgoFlag } if prefixFlagSet { config.Build.Prefix = *prefixFlag } var ( cgo = config.Go.CGo prefix = config.Build.Prefix ext string binaries = config.Build.Binaries ldflags string ) if goos == "windows" { ext = ".exe" } ldflags = getLdflags(projInfo) os.Setenv("CGO_ENABLED", "0") if cgo { os.Setenv("CGO_ENABLED", "1") } defer os.Unsetenv("CGO_ENABLED") if binariesString == "all" { buildAll(ext, prefix, ldflags, getTags(config.Build.Tags), binaries) return } binariesArray := strings.Split(binariesString, ",") binariesToBuild, err := validateBinaryNames(binariesArray, binaries) if err != nil { fatal(fmt.Errorf("validation of given binary names for build command failed: %w", err)) } for _, binary := range binariesToBuild { buildBinary(ext, prefix, ldflags, getTags(config.Build.Tags), binary) } } func getLdflags(info repository.Info) string { var ldflags []string if len(strings.TrimSpace(config.Build.LDFlags)) > 0 { buildDate := getBuildDate() var ( tmplOutput = new(bytes.Buffer) fnMap = template.FuncMap{ "date": buildDate.UTC().Format, "host": HostFunc, "repoPath": RepoPathFunc, "user": UserFunc, } ldflagsTmpl = config.Build.LDFlags ) tmpl, err := template.New("ldflags").Funcs(fnMap).Parse(ldflagsTmpl) if err != nil { fatal(fmt.Errorf("Failed to parse ldflags text/template: %w", err)) } if err := tmpl.Execute(tmplOutput, info); err != nil { fatal(fmt.Errorf("Failed to execute ldflags text/template: %w", err)) } ldflags = append(ldflags, strings.Split(tmplOutput.String(), "\n")...) } else { ldflags = append(ldflags, fmt.Sprintf("-X main.Version=%s", info.Version)) } extLDFlags := config.Build.ExtLDFlags if config.Build.Static && goos != "darwin" && goos != "solaris" && goos != "illumos" && !stringInSlice("-static", extLDFlags) { extLDFlags = append(extLDFlags, "-static") } if len(extLDFlags) > 0 { ldflags = append(ldflags, fmt.Sprintf("-extldflags '%s'", strings.Join(extLDFlags, " "))) } return strings.Join(ldflags, " ") } func getBuildDate() time.Time { var buildDate time.Time sourceDate := os.Getenv(sourceDateEpoch) if sourceDate == "" { buildDate = time.Now() } else { unixBuildDate, err := strconv.ParseInt(sourceDate, 10, 64) if err != nil { fatal(fmt.Errorf("Failed to parse %s: %w", sourceDateEpoch, err)) } else { buildDate = time.Unix(unixBuildDate, 0) } } return buildDate } func HostFunc() string { if isReproducibleBuild() { return "reproducible" } hostname, err := os.Hostname() if err != nil { return "unknown-host" } return hostname } // UserFunc returns the current username. func UserFunc() (interface{}, error) { if isReproducibleBuild() { return "reproducible", nil } // os/user.Current() doesn't always work without CGO return shellOutput("whoami"), nil } func isReproducibleBuild() bool { return os.Getenv(sourceDateEpoch) != "" } // RepoPathFunc returns the repository path. func RepoPathFunc() interface{} { return config.Repository.Path } func getTags(allTags map[string][]string) []string { if tags, ok := allTags[envOr("GOOS", goos)]; ok { return tags } return allTags["all"] } 07070100000016000081A4000000000000000000000001660D2C8E00001002000000000000000000000000000000000000001A00000000promu-0.16.0/cmd/check.go// Copyright © 2017 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "bufio" "fmt" "os" "path/filepath" "strings" "github.com/prometheus/promu/pkg/changelog" ) var ( // validHeaderStrings is a slice of strings that must exist in a header. validHeaderStrings = []string{"copyright", "generated"} checkcmd = app.Command("check", "Check the resources for validity") checkLicensescmd = checkcmd.Command("licenses", "Inspect source files for each file in a given directory") sourceExtensions = checkLicensescmd.Flag("extensions", "Comma separated list of valid source code extensions (default is .go)"). Default(".go").Strings() headerLength = checkLicensescmd.Flag("length", "The number of lines to read from the head of the file"). Short('n').Default("10").Int() checkLicLocation = checkLicensescmd.Arg("location", "Directory path to check licenses"). Default(".").Strings() checkChangelogcmd = checkcmd.Command("changelog", "Check that CHANGELOG.md follows the guidelines") checkChangelogPath = checkChangelogcmd.Flag("location", "Path to CHANGELOG.md"). Default("CHANGELOG.md").String() checkChangelogVersion = checkChangelogcmd.Flag("version", "Version to check (defaults to the current version)"). Default("").String() ) func runCheckLicenses(path string, n int, extensions []string) { path = fmt.Sprintf("%s%c", filepath.Clean(path), filepath.Separator) filesMissingHeaders, err := checkLicenses(path, n, extensions) if err != nil { fatal(fmt.Errorf("Failed to check files for license header: %w", err)) } for _, file := range filesMissingHeaders { fmt.Println(file) } } func checkLicenses(path string, n int, extensions []string) ([]string, error) { var missingHeaders []string walkFunc := func(filepath string, f os.FileInfo, err error) error { if err != nil { return err } if f.IsDir() { return nil } if strings.HasPrefix(filepath, "vendor/") { return nil } if !suffixInSlice(f.Name(), extensions) { return nil } file, err := os.Open(filepath) if err != nil { return err } defer file.Close() pass := false scanner := bufio.NewScanner(file) for i := 0; i < n; i++ { scanner.Scan() if err = scanner.Err(); err != nil { return err } if stringContainedInSlice(strings.ToLower(scanner.Text()), validHeaderStrings) { pass = true } } if !pass { missingHeaders = append(missingHeaders, filepath) } return nil } err := filepath.Walk(path, walkFunc) if err != nil { return nil, err } return missingHeaders, nil } func stringContainedInSlice(needle string, haystack []string) bool { exists := false for _, h := range haystack { if strings.Contains(needle, h) { exists = true break } } return exists } func suffixInSlice(needle string, haystack []string) bool { exists := false for _, h := range haystack { if strings.HasSuffix(needle, h) { exists = true break } } return exists } func runCheckChangelog(path string, version string) error { if version == "" { _, err := projInfo.ToSemver() if err != nil { return fmt.Errorf("invalid semver version: %w", err) } version = projInfo.Version } f, err := os.Open(path) if err != nil { return err } defer f.Close() entry, err := changelog.ReadEntry(f, version) if err != nil { return fmt.Errorf("%s: %w", path, err) } // Check that the changes are ordered correctly. err = entry.Changes.Sorted() if err != nil { return fmt.Errorf("invalid changelog entry: %w", err) } return nil } 07070100000017000081A4000000000000000000000001660D2C8E0000098C000000000000000000000000000000000000001D00000000promu-0.16.0/cmd/checksum.go// Copyright © 2017 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "crypto/sha256" "fmt" "io" "os" "path/filepath" "strings" ) const ( checksumsFilename = "sha256sums.txt" ) var ( checksumcmd = app.Command("checksum", "Calculate the SHA256 checksum for each file in the given location") checksumLocation = checksumcmd.Arg("location", "Location to checksum").Default(".").Strings() ) func runChecksum(path string) { checksums, err := calculateSHA256s(path) if err != nil { fatal(fmt.Errorf("Failed to calculate checksums: %w", err)) } file, err := os.Create(filepath.Join(path, checksumsFilename)) if err != nil { fatal(fmt.Errorf("Failed to create checksums file: %w", err)) } defer file.Close() for _, c := range checksums { if _, err := fmt.Fprintf(file, "%x %s\n", c.checksum, c.filename); err != nil { fatal(fmt.Errorf("Failed to write to checksums file: %w", err)) } } } type checksumSHA256 struct { filename string checksum []byte } // calculateSHA256s calculates the sha256 checksum for each file in the given // path and returns a checksumSHA256 type in the order returned of // filepath.Walk. func calculateSHA256s(path string) ([]checksumSHA256, error) { var checksums []checksumSHA256 path = fmt.Sprintf("%s%c", filepath.Clean(path), filepath.Separator) calculateSHA256 := func(filepath string, f os.FileInfo, err error) error { if err != nil { return err } if f.IsDir() { return nil } file, err := os.Open(filepath) if err != nil { return err } defer file.Close() hash := sha256.New() if _, err = io.Copy(hash, file); err != nil { return err } checksums = append(checksums, checksumSHA256{ filename: strings.TrimPrefix(filepath, path), checksum: hash.Sum(nil), }) return nil } if err := filepath.Walk(path, calculateSHA256); err != nil { return nil, err } return checksums, nil } 07070100000018000081A4000000000000000000000001660D2C8E00000527000000000000000000000000000000000000002200000000promu-0.16.0/cmd/checksum_test.go// Copyright © 2017 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "crypto/sha256" "os" "path/filepath" "reflect" "testing" ) func TestCalculateSHA256s(t *testing.T) { dir, err := os.MkdirTemp("", "promu") if err != nil { t.Fatal(err) } defer os.RemoveAll(dir) var ( filename = "testfile" location = filepath.Join(dir, filename) content = []byte("temporary file's content") checksum = sha256.Sum256(content) ) if err = os.WriteFile(location, content, 0o666); err != nil { t.Fatal(err) } got, err := calculateSHA256s(dir) if err != nil { t.Fatal(err) } want := []checksumSHA256{ { filename: filename, checksum: checksum[:], }, } if !reflect.DeepEqual(want, got) { t.Errorf("want checksums %+v, got %+v", want, got) } } 07070100000019000081A4000000000000000000000001660D2C8E00001B89000000000000000000000000000000000000001F00000000promu-0.16.0/cmd/crossbuild.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "fmt" "log" "os" "regexp" "sort" "strconv" "strings" "sync" "time" kingpin "github.com/alecthomas/kingpin/v2" "go.uber.org/atomic" "github.com/prometheus/promu/util/sh" ) var ( dockerBuilderImageName = "quay.io/prometheus/golang-builder" defaultPlatforms = []string{ "aix/ppc64", "darwin/amd64", "darwin/arm64", "dragonfly/amd64", "freebsd/386", "freebsd/amd64", "freebsd/arm64", "freebsd/armv6", "freebsd/armv7", "illumos/amd64", "linux/386", "linux/amd64", "linux/arm64", "linux/armv5", "linux/armv6", "linux/armv7", "linux/mips", "linux/mips64", "linux/mips64le", "linux/mipsle", "linux/ppc64", "linux/ppc64le", "linux/riscv64", "linux/s390x", "netbsd/386", "netbsd/amd64", "netbsd/arm64", "netbsd/armv6", "netbsd/armv7", "openbsd/386", "openbsd/amd64", "openbsd/arm64", "openbsd/armv7", "windows/386", "windows/amd64", "windows/arm64", } ) var ( crossbuildcmd = app.Command("crossbuild", "Crossbuild a Go project using Golang builder Docker images") crossBuildCgoFlagSet bool crossBuildCgoFlag = crossbuildcmd.Flag("cgo", "Enable CGO using several docker images with different crossbuild toolchains."). PreAction(func(c *kingpin.ParseContext) error { crossBuildCgoFlagSet = true return nil }).Default("false").Bool() parallelFlag = crossbuildcmd.Flag("parallelism", "How many builds to run in parallel").Default("1").Int() parallelThreadFlag = crossbuildcmd.Flag("parallelism-thread", "Index of the parallel build").Default("-1").Int() goFlagSet bool goFlag = crossbuildcmd.Flag("go", "Golang builder version to use (e.g. 1.11)"). PreAction(func(c *kingpin.ParseContext) error { goFlagSet = true return nil }).String() platformsFlagSet bool platformsFlag = crossbuildcmd.Flag("platforms", "Regexp match platforms to build, may be used multiple times.").Short('p'). PreAction(func(c *kingpin.ParseContext) error { platformsFlagSet = true return nil }).Strings() // kingpin doesn't currently support using the crossbuild command and the // crossbuild tarball subcommand at the same time, so we treat the // tarball subcommand as an optional arg tarballsSubcommand = crossbuildcmd.Arg("tarballs", "Optionally pass the string \"tarballs\" from cross-built binaries").String() ) func runCrossbuild() { // Check required configuration if len(strings.TrimSpace(config.Repository.Path)) == 0 { log.Fatalf("missing required '%s' configuration", "repository.path") } if *tarballsSubcommand == "tarballs" { runCrossbuildTarballs() return } if crossBuildCgoFlagSet { config.Go.CGo = *crossBuildCgoFlag } if goFlagSet { config.Go.Version = *goFlag } if platformsFlagSet { config.Crossbuild.Platforms = *platformsFlag } var ( allPlatforms []string unknownPlatforms []string cgo = config.Go.CGo goVersion = config.Go.Version repoPath = config.Repository.Path platforms = config.Crossbuild.Platforms dockerBaseBuilderImage = fmt.Sprintf("%s:%s-base", dockerBuilderImageName, goVersion) dockerMainBuilderImage = fmt.Sprintf("%s:%s-main", dockerBuilderImageName, goVersion) ) var filteredPlatforms []string for _, platform := range platforms { p := regexp.MustCompile(platform) if filteredPlatforms = inSliceRE(p, defaultPlatforms); len(filteredPlatforms) > 0 { allPlatforms = append(allPlatforms, filteredPlatforms...) } else { unknownPlatforms = append(unknownPlatforms, platform) } } // Remove duplicates, e.g. if linux/arm and linux/arm64 is specified, there // would be linux/arm64 twice in the platforms without this. allPlatforms = removeDuplicates(allPlatforms) if len(unknownPlatforms) > 0 { warn(fmt.Errorf("unknown/unhandled platforms: %s", unknownPlatforms)) } if !cgo { // In non-CGO, use the `base` image without any crossbuild toolchain. pg := &platformGroup{"base", dockerBaseBuilderImage, allPlatforms} if err := pg.Build(repoPath); err != nil { fatal(fmt.Errorf("The %s builder docker image exited unexpectedly: %w", pg.Name, err)) } } else { // In CGO, use the `main` image with crossbuild toolchain. pg := &platformGroup{"main", dockerMainBuilderImage, allPlatforms} if err := pg.Build(repoPath); err != nil { fatal(fmt.Errorf("The %s builder docker image exited unexpectedly: %w", pg.Name, err)) } } } type platformGroup struct { Name string DockerImage string Platforms []string } func (pg platformGroup) Build(repoPath string) error { if *parallelThreadFlag != -1 { return pg.buildThread(repoPath, *parallelThreadFlag) } err := sh.RunCommand("docker", "pull", pg.DockerImage) if err != nil { return err } var wg sync.WaitGroup wg.Add(*parallelFlag) atomicErr := atomic.NewError(nil) for p := 0; p < *parallelFlag; p++ { go func(p int) { defer wg.Done() if err := pg.buildThread(repoPath, p); err != nil { atomicErr.Store(err) } }(p) } wg.Wait() return atomicErr.Load() } func (pg platformGroup) buildThread(repoPath string, p int) error { minb := p * len(pg.Platforms) / *parallelFlag maxb := (p + 1) * len(pg.Platforms) / *parallelFlag if maxb > len(pg.Platforms) { maxb = len(pg.Platforms) } platformsParam := strings.Join(pg.Platforms[minb:maxb], " ") if len(platformsParam) == 0 { return nil } fmt.Printf("> running the %s builder docker image\n", pg.Name) cwd, err := os.Getwd() if err != nil { return fmt.Errorf("couldn't get current working directory: %w", err) } ctrName := "promu-crossbuild-" + pg.Name + strconv.FormatInt(time.Now().Unix(), 10) + "-" + strconv.Itoa(p) err = sh.RunCommand("docker", "create", "-t", "--name", ctrName, pg.DockerImage, "-i", repoPath, "-p", platformsParam) if err != nil { return err } err = sh.RunCommand("docker", "cp", cwd+"/.", ctrName+":/app/") if err != nil { return err } err = sh.RunCommand("docker", "start", "-a", ctrName) if err != nil { return err } err = sh.RunCommand("docker", "cp", "-a", ctrName+":/app/.build/.", cwd+"/.build") if err != nil { return err } return sh.RunCommand("docker", "rm", "-f", ctrName) } func removeDuplicates(strings []string) []string { keys := map[string]struct{}{} list := []string{} for _, s := range strings { if _, ok := keys[s]; !ok { list = append(list, s) keys[s] = struct{}{} } } sort.Strings(list) return list } 0707010000001A000081A4000000000000000000000001660D2C8E000004F7000000000000000000000000000000000000002800000000promu-0.16.0/cmd/crossbuild_tarballs.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "fmt" "os" "path/filepath" "strings" ) func runCrossbuildTarballs() { dirs, err := os.ReadDir(".build") if err != nil { fatal(err) } fmt.Println(">> building release tarballs") for _, dir := range dirs { config.Tarball.Prefix = ".tarballs" if platform := strings.Split(dir.Name(), "-"); len(platform) == 2 { os.Setenv("GOOS", platform[0]) os.Setenv("GOARCH", platform[1]) } else { if err := fmt.Errorf("bad .build/%s directory naming, should be <GOOS>-<GOARCH>", platform); err != nil { fatal(err) } } runTarball(filepath.Join(".build", dir.Name())) } defer os.Unsetenv("GOOS") defer os.Unsetenv("GOARCH") } 0707010000001B000081A4000000000000000000000001660D2C8E00000387000000000000000000000000000000000000002400000000promu-0.16.0/cmd/crossbuild_test.go// Copyright 2021 The Prometheus Authors // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "reflect" "testing" ) func TestRemoveDuplicates(t *testing.T) { input := []string{"linux", "linux", "darwin"} output := []string{"darwin", "linux"} if deduplicate := removeDuplicates(input); !reflect.DeepEqual(deduplicate, output) { t.Fatalf("%q != %q", deduplicate, output) } } 0707010000001C000081A4000000000000000000000001660D2C8E000003C9000000000000000000000000000000000000001900000000promu-0.16.0/cmd/info.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "fmt" ) var infocmd = app.Command("info", "Print info about current project and exit") func runInfo() { fmt.Println("Name:", projInfo.Name) fmt.Println("Version:", projInfo.Version) fmt.Println("Owner:", projInfo.Owner) fmt.Println("Repo:", projInfo.Repo) fmt.Println("Branch:", projInfo.Branch) fmt.Println("Revision:", projInfo.Revision) } 0707010000001D000081A4000000000000000000000001660D2C8E00001873000000000000000000000000000000000000001A00000000promu-0.16.0/cmd/promu.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "fmt" "go/build" "os" "os/exec" "path/filepath" "regexp" "strings" kingpin "github.com/alecthomas/kingpin/v2" yaml "gopkg.in/yaml.v2" "github.com/prometheus/promu/pkg/repository" "github.com/prometheus/promu/util/sh" ) const ( // DefaultConfigFilename contains the default filename of the promu config file DefaultConfigFilename = ".promu.yml" ) // Binary represents a built binary. type Binary struct { Name string Path string } // Config contains the Promu Command Configuration type Config struct { Build struct { Binaries []Binary Flags string LDFlags string ExtLDFlags []string Tags map[string][]string Prefix string Static bool } Crossbuild struct { Platforms []string } Repository struct { Path string } Go struct { CGo bool Version string } Tarball struct { Files []string Prefix string } } // NewConfig creates a Config initialized with default values // some values may be overridden by CLI args func NewConfig() *Config { config := &Config{} config.Build.Binaries = []Binary{{Name: projInfo.Name, Path: "."}} config.Build.Prefix = "." config.Build.Static = true config.Crossbuild.Platforms = defaultPlatforms config.Tarball.Prefix = "." config.Go.Version = "1.12" config.Go.CGo = false config.Repository.Path = projInfo.Repo return config } var ( buildContext = build.Default goos = buildContext.GOOS goarch = buildContext.GOARCH configFile = app.Flag("config", "Path to config file").Short('c'). Default(DefaultConfigFilename).String() verbose = app.Flag("verbose", "Verbose output").Short('v').Bool() config *Config projInfo repository.Info // app represents the base command app = kingpin.New("promu", "promu is the utility tool for building and releasing Prometheus projects") ) // init prepares flags func init() { app.HelpFlag.Short('h') } // Execute adds all child commands to the root command sets flags appropriately. // This is called by main.main(). It only needs to happen once to the rootCmd. func Execute() { var err error projInfo, err = repository.NewInfo(warn) checkError(err, "Unable to initialize project info") command := kingpin.MustParse(app.Parse(os.Args[1:])) sh.Verbose = *verbose initConfig(*configFile) info(fmt.Sprintf("Running command: %v %v", command, os.Args[2:])) switch command { case buildcmd.FullCommand(): runBuild(optArg(*binariesArg, 0, "all")) case checkLicensescmd.FullCommand(): runCheckLicenses(optArg(*checkLicLocation, 0, "."), *headerLength, *sourceExtensions) case checkChangelogcmd.FullCommand(): if err := runCheckChangelog(*checkChangelogPath, *checkChangelogVersion); err != nil { fatal(err) } case checksumcmd.FullCommand(): runChecksum(optArg(*checksumLocation, 0, ".")) case crossbuildcmd.FullCommand(): runCrossbuild() case infocmd.FullCommand(): runInfo() case releasecmd.FullCommand(): runRelease(optArg(*releaseLocation, 0, ".")) case tarballcmd.FullCommand(): runTarball(optArg(*tarBinariesLocation, 0, ".")) case versioncmd.FullCommand(): runVersion() } } // initConfig reads the given config file into the Config object func initConfig(filename string) { info(fmt.Sprintf("Using config file: %v", filename)) configData, err := os.ReadFile(filename) checkError(err, "Unable to read config file: "+filename) config = NewConfig() err = yaml.UnmarshalStrict(configData, config) checkError(err, "Unable to parse config file: "+filename) } // info prints the given message only if running in verbose mode func info(message string) { if *verbose { fmt.Println(message) } } // warn prints a non-fatal error func warn(err error) { if *verbose { fmt.Fprintf(os.Stderr, `/!\ %+v\n`, err) } else { fmt.Fprintln(os.Stderr, `/!\`, err) } } // printErr prints a error func printErr(err error) { if *verbose { fmt.Fprintf(os.Stderr, "!! %+v\n", err) } else { fmt.Fprintln(os.Stderr, "!!", err) } } // fatal prints a error and exit func fatal(err error) { printErr(err) os.Exit(1) } // shellOutput executes a shell command and returns the trimmed output func shellOutput(cmd string) string { args := strings.Fields(cmd) out, _ := exec.Command(args[0], args[1:]...).Output() return strings.Trim(string(out), " \n\r") } // fileExists checks if a file exists and is not a directory func fileExists(path ...string) bool { finfo, err := os.Stat(filepath.Join(path...)) if err == nil && !finfo.IsDir() { return true } if os.IsNotExist(err) || finfo.IsDir() { return false } if err != nil { fatal(err) } return true } // readFile reads a file and return the trimmed output func readFile(path string) string { data, err := os.ReadFile(path) if err != nil { return "" } return strings.Trim(string(data), "\n\r ") } func optArg(args []string, i int, def string) string { if i+1 > len(args) { return def } return args[i] } func envOr(name, def string) string { s := os.Getenv(name) if s == "" { return def } return s } func stringInSlice(needle string, haystack []string) bool { for _, hay := range haystack { if hay == needle { return true } } return false } func inSliceRE(needle *regexp.Regexp, haystack []string) []string { var list []string for _, hay := range haystack { if needle.MatchString(hay) { list = append(list, hay) } } return list } func stringInMapKeys(needle string, haystack map[string][]string) bool { _, ok := haystack[needle] return ok } // checkError prints the message and exits if the error is not nil func checkError(e error, message string) { if e != nil { fmt.Println(message) fatal(e) } } 0707010000001E000081A4000000000000000000000001660D2C8E00001691000000000000000000000000000000000000001C00000000promu-0.16.0/cmd/release.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "context" "errors" "fmt" "os" "path/filepath" "strings" "time" "github.com/google/go-github/v25/github" "golang.org/x/oauth2" "github.com/prometheus/promu/pkg/changelog" "github.com/prometheus/promu/util/retry" ) var ( releasecmd = app.Command("release", "Upload all release files to the Github release") timeout = releasecmd.Flag("timeout", "Upload timeout").Duration() allowedRetries = releasecmd.Flag("retry", "Number of retries to perform when upload fails"). Default("2").Int() releaseLocation = releasecmd.Arg("location", "Location of files to release").Default(".").Strings() ) func runRelease(location string) { token := os.Getenv("GITHUB_TOKEN") if len(token) == 0 { fatal(errors.New("GITHUB_TOKEN not defined")) } ctx := context.Background() if *timeout != time.Duration(0) { var cancel context.CancelFunc ctx, cancel = context.WithTimeout(ctx, *timeout) defer cancel() } client := github.NewClient( oauth2.NewClient( ctx, oauth2.StaticTokenSource( &oauth2.Token{AccessToken: token}, ), ), ) semVer, err := projInfo.ToSemver() if err != nil { fatal(err) } // Find the GitHub release matching with the tag. We need to list all // releases because it is the only way to get draft releases too. var ( release *github.RepositoryRelease opts = &github.ListOptions{} tag = fmt.Sprintf("v%s", projInfo.Version) ) for { releases, resp, err := client.Repositories.ListReleases(ctx, projInfo.Owner, projInfo.Name, opts) if err != nil { fatal(fmt.Errorf("failed to list releases: %w", err)) } for _, r := range releases { if r.GetTagName() == tag { release = r break } } if release != nil || resp.NextPage == 0 { break } opts.Page = resp.NextPage } if release == nil { f, err := os.Open("CHANGELOG.md") if err != nil { fatal(err) } defer f.Close() entry, err := changelog.ReadEntry(f, projInfo.Version) if err != nil { fatal(err) } name := entry.Name() // Create a draft release if none exists already. draft := true prerelease := semVer.Prerelease() != "" release, _, err = client.Repositories.CreateRelease( ctx, projInfo.Owner, projInfo.Name, &github.RepositoryRelease{ TagName: &tag, TargetCommitish: &projInfo.Revision, Name: &name, Body: &entry.Text, Draft: &draft, Prerelease: &prerelease, }) if err != nil { fatal(fmt.Errorf("failed to create a draft release for %s: %w", projInfo.Version, err)) } } if err := filepath.Walk(location, releaseFile(ctx, client, release)); err != nil { // Remove incomplete assets. // See https://developer.github.com/v3/repos/releases/#response-for-upstream-failure opts = &github.ListOptions{} for { assets, resp, err := client.Repositories.ListReleaseAssets(ctx, projInfo.Owner, projInfo.Name, release.GetID(), opts) if err != nil { break } for _, asset := range assets { if strings.EqualFold(asset.GetState(), "starter") { _, _ = client.Repositories.DeleteReleaseAsset(ctx, projInfo.Owner, projInfo.Name, asset.GetID()) } } if resp.NextPage == 0 { break } opts.Page = resp.NextPage } fatal(fmt.Errorf("failed to upload all files: %w", err)) } } func releaseFile(ctx context.Context, client *github.Client, release *github.RepositoryRelease) func(string, os.FileInfo, error) error { return func(path string, fi os.FileInfo, err error) error { if err != nil { return err } if fi.IsDir() { return nil } // Check if the asset has already been uploaded and remove it if it is a draft release. filename := filepath.Base(path) opts := &github.ListOptions{} for { assets, resp, err := client.Repositories.ListReleaseAssets(ctx, projInfo.Owner, projInfo.Name, release.GetID(), opts) if err != nil { return fmt.Errorf("failed to list release assets: %w", err) } var stop bool for _, asset := range assets { if asset.GetName() == filename { var err error stop = true if release.GetDraft() { _, err = client.Repositories.DeleteReleaseAsset(ctx, projInfo.Owner, projInfo.Name, asset.GetID()) if err != nil { err = fmt.Errorf("failed to delete existing asset %q: %w", filename, err) } } else { err = fmt.Errorf("%q already exists", filename) } if err != nil { return err } break } } if stop || resp.NextPage == 0 { break } opts.Page = resp.NextPage } maxAttempts := *allowedRetries + 1 err = retry.Do(func(attempt int) (bool, error) { again := attempt < maxAttempts f, err := os.Open(path) if err != nil { return again, err } _, _, err = client.Repositories.UploadReleaseAsset( ctx, projInfo.Owner, projInfo.Name, release.GetID(), &github.UploadOptions{Name: filename}, f) if err != nil { time.Sleep(2 * time.Second) } return again, err }) if err != nil { return fmt.Errorf("failed to upload %q after %d attempts: %w", filename, maxAttempts, err) } fmt.Println(" > uploaded", filename) return nil } } 0707010000001F000081A4000000000000000000000001660D2C8E00000D18000000000000000000000000000000000000001C00000000promu-0.16.0/cmd/tarball.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "archive/zip" "fmt" "io" "os" "path/filepath" "strings" kingpin "github.com/alecthomas/kingpin/v2" "github.com/prometheus/promu/util/sh" ) var ( tarballcmd = app.Command("tarball", "Create a tarball from the built Go project") tarballPrefixSet bool tarballPrefix = tarballcmd.Flag("prefix", "Specific dir to store tarballs"). PreAction(func(c *kingpin.ParseContext) error { tarballPrefixSet = true return nil }). Default(".").String() tarBinariesLocation = tarballcmd.Arg("location", "location of binaries to tar").Default(".").Strings() ) func runTarball(binariesLocation string) { if tarballPrefixSet { config.Tarball.Prefix = *tarballPrefix } var ( prefix = config.Tarball.Prefix tmpDir = ".release" goos = envOr("GOOS", goos) goarch = envOr("GOARCH", goarch) name = fmt.Sprintf("%s-%s.%s-%s", projInfo.Name, projInfo.Version, goos, goarch) binaries = config.Build.Binaries ext string ) if goos == "windows" { ext = ".exe" } dir := filepath.Join(tmpDir, name) if err := os.MkdirAll(dir, 0o777); err != nil { fatal(fmt.Errorf("Failed to create directory: %w", err)) } defer sh.RunCommand("rm", "-rf", tmpDir) projectFiles := config.Tarball.Files for _, file := range projectFiles { sh.RunCommand("cp", "-a", file, dir) } for _, binary := range binaries { binaryName := fmt.Sprintf("%s%s", binary.Name, ext) sh.RunCommand("cp", "-a", filepath.Join(binariesLocation, binaryName), dir) } if !fileExists(prefix) { os.Mkdir(prefix, 0o777) } tar := fmt.Sprintf("%s.tar.gz", name) fmt.Println(" > ", tar) sh.RunCommand("tar", "zcf", filepath.Join(prefix, tar), "-C", tmpDir, name) // Windows systems don't have tar available by default. Produce archives in // the common zip format additionally. if goos == "windows" { archive := name + ".zip" fmt.Println(" > ", archive) if err := createZIP(filepath.Join(prefix, archive), dir); err != nil { fatal(fmt.Errorf("Could not create ZIP archive: %w", err)) } } } // createZIP creates a ZIP archive at the given path containing the specified // directory. func createZIP(path, dir string) error { f, err := os.Create(path) if err != nil { return err } defer f.Close() w := zip.NewWriter(f) defer w.Close() prefix := filepath.Dir(dir) walker := func(path string, info os.FileInfo, err error) error { if err != nil { return err } if info.IsDir() { return nil } r, err := os.Open(path) if err != nil { return err } defer r.Close() name := strings.TrimLeft(strings.TrimPrefix(path, prefix), "/") f, err := w.Create(name) if err != nil { return err } _, err = io.Copy(f, r) return err } return filepath.Walk(dir, walker) } 07070100000020000081A4000000000000000000000001660D2C8E000003AF000000000000000000000000000000000000001C00000000promu-0.16.0/cmd/version.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package cmd import ( "fmt" "github.com/prometheus/common/version" ) var ( versioncmd = app.Command("version", "Print the version and exit") short = versioncmd.Flag("short", "Print shorter version").Short('s').Bool() ) func runVersion() { if *short { fmt.Printf(version.Version) } else { fmt.Println(version.Print("promu")) } } 07070100000021000041ED000000000000000000000003660D2C8E00000000000000000000000000000000000000000000001100000000promu-0.16.0/doc07070100000022000041ED000000000000000000000006660D2C8E00000000000000000000000000000000000000000000001A00000000promu-0.16.0/doc/examples07070100000023000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000002000000000promu-0.16.0/doc/examples/basic07070100000024000081A4000000000000000000000001660D2C8E00000088000000000000000000000000000000000000002B00000000promu-0.16.0/doc/examples/basic/.promu.ymlrepository: path: github.com/prometheus/promu build: binaries: - name: basic-example path: doc/examples/basic 07070100000025000081A4000000000000000000000001660D2C8E00000044000000000000000000000000000000000000002A00000000promu-0.16.0/doc/examples/basic/README.mdBasic example Run `promu build` to build the `basic-example` binary07070100000026000081A4000000000000000000000001660D2C8E00000004000000000000000000000000000000000000002800000000promu-0.16.0/doc/examples/basic/VERSION0.1 07070100000027000081A4000000000000000000000001660D2C8E0000008C000000000000000000000000000000000000002E00000000promu-0.16.0/doc/examples/basic/alt-promu.ymlrepository: path: github.com/prometheus/promu build: binaries: - name: alt-basic-example path: doc/examples/basic 07070100000028000081A4000000000000000000000001660D2C8E000000BC000000000000000000000000000000000000002F00000000promu-0.16.0/doc/examples/basic/extldflags.ymlrepository: path: github.com/prometheus/promu build: binaries: - name: extldflags path: doc/examples/basic extldflags: - -ltesting - -ltesting01 07070100000029000081A4000000000000000000000001660D2C8E000002A3000000000000000000000000000000000000002800000000promu-0.16.0/doc/examples/basic/main.go// Copyright © 2018 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package main import ( "fmt" ) func main() { fmt.Println("Promu example") } 0707010000002A000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000002500000000promu-0.16.0/doc/examples/crossbuild0707010000002B000081A4000000000000000000000001660D2C8E00000117000000000000000000000000000000000000003000000000promu-0.16.0/doc/examples/crossbuild/.promu.ymlrepository: path: github.com/prometheus/promu build: binaries: - name: crossbuild-example path: doc/examples/crossbuild crossbuild: platforms: - linux/amd64 - linux/386 - windows/amd64 tarball: files: - README.md 0707010000002C000081A4000000000000000000000001660D2C8E00000380000000000000000000000000000000000000002E00000000promu-0.16.0/doc/examples/crossbuild/Makefile# Copyright 2018 The Prometheus Authors # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # The crossbuild command sets the PREFIX env var for each platform PREFIX ?= $(shell pwd) # Workaround to mimic promu's build behavior which usually handles the extension. ifeq ($(GOOS),windows) EXTENSION = .exe endif build: go build -o $(PREFIX)/crossbuild-example$(EXTENSION) test: go test 0707010000002D000081A4000000000000000000000001660D2C8E00000106000000000000000000000000000000000000002F00000000promu-0.16.0/doc/examples/crossbuild/README.mdCrossbuild example Run `promu crossbuild` to crossbuild for linux-386, linux-amd64, and windows-amd64 platforms. Output will be in the `.build` directory. Run `promu crossbuild tarballs` to build platform tarballs. Output will be in the `.tarballs` directory. 0707010000002E000081A4000000000000000000000001660D2C8E00000004000000000000000000000000000000000000002D00000000promu-0.16.0/doc/examples/crossbuild/VERSION0.1 0707010000002F000081A4000000000000000000000001660D2C8E000002AE000000000000000000000000000000000000002D00000000promu-0.16.0/doc/examples/crossbuild/main.go// Copyright © 2018 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package main import ( "fmt" ) func main() { fmt.Println("Promu crossbuild example") } 07070100000030000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000002500000000promu-0.16.0/doc/examples/prometheus07070100000031000081A4000000000000000000000001660D2C8E0000050E000000000000000000000000000000000000003000000000promu-0.16.0/doc/examples/prometheus/.promu.ymlgo: version: 1.15.1 cgo: false repository: path: github.com/prometheus/prometheus build: prefix: . binaries: - name: prometheus path: ./cmd/prometheus - name: promtool path: ./cmd/promtool flags: -a -tags netgo ldflags: | -s -X {{repoPath}}/version.Version={{.Version}} -X {{repoPath}}/version.Revision={{.Revision}} -X {{repoPath}}/version.Branch={{.Branch}} -X {{repoPath}}/version.BuildUser={{user}}@{{host}} -X {{repoPath}}/version.BuildDate={{date "20060102-15:04:05"}} tarball: prefix: . files: - consoles - console_libraries - documentation/examples/prometheus.yml - LICENSE - NOTICE crossbuild: platforms: - linux/amd64 - linux/386 - darwin/amd64 - windows/amd64 - windows/386 - freebsd/amd64 - freebsd/386 - openbsd/amd64 - openbsd/386 - netbsd/amd64 - netbsd/386 - dragonfly/amd64 - linux/arm - linux/arm64 - freebsd/arm - openbsd/arm - netbsd/arm - aix/ppc64 - linux/ppc64 - linux/ppc64le - linux/mips64 - linux/mips64le - linux/s390x 07070100000032000081A4000000000000000000000001660D2C8E0000003A000000000000000000000000000000000000002F00000000promu-0.16.0/doc/examples/prometheus/README.mdPrometheus example Run `promu build` to build prometheus 07070100000033000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000002200000000promu-0.16.0/doc/examples/tarball07070100000034000081A4000000000000000000000001660D2C8E000000B4000000000000000000000000000000000000002D00000000promu-0.16.0/doc/examples/tarball/.promu.ymlrepository: path: github.com/prometheus/promu build: binaries: - name: tarball-example path: doc/examples/tarball tarball: files: - README.md 07070100000035000081A4000000000000000000000001660D2C8E0000005E000000000000000000000000000000000000002C00000000promu-0.16.0/doc/examples/tarball/README.mdTarball example Run `promu build && promu tarball` to build the binary and create a tarball. 07070100000036000081A4000000000000000000000001660D2C8E00000004000000000000000000000000000000000000002A00000000promu-0.16.0/doc/examples/tarball/VERSION0.1 07070100000037000081A4000000000000000000000001660D2C8E000002AB000000000000000000000000000000000000002A00000000promu-0.16.0/doc/examples/tarball/main.go// Copyright © 2018 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package main import ( "fmt" ) func main() { fmt.Println("Promu tarball example") } 07070100000038000081A4000000000000000000000001660D2C8E000002CC000000000000000000000000000000000000001400000000promu-0.16.0/go.modmodule github.com/prometheus/promu go 1.21 require ( github.com/Masterminds/semver v1.5.0 github.com/alecthomas/kingpin/v2 v2.4.0 github.com/google/go-github/v25 v25.1.3 github.com/prometheus/common v0.51.1 go.uber.org/atomic v1.11.0 golang.org/x/oauth2 v0.18.0 gopkg.in/yaml.v2 v2.4.0 ) require ( github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect github.com/golang/protobuf v1.5.3 // indirect github.com/google/go-querystring v1.0.0 // indirect github.com/kr/pretty v0.3.1 // indirect github.com/xhit/go-str2duration/v2 v2.1.0 // indirect golang.org/x/net v0.22.0 // indirect google.golang.org/appengine v1.6.7 // indirect google.golang.org/protobuf v1.33.0 // indirect ) 07070100000039000081A4000000000000000000000001660D2C8E00001900000000000000000000000000000000000000001400000000promu-0.16.0/go.sumgithub.com/Masterminds/semver v1.5.0 h1:H65muMkzWKEuNDnfl9d70GUjFniHKHRbFPGBuZ3QEww= github.com/Masterminds/semver v1.5.0/go.mod h1:MB6lktGJrhw8PrUyiEoblNEGEQ+RzHPF078ddwwvV3Y= github.com/alecthomas/kingpin/v2 v2.4.0 h1:f48lwail6p8zpO1bC4TxtqACaGqHYA22qkHjHpqDjYY= github.com/alecthomas/kingpin/v2 v2.4.0/go.mod h1:0gyi0zQnjuFk8xrkNKamJoyUo382HRL7ATRpFZCw6tE= github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 h1:s6gZFSlWYmbqAuRjVTiNNhvNRfY2Wxp9nhfyel4rklc= github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137/go.mod h1:OMCwj8VM1Kc9e19TLln2VL61YJF0x1XFtfdL4JdbSyE= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-github/v25 v25.1.3 h1:Ht4YIQgUh4l4lc80fvGnw60khXysXvlgPxPP8uJG3EA= github.com/google/go-github/v25 v25.1.3/go.mod h1:6z5pC69qHtrPJ0sXPsj4BLnd82b+r6sLB7qcBoRZqpw= github.com/google/go-querystring v1.0.0 h1:Xkwi/a1rcvNg1PPYe5vI8GbeBY/jrVuDX5ASuANWTrk= github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/prometheus/common v0.51.1 h1:eIjN50Bwglz6a/c3hAgSMcofL3nD+nFQkV6Dd4DsQCw= github.com/prometheus/common v0.51.1/go.mod h1:lrWtQx+iDfn2mbH5GUzlH9TSHyfZpHkSiG1W7y3sF2Q= github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ= github.com/rogpeppe/go-internal v1.10.0/go.mod h1:UQnix2H7Ngw/k4C5ijL5+65zddjncjaFoBhdsK/akog= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= github.com/xhit/go-str2duration/v2 v2.1.0 h1:lxklc02Drh6ynqX+DdPyp5pCKLUQpRT8bp8Ydu2Bstc= github.com/xhit/go-str2duration/v2 v2.1.0/go.mod h1:ohY8p+0f07DiV6Em5LKB0s2YpLtXVyJfNt1+BlmyAsU= go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE= go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc= golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/oauth2 v0.18.0 h1:09qnuIAgzdx1XplqJvW6CQqMCtGZykZWcXzPMPUusvI= golang.org/x/oauth2 v0.18.0/go.mod h1:Wf7knwG0MPoWIMMBgFlEaSUDaKskp0dCfrlJRJXbBi8= golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c= google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI= google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= 0707010000003A000081A4000000000000000000000001660D2C8E000002AB000000000000000000000000000000000000001500000000promu-0.16.0/main.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package main import "github.com/prometheus/promu/cmd" func main() { cmd.Execute() } 0707010000003B000081A4000000000000000000000001660D2C8E0000184A000000000000000000000000000000000000001A00000000promu-0.16.0/main_test.go// Copyright © 2018 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package main import ( "fmt" "go/build" "log" "os" "os/exec" "path" "path/filepath" "strings" "testing" ) const ( examplesDir = "doc/examples" testOutputDir = "testoutput" promuCmdBinary = "promu" ) var ( goos = build.Default.GOOS goarch = build.Default.GOARCH promuBinaryRelPath = path.Join(testOutputDir, promuCmdBinary) promuBinaryAbsPath, _ = filepath.Abs(promuBinaryRelPath) promuExamplesBasic = path.Join(examplesDir, "basic") promuExamplesCrossbuild = path.Join(examplesDir, "crossbuild") promuExamplesTarball = path.Join(examplesDir, "tarball") ) func TestMain(m *testing.M) { setup() result := m.Run() os.Exit(result) } // setup any prerequisites for the tests func setup() { err := os.Mkdir(examplesDir, os.ModePerm) if !os.IsExist(err) && err != nil { log.Fatal(err) } cmd := exec.Command("go", "build", "-o", promuBinaryAbsPath) output, err := cmd.Output() if err != nil { log.Fatal(err, string(output)) } } func errcheck(t *testing.T, err error, output string) { if err != nil { log.Print(output) t.Error(err) } } func assertTrue(t *testing.T, cond bool) { if !cond { t.Error("condition isn't true") } } func assertFileExists(t *testing.T, filepath string) { if _, err := os.Stat(filepath); os.IsNotExist(err) { log.Print("file does not exist: ", filepath) t.Error(err) } } func createSymlink(t *testing.T, target, newlink string) { if _, err := os.Stat(newlink); os.IsNotExist(err) { err = os.Symlink(target, newlink) errcheck(t, err, "Unable to create symlink "+newlink) } } func dockerAvailable() bool { cmd := exec.Command("docker", "info") err := cmd.Run() return err == nil } func TestPromuInfo(t *testing.T) { cmd := exec.Command(promuBinaryAbsPath, "info") output, err := cmd.CombinedOutput() errcheck(t, err, string(output)) if !strings.HasPrefix(string(output), "Name: promu") { t.Error("incorrect output for 'info' command: ", string(output)) } } func TestPromuBuild_Basic(t *testing.T) { outputDir := path.Join(testOutputDir, "basic") err := os.MkdirAll(outputDir, os.ModePerm) errcheck(t, err, "Unable to create output dir") createSymlink(t, path.Join("..", "..", promuExamplesBasic, ".promu.yml"), path.Join(outputDir, ".promu.yml")) cmd := exec.Command(promuBinaryAbsPath, "build") cmd.Dir = outputDir output, err := cmd.CombinedOutput() errcheck(t, err, string(output)) assertFileExists(t, path.Join(outputDir, "basic-example")) } func TestPromuBuild_AltConfig(t *testing.T) { outputDir := path.Join(testOutputDir, "altconf") promuConfig := path.Join(promuExamplesBasic, "alt-promu.yml") cmd := exec.Command(promuBinaryAbsPath, "build", "--config", promuConfig, "--prefix", outputDir) output, err := cmd.CombinedOutput() errcheck(t, err, string(output)) assertFileExists(t, path.Join(outputDir, "alt-basic-example")) } func TestPromuBuild_ExtLDFlags(t *testing.T) { outputDir := path.Join(testOutputDir, "extldflags") promuConfig := path.Join(promuExamplesBasic, "extldflags.yml") cmd := exec.Command(promuBinaryAbsPath, "build", "-v", "--config", promuConfig, "--prefix", outputDir) output, err := cmd.CombinedOutput() assertTrue(t, strings.Contains(string(output), "-extldflags '-ltesting -ltesting01 -static'")) errcheck(t, err, string(output)) assertFileExists(t, path.Join(outputDir, "extldflags")) } func TestTarball(t *testing.T) { outputDir := path.Join(testOutputDir, "tarball") err := os.MkdirAll(outputDir, os.ModePerm) errcheck(t, err, "Unable to create output dir") createSymlink(t, path.Join("..", "..", promuExamplesBasic, ".promu.yml"), path.Join(outputDir, ".promu.yml")) createSymlink(t, path.Join("..", "..", promuExamplesTarball, "README.md"), path.Join(outputDir, "README.md")) createSymlink(t, path.Join("..", "..", promuExamplesTarball, "VERSION"), path.Join(outputDir, "VERSION")) cmd := exec.Command(promuBinaryAbsPath, "build") cmd.Dir = outputDir output, err := cmd.CombinedOutput() errcheck(t, err, string(output)) cmd = exec.Command(promuBinaryAbsPath, "tarball") cmd.Dir = outputDir output, err = cmd.CombinedOutput() errcheck(t, err, string(output)) tarfileName := fmt.Sprintf("promu-0.1.%s-%s.tar.gz", goos, goarch) assertFileExists(t, path.Join(outputDir, tarfileName)) } func TestPromuCrossbuild(t *testing.T) { if testing.Short() { t.Skip("skipping crossbuild test in short mode.") } if !dockerAvailable() { t.Error("unable to connect to docker daemon.") return } cmd := exec.Command(promuBinaryAbsPath, "crossbuild") cmd.Dir = promuExamplesCrossbuild output, err := cmd.CombinedOutput() errcheck(t, err, string(output)) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".build", "linux-386", "crossbuild-example")) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".build", "linux-amd64", "crossbuild-example")) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".build", "windows-amd64", "crossbuild-example.exe")) cmd = exec.Command(promuBinaryAbsPath, "crossbuild", "tarballs") cmd.Dir = promuExamplesCrossbuild defer os.RemoveAll(path.Join(promuExamplesCrossbuild, ".tarballs")) output, err = cmd.CombinedOutput() errcheck(t, err, string(output)) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".tarballs", "promu-0.1.linux-386.tar.gz")) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".tarballs", "promu-0.1.linux-amd64.tar.gz")) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".tarballs", "promu-0.1.windows-amd64.tar.gz")) assertFileExists(t, path.Join(promuExamplesCrossbuild, ".tarballs", "promu-0.1.windows-amd64.zip")) } 0707010000003C000041ED000000000000000000000004660D2C8E00000000000000000000000000000000000000000000001100000000promu-0.16.0/pkg0707010000003D000081A4000000000000000000000001660D2C8E0000008B000000000000000000000000000000000000001B00000000promu-0.16.0/pkg/README.mdThe `pkg` directory is deprecated. Please do not add new packages to this directory. Existing packages will be moved elsewhere eventually. 0707010000003E000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001B00000000promu-0.16.0/pkg/changelog0707010000003F000081A4000000000000000000000001660D2C8E000010BC000000000000000000000000000000000000002800000000promu-0.16.0/pkg/changelog/changelog.go// Copyright © 2019 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package changelog import ( "bufio" "fmt" "io" "regexp" "sort" "strings" "time" ) // Kind represents the type of a change. type Kind int const ( kindChange = iota kindFeature kindEnhancement kindBugfix ) func (k Kind) String() string { switch k { case kindChange: return "CHANGE" case kindFeature: return "FEATURE" case kindEnhancement: return "ENHANCEMENT" case kindBugfix: return "BUGFIX" } return "" } // Kinds is a list of Kind which implements sort.Interface. type Kinds []Kind func (k Kinds) Len() int { return len(k) } func (k Kinds) Less(i, j int) bool { return k[i] < k[j] } func (k Kinds) Swap(i, j int) { k[i], k[j] = k[j], k[i] } // ParseKinds converts a slash-separated list of Kind to a list of Kind. func ParseKinds(s string) Kinds { m := make(map[Kind]struct{}) for _, k := range strings.Split(s, "/") { switch k { case "CHANGE": m[kindChange] = struct{}{} case "FEATURE": m[kindFeature] = struct{}{} case "ENHANCEMENT": m[kindEnhancement] = struct{}{} case "BUGFIX": m[kindBugfix] = struct{}{} } } var kinds Kinds for k := range m { kinds = append(kinds, k) } sort.Stable(kinds) return kinds } func (k Kinds) String() string { var s []string for i := range k { s = append(s, k[i].String()) } return strings.Join(s, "/") } // Change represents a change description. type Change struct { Text string Kinds Kinds } type Changes []Change func (c Changes) Sorted() error { less := func(k1, k2 Kinds) bool { if len(k1) == 0 { return len(k2) == 0 } if len(k2) == 0 { return true } n := len(k1) if len(k1) > len(k2) { n = len(k2) } for j := 0; j < n; j++ { if k1[j] == k2[j] { continue } return k1[j] < k2[j] } return len(k1) <= len(k2) } for i := 0; i < len(c)-1; i++ { k1, k2 := c[i].Kinds, c[i+1].Kinds if !less(k1, k2) { return fmt.Errorf("%q should be after %q", c[i].Text, c[i+1].Text) } } return nil } // Entry represents an entry in the changelog. type Entry struct { Version string Date time.Time Changes Changes Text string } const dateFormat = "2006-01-02" // Name returns the canonical name of the entry. func (c Entry) Name() string { return fmt.Sprintf("%s / %s", c.Version, c.Date.Format(dateFormat)) } // ReadEntry reads the entry for the given version from the changelog file. // It returns an error if the version is not found. func ReadEntry(r io.Reader, version string) (*Entry, error) { reHeader, err := regexp.Compile(fmt.Sprintf(`^#{1,2} %s / (\d{4}-\d{2}-\d{2})`, regexp.QuoteMeta(version))) if err != nil { return nil, err } reChange := regexp.MustCompile(`^\* \[([^\]]+)\]`) var ( reading bool lines []string entry = Entry{Version: version} scanner = bufio.NewScanner(r) ) for (len(lines) == 0 || reading) && scanner.Scan() { line := scanner.Text() m := reHeader.FindStringSubmatch(line) switch { case len(m) > 0: reading = true t, err := time.Parse(dateFormat, m[1]) if err != nil { return nil, fmt.Errorf("invalid changelog date: %w", err) } entry.Date = t case strings.HasPrefix(line, "## "): reading = false case reading: if len(lines) == 0 && strings.TrimSpace(line) == "" { continue } m := reChange.FindStringSubmatch(line) if len(m) > 1 { entry.Changes = append(entry.Changes, Change{Text: line, Kinds: ParseKinds(m[1])}) } lines = append(lines, line) } } if entry.Date.IsZero() { return nil, fmt.Errorf( "unable to locate release information in changelog for version %q, expected format: %q", version, reHeader) } entry.Text = strings.Join(lines, "\n") return &entry, nil } 07070100000040000081A4000000000000000000000001660D2C8E00001DA3000000000000000000000000000000000000002D00000000promu-0.16.0/pkg/changelog/changelog_test.go// Copyright 2019 The Prometheus Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package changelog import ( "bytes" "reflect" "testing" "time" ) func TestReadEntry(t *testing.T) { mustParse := func(s string) time.Time { d, err := time.Parse(dateFormat, s) if err != nil { t.Fatalf("unexpected error: %v", err) } return d } for _, tc := range []struct { in string version string exp Entry err bool }{ { // Version not found. in: `## 1.0.0 / 2016-01-02 * [BUGFIX] Some fix. * [FEATURE] Some feature.`, version: "1.0.0-notfound", err: true, }, { in: `## 1.0.0 / 2016-01-02 * [BUGFIX] Some fix. * [FEATURE] Some feature.`, version: "1.0.0", exp: Entry{ Version: "1.0.0", Date: mustParse("2016-01-02"), Changes: []Change{ { Kinds: []Kind{kindBugfix}, Text: "* [BUGFIX] Some fix.", }, { Kinds: []Kind{kindFeature}, Text: "* [FEATURE] Some feature.", }, }, Text: `* [BUGFIX] Some fix. * [FEATURE] Some feature.`, }, }, { in: `# 1.0.0 / 2016-01-02 * [BUGFIX] Some fix. * [FEATURE] Some feature.`, version: "1.0.0", exp: Entry{ Version: "1.0.0", Date: mustParse("2016-01-02"), Changes: []Change{ { Kinds: []Kind{kindBugfix}, Text: "* [BUGFIX] Some fix.", }, { Kinds: []Kind{kindFeature}, Text: "* [FEATURE] Some feature.", }, }, Text: `* [BUGFIX] Some fix. * [FEATURE] Some feature.`, }, }, { in: `## 1.0.0 / 2016-01-02 * [BUGFIX] Some fix. * [FEATURE] Some feature. ## 0.0.1 / 2016-01-02 * [BUGFIX] Another fix.`, version: "1.0.0", exp: Entry{ Version: "1.0.0", Date: mustParse("2016-01-02"), Changes: []Change{ { Kinds: []Kind{kindBugfix}, Text: "* [BUGFIX] Some fix.", }, { Kinds: []Kind{kindFeature}, Text: "* [FEATURE] Some feature.", }, }, Text: `* [BUGFIX] Some fix. * [FEATURE] Some feature. `, }, }, { in: `## 1.0.0 / 2016-01-02 [BUGFIX] Some fix. [FEATURE] Some feature. ## 0.0.1 / 2016-01-01 * [BUGFIX] Another fix.`, version: "0.0.1", exp: Entry{ Version: "0.0.1", Date: mustParse("2016-01-01"), Changes: []Change{ { Kinds: []Kind{kindBugfix}, Text: "* [BUGFIX] Another fix.", }, }, Text: `* [BUGFIX] Another fix.`, }, }, { in: `## 1.0.0 / 2016-01-03 This is the first stable release. * [CHANGE/BUGFIX] Some fix. * [FEATURE] Some feature. ## 0.0.1 / 2016-01-02 * [BUGFIX] Another fix.`, version: "1.0.0", exp: Entry{ Version: "1.0.0", Date: mustParse("2016-01-03"), Changes: []Change{ { Kinds: []Kind{kindChange, kindBugfix}, Text: "* [CHANGE/BUGFIX] Some fix.", }, { Kinds: []Kind{kindFeature}, Text: "* [FEATURE] Some feature.", }, }, Text: `This is the first stable release. * [CHANGE/BUGFIX] Some fix. * [FEATURE] Some feature. `, }, }, { in: `## 1.0.0 / 2016-01-04 ### Breaking changes! * [CHANGE] Some change. * [FEATURE] Some feature. * [ENHANCEMENT] Some enhancement. ## 0.0.1 / 2016-01-02 * [BUGFIX] Another fix.`, version: "1.0.0", exp: Entry{ Version: "1.0.0", Date: mustParse("2016-01-04"), Changes: []Change{ { Kinds: []Kind{kindChange}, Text: "* [CHANGE] Some change.", }, { Kinds: []Kind{kindFeature}, Text: "* [FEATURE] Some feature.", }, { Kinds: []Kind{kindEnhancement}, Text: "* [ENHANCEMENT] Some enhancement.", }, }, Text: `### Breaking changes! * [CHANGE] Some change. * [FEATURE] Some feature. * [ENHANCEMENT] Some enhancement. `, }, }, { // Invalid date. in: "## 1.0.0 / 2006-19-02", version: "1.0.0", err: true, }, } { tc := tc t.Run("", func(t *testing.T) { got, err := ReadEntry(bytes.NewBufferString(tc.in), tc.version) if tc.err { if err == nil { t.Fatal("expected error, got none") } return } if err != nil { t.Fatalf("expected no error, got %v", err) } if !reflect.DeepEqual(&tc.exp, got) { t.Fatalf("expected:\n%v\ngot:\n%v", tc.exp, got) } }) } } func TestKinds(t *testing.T) { for _, tc := range []struct { in string exp Kinds }{ { in: "CHANGE", exp: Kinds{kindChange}, }, { in: "BUGFIX/CHANGE", exp: Kinds{kindChange, kindBugfix}, }, { in: "BUGFIX/BUGFIX", exp: Kinds{kindBugfix}, }, { in: "BUGFIX/INVALID", exp: Kinds{kindBugfix}, }, { in: "INVALID", }, } { t.Run("", func(t *testing.T) { got := ParseKinds(tc.in) if !reflect.DeepEqual(&tc.exp, &got) { t.Fatalf("expected:\n%v\ngot:\n%v", tc.exp, got) } }) } } func TestChangesSorted(t *testing.T) { for _, tc := range []struct { in Changes err bool }{ { in: Changes{ { Kinds: Kinds{}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{}, }, { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{kindFeature}, }, }, err: true, }, { in: Changes{ { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{kindChange}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{kindBugfix}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{kindBugfix}, }, { Kinds: Kinds{kindFeature}, }, }, err: true, }, { in: Changes{ { Kinds: Kinds{kindChange}, }, { Kinds: Kinds{kindChange, kindBugfix}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange, kindBugfix}, }, { Kinds: Kinds{kindChange}, }, }, err: true, }, { in: Changes{ { Kinds: Kinds{kindChange, kindFeature, kindBugfix}, }, { Kinds: Kinds{kindChange, kindBugfix}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange, kindBugfix}, }, { Kinds: Kinds{kindChange, kindFeature, kindBugfix}, }, }, err: true, }, { in: Changes{ { Kinds: Kinds{kindChange, kindFeature, kindBugfix}, }, { Kinds: Kinds{kindChange, kindBugfix}, }, { Kinds: Kinds{}, }, { Kinds: Kinds{}, }, }, err: false, }, { in: Changes{ { Kinds: Kinds{kindChange, kindFeature, kindBugfix}, }, { Kinds: Kinds{}, }, { Kinds: Kinds{kindChange, kindBugfix}, }, { Kinds: Kinds{}, }, }, err: true, }, } { t.Run("", func(t *testing.T) { err := tc.in.Sorted() if tc.err { if err == nil { t.Fatal("expected error but got nil") } return } if err != nil { t.Fatalf("expected no error but got: %v", err) } }) } } 07070100000041000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001C00000000promu-0.16.0/pkg/repository07070100000042000081A4000000000000000000000001660D2C8E00001293000000000000000000000000000000000000002400000000promu-0.16.0/pkg/repository/info.go// Copyright © 2019 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package repository import ( "fmt" "net/url" "os" "os/exec" "os/user" "path/filepath" "regexp" "strings" "github.com/Masterminds/semver" ) // Info represents current project useful information. type Info struct { Branch string Name string Owner string Repo string Revision string Version string } // shellOutput executes a shell command and returns the trimmed output. func shellOutput(cmd string, arg ...string) string { out, _ := shellOutputWithError(cmd, arg...) return out } // shellOutputWithError executes a shell command and returns the trimmed output and error. func shellOutputWithError(cmd string, arg ...string) (string, error) { out, err := exec.Command(cmd, arg...).Output() return strings.Trim(string(out), " \n\r"), err } // NewInfo returns a new Info. func NewInfo(warnf func(error)) (Info, error) { if warnf == nil { warnf = func(error) {} } var ( info Info err error ) cmd := exec.Command("git", "rev-parse", "--show-toplevel") cmd.Stdout, cmd.Stderr = nil, nil if err := cmd.Run(); err != nil { // Not a git repository. repo, err := os.Getwd() if err != nil { return info, fmt.Errorf("couldn't get current working directory: %w", err) } repo = strings.TrimPrefix(repo, os.Getenv("GOPATH")) repo = strings.TrimPrefix(repo, "/src/") user, err := user.Current() if err != nil { return info, fmt.Errorf("couldn't get current user: %w", err) } info = Info{ Branch: "non-git", Name: filepath.Base(repo), Owner: user.Username, Repo: repo, Revision: "non-git", } } else { branch, err := shellOutputWithError("git", "rev-parse", "--abbrev-ref", "HEAD") if err != nil { return info, fmt.Errorf("unable to get the current branch: %w", err) } remote, err := shellOutputWithError("git", "config", "--get", fmt.Sprintf("branch.%s.remote", branch)) if err != nil { // default to origin. remote = "origin" } repoURL, err := shellOutputWithError("git", "config", "--get", fmt.Sprintf("remote.%s.url", remote)) if err != nil { warnf(fmt.Errorf("unable to get repository location for remote %q: %w", remote, err)) } repo, err := repoLocation(repoURL) if err != nil { return info, fmt.Errorf("couldn't parse repository location: %q: %w", repoURL, err) } info = Info{ Branch: branch, Name: filepath.Base(repo), Owner: filepath.Base(filepath.Dir(repo)), Repo: repo, Revision: shellOutput("git", "rev-parse", "HEAD"), } } info.Version, err = findVersion() if err != nil { warnf(fmt.Errorf("unable to find project's version: %w", err)) } return info, nil } // Convert SCP-like URL to SSH URL(e.g. [user@]host.xz:path/to/repo.git/) // ref. http://git-scm.com/docs/git-fetch#_git_urls // (golang hasn't supported Perl-like negative look-behind match) var ( hasSchemePattern = regexp.MustCompile("^[^:]+://") scpLikeURLPattern = regexp.MustCompile("^([^@]+@)?([^:]+):/?(.+)$") ) func repoLocation(repo string) (string, error) { if !hasSchemePattern.MatchString(repo) && scpLikeURLPattern.MatchString(repo) { matched := scpLikeURLPattern.FindStringSubmatch(repo) user := matched[1] host := matched[2] path := matched[3] repo = fmt.Sprintf("ssh://%s%s/%s", user, host, path) } u, err := url.Parse(repo) if err != nil { return "", err } repo = fmt.Sprintf("%s%s", strings.Split(u.Host, ":")[0], u.Path) repo = strings.TrimSuffix(repo, ".git") return repo, nil } func findVersion() (string, error) { for _, file := range []string{"VERSION", "version/VERSION"} { b, err := os.ReadFile(file) if err != nil { continue } return strings.Trim(string(b), "\n\r "), nil } return strings.TrimPrefix(shellOutput("git", "describe", "--tags", "--always", "--dirty"), "v"), nil } // ToSemver returns a *semver.Version from Info. func (i Info) ToSemver() (*semver.Version, error) { if strings.HasPrefix(i.Version, "v") { return nil, fmt.Errorf("version %q shouldn't start with 'v'", i.Version) } semVer, err := semver.NewVersion(i.Version) if err != nil { return nil, fmt.Errorf("invalid semver version: %w", err) } return semVer, nil } 07070100000043000081A4000000000000000000000001660D2C8E00000796000000000000000000000000000000000000002900000000promu-0.16.0/pkg/repository/info_test.go// Copyright © 2017 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package repository import ( "testing" ) func TestToSemver(t *testing.T) { for _, tc := range []struct { version string exp string }{ { version: "1.0.0", exp: "1.0.0", }, { version: "1.0.0-rc0", exp: "1.0.0-rc0", }, { version: "x1.0.0-rc0", }, { version: "v1.0.0-rc0", }, } { tc := tc t.Run("", func(t *testing.T) { got, err := Info{Version: tc.version}.ToSemver() if err != nil { if tc.exp != "" { t.Fatalf("expected no error, got %v", err) } return } if got.String() != tc.exp { t.Fatalf("expected %q, got %q", tc.exp, got) } }) } } func TestRepoLocation(t *testing.T) { repoTests := []struct { s string // input expected string // expected result }{ {"git@github.com:prometheus/promu.git", "github.com/prometheus/promu"}, {"https://github.com/prometheus/promu.git", "github.com/prometheus/promu"}, {"ssh://git@gitlab.fr:22443/prometheus/promu.git", "gitlab.fr/prometheus/promu"}, {"https://sdurrheimer@gitlab.fr/prometheus/promu.git", "gitlab.fr/prometheus/promu"}, } for _, tc := range repoTests { actual, err := repoLocation(tc.s) if err != nil { t.Errorf("repoLocation(%s): %+v", tc.s, err) } if actual != tc.expected { t.Errorf("repoLocation(%s): expected %s, got %s", tc.s, tc.expected, actual) } } } 07070100000044000041ED000000000000000000000004660D2C8E00000000000000000000000000000000000000000000001200000000promu-0.16.0/util07070100000045000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001800000000promu-0.16.0/util/retry07070100000046000081A4000000000000000000000001660D2C8E000003DA000000000000000000000000000000000000002100000000promu-0.16.0/util/retry/retry.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package retry // Func represents functions that can be retried. type Func func(attempt int) (retry bool, err error) // Do keeps trying the function until the second argument // returns false, or no error is returned. func Do(fn Func) error { var err error var cont bool attempt := 1 for { cont, err = fn(attempt) if !cont || err == nil { break } attempt++ } return err } 07070100000047000041ED000000000000000000000002660D2C8E00000000000000000000000000000000000000000000001500000000promu-0.16.0/util/sh07070100000048000081A4000000000000000000000001660D2C8E000004D7000000000000000000000000000000000000001B00000000promu-0.16.0/util/sh/sh.go// Copyright © 2016 Prometheus Team // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package sh import ( "fmt" "os" "os/exec" "regexp" "strings" ) // Verbose enables verbose output var Verbose bool // RunCommand executes a shell command. func RunCommand(name string, arg ...string) error { if Verbose { cmdText := name + " " + strings.Join(arg, " ") fmt.Fprintln(os.Stderr, " + ", cmdText) } cmd := exec.Command(name, arg...) cmd.Stdout = os.Stdout cmd.Stdin = os.Stdin cmd.Stderr = os.Stderr return cmd.Run() } // SplitParameters splits shell command parameters, taking quoting in account. func SplitParameters(s string) []string { r := regexp.MustCompile(`'[^']*'|[^ ]+`) return r.FindAllString(s, -1) } 07070100000049000081A4000000000000000000000001660D2C8E000003B3000000000000000000000000000000000000002000000000promu-0.16.0/util/sh/sh_test.go// Copyright 2018 The Prometheus Authors // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package sh import ( "strings" "testing" ) func TestSplitParameters(t *testing.T) { in := `-a -tags 'netgo static_build'` expect := []string{"-a", "-tags", `'netgo static_build'`} got := SplitParameters(in) for i, g := range got { if expect[i] != g { t.Error("expected", expect[i], "got", g, "full output: ", strings.Join(got, "#")) } } } 07070100000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000B00000000TRAILER!!!245 blocks
Locations
Projects
Search
Status Monitor
Help
OpenBuildService.org
Documentation
API Documentation
Code of Conduct
Contact
Support
@OBShq
Terms
openSUSE Build Service is sponsored by
The Open Build Service is an
openSUSE project
.
Sign Up
Log In
Places
Places
All Projects
Status Monitor