Compare commits
154 Commits
Author | SHA1 | Date | |
---|---|---|---|
439d5cf125 | |||
e02f291db2 | |||
4e64f3f994 | |||
c0b834b0b4 | |||
f937328147 | |||
|
883569bf14 | ||
e32c70a27d | |||
|
8c9abca56e | ||
09c9cbb39f | |||
|
1075d8cf42 | ||
92db4e4209 | |||
|
06ebcd2764 | ||
66878c406f | |||
|
0561824ca3 | ||
|
6210e2ad0c | ||
89c00bb59f | |||
|
8d20f74ec9 | ||
|
e0634128f1 | ||
|
873801833b | ||
160217d9a5 | |||
|
6053ceecc3 | ||
34e909dc98 | |||
|
ae51f8ff7c | ||
f561c77a0d | |||
|
aba9197d2b | ||
f2c3f3d5d1 | |||
|
a702bb558d | ||
aa685064bc | |||
|
7f5ce2e66c | ||
11f849b92d | |||
|
e0c8b29893 | ||
c57e4eedbd | |||
|
b8b9b8ae6b | ||
|
d9f6ca8fae | ||
5ca911c33a | |||
|
09205abdcf | ||
63ab0dbc88 | |||
|
8e21533a35 | ||
|
5c3028d217 | ||
997de86f0f | |||
43d28eca6f | |||
|
49fab99ef6 | ||
|
8f64a205a7 | ||
|
b629031cdc | ||
|
b1082245a6 | ||
|
6b11525d4a | ||
|
f0847578a4 | ||
a0ea7509f7 | |||
|
e30bd45b92 | ||
b6da2df6dc | |||
|
1c8a9d87e8 | ||
|
dbc53d520a | ||
|
ee1616c05b | ||
|
e6cd8bb731 | ||
|
4be3245371 | ||
da1ebc38ca | |||
|
14acb77ab8 | ||
|
9ef586a818 | ||
|
44a26918f9 | ||
95ae9cd371 | |||
|
7cc49c92cd | ||
5517b36e40 | |||
|
7ce39c2bcc | ||
cecd5801cc | |||
|
99063751e1 | ||
6dfd19692d | |||
|
7cc55694f9 | ||
e9bdb73e79 | |||
|
558e2b2360 | ||
d13f36d444 | |||
|
2f6780544e | ||
ce1eeb4af1 | |||
|
79626cb7fb | ||
9afed0bfb0 | |||
eac6b15775 | |||
85319f9bd0 | |||
e4f2c77510 | |||
b77f70aeb0 | |||
bae8886836 | |||
8ad0258828 | |||
d7a9e96561 | |||
4a8caabc52 | |||
5d50b02888 | |||
|
c793eeafda | ||
|
4bad2f0a5f | ||
|
1dac3b9598 | ||
|
c77ac5eed2 | ||
dc498edc1b | |||
326eceddc5 | |||
|
071a135a70 | ||
336b128033 | |||
4579965c49 | |||
944294f19f | |||
|
2a6371c797 | ||
|
629a3fee48 | ||
6dc3d858d6 | |||
|
e7797e2e46 | ||
1a03f7e25f | |||
50d21479fd | |||
2f8021fe00 | |||
0571f46d75 | |||
13843fbcf4 | |||
|
d4d0604412 | ||
fe897ee428 | |||
|
d142fc6e91 | ||
2d918e9f5a | |||
|
6d80445682 | ||
bb514b5dc2 | |||
|
b79261650a | ||
62c424c47f | |||
|
078a9c9391 | ||
45a8351d5e | |||
|
4eb66bfe99 | ||
54aee07622 | |||
|
357455f07d | ||
9b86861918 | |||
|
5fa2e1bef0 | ||
4f7030af75 | |||
599c915ed2 | |||
cf0406a859 | |||
|
8fd9565d9f | ||
|
92bd35c6a2 | ||
|
23db1049a6 | ||
|
09b5fa42e1 | ||
|
0301fdddbc | ||
|
e00a5746b6 | ||
|
100aa3c791 | ||
|
aaebddde88 | ||
|
13077b0388 | ||
|
a9d4f8f4e0 | ||
|
24808c5145 | ||
|
aa6b07bb1c | ||
|
39264eceec | ||
|
22ab862254 | ||
|
4ec25842c1 | ||
|
9a88bafc09 | ||
|
c21748cfe1 | ||
|
ef3c770575 | ||
|
8c274abb26 | ||
|
56087a4f32 | ||
|
0fedf0ac17 | ||
|
689a1f2c86 | ||
|
00a83f67b5 | ||
0447949b54 | |||
|
f2b0531c7a | ||
f0c3617ad7 | |||
01a951c718 | |||
fa5e714cf9 | |||
|
0a907f22b7 | ||
|
1ca8970a93 | ||
8e03924ed9 | |||
8b608892e0 | |||
|
f15d93f16b | ||
bea52c961f |
24
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
24
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: For reporting bugs in go-micro
|
||||||
|
title: "[BUG]"
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
|
||||||
|
1. What are you trying to do?
|
||||||
|
2. What did you expect to happen?
|
||||||
|
3. What happens instead?
|
||||||
|
|
||||||
|
**How to reproduce the bug:**
|
||||||
|
|
||||||
|
If possible, please include a minimal code snippet here.
|
||||||
|
|
||||||
|
**Environment:**
|
||||||
|
Go Version: please paste `go version` output here
|
||||||
|
```
|
||||||
|
please paste `go env` output here
|
||||||
|
```
|
17
.github/ISSUE_TEMPLATE/feature-request---enhancement.md
vendored
Normal file
17
.github/ISSUE_TEMPLATE/feature-request---enhancement.md
vendored
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
---
|
||||||
|
name: Feature request / Enhancement
|
||||||
|
about: If you have a need not served by go-micro
|
||||||
|
title: "[FEATURE]"
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
14
.github/ISSUE_TEMPLATE/question.md
vendored
Normal file
14
.github/ISSUE_TEMPLATE/question.md
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
---
|
||||||
|
name: Question
|
||||||
|
about: Ask a question about go-micro
|
||||||
|
title: ''
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Before asking, please check if your question has already been answered:
|
||||||
|
|
||||||
|
1. Check the documentation - https://micro.mu/docs/
|
||||||
|
2. Check the examples and plugins - https://github.com/micro/examples & https://github.com/micro/go-plugins
|
||||||
|
3. Search existing issues
|
9
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
9
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
## Pull Request template
|
||||||
|
Please, go through these steps before clicking submit on this PR.
|
||||||
|
|
||||||
|
1. Give a descriptive title to your PR.
|
||||||
|
2. Provide a description of your changes.
|
||||||
|
3. Make sure you have some relevant tests.
|
||||||
|
4. Put `closes #XXXX` in your comment to auto-close the issue that your PR fixes (if applicable).
|
||||||
|
|
||||||
|
**PLEASE REMOVE THIS TEMPLATE BEFORE SUBMITTING**
|
19
.github/dependabot.yml
vendored
Normal file
19
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# To get started with Dependabot version updates, you'll need to specify which
|
||||||
|
# package ecosystems to update and where the package manifests are located.
|
||||||
|
# Please see the documentation for all configuration options:
|
||||||
|
# https://help.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
|
||||||
|
# Maintain dependencies for GitHub Actions
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "daily"
|
||||||
|
|
||||||
|
# Maintain dependencies for Golang
|
||||||
|
- package-ecosystem: "gomod"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "daily"
|
20
.github/workflows/autoapprove.yml
vendored
Normal file
20
.github/workflows/autoapprove.yml
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
name: "autoapprove"
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_target:
|
||||||
|
types: [assigned, opened, synchronize, reopened]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
pull-requests: write
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
autoapprove:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: approve
|
||||||
|
uses: hmarr/auto-approve-action@v3
|
||||||
|
if: github.actor == 'vtolstov' || github.actor == 'dependabot[bot]'
|
||||||
|
id: approve
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
21
.github/workflows/automerge.yml
vendored
Normal file
21
.github/workflows/automerge.yml
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
name: "automerge"
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_target:
|
||||||
|
types: [assigned, opened, synchronize, reopened]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
pull-requests: write
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
automerge:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.actor == 'vtolstov'
|
||||||
|
steps:
|
||||||
|
- name: merge
|
||||||
|
id: merge
|
||||||
|
run: gh pr merge --auto --merge "$PR_URL"
|
||||||
|
env:
|
||||||
|
PR_URL: ${{github.event.pull_request.html_url}}
|
||||||
|
GITHUB_TOKEN: ${{secrets.TOKEN}}
|
47
.github/workflows/build.yml
vendored
Normal file
47
.github/workflows/build.yml
vendored
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
name: build
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
- v3
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: test
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: setup
|
||||||
|
uses: actions/setup-go@v3
|
||||||
|
with:
|
||||||
|
go-version: 1.17
|
||||||
|
- name: checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/go/pkg/mod
|
||||||
|
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||||
|
restore-keys: ${{ runner.os }}-go-
|
||||||
|
- name: deps
|
||||||
|
run: go get -v -t -d ./...
|
||||||
|
- name: test
|
||||||
|
env:
|
||||||
|
INTEGRATION_TESTS: yes
|
||||||
|
run: go test -mod readonly -v ./...
|
||||||
|
lint:
|
||||||
|
name: lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: lint
|
||||||
|
uses: golangci/golangci-lint-action@v3.4.0
|
||||||
|
continue-on-error: true
|
||||||
|
with:
|
||||||
|
# Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
|
||||||
|
version: v1.30
|
||||||
|
# Optional: working directory, useful for monorepos
|
||||||
|
# working-directory: somedir
|
||||||
|
# Optional: golangci-lint command line arguments.
|
||||||
|
# args: --issues-exit-code=0
|
||||||
|
# Optional: show only new issues if it's a pull request. The default value is `false`.
|
||||||
|
# only-new-issues: true
|
78
.github/workflows/codeql-analysis.yml
vendored
Normal file
78
.github/workflows/codeql-analysis.yml
vendored
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
# For most projects, this workflow file will not need changing; you simply need
|
||||||
|
# to commit it to your repository.
|
||||||
|
#
|
||||||
|
# You may wish to alter this file to override the set of languages analyzed,
|
||||||
|
# or to provide custom queries or build logic.
|
||||||
|
#
|
||||||
|
# ******** NOTE ********
|
||||||
|
# We have attempted to detect the languages in your repository. Please check
|
||||||
|
# the `language` matrix defined below to confirm you have the correct set of
|
||||||
|
# supported CodeQL languages.
|
||||||
|
#
|
||||||
|
name: "codeql"
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_run:
|
||||||
|
workflows: ["prbuild"]
|
||||||
|
types:
|
||||||
|
- completed
|
||||||
|
push:
|
||||||
|
branches: [ master, v3 ]
|
||||||
|
pull_request:
|
||||||
|
# The branches below must be a subset of the branches above
|
||||||
|
branches: [ master, v3 ]
|
||||||
|
schedule:
|
||||||
|
- cron: '34 1 * * 0'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
analyze:
|
||||||
|
name: analyze
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
actions: read
|
||||||
|
contents: read
|
||||||
|
security-events: write
|
||||||
|
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
language: [ 'go' ]
|
||||||
|
# CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python' ]
|
||||||
|
# Learn more:
|
||||||
|
# https://docs.github.com/en/free-pro-team@latest/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#changing-the-languages-that-are-analyzed
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: setup
|
||||||
|
uses: actions/setup-go@v3
|
||||||
|
with:
|
||||||
|
go-version: 1.17
|
||||||
|
# Initializes the CodeQL tools for scanning.
|
||||||
|
- name: init
|
||||||
|
uses: github/codeql-action/init@v2
|
||||||
|
with:
|
||||||
|
languages: ${{ matrix.language }}
|
||||||
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
|
# By default, queries listed here will override any specified in a config file.
|
||||||
|
# Prefix the list here with "+" to use these queries and those in the config file.
|
||||||
|
# queries: ./path/to/local/query, your-org/your-repo/queries@main
|
||||||
|
|
||||||
|
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||||
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
|
- name: autobuild
|
||||||
|
uses: github/codeql-action/autobuild@v2
|
||||||
|
|
||||||
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
|
# 📚 https://git.io/JvXDl
|
||||||
|
|
||||||
|
# ✏️ If the Autobuild fails above, remove it and uncomment the following three lines
|
||||||
|
# and modify them (or add more) to build your code if your project
|
||||||
|
# uses a compiled language
|
||||||
|
|
||||||
|
#- run: |
|
||||||
|
# make bootstrap
|
||||||
|
# make release
|
||||||
|
|
||||||
|
- name: analyze
|
||||||
|
uses: github/codeql-action/analyze@v2
|
27
.github/workflows/dependabot-automerge.yml
vendored
Normal file
27
.github/workflows/dependabot-automerge.yml
vendored
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
name: "dependabot-automerge"
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_target:
|
||||||
|
types: [assigned, opened, synchronize, reopened]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
pull-requests: write
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
automerge:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.actor == 'dependabot[bot]'
|
||||||
|
steps:
|
||||||
|
- name: metadata
|
||||||
|
id: metadata
|
||||||
|
uses: dependabot/fetch-metadata@v1.3.6
|
||||||
|
with:
|
||||||
|
github-token: "${{ secrets.TOKEN }}"
|
||||||
|
- name: merge
|
||||||
|
id: merge
|
||||||
|
if: ${{contains(steps.metadata.outputs.dependency-names, 'go.unistack.org')}}
|
||||||
|
run: gh pr merge --auto --merge "$PR_URL"
|
||||||
|
env:
|
||||||
|
PR_URL: ${{github.event.pull_request.html_url}}
|
||||||
|
GITHUB_TOKEN: ${{secrets.TOKEN}}
|
47
.github/workflows/pr.yml
vendored
Normal file
47
.github/workflows/pr.yml
vendored
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
name: prbuild
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
- v3
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: test
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: setup
|
||||||
|
uses: actions/setup-go@v3
|
||||||
|
with:
|
||||||
|
go-version: 1.17
|
||||||
|
- name: checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: cache
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/go/pkg/mod
|
||||||
|
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||||
|
restore-keys: ${{ runner.os }}-go-
|
||||||
|
- name: deps
|
||||||
|
run: go get -v -t -d ./...
|
||||||
|
- name: test
|
||||||
|
env:
|
||||||
|
INTEGRATION_TESTS: yes
|
||||||
|
run: go test -mod readonly -v ./...
|
||||||
|
lint:
|
||||||
|
name: lint
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: lint
|
||||||
|
uses: golangci/golangci-lint-action@v3.4.0
|
||||||
|
continue-on-error: true
|
||||||
|
with:
|
||||||
|
# Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
|
||||||
|
version: v1.30
|
||||||
|
# Optional: working directory, useful for monorepos
|
||||||
|
# working-directory: somedir
|
||||||
|
# Optional: golangci-lint command line arguments.
|
||||||
|
# args: --issues-exit-code=0
|
||||||
|
# Optional: show only new issues if it's a pull request. The default value is `false`.
|
||||||
|
# only-new-issues: true
|
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.idea
|
201
LICENSE
Normal file
201
LICENSE
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
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.
|
23
README.md
23
README.md
@@ -1,23 +0,0 @@
|
|||||||
# Prometheus
|
|
||||||
|
|
||||||
Wrappers are a form of middleware that can be used with go-micro services. They can wrap both the Client and Server handlers.
|
|
||||||
This plugin implements the HandlerWrapper interface to provide automatic prometheus metric handling
|
|
||||||
for each microservice method execution time and operation count for success and failed cases.
|
|
||||||
|
|
||||||
This handler will export two metrics to prometheus:
|
|
||||||
* **micro_request_total**. How many go-micro requests processed, partitioned by method and status.
|
|
||||||
* **micro_request_duration_microseconds**. Service method request latencies in microseconds, partitioned by method.
|
|
||||||
|
|
||||||
# Usage
|
|
||||||
|
|
||||||
When creating your service, add the wrapper like so.
|
|
||||||
|
|
||||||
```go
|
|
||||||
service := micro.NewService(
|
|
||||||
micro.Name("service name"),
|
|
||||||
micro.Version("latest"),
|
|
||||||
micro.WrapHandler(prometheus.NewHandlerWrapper()),
|
|
||||||
)
|
|
||||||
|
|
||||||
service.Init()
|
|
||||||
```
|
|
326
go.mod
326
go.mod
@@ -1,12 +1,324 @@
|
|||||||
module github.com/micro/go-plugins/wrapper/monitoring/prometheus/v2
|
module go.unistack.org/micro-meter-prometheus/v3
|
||||||
|
|
||||||
go 1.13
|
go 1.21
|
||||||
|
|
||||||
|
toolchain go1.22.0
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/micro/go-micro/v2 v2.9.1-0.20200716153311-f9bf56239306
|
github.com/prometheus/client_golang v1.19.0
|
||||||
github.com/prometheus/client_golang v1.5.1
|
github.com/prometheus/client_model v0.6.0
|
||||||
github.com/prometheus/client_model v0.2.0
|
github.com/prometheus/common v0.51.1
|
||||||
github.com/stretchr/testify v1.4.0
|
go.unistack.org/micro/v3 v3.10.53
|
||||||
)
|
)
|
||||||
|
|
||||||
replace github.com/coreos/etcd => github.com/ozonru/etcd v3.3.20-grpc1.27-origmodule+incompatible
|
require (
|
||||||
|
cloud.google.com/go v0.110.2 // indirect
|
||||||
|
cloud.google.com/go/accessapproval v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/accesscontextmanager v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/aiplatform v1.37.0 // indirect
|
||||||
|
cloud.google.com/go/analytics v0.19.0 // indirect
|
||||||
|
cloud.google.com/go/apigateway v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/apigeeconnect v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/apigeeregistry v0.6.0 // indirect
|
||||||
|
cloud.google.com/go/apikeys v0.6.0 // indirect
|
||||||
|
cloud.google.com/go/appengine v1.7.1 // indirect
|
||||||
|
cloud.google.com/go/area120 v0.7.1 // indirect
|
||||||
|
cloud.google.com/go/artifactregistry v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/asset v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/assuredworkloads v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/automl v1.12.0 // indirect
|
||||||
|
cloud.google.com/go/baremetalsolution v0.5.0 // indirect
|
||||||
|
cloud.google.com/go/batch v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/beyondcorp v0.5.0 // indirect
|
||||||
|
cloud.google.com/go/bigquery v1.50.0 // indirect
|
||||||
|
cloud.google.com/go/billing v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/binaryauthorization v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/certificatemanager v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/channel v1.12.0 // indirect
|
||||||
|
cloud.google.com/go/cloudbuild v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/clouddms v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/cloudtasks v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/compute v1.20.1 // indirect
|
||||||
|
cloud.google.com/go/compute/metadata v0.2.3 // indirect
|
||||||
|
cloud.google.com/go/contactcenterinsights v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/container v1.15.0 // indirect
|
||||||
|
cloud.google.com/go/containeranalysis v0.9.0 // indirect
|
||||||
|
cloud.google.com/go/datacatalog v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/dataflow v0.8.0 // indirect
|
||||||
|
cloud.google.com/go/dataform v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/datafusion v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/datalabeling v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/dataplex v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/dataproc v1.12.0 // indirect
|
||||||
|
cloud.google.com/go/dataqna v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/datastore v1.11.0 // indirect
|
||||||
|
cloud.google.com/go/datastream v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/deploy v1.8.0 // indirect
|
||||||
|
cloud.google.com/go/dialogflow v1.32.0 // indirect
|
||||||
|
cloud.google.com/go/dlp v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/documentai v1.18.0 // indirect
|
||||||
|
cloud.google.com/go/domains v0.8.0 // indirect
|
||||||
|
cloud.google.com/go/edgecontainer v1.0.0 // indirect
|
||||||
|
cloud.google.com/go/errorreporting v0.3.0 // indirect
|
||||||
|
cloud.google.com/go/essentialcontacts v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/eventarc v1.11.0 // indirect
|
||||||
|
cloud.google.com/go/filestore v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/firestore v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/functions v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/gaming v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/gkebackup v0.4.0 // indirect
|
||||||
|
cloud.google.com/go/gkeconnect v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/gkehub v0.12.0 // indirect
|
||||||
|
cloud.google.com/go/gkemulticloud v0.5.0 // indirect
|
||||||
|
cloud.google.com/go/grafeas v0.2.0 // indirect
|
||||||
|
cloud.google.com/go/gsuiteaddons v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/iam v0.13.0 // indirect
|
||||||
|
cloud.google.com/go/iap v1.7.1 // indirect
|
||||||
|
cloud.google.com/go/ids v1.3.0 // indirect
|
||||||
|
cloud.google.com/go/iot v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/kms v1.10.1 // indirect
|
||||||
|
cloud.google.com/go/language v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/lifesciences v0.8.0 // indirect
|
||||||
|
cloud.google.com/go/logging v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/longrunning v0.4.1 // indirect
|
||||||
|
cloud.google.com/go/managedidentities v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/maps v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/mediatranslation v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/memcache v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/metastore v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/monitoring v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/networkconnectivity v1.11.0 // indirect
|
||||||
|
cloud.google.com/go/networkmanagement v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/networksecurity v0.8.0 // indirect
|
||||||
|
cloud.google.com/go/notebooks v1.8.0 // indirect
|
||||||
|
cloud.google.com/go/optimization v1.3.1 // indirect
|
||||||
|
cloud.google.com/go/orchestration v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/orgpolicy v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/osconfig v1.11.0 // indirect
|
||||||
|
cloud.google.com/go/oslogin v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/phishingprotection v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/policytroubleshooter v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/privatecatalog v0.8.0 // indirect
|
||||||
|
cloud.google.com/go/pubsub v1.30.0 // indirect
|
||||||
|
cloud.google.com/go/pubsublite v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/recaptchaenterprise v1.3.1 // indirect
|
||||||
|
cloud.google.com/go/recaptchaenterprise/v2 v2.7.0 // indirect
|
||||||
|
cloud.google.com/go/recommendationengine v0.7.0 // indirect
|
||||||
|
cloud.google.com/go/recommender v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/redis v1.11.0 // indirect
|
||||||
|
cloud.google.com/go/resourcemanager v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/resourcesettings v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/retail v1.12.0 // indirect
|
||||||
|
cloud.google.com/go/run v0.9.0 // indirect
|
||||||
|
cloud.google.com/go/scheduler v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/secretmanager v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/security v1.13.0 // indirect
|
||||||
|
cloud.google.com/go/securitycenter v1.19.0 // indirect
|
||||||
|
cloud.google.com/go/servicecontrol v1.11.1 // indirect
|
||||||
|
cloud.google.com/go/servicedirectory v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/servicemanagement v1.8.0 // indirect
|
||||||
|
cloud.google.com/go/serviceusage v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/shell v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/spanner v1.45.0 // indirect
|
||||||
|
cloud.google.com/go/speech v1.15.0 // indirect
|
||||||
|
cloud.google.com/go/storage v1.29.0 // indirect
|
||||||
|
cloud.google.com/go/storagetransfer v1.8.0 // indirect
|
||||||
|
cloud.google.com/go/talent v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/texttospeech v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/tpu v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/trace v1.9.0 // indirect
|
||||||
|
cloud.google.com/go/translate v1.7.0 // indirect
|
||||||
|
cloud.google.com/go/video v1.15.0 // indirect
|
||||||
|
cloud.google.com/go/videointelligence v1.10.0 // indirect
|
||||||
|
cloud.google.com/go/vision v1.2.0 // indirect
|
||||||
|
cloud.google.com/go/vision/v2 v2.7.0 // indirect
|
||||||
|
cloud.google.com/go/vmmigration v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/vmwareengine v0.3.0 // indirect
|
||||||
|
cloud.google.com/go/vpcaccess v1.6.0 // indirect
|
||||||
|
cloud.google.com/go/webrisk v1.8.0 // indirect
|
||||||
|
cloud.google.com/go/websecurityscanner v1.5.0 // indirect
|
||||||
|
cloud.google.com/go/workflows v1.10.0 // indirect
|
||||||
|
dario.cat/mergo v1.0.0 // indirect
|
||||||
|
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9 // indirect
|
||||||
|
gioui.org v0.0.0-20210308172011-57750fc8a0a6 // indirect
|
||||||
|
git.sr.ht/~sbinet/gg v0.3.1 // indirect
|
||||||
|
github.com/BurntSushi/toml v0.3.1 // indirect
|
||||||
|
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802 // indirect
|
||||||
|
github.com/DATA-DOG/go-sqlmock v1.5.0 // indirect
|
||||||
|
github.com/JohnCGriffin/overflow v0.0.0-20211019200055-46fa312c352c // indirect
|
||||||
|
github.com/OneOfOne/xxhash v1.2.2 // indirect
|
||||||
|
github.com/ajstarks/deck v0.0.0-20200831202436-30c9fc6549a9 // indirect
|
||||||
|
github.com/ajstarks/deck/generate v0.0.0-20210309230005-c3f852c02e19 // indirect
|
||||||
|
github.com/ajstarks/svgo v0.0.0-20211024235047-1546f124cd8b // indirect
|
||||||
|
github.com/alecthomas/kingpin/v2 v2.4.0 // indirect
|
||||||
|
github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751 // indirect
|
||||||
|
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 // indirect
|
||||||
|
github.com/andybalholm/brotli v1.0.4 // indirect
|
||||||
|
github.com/antihax/optional v1.0.0 // indirect
|
||||||
|
github.com/apache/arrow/go/v10 v10.0.1 // indirect
|
||||||
|
github.com/apache/arrow/go/v11 v11.0.0 // indirect
|
||||||
|
github.com/apache/thrift v0.16.0 // indirect
|
||||||
|
github.com/beorn7/perks v1.0.1 // indirect
|
||||||
|
github.com/boombuler/barcode v1.0.1 // indirect
|
||||||
|
github.com/census-instrumentation/opencensus-proto v0.4.1 // indirect
|
||||||
|
github.com/cespare/xxhash v1.1.0 // indirect
|
||||||
|
github.com/cespare/xxhash/v2 v2.2.0 // indirect
|
||||||
|
github.com/chzyer/logex v1.1.10 // indirect
|
||||||
|
github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e // indirect
|
||||||
|
github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1 // indirect
|
||||||
|
github.com/client9/misspell v0.3.4 // indirect
|
||||||
|
github.com/cncf/udpa/go v0.0.0-20220112060539-c52dc94e7fbe // indirect
|
||||||
|
github.com/cncf/xds/go v0.0.0-20230607035331-e9ce68804cb4 // indirect
|
||||||
|
github.com/creack/pty v1.1.9 // indirect
|
||||||
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
|
github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815 // indirect
|
||||||
|
github.com/dustin/go-humanize v1.0.0 // indirect
|
||||||
|
github.com/envoyproxy/go-control-plane v0.11.1-0.20230524094728-9239064ad72f // indirect
|
||||||
|
github.com/envoyproxy/protoc-gen-validate v0.10.1 // indirect
|
||||||
|
github.com/fogleman/gg v1.3.0 // indirect
|
||||||
|
github.com/ghodss/yaml v1.0.0 // indirect
|
||||||
|
github.com/go-fonts/dejavu v0.1.0 // indirect
|
||||||
|
github.com/go-fonts/latin-modern v0.2.0 // indirect
|
||||||
|
github.com/go-fonts/liberation v0.2.0 // indirect
|
||||||
|
github.com/go-fonts/stix v0.1.0 // indirect
|
||||||
|
github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1 // indirect
|
||||||
|
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4 // indirect
|
||||||
|
github.com/go-kit/kit v0.9.0 // indirect
|
||||||
|
github.com/go-kit/log v0.2.1 // indirect
|
||||||
|
github.com/go-latex/latex v0.0.0-20210823091927-c0d11ff05a81 // indirect
|
||||||
|
github.com/go-logfmt/logfmt v0.5.1 // indirect
|
||||||
|
github.com/go-pdf/fpdf v0.6.0 // indirect
|
||||||
|
github.com/go-stack/stack v1.8.0 // indirect
|
||||||
|
github.com/goccy/go-json v0.9.11 // indirect
|
||||||
|
github.com/gogo/protobuf v1.1.1 // indirect
|
||||||
|
github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 // indirect
|
||||||
|
github.com/golang/glog v1.1.0 // indirect
|
||||||
|
github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect
|
||||||
|
github.com/golang/mock v1.6.0 // indirect
|
||||||
|
github.com/golang/protobuf v1.5.4 // indirect
|
||||||
|
github.com/golang/snappy v0.0.4 // indirect
|
||||||
|
github.com/google/btree v1.0.0 // indirect
|
||||||
|
github.com/google/flatbuffers v2.0.8+incompatible // indirect
|
||||||
|
github.com/google/go-cmp v0.6.0 // indirect
|
||||||
|
github.com/google/gofuzz v1.0.0 // indirect
|
||||||
|
github.com/google/martian v2.1.0+incompatible // indirect
|
||||||
|
github.com/google/martian/v3 v3.3.2 // indirect
|
||||||
|
github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1 // indirect
|
||||||
|
github.com/google/renameio v0.1.0 // indirect
|
||||||
|
github.com/google/s2a-go v0.1.4 // indirect
|
||||||
|
github.com/google/uuid v1.3.0 // indirect
|
||||||
|
github.com/googleapis/enterprise-certificate-proxy v0.2.3 // indirect
|
||||||
|
github.com/googleapis/gax-go/v2 v2.11.0 // indirect
|
||||||
|
github.com/googleapis/go-type-adapters v1.0.0 // indirect
|
||||||
|
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8 // indirect
|
||||||
|
github.com/grpc-ecosystem/grpc-gateway v1.16.0 // indirect
|
||||||
|
github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3 // indirect
|
||||||
|
github.com/hashicorp/golang-lru v0.5.1 // indirect
|
||||||
|
github.com/iancoleman/strcase v0.2.0 // indirect
|
||||||
|
github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639 // indirect
|
||||||
|
github.com/imdario/mergo v0.3.13 // indirect
|
||||||
|
github.com/jpillora/backoff v1.0.0 // indirect
|
||||||
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
|
github.com/jstemmer/go-junit-report v0.9.1 // indirect
|
||||||
|
github.com/julienschmidt/httprouter v1.3.0 // indirect
|
||||||
|
github.com/jung-kurt/gofpdf v1.0.3-0.20190309125859-24315acbbda5 // indirect
|
||||||
|
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect
|
||||||
|
github.com/kisielk/gotool v1.0.0 // indirect
|
||||||
|
github.com/klauspost/asmfmt v1.3.2 // indirect
|
||||||
|
github.com/klauspost/compress v1.15.9 // indirect
|
||||||
|
github.com/klauspost/cpuid/v2 v2.0.9 // indirect
|
||||||
|
github.com/konsorten/go-windows-terminal-sequences v1.0.3 // indirect
|
||||||
|
github.com/kr/fs v0.1.0 // indirect
|
||||||
|
github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515 // indirect
|
||||||
|
github.com/kr/pretty v0.3.1 // indirect
|
||||||
|
github.com/kr/pty v1.1.1 // indirect
|
||||||
|
github.com/kr/text v0.2.0 // indirect
|
||||||
|
github.com/lyft/protoc-gen-star v0.6.1 // indirect
|
||||||
|
github.com/lyft/protoc-gen-star/v2 v2.0.1 // indirect
|
||||||
|
github.com/mattn/go-isatty v0.0.16 // indirect
|
||||||
|
github.com/mattn/go-sqlite3 v1.14.14 // indirect
|
||||||
|
github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect
|
||||||
|
github.com/matttproud/golang_protobuf_extensions/v2 v2.0.0 // indirect
|
||||||
|
github.com/minio/asm2plan9s v0.0.0-20200509001527-cdd76441f9d8 // indirect
|
||||||
|
github.com/minio/c2goasm v0.0.0-20190812172519-36a3d3bbc4f3 // indirect
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
|
github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f // indirect
|
||||||
|
github.com/patrickmn/go-cache v2.1.0+incompatible // indirect
|
||||||
|
github.com/phpdave11/gofpdf v1.4.2 // indirect
|
||||||
|
github.com/phpdave11/gofpdi v1.0.13 // indirect
|
||||||
|
github.com/pierrec/lz4/v4 v4.1.15 // indirect
|
||||||
|
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e // indirect
|
||||||
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
|
github.com/pkg/sftp v1.13.1 // indirect
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
|
github.com/prometheus/procfs v0.13.0 // indirect
|
||||||
|
github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 // indirect
|
||||||
|
github.com/rogpeppe/fastuuid v1.2.0 // indirect
|
||||||
|
github.com/rogpeppe/go-internal v1.10.0 // indirect
|
||||||
|
github.com/ruudk/golang-pdf417 v0.0.0-20201230142125-a7e3863a1245 // indirect
|
||||||
|
github.com/silas/dag v0.0.0-20220518035006-a7e85ada93c5 // indirect
|
||||||
|
github.com/sirupsen/logrus v1.6.0 // indirect
|
||||||
|
github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72 // indirect
|
||||||
|
github.com/spf13/afero v1.9.2 // indirect
|
||||||
|
github.com/stretchr/objx v0.5.0 // indirect
|
||||||
|
github.com/stretchr/testify v1.8.3 // indirect
|
||||||
|
github.com/xhit/go-str2duration v1.2.0 // indirect
|
||||||
|
github.com/xhit/go-str2duration/v2 v2.1.0 // indirect
|
||||||
|
github.com/yuin/goldmark v1.4.13 // indirect
|
||||||
|
github.com/zeebo/assert v1.3.0 // indirect
|
||||||
|
github.com/zeebo/xxh3 v1.0.2 // indirect
|
||||||
|
go.opencensus.io v0.24.0 // indirect
|
||||||
|
go.opentelemetry.io/proto/otlp v0.19.0 // indirect
|
||||||
|
golang.org/x/crypto v0.21.0 // indirect
|
||||||
|
golang.org/x/exp v0.0.0-20220827204233-334a2380cb91 // indirect
|
||||||
|
golang.org/x/image v0.0.0-20220302094943-723b81ca9867 // indirect
|
||||||
|
golang.org/x/lint v0.0.0-20210508222113-6edffad5e616 // indirect
|
||||||
|
golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028 // indirect
|
||||||
|
golang.org/x/mod v0.9.0 // indirect
|
||||||
|
golang.org/x/net v0.22.0 // indirect
|
||||||
|
golang.org/x/oauth2 v0.18.0 // indirect
|
||||||
|
golang.org/x/sync v0.6.0 // indirect
|
||||||
|
golang.org/x/sys v0.18.0 // indirect
|
||||||
|
golang.org/x/term v0.18.0 // indirect
|
||||||
|
golang.org/x/text v0.14.0 // indirect
|
||||||
|
golang.org/x/time v0.3.0 // indirect
|
||||||
|
golang.org/x/tools v0.7.0 // indirect
|
||||||
|
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect
|
||||||
|
gonum.org/v1/gonum v0.11.0 // indirect
|
||||||
|
gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0 // indirect
|
||||||
|
gonum.org/v1/plot v0.10.1 // indirect
|
||||||
|
google.golang.org/api v0.126.0 // indirect
|
||||||
|
google.golang.org/appengine v1.6.7 // indirect
|
||||||
|
google.golang.org/genproto v0.0.0-20230530153820-e85fd2cbaebc // indirect
|
||||||
|
google.golang.org/genproto/googleapis/api v0.0.0-20230530153820-e85fd2cbaebc // indirect
|
||||||
|
google.golang.org/genproto/googleapis/bytestream v0.0.0-20230530153820-e85fd2cbaebc // indirect
|
||||||
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20230530153820-e85fd2cbaebc // indirect
|
||||||
|
google.golang.org/grpc v1.57.0 // indirect
|
||||||
|
google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0 // indirect
|
||||||
|
google.golang.org/protobuf v1.33.0 // indirect
|
||||||
|
gopkg.in/alecthomas/kingpin.v2 v2.2.6 // indirect
|
||||||
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect
|
||||||
|
gopkg.in/errgo.v2 v2.1.0 // indirect
|
||||||
|
gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
|
honnef.co/go/tools v0.1.3 // indirect
|
||||||
|
lukechampine.com/uint128 v1.2.0 // indirect
|
||||||
|
modernc.org/cc/v3 v3.36.3 // indirect
|
||||||
|
modernc.org/ccgo/v3 v3.16.9 // indirect
|
||||||
|
modernc.org/ccorpus v1.11.6 // indirect
|
||||||
|
modernc.org/httpfs v1.0.6 // indirect
|
||||||
|
modernc.org/libc v1.17.1 // indirect
|
||||||
|
modernc.org/mathutil v1.5.0 // indirect
|
||||||
|
modernc.org/memory v1.2.1 // indirect
|
||||||
|
modernc.org/opt v0.1.3 // indirect
|
||||||
|
modernc.org/sqlite v1.18.1 // indirect
|
||||||
|
modernc.org/strutil v1.1.3 // indirect
|
||||||
|
modernc.org/tcl v1.13.1 // indirect
|
||||||
|
modernc.org/token v1.0.0 // indirect
|
||||||
|
modernc.org/z v1.5.1 // indirect
|
||||||
|
rsc.io/binaryregexp v0.2.0 // indirect
|
||||||
|
rsc.io/pdf v0.1.1 // indirect
|
||||||
|
rsc.io/quote/v3 v3.1.0 // indirect
|
||||||
|
rsc.io/sampler v1.3.0 // indirect
|
||||||
|
)
|
||||||
|
725
prometheus.go
725
prometheus.go
@@ -1,310 +1,541 @@
|
|||||||
package prometheus
|
package prometheus
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"hash/fnv"
|
||||||
|
"io"
|
||||||
|
"regexp"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/micro/go-micro/v2/client"
|
|
||||||
"github.com/micro/go-micro/v2/logger"
|
|
||||||
"github.com/micro/go-micro/v2/registry"
|
|
||||||
"github.com/micro/go-micro/v2/server"
|
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
"github.com/prometheus/client_golang/prometheus/collectors"
|
||||||
|
dto "github.com/prometheus/client_model/go"
|
||||||
|
"github.com/prometheus/common/expfmt"
|
||||||
|
"go.unistack.org/micro/v3/meter"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var _ meter.Meter = &prometheusMeter{}
|
||||||
// default metric prefix
|
|
||||||
DefaultMetricPrefix = "micro_"
|
|
||||||
// default label prefix
|
|
||||||
DefaultLabelPrefix = "micro_"
|
|
||||||
|
|
||||||
opsCounter *prometheus.CounterVec
|
type prometheusMeter struct {
|
||||||
timeCounterSummary *prometheus.SummaryVec
|
opts meter.Options
|
||||||
timeCounterHistogram *prometheus.HistogramVec
|
set prometheus.Registerer
|
||||||
|
counter map[string]*counters
|
||||||
mu sync.Mutex
|
floatCounter map[string]*floatCounters
|
||||||
)
|
gauge map[string]*gauges
|
||||||
|
histogram map[string]*histograms
|
||||||
type Options struct {
|
summary map[string]*summaries
|
||||||
Name string
|
sync.Mutex
|
||||||
Version string
|
|
||||||
ID string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type Option func(*Options)
|
type counters struct {
|
||||||
|
cs map[uint64]*prometheusCounter
|
||||||
|
}
|
||||||
|
|
||||||
func ServiceName(name string) Option {
|
type gauges struct {
|
||||||
return func(opts *Options) {
|
cs map[uint64]*prometheusGauge
|
||||||
opts.Name = name
|
}
|
||||||
|
|
||||||
|
type histograms struct {
|
||||||
|
cs map[uint64]*prometheusHistogram
|
||||||
|
}
|
||||||
|
|
||||||
|
type summaries struct {
|
||||||
|
cs map[uint64]*prometheusSummary
|
||||||
|
}
|
||||||
|
|
||||||
|
type floatCounters struct {
|
||||||
|
cs map[uint64]*prometheusFloatCounter
|
||||||
|
}
|
||||||
|
|
||||||
|
func newFloat64(v float64) *float64 {
|
||||||
|
nv := v
|
||||||
|
return &nv
|
||||||
|
}
|
||||||
|
|
||||||
|
func newString(v string) *string {
|
||||||
|
nv := v
|
||||||
|
return &nv
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMeter(opts ...meter.Option) *prometheusMeter {
|
||||||
|
return &prometheusMeter{
|
||||||
|
set: prometheus.NewRegistry(), // prometheus.DefaultRegisterer,
|
||||||
|
opts: meter.NewOptions(opts...),
|
||||||
|
counter: make(map[string]*counters),
|
||||||
|
floatCounter: make(map[string]*floatCounters),
|
||||||
|
gauge: make(map[string]*gauges),
|
||||||
|
histogram: make(map[string]*histograms),
|
||||||
|
summary: make(map[string]*summaries),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ServiceVersion(version string) Option {
|
func (m *prometheusMeter) buildMetric(name string, labels ...string) string {
|
||||||
return func(opts *Options) {
|
if len(m.opts.MetricPrefix) > 0 {
|
||||||
opts.Version = version
|
name = m.opts.MetricPrefix + name
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nl := len(m.opts.Labels) + len(labels)
|
||||||
|
if nl == 0 {
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
nlabels := make([]string, 0, nl)
|
||||||
|
nlabels = append(nlabels, m.opts.Labels...)
|
||||||
|
nlabels = append(nlabels, labels...)
|
||||||
|
|
||||||
|
if len(m.opts.LabelPrefix) == 0 {
|
||||||
|
return meter.BuildName(name, nlabels...)
|
||||||
|
}
|
||||||
|
|
||||||
|
for idx := 0; idx < nl; idx += 2 {
|
||||||
|
nlabels[idx] = m.opts.LabelPrefix + nlabels[idx]
|
||||||
|
}
|
||||||
|
return meter.BuildName(name, nlabels...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ServiceID(id string) Option {
|
func (m *prometheusMeter) buildName(name string) string {
|
||||||
return func(opts *Options) {
|
if len(m.opts.MetricPrefix) > 0 {
|
||||||
opts.ID = id
|
name = m.opts.MetricPrefix + name
|
||||||
}
|
}
|
||||||
|
return name
|
||||||
}
|
}
|
||||||
|
|
||||||
func registerMetrics() {
|
func (m *prometheusMeter) buildLabels(labels ...string) []string {
|
||||||
mu.Lock()
|
nl := len(labels)
|
||||||
defer mu.Unlock()
|
if nl == 0 {
|
||||||
|
return nil
|
||||||
if opsCounter == nil {
|
|
||||||
opsCounter = prometheus.NewCounterVec(
|
|
||||||
prometheus.CounterOpts{
|
|
||||||
Name: fmt.Sprintf("%srequest_total", DefaultMetricPrefix),
|
|
||||||
Help: "Requests processed, partitioned by endpoint and status",
|
|
||||||
},
|
|
||||||
[]string{
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "name"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "version"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "id"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "endpoint"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "status"),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if timeCounterSummary == nil {
|
nlabels := make([]string, 0, nl)
|
||||||
timeCounterSummary = prometheus.NewSummaryVec(
|
|
||||||
prometheus.SummaryOpts{
|
|
||||||
Name: fmt.Sprintf("%slatency_microseconds", DefaultMetricPrefix),
|
|
||||||
Help: "Request latencies in microseconds, partitioned by endpoint",
|
|
||||||
},
|
|
||||||
[]string{
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "name"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "version"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "id"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "endpoint"),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if timeCounterHistogram == nil {
|
for idx := 0; idx < nl; idx += 2 {
|
||||||
timeCounterHistogram = prometheus.NewHistogramVec(
|
nlabels = append(nlabels, m.opts.LabelPrefix+labels[idx])
|
||||||
prometheus.HistogramOpts{
|
nlabels = append(nlabels, labels[idx+1])
|
||||||
Name: fmt.Sprintf("%srequest_duration_seconds", DefaultMetricPrefix),
|
|
||||||
Help: "Request time in seconds, partitioned by endpoint",
|
|
||||||
},
|
|
||||||
[]string{
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "name"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "version"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "id"),
|
|
||||||
fmt.Sprintf("%s%s", DefaultLabelPrefix, "endpoint"),
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
return nlabels
|
||||||
for _, collector := range []prometheus.Collector{opsCounter, timeCounterSummary, timeCounterHistogram} {
|
|
||||||
if err := prometheus.DefaultRegisterer.Register(collector); err != nil {
|
|
||||||
// if already registered, skip fatal
|
|
||||||
if _, ok := err.(prometheus.AlreadyRegisteredError); !ok {
|
|
||||||
logger.Fatal(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type wrapper struct {
|
func (m *prometheusMeter) Name() string {
|
||||||
options Options
|
return m.opts.Name
|
||||||
callFunc client.CallFunc
|
|
||||||
client.Client
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClientWrapper(opts ...Option) client.Wrapper {
|
func (m *prometheusMeter) Counter(name string, labels ...string) meter.Counter {
|
||||||
registerMetrics()
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
options := Options{}
|
nm := m.buildName(name)
|
||||||
for _, opt := range opts {
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
opt(&options)
|
cd, ok := m.counter[nm]
|
||||||
|
h := newHash(labels)
|
||||||
|
if !ok {
|
||||||
|
cd = &counters{cs: make(map[uint64]*prometheusCounter)}
|
||||||
|
c := &prometheusCounter{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.counter[nm] = cd
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return func(c client.Client) client.Client {
|
if !ok {
|
||||||
handler := &wrapper{
|
c = &prometheusCounter{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
options: options,
|
cd.cs[h] = c
|
||||||
Client: c,
|
m.counter[nm] = cd
|
||||||
}
|
|
||||||
|
|
||||||
return handler
|
|
||||||
}
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewCallWrapper(opts ...Option) client.CallWrapper {
|
func (m *prometheusMeter) FloatCounter(name string, labels ...string) meter.FloatCounter {
|
||||||
registerMetrics()
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
options := Options{}
|
nm := m.buildName(name)
|
||||||
for _, opt := range opts {
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
opt(&options)
|
cd, ok := m.floatCounter[nm]
|
||||||
|
h := newHash(labels)
|
||||||
|
if !ok {
|
||||||
|
cd = &floatCounters{cs: make(map[uint64]*prometheusFloatCounter)}
|
||||||
|
c := &prometheusFloatCounter{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.floatCounter[nm] = cd
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return func(fn client.CallFunc) client.CallFunc {
|
if !ok {
|
||||||
handler := &wrapper{
|
c = &prometheusFloatCounter{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
options: options,
|
cd.cs[h] = c
|
||||||
callFunc: fn,
|
m.floatCounter[nm] = cd
|
||||||
}
|
|
||||||
|
|
||||||
return handler.CallFunc
|
|
||||||
}
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *wrapper) CallFunc(ctx context.Context, node *registry.Node, req client.Request, rsp interface{}, opts client.CallOptions) error {
|
func (m *prometheusMeter) Gauge(name string, fn func() float64, labels ...string) meter.Gauge {
|
||||||
endpoint := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
nm := m.buildName(name)
|
||||||
us := v * 1000000 // make microseconds
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
cd, ok := m.gauge[nm]
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
h := newHash(labels)
|
||||||
}))
|
if !ok {
|
||||||
defer timer.ObserveDuration()
|
cd = &gauges{cs: make(map[uint64]*prometheusGauge)}
|
||||||
|
c := &prometheusGauge{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
err := w.callFunc(ctx, node, req, rsp, opts)
|
cd.cs[h] = c
|
||||||
if err == nil {
|
m.gauge[nm] = cd
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
return c
|
||||||
} else {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return err
|
if !ok {
|
||||||
|
c = &prometheusGauge{c: prometheus.NewGauge(prometheus.GaugeOpts{Name: nm}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.gauge[nm] = cd
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *wrapper) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
|
func (m *prometheusMeter) Histogram(name string, labels ...string) meter.Histogram {
|
||||||
endpoint := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
nm := m.buildName(name)
|
||||||
us := v * 1000000 // make microseconds
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
cd, ok := m.histogram[nm]
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
h := newHash(labels)
|
||||||
}))
|
if !ok {
|
||||||
defer timer.ObserveDuration()
|
cd = &histograms{cs: make(map[uint64]*prometheusHistogram)}
|
||||||
|
c := &prometheusHistogram{c: prometheus.NewHistogram(prometheus.HistogramOpts{Name: nm}), labels: labels}
|
||||||
err := w.Client.Call(ctx, req, rsp, opts...)
|
cd.cs[h] = c
|
||||||
if err == nil {
|
m.histogram[nm] = cd
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
return c
|
||||||
} else {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return err
|
if !ok {
|
||||||
|
c = &prometheusHistogram{c: prometheus.NewHistogram(prometheus.HistogramOpts{Name: nm}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.histogram[nm] = cd
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *wrapper) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
|
func (m *prometheusMeter) Summary(name string, labels ...string) meter.Summary {
|
||||||
endpoint := fmt.Sprintf("%s.%s", req.Service(), req.Endpoint())
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
nm := m.buildName(name)
|
||||||
us := v * 1000000 // make microseconds
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
cd, ok := m.summary[nm]
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
h := newHash(labels)
|
||||||
}))
|
if !ok {
|
||||||
defer timer.ObserveDuration()
|
cd = &summaries{cs: make(map[uint64]*prometheusSummary)}
|
||||||
|
c := &prometheusSummary{c: prometheus.NewSummary(prometheus.SummaryOpts{Name: nm}), labels: labels}
|
||||||
stream, err := w.Client.Stream(ctx, req, opts...)
|
cd.cs[h] = c
|
||||||
if err == nil {
|
m.summary[nm] = cd
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
return c
|
||||||
} else {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return stream, err
|
if !ok {
|
||||||
|
c = &prometheusSummary{c: prometheus.NewSummary(prometheus.SummaryOpts{Name: nm}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.summary[nm] = cd
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *wrapper) Publish(ctx context.Context, p client.Message, opts ...client.PublishOption) error {
|
func (m *prometheusMeter) SummaryExt(name string, window time.Duration, quantiles []float64, labels ...string) meter.Summary {
|
||||||
endpoint := p.Topic()
|
m.Lock()
|
||||||
|
defer m.Unlock()
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
nm := m.buildName(name)
|
||||||
us := v * 1000000 // make microseconds
|
labels = m.buildLabels(append(m.opts.Labels, labels...)...)
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
cd, ok := m.summary[nm]
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
h := newHash(labels)
|
||||||
}))
|
if !ok {
|
||||||
defer timer.ObserveDuration()
|
cd = &summaries{cs: make(map[uint64]*prometheusSummary)}
|
||||||
|
c := &prometheusSummary{c: prometheus.NewSummary(prometheus.SummaryOpts{
|
||||||
err := w.Client.Publish(ctx, p, opts...)
|
Name: nm,
|
||||||
if err == nil {
|
MaxAge: window,
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
Objectives: map[float64]float64{0.5: 0.05, 0.9: 0.01, 0.99: 0.001},
|
||||||
} else {
|
}), labels: labels}
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
cd.cs[h] = c
|
||||||
|
m.summary[nm] = cd
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
c, ok := cd.cs[h]
|
||||||
return err
|
if !ok {
|
||||||
|
c = &prometheusSummary{c: prometheus.NewSummary(prometheus.SummaryOpts{
|
||||||
|
Name: nm,
|
||||||
|
MaxAge: window,
|
||||||
|
Objectives: map[float64]float64{0.5: 0.05, 0.9: 0.01, 0.99: 0.001},
|
||||||
|
}), labels: labels}
|
||||||
|
cd.cs[h] = c
|
||||||
|
m.summary[nm] = cd
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHandlerWrapper(opts ...Option) server.HandlerWrapper {
|
func (m *prometheusMeter) Init(opts ...meter.Option) error {
|
||||||
registerMetrics()
|
for _, o := range opts {
|
||||||
|
o(&m.opts)
|
||||||
options := Options{}
|
|
||||||
for _, opt := range opts {
|
|
||||||
opt(&options)
|
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
handler := &wrapper{
|
|
||||||
options: options,
|
|
||||||
}
|
|
||||||
|
|
||||||
return handler.HandlerFunc
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *wrapper) HandlerFunc(fn server.HandlerFunc) server.HandlerFunc {
|
func (m *prometheusMeter) Write(w io.Writer, opts ...meter.Option) error {
|
||||||
return func(ctx context.Context, req server.Request, rsp interface{}) error {
|
options := m.opts
|
||||||
endpoint := req.Endpoint()
|
for _, o := range opts {
|
||||||
|
o(&options)
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
|
||||||
us := v * 1000000 // make microseconds
|
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
|
||||||
}))
|
|
||||||
defer timer.ObserveDuration()
|
|
||||||
|
|
||||||
err := fn(ctx, req, rsp)
|
|
||||||
if err == nil {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
|
||||||
} else {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if options.WriteProcessMetrics || options.WriteFDMetrics {
|
||||||
|
pc := collectors.NewProcessCollector(collectors.ProcessCollectorOpts{})
|
||||||
|
_ = m.set.Register(pc)
|
||||||
|
gc := collectors.NewGoCollector(collectors.WithGoCollectorRuntimeMetrics(collectors.GoRuntimeMetricsRule{Matcher: regexp.MustCompile("/.*")}))
|
||||||
|
_ = m.set.Register(gc)
|
||||||
|
}
|
||||||
|
|
||||||
|
g, ok := m.set.(prometheus.Gatherer)
|
||||||
|
if !ok {
|
||||||
|
return fmt.Errorf("set type %T not prometheus.Gatherer", m.set)
|
||||||
|
}
|
||||||
|
|
||||||
|
mfs, err := g.Gather()
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enc := expfmt.NewEncoder(w, expfmt.NewFormat(expfmt.TypeTextPlain))
|
||||||
|
|
||||||
|
for name, metrics := range m.counter {
|
||||||
|
mf := &dto.MetricFamily{
|
||||||
|
Name: newString(name),
|
||||||
|
Type: dto.MetricType_GAUGE.Enum(),
|
||||||
|
Metric: make([]*dto.Metric, 0, len(metrics.cs)),
|
||||||
|
}
|
||||||
|
for _, c := range metrics.cs {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
_ = c.c.Write(m)
|
||||||
|
fillMetric(m, c.labels)
|
||||||
|
mf.Metric = append(mf.Metric, m)
|
||||||
|
}
|
||||||
|
mfs = append(mfs, mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
for name, metrics := range m.gauge {
|
||||||
|
mf := &dto.MetricFamily{
|
||||||
|
Name: newString(name),
|
||||||
|
Type: dto.MetricType_GAUGE.Enum(),
|
||||||
|
Metric: make([]*dto.Metric, 0, len(metrics.cs)),
|
||||||
|
}
|
||||||
|
for _, c := range metrics.cs {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
_ = c.c.Write(m)
|
||||||
|
fillMetric(m, c.labels)
|
||||||
|
mf.Metric = append(mf.Metric, m)
|
||||||
|
}
|
||||||
|
mfs = append(mfs, mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
for name, metrics := range m.floatCounter {
|
||||||
|
mf := &dto.MetricFamily{
|
||||||
|
Name: newString(name),
|
||||||
|
Type: dto.MetricType_GAUGE.Enum(),
|
||||||
|
Metric: make([]*dto.Metric, 0, len(metrics.cs)),
|
||||||
|
}
|
||||||
|
for _, c := range metrics.cs {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
_ = c.c.Write(m)
|
||||||
|
fillMetric(m, c.labels)
|
||||||
|
mf.Metric = append(mf.Metric, m)
|
||||||
|
}
|
||||||
|
mfs = append(mfs, mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
for name, metrics := range m.histogram {
|
||||||
|
mf := &dto.MetricFamily{
|
||||||
|
Name: newString(name),
|
||||||
|
Type: dto.MetricType_HISTOGRAM.Enum(),
|
||||||
|
Metric: make([]*dto.Metric, 0, len(metrics.cs)),
|
||||||
|
}
|
||||||
|
for _, c := range metrics.cs {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
_ = c.c.Write(m)
|
||||||
|
fillMetric(m, c.labels)
|
||||||
|
mf.Metric = append(mf.Metric, m)
|
||||||
|
}
|
||||||
|
mfs = append(mfs, mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
for name, metrics := range m.summary {
|
||||||
|
mf := &dto.MetricFamily{
|
||||||
|
Name: newString(name),
|
||||||
|
Type: dto.MetricType_SUMMARY.Enum(),
|
||||||
|
Metric: make([]*dto.Metric, 0, len(metrics.cs)),
|
||||||
|
}
|
||||||
|
for _, c := range metrics.cs {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
_ = c.c.Write(m)
|
||||||
|
fillMetric(m, c.labels)
|
||||||
|
mf.Metric = append(mf.Metric, m)
|
||||||
|
}
|
||||||
|
mfs = append(mfs, mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, mf := range mfs {
|
||||||
|
_ = enc.Encode(mf)
|
||||||
|
}
|
||||||
|
|
||||||
|
if closer, ok := enc.(io.Closer); ok {
|
||||||
|
_ = closer.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSubscriberWrapper(opts ...Option) server.SubscriberWrapper {
|
func (m *prometheusMeter) Clone(opts ...meter.Option) meter.Meter {
|
||||||
registerMetrics()
|
options := m.opts
|
||||||
|
for _, o := range opts {
|
||||||
options := Options{}
|
o(&options)
|
||||||
for _, opt := range opts {
|
|
||||||
opt(&options)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
handler := &wrapper{
|
return &prometheusMeter{
|
||||||
options: options,
|
set: m.set,
|
||||||
}
|
opts: options,
|
||||||
|
floatCounter: m.floatCounter,
|
||||||
return handler.SubscriberFunc
|
counter: m.counter,
|
||||||
}
|
gauge: m.gauge,
|
||||||
|
histogram: m.histogram,
|
||||||
func (w *wrapper) SubscriberFunc(fn server.SubscriberFunc) server.SubscriberFunc {
|
summary: m.summary,
|
||||||
return func(ctx context.Context, msg server.Message) error {
|
|
||||||
endpoint := msg.Topic()
|
|
||||||
|
|
||||||
timer := prometheus.NewTimer(prometheus.ObserverFunc(func(v float64) {
|
|
||||||
us := v * 1000000 // make microseconds
|
|
||||||
timeCounterSummary.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(us)
|
|
||||||
timeCounterHistogram.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint).Observe(v)
|
|
||||||
}))
|
|
||||||
defer timer.ObserveDuration()
|
|
||||||
|
|
||||||
err := fn(ctx, msg)
|
|
||||||
if err == nil {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "success").Inc()
|
|
||||||
} else {
|
|
||||||
opsCounter.WithLabelValues(w.options.Name, w.options.Version, w.options.ID, endpoint, "failure").Inc()
|
|
||||||
}
|
|
||||||
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *prometheusMeter) Options() meter.Options {
|
||||||
|
return m.opts
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *prometheusMeter) String() string {
|
||||||
|
return "prometheus"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *prometheusMeter) Set(opts ...meter.Option) meter.Meter {
|
||||||
|
nm := &prometheusMeter{opts: m.opts}
|
||||||
|
for _, o := range opts {
|
||||||
|
o(&nm.opts)
|
||||||
|
}
|
||||||
|
nm.set = prometheus.NewRegistry()
|
||||||
|
return nm
|
||||||
|
}
|
||||||
|
|
||||||
|
type prometheusCounter struct {
|
||||||
|
c prometheus.Gauge
|
||||||
|
labels []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *prometheusCounter) Add(n int) {
|
||||||
|
c.c.Add(float64(n))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *prometheusCounter) Dec() {
|
||||||
|
c.c.Dec()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *prometheusCounter) Inc() {
|
||||||
|
c.c.Inc()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *prometheusCounter) Get() uint64 {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
if err := c.c.Write(m); err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return uint64(m.GetGauge().GetValue())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *prometheusCounter) Set(n uint64) {
|
||||||
|
c.c.Set(float64(n))
|
||||||
|
}
|
||||||
|
|
||||||
|
type prometheusFloatCounter struct {
|
||||||
|
c prometheus.Gauge
|
||||||
|
labels []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusFloatCounter) Add(n float64) {
|
||||||
|
c.c.Add(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusFloatCounter) Get() float64 {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
if err := c.c.Write(m); err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return m.GetGauge().GetValue()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusFloatCounter) Set(n float64) {
|
||||||
|
c.c.Set(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusFloatCounter) Sub(n float64) {
|
||||||
|
c.c.Add(-n)
|
||||||
|
}
|
||||||
|
|
||||||
|
type prometheusGauge struct {
|
||||||
|
c prometheus.Gauge
|
||||||
|
labels []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusGauge) Get() float64 {
|
||||||
|
m := &dto.Metric{}
|
||||||
|
if err := c.c.Write(m); err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return float64(m.GetGauge().GetValue())
|
||||||
|
}
|
||||||
|
|
||||||
|
type prometheusHistogram struct {
|
||||||
|
c prometheus.Histogram
|
||||||
|
labels []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusHistogram) Reset() {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusHistogram) Update(n float64) {
|
||||||
|
c.c.Observe(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusHistogram) UpdateDuration(n time.Time) {
|
||||||
|
c.c.Observe(time.Since(n).Seconds())
|
||||||
|
}
|
||||||
|
|
||||||
|
type prometheusSummary struct {
|
||||||
|
c prometheus.Summary
|
||||||
|
labels []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusSummary) Update(n float64) {
|
||||||
|
c.c.Observe(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c prometheusSummary) UpdateDuration(n time.Time) {
|
||||||
|
c.c.Observe(time.Since(n).Seconds())
|
||||||
|
}
|
||||||
|
|
||||||
|
func newHash(labels []string) uint64 {
|
||||||
|
labels = meter.BuildLabels(labels...)
|
||||||
|
h := fnv.New64a()
|
||||||
|
for _, l := range labels {
|
||||||
|
h.Write([]byte(l))
|
||||||
|
}
|
||||||
|
return h.Sum64()
|
||||||
|
}
|
||||||
|
|
||||||
|
func fillMetric(m *dto.Metric, labels []string) *dto.Metric {
|
||||||
|
var ok bool
|
||||||
|
seen := make(map[string]bool, len(labels)/2)
|
||||||
|
m.Label = make([]*dto.LabelPair, 0, len(labels)/2)
|
||||||
|
for idx := 0; idx < len(labels); idx += 2 {
|
||||||
|
if _, ok = seen[labels[idx]]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
m.Label = append(m.Label, &dto.LabelPair{
|
||||||
|
Name: newString(labels[idx]),
|
||||||
|
Value: newString(labels[idx+1]),
|
||||||
|
})
|
||||||
|
seen[labels[idx]] = true
|
||||||
|
}
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
@@ -1,178 +1,82 @@
|
|||||||
package prometheus_test
|
package prometheus
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/micro/go-micro/v2/broker"
|
"go.unistack.org/micro/v3/client"
|
||||||
bmemory "github.com/micro/go-micro/v2/broker/memory"
|
"go.unistack.org/micro/v3/codec"
|
||||||
"github.com/micro/go-micro/v2/client"
|
"go.unistack.org/micro/v3/meter"
|
||||||
"github.com/micro/go-micro/v2/registry/memory"
|
"go.unistack.org/micro/v3/meter/wrapper"
|
||||||
"github.com/micro/go-micro/v2/router"
|
|
||||||
rrouter "github.com/micro/go-micro/v2/router/registry"
|
|
||||||
"github.com/micro/go-micro/v2/server"
|
|
||||||
promwrapper "github.com/micro/go-plugins/wrapper/monitoring/prometheus/v2"
|
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
|
||||||
dto "github.com/prometheus/client_model/go"
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Test interface {
|
func TestStd(t *testing.T) {
|
||||||
Method(ctx context.Context, in *TestRequest, opts ...client.CallOption) (*TestResponse, error)
|
m := NewMeter(meter.WriteProcessMetrics(true), meter.WriteFDMetrics(true))
|
||||||
}
|
if err := m.Init(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
type TestRequest struct {
|
}
|
||||||
IsError bool
|
buf := bytes.NewBuffer(nil)
|
||||||
}
|
_ = m.Write(buf)
|
||||||
type TestResponse struct{}
|
if !bytes.Contains(buf.Bytes(), []byte(`go_goroutine`)) {
|
||||||
|
t.Fatalf("invalid metrics output: %s", buf.Bytes())
|
||||||
type testHandler struct{}
|
|
||||||
|
|
||||||
func (t *testHandler) Method(ctx context.Context, req *TestRequest, rsp *TestResponse) error {
|
|
||||||
if req.IsError {
|
|
||||||
return fmt.Errorf("test error")
|
|
||||||
}
|
}
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPrometheusMetrics(t *testing.T) {
|
func TestBuildName(t *testing.T) {
|
||||||
// setup
|
m := NewMeter()
|
||||||
reg := memory.NewRegistry()
|
check := `micro_foo{micro_aaa="b",micro_bar="baz",micro_ccc="d"}`
|
||||||
brk := bmemory.NewBroker(broker.Registry(reg))
|
name := m.buildMetric("foo", "bar", "baz", "aaa", "b", "ccc", "d")
|
||||||
|
if name != check {
|
||||||
|
t.Fatalf("metric name error: %s != %s", name, check)
|
||||||
|
}
|
||||||
|
|
||||||
name := "test"
|
cnt := m.Counter("counter", "key", "val")
|
||||||
id := "id-1234567890"
|
cnt.Inc()
|
||||||
version := "1.2.3.4"
|
}
|
||||||
|
|
||||||
c := client.NewClient(
|
func TestWrapper(t *testing.T) {
|
||||||
client.Router(rrouter.NewRouter(router.Registry(registry))),
|
m := NewMeter() // meter.Labels("test_key", "test_val"))
|
||||||
)
|
|
||||||
s := server.NewServer(
|
w := wrapper.NewClientWrapper(
|
||||||
server.Name(name),
|
wrapper.ServiceName("svc1"),
|
||||||
server.Version(version),
|
wrapper.ServiceVersion("0.0.1"),
|
||||||
server.Id(id),
|
wrapper.ServiceID("12345"),
|
||||||
server.Registry(reg),
|
wrapper.Meter(m),
|
||||||
server.Broker(brk),
|
|
||||||
server.WrapHandler(
|
|
||||||
promwrapper.NewHandlerWrapper(
|
|
||||||
promwrapper.ServiceName(name),
|
|
||||||
promwrapper.ServiceVersion(version),
|
|
||||||
promwrapper.ServiceID(id),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
defer s.Stop()
|
ctx := context.Background()
|
||||||
|
|
||||||
type Test struct {
|
c := client.NewClient(client.Wrap(w))
|
||||||
*testHandler
|
if err := c.Init(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
rsp := &codec.Frame{}
|
||||||
s.Handle(
|
req := &codec.Frame{}
|
||||||
s.NewHandler(&Test{new(testHandler)}),
|
err := c.Call(ctx, c.NewRequest("svc2", "Service.Method", req), rsp)
|
||||||
)
|
_, _ = rsp, err
|
||||||
|
buf := bytes.NewBuffer(nil)
|
||||||
if err := s.Start(); err != nil {
|
_ = m.Write(buf, meter.WriteProcessMetrics(false), meter.WriteFDMetrics(false))
|
||||||
t.Fatalf("Unexpected error starting server: %v", err)
|
if !bytes.Contains(buf.Bytes(), []byte(`micro_client_request_inflight{micro_endpoint="svc2.Service.Method"} 0`)) {
|
||||||
|
t.Fatalf("invalid metrics output: %s", buf.Bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
req := c.NewRequest(name, "Test.Method", &TestRequest{IsError: false}, client.WithContentType("application/json"))
|
|
||||||
rsp := TestResponse{}
|
|
||||||
|
|
||||||
assert.NoError(t, c.Call(context.TODO(), req, &rsp))
|
|
||||||
|
|
||||||
req = c.NewRequest(name, "Test.Method", &TestRequest{IsError: true}, client.WithContentType("application/json"))
|
|
||||||
assert.Error(t, c.Call(context.TODO(), req, &rsp))
|
|
||||||
|
|
||||||
list, _ := prometheus.DefaultGatherer.Gather()
|
|
||||||
|
|
||||||
metric := findMetricByName(list, dto.MetricType_SUMMARY, "micro_latency_microseconds")
|
|
||||||
|
|
||||||
if metric == nil || metric.Metric == nil || len(metric.Metric) == 0 {
|
|
||||||
t.Fatalf("no metrics returned")
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, v := range metric.Metric[0].Label {
|
|
||||||
switch *v.Name {
|
|
||||||
case "micro_version":
|
|
||||||
assert.Equal(t, version, *v.Value)
|
|
||||||
case "micro_id":
|
|
||||||
assert.Equal(t, id, *v.Value)
|
|
||||||
case "micro_name":
|
|
||||||
assert.Equal(t, name, *v.Value)
|
|
||||||
case "micro_endpoint":
|
|
||||||
assert.Equal(t, "Test.Method", *v.Value)
|
|
||||||
default:
|
|
||||||
t.Fatalf("unknown %v with %v", *v.Name, *v.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.Equal(t, uint64(2), *metric.Metric[0].Summary.SampleCount)
|
|
||||||
assert.True(t, *metric.Metric[0].Summary.SampleSum > 0)
|
|
||||||
|
|
||||||
metric = findMetricByName(list, dto.MetricType_HISTOGRAM, "micro_request_duration_seconds")
|
|
||||||
|
|
||||||
for _, v := range metric.Metric[0].Label {
|
|
||||||
switch *v.Name {
|
|
||||||
case "micro_version":
|
|
||||||
assert.Equal(t, version, *v.Value)
|
|
||||||
case "micro_id":
|
|
||||||
assert.Equal(t, id, *v.Value)
|
|
||||||
case "micro_name":
|
|
||||||
assert.Equal(t, name, *v.Value)
|
|
||||||
case "micro_endpoint":
|
|
||||||
assert.Equal(t, "Test.Method", *v.Value)
|
|
||||||
default:
|
|
||||||
t.Fatalf("unknown %v with %v", *v.Name, *v.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.Equal(t, uint64(2), *metric.Metric[0].Histogram.SampleCount)
|
|
||||||
assert.True(t, *metric.Metric[0].Histogram.SampleSum > 0)
|
|
||||||
|
|
||||||
metric = findMetricByName(list, dto.MetricType_COUNTER, "micro_request_total")
|
|
||||||
|
|
||||||
for _, v := range metric.Metric[0].Label {
|
|
||||||
switch *v.Name {
|
|
||||||
case "micro_version":
|
|
||||||
assert.Equal(t, version, *v.Value)
|
|
||||||
case "micro_id":
|
|
||||||
assert.Equal(t, id, *v.Value)
|
|
||||||
case "micro_name":
|
|
||||||
assert.Equal(t, name, *v.Value)
|
|
||||||
case "micro_endpoint":
|
|
||||||
assert.Equal(t, "Test.Method", *v.Value)
|
|
||||||
case "micro_status":
|
|
||||||
assert.Equal(t, "failure", *v.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assert.Equal(t, *metric.Metric[0].Counter.Value, float64(1))
|
|
||||||
|
|
||||||
for _, v := range metric.Metric[1].Label {
|
|
||||||
switch *v.Name {
|
|
||||||
case "micro_version":
|
|
||||||
assert.Equal(t, version, *v.Value)
|
|
||||||
case "micro_id":
|
|
||||||
assert.Equal(t, id, *v.Value)
|
|
||||||
case "micro_name":
|
|
||||||
assert.Equal(t, name, *v.Value)
|
|
||||||
case "micro_endpoint":
|
|
||||||
assert.Equal(t, "Test.Method", *v.Value)
|
|
||||||
case "micro_status":
|
|
||||||
assert.Equal(t, "success", *v.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.Equal(t, *metric.Metric[1].Counter.Value, float64(1))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func findMetricByName(list []*dto.MetricFamily, tp dto.MetricType, name string) *dto.MetricFamily {
|
func TestMultiple(t *testing.T) {
|
||||||
for _, metric := range list {
|
m := NewMeter() // meter.Labels("test_key", "test_val"))
|
||||||
if *metric.Name == name && *metric.Type == tp {
|
|
||||||
return metric
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
m.Counter("server", "endpoint", "ep1", "path", "/path1").Inc()
|
||||||
|
m.Counter("server", "endpoint", "ep1", "path", "/path1").Inc()
|
||||||
|
|
||||||
|
m.Counter("server", "endpoint", "ep2", "path", "/path2").Inc()
|
||||||
|
m.Counter("server", "endpoint", "ep2", "path", "/path2").Inc()
|
||||||
|
|
||||||
|
m.Counter("server", "endpoint", "ep3", "path", "/path3", "status", "success").Inc()
|
||||||
|
|
||||||
|
buf := bytes.NewBuffer(nil)
|
||||||
|
_ = m.Write(buf, meter.WriteProcessMetrics(false), meter.WriteFDMetrics(false))
|
||||||
|
if !bytes.Contains(buf.Bytes(), []byte(`micro_server{micro_endpoint="ep1",micro_path="/path1"} 2`)) {
|
||||||
|
// t.Fatal("XXXX")
|
||||||
|
t.Fatalf("invalid metrics output: %s", buf.Bytes())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user