mirror of
				https://github.com/ClusterCockpit/cc-metric-collector.git
				synced 2025-10-25 07:15:06 +02:00 
			
		
		
		
	Merge branch 'develop' into main
This commit is contained in:
		
							
								
								
									
										6
									
								
								.github/ci-collectors.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.github/ci-collectors.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| { | ||||
|     "tempstat": {}, | ||||
|     "diskstat": {}, | ||||
|     "memstat": {}, | ||||
|     "cpustat": {} | ||||
| } | ||||
							
								
								
									
										56
									
								
								.github/ci-config.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										56
									
								
								.github/ci-config.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,52 +1,8 @@ | ||||
| { | ||||
|   "sink": { | ||||
|     "user": "testuser", | ||||
|     "password": "testpass", | ||||
|     "host": "127.0.0.1", | ||||
|     "port": "9090", | ||||
|     "database": "testdb", | ||||
|     "organization": "testorg", | ||||
|     "type": "stdout" | ||||
|   }, | ||||
|   "interval": 3, | ||||
|   "duration": 1, | ||||
|   "collectors": [ | ||||
|     "tempstat", | ||||
|     "loadavg", | ||||
|     "memstat", | ||||
|     "netstat", | ||||
|     "ibstat", | ||||
|     "lustrestat", | ||||
|     "cpustat", | ||||
|     "topprocs", | ||||
|     "nvidia", | ||||
|     "diskstat", | ||||
|     "ipmistat", | ||||
|     "gpfs", | ||||
|     "cpufreq", | ||||
|     "cpufreq_cpuinfo" | ||||
|   ], | ||||
|   "default_tags": { | ||||
|     "cluster": "testcluster" | ||||
|   }, | ||||
|   "receiver": { | ||||
|     "type": "none" | ||||
|   }, | ||||
|   "collect_config": { | ||||
|     "topprocs": { | ||||
|       "num_procs": 2 | ||||
|     }, | ||||
|     "tempstat": { | ||||
|       "tag_override": { | ||||
|         "hwmon0": { | ||||
|           "type": "socket", | ||||
|           "type-id": "0" | ||||
|         }, | ||||
|         "hwmon1": { | ||||
|           "type": "socket", | ||||
|           "type-id": "1" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   "sinks": ".github/ci-sinks.json", | ||||
|   "collectors" : ".github/ci-collectors.json", | ||||
|   "receivers" : ".github/ci-receivers.json", | ||||
|   "router" : ".github/ci-router.json", | ||||
|   "interval": 5, | ||||
|   "duration": 1 | ||||
| } | ||||
							
								
								
									
										1
									
								
								.github/ci-receivers.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/ci-receivers.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| {} | ||||
							
								
								
									
										37
									
								
								.github/ci-router.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								.github/ci-router.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| { | ||||
|   "add_tags": [ | ||||
|     { | ||||
|       "key": "cluster", | ||||
|       "value": "testcluster", | ||||
|       "if": "*" | ||||
|     }, | ||||
|     { | ||||
|       "key": "test", | ||||
|       "value": "testing", | ||||
|       "if": "name == 'temp_package_id_0'" | ||||
|     } | ||||
|   ], | ||||
|   "delete_tags": [ | ||||
|     { | ||||
|       "key": "unit", | ||||
|       "value": "*", | ||||
|       "if": "*" | ||||
|     } | ||||
|   ], | ||||
|   "interval_aggregates": [ | ||||
|     { | ||||
|       "name": "temp_cores_avg", | ||||
|       "function": "avg(values)", | ||||
|       "if": "match('temp_core_%d+', metric.Name())", | ||||
|       "tags": { | ||||
|         "type": "node" | ||||
|       }, | ||||
|       "meta": { | ||||
|         "group": "<copy>", | ||||
|         "unit": "<copy>", | ||||
|         "source": "MetricAggregator" | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "interval_timestamp": true | ||||
| } | ||||
							
								
								
									
										6
									
								
								.github/ci-sinks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.github/ci-sinks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| { | ||||
|   "testoutput" : { | ||||
|     "type" : "stdout", | ||||
|     "meta_as_tags" : true | ||||
|   } | ||||
| } | ||||
							
								
								
									
										47
									
								
								.github/workflows/rpmbuild.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										47
									
								
								.github/workflows/rpmbuild.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,16 @@ | ||||
| name: Run RPM Build | ||||
| on: push | ||||
| on: | ||||
|  push: | ||||
|   tags: | ||||
|   - '**' | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|   build-centos8: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|       with: | ||||
|         submodules: recursive | ||||
|     - uses: TomTheBear/rpmbuild@master | ||||
|       id: rpm | ||||
|       name: Build RPM package on CentOS8 | ||||
| @@ -21,3 +26,41 @@ jobs: | ||||
|       with: | ||||
|         name: cc-metric-collector SRPM CentOS8 | ||||
|         path: ${{ steps.rpm.outputs.source_rpm_path }} | ||||
|   build-centos-latest: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: TomTheBear/rpmbuild@centos_latest | ||||
|       id: rpm | ||||
|       name: Build RPM package on CentOS 'Latest' | ||||
|       with: | ||||
|         spec_file: "./scripts/cc-metric-collector.spec" | ||||
|     - name: Save RPM as artifact | ||||
|       uses: actions/upload-artifact@v1.0.0 | ||||
|       with: | ||||
|         name: cc-metric-collector RPM CentOS 'Latest' | ||||
|         path: ${{ steps.rpm.outputs.rpm_dir_path }} | ||||
|     - name: Save SRPM as artifact | ||||
|       uses: actions/upload-artifact@v1.0.0 | ||||
|       with: | ||||
|         name: cc-metric-collector SRPM CentOS 'Latest' | ||||
|         path: ${{ steps.rpm.outputs.source_rpm_path }} | ||||
|   build-alma-8_5: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: TomTheBear/rpmbuild@alma8.5 | ||||
|       id: rpm | ||||
|       name: Build RPM package on AlmaLinux 8.5 | ||||
|       with: | ||||
|         spec_file: "./scripts/cc-metric-collector.spec" | ||||
|     - name: Save RPM as artifact | ||||
|       uses: actions/upload-artifact@v1.0.0 | ||||
|       with: | ||||
|         name: cc-metric-collector RPM AlmaLinux 8.5 | ||||
|         path: ${{ steps.rpm.outputs.rpm_dir_path }} | ||||
|     - name: Save SRPM as artifact | ||||
|       uses: actions/upload-artifact@v1.0.0 | ||||
|       with: | ||||
|         name: cc-metric-collector SRPM AlmaLinux 8.5 | ||||
|         path: ${{ steps.rpm.outputs.source_rpm_path }} | ||||
|   | ||||
							
								
								
									
										28
									
								
								.github/workflows/runonce.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/runonce.yml
									
									
									
									
										vendored
									
									
								
							| @@ -2,10 +2,12 @@ name: Run Test | ||||
| on: push | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|   build-1-17: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|       with: | ||||
|         submodules: recursive | ||||
|  | ||||
|     # See: https://github.com/marketplace/actions/setup-go-environment | ||||
|     - name: Setup Golang | ||||
| @@ -13,8 +15,32 @@ jobs: | ||||
|       with: | ||||
|         go-version: '^1.17.6' | ||||
|  | ||||
|     - name: Setup Ganglia | ||||
|       run: sudo apt install ganglia-monitor libganglia1 | ||||
|  | ||||
|     - name: Build MetricCollector | ||||
|       run: make | ||||
|  | ||||
|     - name: Run MetricCollector | ||||
|       run: ./cc-metric-collector --once --config .github/ci-config.json | ||||
|   build-1-16: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|       with: | ||||
|         submodules: recursive | ||||
|  | ||||
|     # See: https://github.com/marketplace/actions/setup-go-environment | ||||
|     - name: Setup Golang | ||||
|       uses: actions/setup-go@v2.1.5 | ||||
|       with: | ||||
|         go-version: '^1.16.7' # The version AlmaLinux 8.5 uses | ||||
|  | ||||
|     - name: Setup Ganglia | ||||
|       run: sudo apt install ganglia-monitor libganglia1 | ||||
|  | ||||
|     - name: Build MetricCollector | ||||
|       run: make | ||||
|  | ||||
|     - name: Run MetricCollectorlibganglia1 | ||||
|       run: ./cc-metric-collector --once --config .github/ci-config.json | ||||
|   | ||||
							
								
								
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| [submodule ".github/actions/rpmbuild-centos8-golang"] | ||||
| 	path = .github/actions/rpmbuild-centos8-golang | ||||
| 	url = https://github.com/naveenrajm7/rpmbuild.git | ||||
| 	branch = centos8 | ||||
							
								
								
									
										18
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								Makefile
									
									
									
									
									
								
							| @@ -3,19 +3,31 @@ GOSRC_APP        := metric-collector.go | ||||
| GOSRC_COLLECTORS := $(wildcard collectors/*.go) | ||||
| GOSRC_SINKS      := $(wildcard sinks/*.go) | ||||
| GOSRC_RECEIVERS  := $(wildcard receivers/*.go) | ||||
| GOSRC            := $(GOSRC_APP) $(GOSRC_COLLECTORS) $(GOSRC_SINKS) $(GOSRC_RECEIVERS) | ||||
| GOSRC_INTERNAL   := $(wildcard internal/*/*.go) | ||||
| GOSRC            := $(GOSRC_APP) $(GOSRC_COLLECTORS) $(GOSRC_SINKS) $(GOSRC_RECEIVERS) $(GOSRC_INTERNAL) | ||||
| COMPONENT_DIRS   := collectors \ | ||||
| 			sinks \ | ||||
| 			receivers \ | ||||
| 			internal/metricRouter \ | ||||
| 			internal/ccMetric \ | ||||
| 			internal/metricAggregator \ | ||||
| 			internal/ccLogger \ | ||||
| 			internal/ccTopology \ | ||||
| 			internal/multiChanTicker | ||||
|  | ||||
|  | ||||
| .PHONY: all | ||||
| all: $(APP) | ||||
|  | ||||
| $(APP): $(GOSRC) | ||||
| 	make -C collectors | ||||
| 	make -C sinks | ||||
| 	go get | ||||
| 	go build -o $(APP) $(GOSRC_APP) | ||||
|  | ||||
| .PHONY: clean | ||||
| clean: | ||||
| 	make -C collectors clean | ||||
| 	@for COMP in $(COMPONENT_DIRS); do if [ -e $$COMP/Makefile ]; then make -C $$COMP clean; fi; done | ||||
| 	rm -f $(APP) | ||||
|  | ||||
| .PHONY: fmt | ||||
| @@ -24,6 +36,8 @@ fmt: | ||||
| 	go fmt $(GOSRC_SINKS) | ||||
| 	go fmt $(GOSRC_RECEIVERS) | ||||
| 	go fmt $(GOSRC_APP) | ||||
| 	@for F in $(GOSRC_INTERNAL); do go fmt $$F; done | ||||
|  | ||||
|  | ||||
| # Examine Go source code and reports suspicious constructs | ||||
| .PHONY: vet | ||||
|   | ||||
							
								
								
									
										102
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										102
									
								
								README.md
									
									
									
									
									
								
							| @@ -12,87 +12,48 @@ The receiver runs as a go routine side-by-side with the timer loop and asynchron | ||||
| Configuration is implemented using a single json document that is distributed over network and may be persisted as file. | ||||
| Supported metrics are documented [here](https://github.com/ClusterCockpit/cc-specifications/blob/master/metrics/lineprotocol_alternative.md). | ||||
|  | ||||
| There is a main configuration file with basic settings that point to the other configuration files for the different components. | ||||
|  | ||||
| ``` json | ||||
| { | ||||
|   "interval": 3, | ||||
|   "duration": 1, | ||||
|   "collectors": [ | ||||
|     "memstat", | ||||
|     "likwid", | ||||
|     "loadavg", | ||||
|     "netstat", | ||||
|     "ibstat", | ||||
|     "lustrestat", | ||||
|     "topprocs", | ||||
|     "cpustat", | ||||
|     "nvidia" | ||||
|   ], | ||||
|   "sink": { | ||||
|     "user": "admin", | ||||
|     "password": "12345", | ||||
|     "host": "localhost", | ||||
|     "port": "8080", | ||||
|     "database": "testdb", | ||||
|     "organisation": "testorg", | ||||
|     "type": "stdout" | ||||
|   }, | ||||
|   "default_tags": { | ||||
|     "cluster": "testcluster" | ||||
|   }, | ||||
|   "receiver": { | ||||
|     "type": "none", | ||||
|     "address": "127.0.0.1", | ||||
|     "port": "4222", | ||||
|     "database": "testdb" | ||||
|   }, | ||||
|   "collect_config": { | ||||
|     "tempstat": { | ||||
|       "tag_override": { | ||||
|         "hwmon0": { | ||||
|           "type": "socket", | ||||
|           "type-id": "0" | ||||
|         }, | ||||
|         "hwmon1": { | ||||
|           "type": "socket", | ||||
|           "type-id": "1" | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "diskstat": { | ||||
|       "exclude_metrics": [ | ||||
|         "read_ms" | ||||
|       ] | ||||
|     } | ||||
|   } | ||||
|   "sinks": "sinks.json", | ||||
|   "collectors" : "collectors.json", | ||||
|   "receivers" : "receivers.json", | ||||
|   "router" : "router.json", | ||||
|   "interval": 10, | ||||
|   "duration": 1 | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The `interval` defines how often the metrics should be read and send to the sink. The `duration` tells collectors how long one measurement has to take. An example for this is the `likwid` collector which starts the hardware performance counter, waits for `duration` seconds and stops the counters again. If you configure a collector to do two measurments, the `duration` must be at least half the `interval`. | ||||
| The `interval` defines how often the metrics should be read and send to the sink. The `duration` tells collectors how long one measurement has to take. This is important for some collectors, like the `likwid` collector. | ||||
|  | ||||
| The `collectors` contains all collectors executed collectors. Each collector can be configured in the `collect_config` section. A more detailed list of all collectors and their configuration options can be found in the [README for collectors](./collectors/README.md). | ||||
| See the component READMEs for their configuration: | ||||
| * [`collectors`](./collectors/README.md) | ||||
| * [`sinks`](./sinks/README.md) | ||||
| * [`receivers`](./receivers/README.md) | ||||
| * [`router`](./internal/metricRouter/README.md) | ||||
|  | ||||
| The `sink` section contains the configuration where the data should be transmitted to. There are currently four sinks supported `influxdb`, `nats`, `http` and `stdout`. See [README for sinks](./sinks/README.md) for more information about the individual sinks and which configuration field they are using. | ||||
|  | ||||
| In the `default_tags` section, one can define key-value-pairs (only strings) that are added to each sent out metric. This can be useful for cluster names like in the example JSON or information like rank or island for orientation. | ||||
|  | ||||
| With `receiver`, the collector can be used as a router by receiving metrics and forwarding them to the configured sink. There are currently only types `none` (for no receiver) and `nats`. For more information see the [README in receivers](./receivers/README.md). | ||||
|  | ||||
| # Installation | ||||
|  | ||||
| ``` | ||||
| $ git clone git@github.com:ClusterCockpit/cc-metric-collector.git | ||||
| $ cd cc-metric-collector/collectors | ||||
| $ edit Makefile (for LIKWID collector) | ||||
| $ make (downloads LIKWID, builds it as static library and copies all required files for the collector. Uses sudo in case of own accessdaemon) | ||||
| $ cd .. | ||||
| $ go get (requires at least golang 1.13) | ||||
| $ go build metric-collector | ||||
| $ make (downloads LIKWID, builds it as static library with 'direct' accessmode and copies all required files for the collector) | ||||
| $ go get (requires at least golang 1.16) | ||||
| $ make tags | ||||
| Available tags: | ||||
| ganglia | ||||
| [...] | ||||
| $ make # calls go build (-tags ganglia,...) -o cc-metric-collector | ||||
| ``` | ||||
|  | ||||
| ## `ganglia` build tag | ||||
| If you want support for the [Ganglia Monitoring System](http://ganglia.info/), you have to add `-tags ganglia` to the build command line. This enables two metric sinks. One is using the command line application `gmetric` (see [`ganglia`](./sinks/gangliaSink.md) sink), the other one interacts directly with `libganglia` the main Ganglia library that is commonly installed on each compute node (see [`libganglia`](./sinks/libgangliaSink.md) sink). The later one requires configuration before building, so use `make` instead of `go build` directly. | ||||
|  | ||||
| # Running | ||||
|  | ||||
| ``` | ||||
| $ ./metric-collector --help | ||||
| $ ./cc-metric-collector --help | ||||
| Usage of metric-collector: | ||||
|   -config string | ||||
|     	Path to configuration file (default "./config.json") | ||||
| @@ -100,17 +61,8 @@ Usage of metric-collector: | ||||
|     	Path for logfile (default "stderr") | ||||
|   -once | ||||
|     	Run all collectors only once | ||||
|   -pidfile string | ||||
|     	Path for PID file (default "/var/run/cc-metric-collector.pid") | ||||
| ``` | ||||
|  | ||||
| # Todos | ||||
|  | ||||
| - [ ] Use only non-blocking APIs for the sinks | ||||
| - [x] Collector specific configuration in global JSON file? Changing the configuration inside the Go code is not user-friendly. | ||||
| - [ ] Mark collectors as 'can-run-in-parallel' and use goroutines for them. There are only a few collectors that should run serially (e.g. LIKWID) | ||||
| - [ ] Configuration option for receivers to add other tags. Additonal flag to tell whether default tags should be added as well. | ||||
| - [ ] CLI option to get help output for collectors, sinks and receivers about their configuration options and metrics | ||||
|  | ||||
| # Contributing | ||||
| The ClusterCockpit ecosystem is designed to be used by different HPC computing centers. Since configurations and setups differ between the centers, the centers likely have to put some work into the cc-metric-collector to gather all desired metrics. | ||||
| @@ -119,5 +71,5 @@ You are free to open an issue to request a collector but we would also be happy | ||||
|  | ||||
| # Contact  | ||||
|  | ||||
| [Matrix.org ClusterCockpit General chat](https://matrix.to/#/#clustercockpit-dev:matrix.org) | ||||
| [Matrix.org ClusterCockpit Development chat](https://matrix.to/#/#clustercockpit:matrix.org) | ||||
| * [Matrix.org ClusterCockpit General chat](https://matrix.to/#/#clustercockpit-dev:matrix.org) | ||||
| * [Matrix.org ClusterCockpit Development chat](https://matrix.to/#/#clustercockpit:matrix.org) | ||||
|   | ||||
							
								
								
									
										31
									
								
								collectors.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								collectors.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| { | ||||
|     "cpufreq": {}, | ||||
|     "cpufreq_cpuinfo": {}, | ||||
|     "gpfs": { | ||||
|         "exclude_filesystem": [ | ||||
|             "test_fs" | ||||
|         ] | ||||
|     }, | ||||
|     "ibstat": {}, | ||||
|     "loadavg": { | ||||
|         "exclude_metrics": [ | ||||
|             "proc_total" | ||||
|         ] | ||||
|     }, | ||||
|     "numastats": {}, | ||||
|     "nvidia": {}, | ||||
|     "tempstat": { | ||||
|         "report_max_temperature": true, | ||||
|         "report_critical_temperature": true, | ||||
|         "tag_override": { | ||||
|             "hwmon0": { | ||||
|                 "type": "socket", | ||||
|                 "type-id": "0" | ||||
|             }, | ||||
|             "hwmon1": { | ||||
|                 "type": "socket", | ||||
|                 "type-id": "1" | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,304 +1,59 @@ | ||||
| # CCMetric collectors | ||||
|  | ||||
| This folder contains the collectors for the cc-metric-collector. | ||||
|  | ||||
| # `metricCollector.go` | ||||
| The base class/configuration is located in `metricCollector.go`. | ||||
|  | ||||
| # Collectors | ||||
|  | ||||
| * `memstatMetric.go`: Reads `/proc/meminfo` to calculate **node** metrics. It also combines values to the metric `mem_used` | ||||
| * `loadavgMetric.go`: Reads `/proc/loadavg` and submits **node** metrics: | ||||
| * `netstatMetric.go`: Reads `/proc/net/dev` and submits for all network devices as the **node** metrics. | ||||
| * `lustreMetric.go`: Reads Lustre's stats files and submits **node** metrics: | ||||
| * `infinibandMetric.go`: Reads InfiniBand metrics. It uses the `perfquery` command to read the **node** metrics but can fallback to sysfs counters in case `perfquery` does not work. | ||||
| * `likwidMetric.go`: Reads hardware performance events using LIKWID. It submits **socket** and **cpu** metrics | ||||
| * `cpustatMetric.go`: Read CPU specific values from `/proc/stat` | ||||
| * `topprocsMetric.go`: Reads the TopX processes by their CPU usage. X is configurable | ||||
| * `nvidiaMetric.go`: Read data about Nvidia GPUs using the NVML library | ||||
| * `tempMetric.go`: Read temperature data from `/sys/class/hwmon/hwmon*` | ||||
| * `ipmiMetric.go`: Collect data from `ipmitool` or as fallback `ipmi-sensors` | ||||
| * `customCmdMetric.go`: Run commands or read files and submit the output (output has to be in InfluxDB line protocol!) | ||||
|  | ||||
| If any of the collectors cannot be initialized, it is excluded from all further reads. Like if the Lustre stat file is not a valid path, no Lustre specific metrics will be recorded. | ||||
|  | ||||
| # Collector configuration | ||||
| # Configuration | ||||
|  | ||||
| ```json | ||||
|   "collectors": [ | ||||
|     "tempstat" | ||||
|   ], | ||||
|   "collect_config": { | ||||
|     "tempstat": { | ||||
|       "tag_override": { | ||||
|         "hwmon0" : { | ||||
|             "type" : "socket", | ||||
|             "type-id" : "0" | ||||
|         }, | ||||
|         "hwmon1" : { | ||||
|             "type" : "socket", | ||||
|             "type-id" : "1" | ||||
|         } | ||||
|       } | ||||
| { | ||||
|     "collector_type" : { | ||||
|         <collector specific configuration> | ||||
|     } | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The configuration of the collectors in the main config files consists of two parts: active collectors (`collectors`) and collector configuration (`collect_config`). At startup, all collectors in the `collectors` list is initialized and, if successfully initialized, added to the active collectors for metric retrieval. At initialization the collector-specific configuration from the `collect_config` section is handed over. Each collector has own configuration options, check at the collector-specific section. | ||||
| In contrast to the configuration files for sinks and receivers, the collectors configuration is not a list but a set of dicts. This is required because we didn't manage to partially read the type before loading the remaining configuration. We are eager to change this to the same format. | ||||
|  | ||||
| ## `memstat` | ||||
| # Available collectors | ||||
|  | ||||
| ```json | ||||
|   "memstat": { | ||||
|     "exclude_metrics": [ | ||||
|       "mem_used" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `memstat` collector reads data from `/proc/meminfo` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
|  | ||||
| Metrics: | ||||
| * `mem_total` | ||||
| * `mem_sreclaimable` | ||||
| * `mem_slab` | ||||
| * `mem_free` | ||||
| * `mem_buffers` | ||||
| * `mem_cached` | ||||
| * `mem_available` | ||||
| * `mem_shared` | ||||
| * `swap_total` | ||||
| * `swap_free` | ||||
| * `mem_used` = `mem_total` - (`mem_free` + `mem_buffers` + `mem_cached`) | ||||
|  | ||||
| ## `loadavg` | ||||
| ```json | ||||
|   "loadavg": { | ||||
|     "exclude_metrics": [ | ||||
|       "proc_run" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `loadavg` collector reads data from `/proc/loadavg` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `load_one` | ||||
| * `load_five` | ||||
| * `load_fifteen` | ||||
| * `proc_run` | ||||
| * `proc_total` | ||||
|  | ||||
| ## `netstat` | ||||
| ```json | ||||
|   "netstat": { | ||||
|     "exclude_devices": [ | ||||
|       "lo" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `netstat` collector reads data from `/proc/net/dev` and outputs a handful **node** metrics. If a device is not required, it can be excluded from forwarding it to the sink. Commonly the `lo` device should be excluded. | ||||
|  | ||||
| Metrics: | ||||
| * `bytes_in` | ||||
| * `bytes_out` | ||||
| * `pkts_in` | ||||
| * `pkts_out` | ||||
|  | ||||
| The device name is added as tag `device`. | ||||
|  | ||||
|  | ||||
| ## `diskstat` | ||||
|  | ||||
| ```json | ||||
|   "diskstat": { | ||||
|     "exclude_metrics": [ | ||||
|       "read_ms" | ||||
|     ], | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `netstat` collector reads data from `/proc/net/dev` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `reads` | ||||
| * `reads_merged` | ||||
| * `read_sectors` | ||||
| * `read_ms` | ||||
| * `writes` | ||||
| * `writes_merged` | ||||
| * `writes_sectors` | ||||
| * `writes_ms` | ||||
| * `ioops` | ||||
| * `ioops_ms` | ||||
| * `ioops_weighted_ms` | ||||
| * `discards` | ||||
| * `discards_merged` | ||||
| * `discards_sectors` | ||||
| * `discards_ms` | ||||
| * `flushes` | ||||
| * `flushes_ms` | ||||
|  | ||||
|  | ||||
| The device name is added as tag `device`. | ||||
|  | ||||
| ## `cpustat` | ||||
| ```json | ||||
|   "netstat": { | ||||
|     "exclude_metrics": [ | ||||
|       "cpu_idle" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `cpustat` collector reads data from `/proc/stats` and outputs a handful **node** and **hwthread** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `cpu_user` | ||||
| * `cpu_nice` | ||||
| * `cpu_system` | ||||
| * `cpu_idle` | ||||
| * `cpu_iowait` | ||||
| * `cpu_irq` | ||||
| * `cpu_softirq` | ||||
| * `cpu_steal` | ||||
| * `cpu_guest` | ||||
| * `cpu_guest_nice` | ||||
|  | ||||
| ## `likwid` | ||||
| ```json | ||||
|   "likwid": { | ||||
|     "eventsets": [ | ||||
|       { | ||||
|         "events": { | ||||
|           "FIXC1": "ACTUAL_CPU_CLOCK", | ||||
|           "FIXC2": "MAX_CPU_CLOCK", | ||||
|           "PMC0": "RETIRED_INSTRUCTIONS", | ||||
|           "PMC1": "CPU_CLOCKS_UNHALTED", | ||||
|           "PMC2": "RETIRED_SSE_AVX_FLOPS_ALL", | ||||
|           "PMC3": "MERGE", | ||||
|           "DFC0": "DRAM_CHANNEL_0", | ||||
|           "DFC1": "DRAM_CHANNEL_1", | ||||
|           "DFC2": "DRAM_CHANNEL_2", | ||||
|           "DFC3": "DRAM_CHANNEL_3" | ||||
|         }, | ||||
|         "metrics": [ | ||||
|           { | ||||
|             "name": "ipc", | ||||
|             "calc": "PMC0/PMC1", | ||||
|             "socket_scope": false, | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "flops_any", | ||||
|             "calc": "0.000001*PMC2/time", | ||||
|             "socket_scope": false, | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "clock_mhz", | ||||
|             "calc": "0.000001*(FIXC1/FIXC2)/inverseClock", | ||||
|             "socket_scope": false, | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "mem1", | ||||
|             "calc": "0.000001*(DFC0+DFC1+DFC2+DFC3)*64.0/time", | ||||
|             "socket_scope": true, | ||||
|             "publish": false | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "events": { | ||||
|           "DFC0": "DRAM_CHANNEL_4", | ||||
|           "DFC1": "DRAM_CHANNEL_5", | ||||
|           "DFC2": "DRAM_CHANNEL_6", | ||||
|           "DFC3": "DRAM_CHANNEL_7", | ||||
|           "PWR0": "RAPL_CORE_ENERGY", | ||||
|           "PWR1": "RAPL_PKG_ENERGY" | ||||
|         }, | ||||
|         "metrics": [ | ||||
|           { | ||||
|             "name": "pwr_core", | ||||
|             "calc": "PWR0/time", | ||||
|             "socket_scope": false, | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "pwr_pkg", | ||||
|             "calc": "PWR1/time", | ||||
|             "socket_scope": true, | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "mem2", | ||||
|             "calc": "0.000001*(DFC0+DFC1+DFC2+DFC3)*64.0/time", | ||||
|             "socket_scope": true, | ||||
|             "publish": false | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "globalmetrics": [ | ||||
|       { | ||||
|         "name": "mem_bw", | ||||
|         "calc": "mem1+mem2", | ||||
|         "socket_scope": true, | ||||
|         "publish": true | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| _Example config suitable for AMD Zen3_ | ||||
|  | ||||
| The `likwid` collector reads hardware performance counters at a **hwthread** and **socket** level. The configuration looks quite complicated but it is basically copy&paste from [LIKWID's performance groups](https://github.com/RRZE-HPC/likwid/tree/master/groups). The collector made multiple iterations and tried to use the performance groups but it lacked flexibility. The current way of configuration provides most flexibility. | ||||
|  | ||||
| The logic is as following: There are multiple eventsets, each consisting of a list of counters+events and a list of metrics. If you compare a common performance group with the example setting above, there is not much difference: | ||||
| ``` | ||||
| EVENTSET                         ->   "events": { | ||||
| FIXC1 ACTUAL_CPU_CLOCK           ->     "FIXC1": "ACTUAL_CPU_CLOCK", | ||||
| FIXC2 MAX_CPU_CLOCK              ->     "FIXC2": "MAX_CPU_CLOCK", | ||||
| PMC0  RETIRED_INSTRUCTIONS       ->     "PMC0" : "RETIRED_INSTRUCTIONS", | ||||
| PMC1  CPU_CLOCKS_UNHALTED        ->     "PMC1" : "CPU_CLOCKS_UNHALTED", | ||||
| PMC2  RETIRED_SSE_AVX_FLOPS_ALL  ->     "PMC2": "RETIRED_SSE_AVX_FLOPS_ALL", | ||||
| PMC3  MERGE                      ->     "PMC3": "MERGE", | ||||
|                                  ->   } | ||||
| ``` | ||||
|  | ||||
| The metrics are following the same procedure: | ||||
|  | ||||
| ``` | ||||
| METRICS                          ->   "metrics": [ | ||||
| IPC   PMC0/PMC1                  ->     { | ||||
|                                  ->       "name" : "IPC", | ||||
|                                  ->       "calc" : "PMC0/PMC1", | ||||
|                                  ->       "socket_scope": false, | ||||
|                                  ->       "publish": true | ||||
|                                  ->     } | ||||
|                                  ->   ] | ||||
| ``` | ||||
|  | ||||
| The `socket_scope` option tells whether it is submitted per socket or per hwthread. If a metric is only used for internal calculations, you can set `publish = false`. | ||||
|  | ||||
| Since some metrics can only be gathered in multiple measurements (like the memory bandwidth on AMD Zen3 chips), configure multiple eventsets like in the example config and use the `globalmetrics` section to combine them. **Be aware** that the combination might be misleading because the "behavior" of a metric changes over time and the multiple measurements might count different computing phases. | ||||
| * [`cpustat`](./cpustatMetric.md) | ||||
| * [`memstat`](./memstatMetric.md) | ||||
| * [`iostat`](./iostatMetric.md) | ||||
| * [`diskstat`](./diskstatMetric.md) | ||||
| * [`loadavg`](./loadavgMetric.md) | ||||
| * [`netstat`](./netstatMetric.md) | ||||
| * [`ibstat`](./infinibandMetric.md) | ||||
| * [`ibstat_perfquery`](./infinibandPerfQueryMetric.md) | ||||
| * [`tempstat`](./tempMetric.md) | ||||
| * [`lustrestat`](./lustreMetric.md) | ||||
| * [`likwid`](./likwidMetric.md) | ||||
| * [`nvidia`](./nvidiaMetric.md) | ||||
| * [`customcmd`](./customCmdMetric.md) | ||||
| * [`ipmistat`](./ipmiMetric.md) | ||||
| * [`topprocs`](./topprocsMetric.md) | ||||
| * [`nfs3stat`](./nfs3Metric.md) | ||||
| * [`nfs4stat`](./nfs4Metric.md) | ||||
| * [`cpufreq`](./cpufreqMetric.md) | ||||
| * [`cpufreq_cpuinfo`](./cpufreqCpuinfoMetric.md) | ||||
| * [`numastat`](./numastatMetric.md) | ||||
| * [`gpfs`](./gpfsMetric.md) | ||||
|  | ||||
| ## Todos | ||||
|  | ||||
| * [ ] Exclude devices for `diskstat` collector | ||||
| * [ ] Aggreate metrics to higher topology entity (sum hwthread metrics to socket metric, ...). Needs to be configurable | ||||
|  | ||||
| # Contributing own collectors | ||||
| A collector reads data from any source, parses it to metrics and submits these metrics to the `metric-collector`. A collector provides three function: | ||||
|  | ||||
| * `Init(config []byte) error`: Initializes the collector using the given collector-specific config in JSON. | ||||
| * `Read(duration time.Duration, out *[]lp.MutableMetric) error`: Read, parse and submit data to the `out` list. If the collector has to measure anything for some duration, use the provided function argument `duration`.  | ||||
| * `Name() string`: Return the name of the collector | ||||
| * `Init(config json.RawMessage) error`: Initializes the collector using the given collector-specific config in JSON. Check if needed files/commands exists, ... | ||||
| * `Initialized() bool`: Check if a collector is successfully initialized | ||||
| * `Read(duration time.Duration, output chan ccMetric.CCMetric)`: Read, parse and submit data to the `output` channel as [`CCMetric`](../internal/ccMetric/README.md). If the collector has to measure anything for some duration, use the provided function argument `duration`.  | ||||
| * `Close()`: Closes down the collector. | ||||
|  | ||||
| It is recommanded to call `setup()` in the `Init()` function. | ||||
|  | ||||
| Finally, the collector needs to be registered in the `metric-collector.go`. There is a list of collectors called `Collectors` which is a map (string -> pointer to collector). Add a new entry with a descriptive name and the new collector. | ||||
| Finally, the collector needs to be registered in the `collectorManager.go`. There is a list of collectors called `AvailableCollectors` which is a map (`collector_type_string` -> `pointer to MetricCollector interface`). Add a new entry with a descriptive name and the new collector. | ||||
|  | ||||
| ## Sample collector | ||||
|  | ||||
| @@ -307,8 +62,9 @@ package collectors | ||||
|  | ||||
| import ( | ||||
|     "encoding/json" | ||||
|     lp "github.com/influxdata/line-protocol" | ||||
|     "time" | ||||
|  | ||||
|     lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| // Struct for the collector-specific JSON config | ||||
| @@ -317,11 +73,11 @@ type SampleCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type SampleCollector struct { | ||||
|     MetricCollector | ||||
|     metricCollector | ||||
|     config SampleCollectorConfig | ||||
| } | ||||
|  | ||||
| func (m *SampleCollector) Init(config []byte) error { | ||||
| func (m *SampleCollector) Init(config json.RawMessage) error { | ||||
|     // Check if already initialized | ||||
|     if m.init { | ||||
|         return nil | ||||
| @@ -335,21 +91,28 @@ func (m *SampleCollector) Init(config []byte) error { | ||||
|             return err | ||||
|         } | ||||
|     } | ||||
|     m.meta = map[string]string{"source": m.name, "group": "Sample"} | ||||
|  | ||||
|     m.init = true | ||||
|     return nil | ||||
| } | ||||
|  | ||||
| func (m *SampleCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *SampleCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
|     if !m.init { | ||||
|         return | ||||
|     } | ||||
|     // tags for the metric, if type != node use proper type and type-id | ||||
|     tags := map[string]string{"type" : "node"} | ||||
|  | ||||
|     x, err := GetMetric() | ||||
|     if err != nil { | ||||
|         cclog.ComponentError(m.name, fmt.Sprintf("Read(): %v", err)) | ||||
|     } | ||||
|  | ||||
|     // Each metric has exactly one field: value ! | ||||
|     value := map[string]interface{}{"value": int(x)} | ||||
|     y, err := lp.New("sample_metric", tags, value, time.Now()) | ||||
|     if err == nil { | ||||
|         *out = append(*out, y) | ||||
|     value := map[string]interface{}{"value": int64(x)} | ||||
|     if y, err := lp.New("sample_metric", tags, m.meta, value, time.Now()); err == nil { | ||||
|         output <- y | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										173
									
								
								collectors/collectorManager.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								collectors/collectorManager.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	mct "github.com/ClusterCockpit/cc-metric-collector/internal/multiChanTicker" | ||||
| ) | ||||
|  | ||||
| // Map of all available metric collectors | ||||
| var AvailableCollectors = map[string]MetricCollector{ | ||||
|  | ||||
| 	"likwid":           new(LikwidCollector), | ||||
| 	"loadavg":          new(LoadavgCollector), | ||||
| 	"memstat":          new(MemstatCollector), | ||||
| 	"netstat":          new(NetstatCollector), | ||||
| 	"ibstat":           new(InfinibandCollector), | ||||
| 	"ibstat_perfquery": new(InfinibandPerfQueryCollector), | ||||
| 	"lustrestat":       new(LustreCollector), | ||||
| 	"cpustat":          new(CpustatCollector), | ||||
| 	"topprocs":         new(TopProcsCollector), | ||||
| 	"nvidia":           new(NvidiaCollector), | ||||
| 	"customcmd":        new(CustomCmdCollector), | ||||
| 	"iostat":           new(IOstatCollector), | ||||
| 	"diskstat":         new(DiskstatCollector), | ||||
| 	"tempstat":         new(TempCollector), | ||||
| 	"ipmistat":         new(IpmiCollector), | ||||
| 	"gpfs":             new(GpfsCollector), | ||||
| 	"cpufreq":          new(CPUFreqCollector), | ||||
| 	"cpufreq_cpuinfo":  new(CPUFreqCpuInfoCollector), | ||||
| 	"nfs3stat":         new(Nfs3Collector), | ||||
| 	"nfs4stat":         new(Nfs4Collector), | ||||
| 	"numastats":        new(NUMAStatsCollector), | ||||
| } | ||||
|  | ||||
| // Metric collector manager data structure | ||||
| type collectorManager struct { | ||||
| 	collectors []MetricCollector          // List of metric collectors to use | ||||
| 	output     chan lp.CCMetric           // Output channels | ||||
| 	done       chan bool                  // channel to finish / stop metric collector manager | ||||
| 	ticker     mct.MultiChanTicker        // periodically ticking once each interval | ||||
| 	duration   time.Duration              // duration (for metrics that measure over a given duration) | ||||
| 	wg         *sync.WaitGroup            // wait group for all goroutines in cc-metric-collector | ||||
| 	config     map[string]json.RawMessage // json encoded config for collector manager | ||||
| } | ||||
|  | ||||
| // Metric collector manager access functions | ||||
| type CollectorManager interface { | ||||
| 	Init(ticker mct.MultiChanTicker, duration time.Duration, wg *sync.WaitGroup, collectConfigFile string) error | ||||
| 	AddOutput(output chan lp.CCMetric) | ||||
| 	Start() | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| // Init initializes a new metric collector manager by setting up: | ||||
| // * output channel | ||||
| // * done channel | ||||
| // * wait group synchronization for goroutines (from variable wg) | ||||
| // * ticker (from variable ticker) | ||||
| // * configuration (read from config file in variable collectConfigFile) | ||||
| // Initialization is done for all configured collectors | ||||
| func (cm *collectorManager) Init(ticker mct.MultiChanTicker, duration time.Duration, wg *sync.WaitGroup, collectConfigFile string) error { | ||||
| 	cm.collectors = make([]MetricCollector, 0) | ||||
| 	cm.output = nil | ||||
| 	cm.done = make(chan bool) | ||||
| 	cm.wg = wg | ||||
| 	cm.ticker = ticker | ||||
| 	cm.duration = duration | ||||
|  | ||||
| 	// Read collector config file | ||||
| 	configFile, err := os.Open(collectConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer configFile.Close() | ||||
| 	jsonParser := json.NewDecoder(configFile) | ||||
| 	err = jsonParser.Decode(&cm.config) | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Initialize configured collectors | ||||
| 	for collectorName, collectorCfg := range cm.config { | ||||
| 		if _, found := AvailableCollectors[collectorName]; !found { | ||||
| 			cclog.ComponentError("CollectorManager", "SKIP unknown collector", collectorName) | ||||
| 			continue | ||||
| 		} | ||||
| 		collector := AvailableCollectors[collectorName] | ||||
|  | ||||
| 		err = collector.Init(collectorCfg) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError("CollectorManager", "Collector", collectorName, "initialization failed:", err.Error()) | ||||
| 			continue | ||||
| 		} | ||||
| 		cclog.ComponentDebug("CollectorManager", "ADD COLLECTOR", collector.Name()) | ||||
| 		cm.collectors = append(cm.collectors, collector) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Start starts the metric collector manager | ||||
| func (cm *collectorManager) Start() { | ||||
| 	tick := make(chan time.Time) | ||||
| 	cm.ticker.AddChannel(tick) | ||||
|  | ||||
| 	cm.wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer cm.wg.Done() | ||||
| 		// Collector manager is done | ||||
| 		done := func() { | ||||
| 			// close all metric collectors | ||||
| 			for _, c := range cm.collectors { | ||||
| 				c.Close() | ||||
| 			} | ||||
| 			close(cm.done) | ||||
| 			cclog.ComponentDebug("CollectorManager", "DONE") | ||||
| 		} | ||||
|  | ||||
| 		// Wait for done signal or timer event | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-cm.done: | ||||
| 				done() | ||||
| 				return | ||||
| 			case t := <-tick: | ||||
| 				for _, c := range cm.collectors { | ||||
| 					// Wait for done signal or execute the collector | ||||
| 					select { | ||||
| 					case <-cm.done: | ||||
| 						done() | ||||
| 						return | ||||
| 					default: | ||||
| 						// Read metrics from collector c | ||||
| 						cclog.ComponentDebug("CollectorManager", c.Name(), t) | ||||
| 						c.Read(cm.duration, cm.output) | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	// Collector manager is started | ||||
| 	cclog.ComponentDebug("CollectorManager", "STARTED") | ||||
| } | ||||
|  | ||||
| // AddOutput adds the output channel to the metric collector manager | ||||
| func (cm *collectorManager) AddOutput(output chan lp.CCMetric) { | ||||
| 	cm.output = output | ||||
| } | ||||
|  | ||||
| // Close finishes / stops the metric collector manager | ||||
| func (cm *collectorManager) Close() { | ||||
| 	cclog.ComponentDebug("CollectorManager", "CLOSE") | ||||
| 	cm.done <- true | ||||
| 	// wait for close of channel cm.done | ||||
| 	<-cm.done | ||||
| } | ||||
|  | ||||
| // New creates a new initialized metric collector manager | ||||
| func New(ticker mct.MultiChanTicker, duration time.Duration, wg *sync.WaitGroup, collectConfigFile string) (CollectorManager, error) { | ||||
| 	cm := new(collectorManager) | ||||
| 	err := cm.Init(ticker, duration, wg, collectConfigFile) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return cm, err | ||||
| } | ||||
| @@ -2,14 +2,16 @@ package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
|  | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| // | ||||
| @@ -21,41 +23,55 @@ import ( | ||||
| type CPUFreqCpuInfoCollectorTopology struct { | ||||
| 	processor               string // logical processor number (continuous, starting at 0) | ||||
| 	coreID                  string // socket local core ID | ||||
| 	coreID_int              int | ||||
| 	coreID_int              int64 | ||||
| 	physicalPackageID       string // socket / package ID | ||||
| 	physicalPackageID_int   int | ||||
| 	physicalPackageID_int   int64 | ||||
| 	numPhysicalPackages     string // number of  sockets / packages | ||||
| 	numPhysicalPackages_int int | ||||
| 	numPhysicalPackages_int int64 | ||||
| 	isHT                    bool | ||||
| 	numNonHT                string // number of non hyperthreading processors | ||||
| 	numNonHT_int            int | ||||
| 	numNonHT_int            int64 | ||||
| 	tagSet                  map[string]string | ||||
| } | ||||
|  | ||||
| type CPUFreqCpuInfoCollector struct { | ||||
| 	MetricCollector | ||||
| 	topology []CPUFreqCpuInfoCollectorTopology | ||||
| 	metricCollector | ||||
| 	topology []*CPUFreqCpuInfoCollectorTopology | ||||
| } | ||||
|  | ||||
| func (m *CPUFreqCpuInfoCollector) Init(config []byte) error { | ||||
| func (m *CPUFreqCpuInfoCollector) Init(config json.RawMessage) error { | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	m.setup() | ||||
|  | ||||
| 	m.name = "CPUFreqCpuInfoCollector" | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "CPU", | ||||
| 		"unit":   "MHz", | ||||
| 	} | ||||
|  | ||||
| 	const cpuInfoFile = "/proc/cpuinfo" | ||||
| 	file, err := os.Open(cpuInfoFile) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("Failed to open '%s': %v", cpuInfoFile, err) | ||||
| 		return fmt.Errorf("Failed to open file '%s': %v", cpuInfoFile, err) | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	// Collect topology information from file cpuinfo | ||||
| 	foundFreq := false | ||||
| 	processor := "" | ||||
| 	numNonHT_int := 0 | ||||
| 	var numNonHT_int int64 = 0 | ||||
| 	coreID := "" | ||||
| 	physicalPackageID := "" | ||||
| 	maxPhysicalPackageID := 0 | ||||
| 	m.topology = make([]CPUFreqCpuInfoCollectorTopology, 0) | ||||
| 	var maxPhysicalPackageID int64 = 0 | ||||
| 	m.topology = make([]*CPUFreqCpuInfoCollectorTopology, 0) | ||||
| 	coreSeenBefore := make(map[string]bool) | ||||
|  | ||||
| 	// Read cpuinfo file, line by line | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		lineSplit := strings.Split(scanner.Text(), ":") | ||||
| @@ -81,39 +97,41 @@ func (m *CPUFreqCpuInfoCollector) Init(config []byte) error { | ||||
| 			len(coreID) > 0 && | ||||
| 			len(physicalPackageID) > 0 { | ||||
|  | ||||
| 			coreID_int, err := strconv.Atoi(coreID) | ||||
| 			topology := new(CPUFreqCpuInfoCollectorTopology) | ||||
|  | ||||
| 			// Processor | ||||
| 			topology.processor = processor | ||||
|  | ||||
| 			// Core ID | ||||
| 			topology.coreID = coreID | ||||
| 			topology.coreID_int, err = strconv.ParseInt(coreID, 10, 64) | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("Unable to convert coreID to int: %v", err) | ||||
| 				return fmt.Errorf("Unable to convert coreID '%s' to int64: %v", coreID, err) | ||||
| 			} | ||||
| 			physicalPackageID_int, err := strconv.Atoi(physicalPackageID) | ||||
|  | ||||
| 			// Physical package ID | ||||
| 			topology.physicalPackageID = physicalPackageID | ||||
| 			topology.physicalPackageID_int, err = strconv.ParseInt(physicalPackageID, 10, 64) | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("Unable to convert physicalPackageID to int: %v", err) | ||||
| 				return fmt.Errorf("Unable to convert physicalPackageID '%s' to int64: %v", physicalPackageID, err) | ||||
| 			} | ||||
|  | ||||
| 			// increase maximun socket / package ID, when required | ||||
| 			if physicalPackageID_int > maxPhysicalPackageID { | ||||
| 				maxPhysicalPackageID = physicalPackageID_int | ||||
| 			if topology.physicalPackageID_int > maxPhysicalPackageID { | ||||
| 				maxPhysicalPackageID = topology.physicalPackageID_int | ||||
| 			} | ||||
|  | ||||
| 			// is hyperthread? | ||||
| 			globalID := physicalPackageID + ":" + coreID | ||||
| 			isHT := coreSeenBefore[globalID] | ||||
| 			topology.isHT = coreSeenBefore[globalID] | ||||
| 			coreSeenBefore[globalID] = true | ||||
| 			if !isHT { | ||||
| 			if !topology.isHT { | ||||
| 				// increase number on non hyper thread cores | ||||
| 				numNonHT_int++ | ||||
| 			} | ||||
|  | ||||
| 			// store collected topology information | ||||
| 			m.topology = append( | ||||
| 				m.topology, | ||||
| 				CPUFreqCpuInfoCollectorTopology{ | ||||
| 					processor:             processor, | ||||
| 					coreID:                coreID, | ||||
| 					coreID_int:            coreID_int, | ||||
| 					physicalPackageID:     physicalPackageID, | ||||
| 					physicalPackageID_int: physicalPackageID_int, | ||||
| 					isHT:                  isHT, | ||||
| 				}) | ||||
| 			m.topology = append(m.topology, topology) | ||||
|  | ||||
| 			// reset topology information | ||||
| 			foundFreq = false | ||||
| @@ -126,18 +144,15 @@ func (m *CPUFreqCpuInfoCollector) Init(config []byte) error { | ||||
| 	numPhysicalPackageID_int := maxPhysicalPackageID + 1 | ||||
| 	numPhysicalPackageID := fmt.Sprint(numPhysicalPackageID_int) | ||||
| 	numNonHT := fmt.Sprint(numNonHT_int) | ||||
| 	for i := range m.topology { | ||||
| 		t := &m.topology[i] | ||||
| 	for _, t := range m.topology { | ||||
| 		t.numPhysicalPackages = numPhysicalPackageID | ||||
| 		t.numPhysicalPackages_int = numPhysicalPackageID_int | ||||
| 		t.numNonHT = numNonHT | ||||
| 		t.numNonHT_int = numNonHT_int | ||||
| 		t.tagSet = map[string]string{ | ||||
| 			"type":        "cpu", | ||||
| 			"type-id":     t.processor, | ||||
| 			"num_core":    t.numNonHT, | ||||
| 			"package_id":  t.physicalPackageID, | ||||
| 			"num_package": t.numPhysicalPackages, | ||||
| 			"type":       "cpu", | ||||
| 			"type-id":    t.processor, | ||||
| 			"package_id": t.physicalPackageID, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -145,14 +160,18 @@ func (m *CPUFreqCpuInfoCollector) Init(config []byte) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *CPUFreqCpuInfoCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *CPUFreqCpuInfoCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	// Check if already initialized | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	const cpuInfoFile = "/proc/cpuinfo" | ||||
| 	file, err := os.Open(cpuInfoFile) | ||||
| 	if err != nil { | ||||
| 		log.Printf("Failed to open '%s': %v", cpuInfoFile, err) | ||||
| 		cclog.ComponentError( | ||||
| 			m.name, | ||||
| 			fmt.Sprintf("Read(): Failed to open file '%s': %v", cpuInfoFile, err)) | ||||
| 		return | ||||
| 	} | ||||
| 	defer file.Close() | ||||
| @@ -167,16 +186,17 @@ func (m *CPUFreqCpuInfoCollector) Read(interval time.Duration, out *[]lp.Mutable | ||||
|  | ||||
| 			// frequency | ||||
| 			if key == "cpu MHz" { | ||||
| 				t := &m.topology[processorCounter] | ||||
| 				t := m.topology[processorCounter] | ||||
| 				if !t.isHT { | ||||
| 					value, err := strconv.ParseFloat(strings.TrimSpace(lineSplit[1]), 64) | ||||
| 					if err != nil { | ||||
| 						log.Printf("Failed to convert cpu MHz to float: %v", err) | ||||
| 						cclog.ComponentError( | ||||
| 							m.name, | ||||
| 							fmt.Sprintf("Read(): Failed to convert cpu MHz '%s' to float64: %v", lineSplit[1], err)) | ||||
| 						return | ||||
| 					} | ||||
| 					y, err := lp.New("cpufreq", t.tagSet, map[string]interface{}{"value": value}, now) | ||||
| 					if err == nil { | ||||
| 						*out = append(*out, y) | ||||
| 					if y, err := lp.New("cpufreq", t.tagSet, m.meta, map[string]interface{}{"value": value}, now); err == nil { | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 				processorCounter++ | ||||
|   | ||||
							
								
								
									
										10
									
								
								collectors/cpufreqCpuinfoMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								collectors/cpufreqCpuinfoMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
|  | ||||
| ## `cpufreq_cpuinfo` collector | ||||
| ```json | ||||
|   "cpufreq_cpuinfo": {} | ||||
| ``` | ||||
|  | ||||
| The `cpufreq_cpuinfo` collector reads the clock frequency from `/proc/cpuinfo` and outputs a handful **cpu** metrics. | ||||
|  | ||||
| Metrics: | ||||
| * `cpufreq` | ||||
| @@ -1,48 +1,30 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"io/ioutil" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	"golang.org/x/sys/unix" | ||||
| ) | ||||
|  | ||||
| // | ||||
| // readOneLine reads one line from a file. | ||||
| // It returns ok when file was successfully read. | ||||
| // In this case text contains the first line of the files contents. | ||||
| // | ||||
| func readOneLine(filename string) (text string, ok bool) { | ||||
| 	file, err := os.Open(filename) | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	defer file.Close() | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	ok = scanner.Scan() | ||||
| 	text = scanner.Text() | ||||
| 	return | ||||
| } | ||||
|  | ||||
| type CPUFreqCollectorTopology struct { | ||||
| 	processor               string // logical processor number (continuous, starting at 0) | ||||
| 	coreID                  string // socket local core ID | ||||
| 	coreID_int              int | ||||
| 	coreID_int              int64 | ||||
| 	physicalPackageID       string // socket / package ID | ||||
| 	physicalPackageID_int   int | ||||
| 	physicalPackageID_int   int64 | ||||
| 	numPhysicalPackages     string // number of  sockets / packages | ||||
| 	numPhysicalPackages_int int | ||||
| 	numPhysicalPackages_int int64 | ||||
| 	isHT                    bool | ||||
| 	numNonHT                string // number of non hyperthreading processors | ||||
| 	numNonHT_int            int | ||||
| 	numNonHT_int            int64 | ||||
| 	scalingCurFreqFile      string | ||||
| 	tagSet                  map[string]string | ||||
| } | ||||
| @@ -56,14 +38,19 @@ type CPUFreqCollectorTopology struct { | ||||
| // See: https://www.kernel.org/doc/html/latest/admin-guide/pm/cpufreq.html | ||||
| // | ||||
| type CPUFreqCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	topology []CPUFreqCollectorTopology | ||||
| 	config   struct { | ||||
| 		ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *CPUFreqCollector) Init(config []byte) error { | ||||
| func (m *CPUFreqCollector) Init(config json.RawMessage) error { | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	m.name = "CPUFreqCollector" | ||||
| 	m.setup() | ||||
| 	if len(config) > 0 { | ||||
| @@ -72,54 +59,61 @@ func (m *CPUFreqCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "CPU", | ||||
| 		"unit":   "MHz", | ||||
| 	} | ||||
|  | ||||
| 	// Loop for all CPU directories | ||||
| 	baseDir := "/sys/devices/system/cpu" | ||||
| 	globPattern := filepath.Join(baseDir, "cpu[0-9]*") | ||||
| 	cpuDirs, err := filepath.Glob(globPattern) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("CPUFreqCollector.Init() unable to glob files with pattern %s: %v", globPattern, err) | ||||
| 		return fmt.Errorf("Unable to glob files with pattern '%s': %v", globPattern, err) | ||||
| 	} | ||||
| 	if cpuDirs == nil { | ||||
| 		return fmt.Errorf("CPUFreqCollector.Init() unable to find any files with pattern %s", globPattern) | ||||
| 		return fmt.Errorf("Unable to find any files with pattern '%s'", globPattern) | ||||
| 	} | ||||
|  | ||||
| 	// Initialize CPU topology | ||||
| 	m.topology = make([]CPUFreqCollectorTopology, len(cpuDirs)) | ||||
| 	for _, cpuDir := range cpuDirs { | ||||
| 		processor := strings.TrimPrefix(cpuDir, "/sys/devices/system/cpu/cpu") | ||||
| 		processor_int, err := strconv.Atoi(processor) | ||||
| 		processor_int, err := strconv.ParseInt(processor, 10, 64) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to convert cpuID to int: %v", err) | ||||
| 			return fmt.Errorf("Unable to convert cpuID '%s' to int64: %v", processor, err) | ||||
| 		} | ||||
|  | ||||
| 		// Read package ID | ||||
| 		physicalPackageIDFile := filepath.Join(cpuDir, "topology", "physical_package_id") | ||||
| 		physicalPackageID, ok := readOneLine(physicalPackageIDFile) | ||||
| 		if !ok { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to read physical package ID from %s", physicalPackageIDFile) | ||||
| 		} | ||||
| 		physicalPackageID_int, err := strconv.Atoi(physicalPackageID) | ||||
| 		line, err := ioutil.ReadFile(physicalPackageIDFile) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to convert packageID to int: %v", err) | ||||
| 			return fmt.Errorf("Unable to read physical package ID from file '%s': %v", physicalPackageIDFile, err) | ||||
| 		} | ||||
| 		physicalPackageID := strings.TrimSpace(string(line)) | ||||
| 		physicalPackageID_int, err := strconv.ParseInt(physicalPackageID, 10, 64) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("Unable to convert packageID '%s' to int64: %v", physicalPackageID, err) | ||||
| 		} | ||||
|  | ||||
| 		// Read core ID | ||||
| 		coreIDFile := filepath.Join(cpuDir, "topology", "core_id") | ||||
| 		coreID, ok := readOneLine(coreIDFile) | ||||
| 		if !ok { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to read core ID from %s", coreIDFile) | ||||
| 		} | ||||
| 		coreID_int, err := strconv.Atoi(coreID) | ||||
| 		line, err = ioutil.ReadFile(coreIDFile) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to convert coreID to int: %v", err) | ||||
| 			return fmt.Errorf("Unable to read core ID from file '%s': %v", coreIDFile, err) | ||||
| 		} | ||||
| 		coreID := strings.TrimSpace(string(line)) | ||||
| 		coreID_int, err := strconv.ParseInt(coreID, 10, 64) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("Unable to convert coreID '%s' to int64: %v", coreID, err) | ||||
| 		} | ||||
|  | ||||
| 		// Check access to current frequency file | ||||
| 		scalingCurFreqFile := filepath.Join(cpuDir, "cpufreq", "scaling_cur_freq") | ||||
| 		err = unix.Access(scalingCurFreqFile, unix.R_OK) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("CPUFreqCollector.Init() unable to access %s: %v", scalingCurFreqFile, err) | ||||
| 			return fmt.Errorf("Unable to access file '%s': %v", scalingCurFreqFile, err) | ||||
| 		} | ||||
|  | ||||
| 		t := &m.topology[processor_int] | ||||
| @@ -142,8 +136,8 @@ func (m *CPUFreqCollector) Init(config []byte) error { | ||||
| 	} | ||||
|  | ||||
| 	// number of non hyper thread cores and packages / sockets | ||||
| 	numNonHT_int := 0 | ||||
| 	maxPhysicalPackageID := 0 | ||||
| 	var numNonHT_int int64 = 0 | ||||
| 	var maxPhysicalPackageID int64 = 0 | ||||
| 	for i := range m.topology { | ||||
| 		t := &m.topology[i] | ||||
|  | ||||
| @@ -167,11 +161,9 @@ func (m *CPUFreqCollector) Init(config []byte) error { | ||||
| 		t.numNonHT = numNonHT | ||||
| 		t.numNonHT_int = numNonHT_int | ||||
| 		t.tagSet = map[string]string{ | ||||
| 			"type":        "cpu", | ||||
| 			"type-id":     t.processor, | ||||
| 			"num_core":    t.numNonHT, | ||||
| 			"package_id":  t.physicalPackageID, | ||||
| 			"num_package": t.numPhysicalPackages, | ||||
| 			"type":       "cpu", | ||||
| 			"type-id":    t.processor, | ||||
| 			"package_id": t.physicalPackageID, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -179,7 +171,8 @@ func (m *CPUFreqCollector) Init(config []byte) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *CPUFreqCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *CPUFreqCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	// Check if already initialized | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -194,20 +187,23 @@ func (m *CPUFreqCollector) Read(interval time.Duration, out *[]lp.MutableMetric) | ||||
| 		} | ||||
|  | ||||
| 		// Read current frequency | ||||
| 		line, ok := readOneLine(t.scalingCurFreqFile) | ||||
| 		if !ok { | ||||
| 			log.Printf("CPUFreqCollector.Read(): Failed to read one line from file '%s'", t.scalingCurFreqFile) | ||||
| 		line, err := ioutil.ReadFile(t.scalingCurFreqFile) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to read file '%s': %v", t.scalingCurFreqFile, err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		cpuFreq, err := strconv.Atoi(line) | ||||
| 		cpuFreq, err := strconv.ParseInt(strings.TrimSpace(string(line)), 10, 64) | ||||
| 		if err != nil { | ||||
| 			log.Printf("CPUFreqCollector.Read(): Failed to convert CPU frequency '%s': %v", line, err) | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert CPU frequency '%s' to int64: %v", line, err)) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		y, err := lp.New("cpufreq", t.tagSet, map[string]interface{}{"value": cpuFreq}, now) | ||||
| 		if err == nil { | ||||
| 			*out = append(*out, y) | ||||
| 		if y, err := lp.New("cpufreq", t.tagSet, m.meta, map[string]interface{}{"value": cpuFreq}, now); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										11
									
								
								collectors/cpufreqMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								collectors/cpufreqMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| ## `cpufreq_cpuinfo` collector | ||||
| ```json | ||||
|   "cpufreq": { | ||||
|     "exclude_metrics": [] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `cpufreq` collector reads the clock frequency from `/sys/devices/system/cpu/cpu*/cpufreq` and outputs a handful **cpu** metrics. | ||||
|  | ||||
| Metrics: | ||||
| * `cpufreq` | ||||
| @@ -1,14 +1,16 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const CPUSTATFILE = `/proc/stat` | ||||
| @@ -18,72 +20,129 @@ type CpustatCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type CpustatCollector struct { | ||||
| 	MetricCollector | ||||
| 	config CpustatCollectorConfig | ||||
| 	metricCollector | ||||
| 	config          CpustatCollectorConfig | ||||
| 	matches         map[string]int | ||||
| 	cputags         map[string]map[string]string | ||||
| 	nodetags        map[string]string | ||||
| 	num_cpus_metric lp.CCMetric | ||||
| } | ||||
|  | ||||
| func (m *CpustatCollector) Init(config []byte) error { | ||||
| func (m *CpustatCollector) Init(config json.RawMessage) error { | ||||
| 	m.name = "CpustatCollector" | ||||
| 	m.setup() | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "CPU", "unit": "Percent"} | ||||
| 	m.nodetags = map[string]string{"type": "node"} | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	matches := map[string]int{ | ||||
| 		"cpu_user":       1, | ||||
| 		"cpu_nice":       2, | ||||
| 		"cpu_system":     3, | ||||
| 		"cpu_idle":       4, | ||||
| 		"cpu_iowait":     5, | ||||
| 		"cpu_irq":        6, | ||||
| 		"cpu_softirq":    7, | ||||
| 		"cpu_steal":      8, | ||||
| 		"cpu_guest":      9, | ||||
| 		"cpu_guest_nice": 10, | ||||
| 	} | ||||
|  | ||||
| 	m.matches = make(map[string]int) | ||||
| 	for match, index := range matches { | ||||
| 		doExclude := false | ||||
| 		for _, exclude := range m.config.ExcludeMetrics { | ||||
| 			if match == exclude { | ||||
| 				doExclude = true | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 		if !doExclude { | ||||
| 			m.matches[match] = index | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Check input file | ||||
| 	file, err := os.Open(string(CPUSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	// Pre-generate tags for all CPUs | ||||
| 	num_cpus := 0 | ||||
| 	m.cputags = make(map[string]map[string]string) | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		linefields := strings.Fields(line) | ||||
| 		if strings.HasPrefix(linefields[0], "cpu") && strings.Compare(linefields[0], "cpu") != 0 { | ||||
| 			cpustr := strings.TrimLeft(linefields[0], "cpu") | ||||
| 			cpu, _ := strconv.Atoi(cpustr) | ||||
| 			m.cputags[linefields[0]] = map[string]string{"type": "cpu", "type-id": fmt.Sprintf("%d", cpu)} | ||||
| 			num_cpus++ | ||||
| 		} | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func ParseStatLine(line string, cpu int, exclude []string, out *[]lp.MutableMetric) { | ||||
| 	ls := strings.Fields(line) | ||||
| 	matches := []string{"", "cpu_user", "cpu_nice", "cpu_system", "cpu_idle", "cpu_iowait", "cpu_irq", "cpu_softirq", "cpu_steal", "cpu_guest", "cpu_guest_nice"} | ||||
| 	for _, ex := range exclude { | ||||
| 		matches, _ = RemoveFromStringList(matches, ex) | ||||
| 	} | ||||
|  | ||||
| 	var tags map[string]string | ||||
| 	if cpu < 0 { | ||||
| 		tags = map[string]string{"type": "node"} | ||||
| 	} else { | ||||
| 		tags = map[string]string{"type": "cpu", "type-id": fmt.Sprintf("%d", cpu)} | ||||
| 	} | ||||
| 	for i, m := range matches { | ||||
| 		if len(m) > 0 { | ||||
| 			x, err := strconv.ParseInt(ls[i], 0, 64) | ||||
| func (m *CpustatCollector) parseStatLine(linefields []string, tags map[string]string, output chan lp.CCMetric) { | ||||
| 	values := make(map[string]float64) | ||||
| 	total := 0.0 | ||||
| 	for match, index := range m.matches { | ||||
| 		if len(match) > 0 { | ||||
| 			x, err := strconv.ParseInt(linefields[index], 0, 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New(m, tags, map[string]interface{}{"value": int(x)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 				values[match] = float64(x) | ||||
| 				total += values[match] | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	t := time.Now() | ||||
| 	for name, value := range values { | ||||
| 		y, err := lp.New(name, tags, m.meta, map[string]interface{}{"value": (value * 100.0) / total}, t) | ||||
| 		if err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *CpustatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *CpustatCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	buffer, err := ioutil.ReadFile(string(CPUSTATFILE)) | ||||
|  | ||||
| 	num_cpus := 0 | ||||
| 	file, err := os.Open(string(CPUSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		linefields := strings.Fields(line) | ||||
| 		if strings.Compare(linefields[0], "cpu") == 0 { | ||||
| 			m.parseStatLine(linefields, m.nodetags, output) | ||||
| 		} else if strings.HasPrefix(linefields[0], "cpu") { | ||||
| 			m.parseStatLine(linefields, m.cputags[linefields[0]], output) | ||||
| 			num_cpus++ | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	ll := strings.Split(string(buffer), "\n") | ||||
| 	for _, line := range ll { | ||||
| 		if len(line) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		ls := strings.Fields(line) | ||||
| 		if strings.Compare(ls[0], "cpu") == 0 { | ||||
| 			ParseStatLine(line, -1, m.config.ExcludeMetrics, out) | ||||
| 		} else if strings.HasPrefix(ls[0], "cpu") { | ||||
| 			cpustr := strings.TrimLeft(ls[0], "cpu") | ||||
| 			cpu, _ := strconv.Atoi(cpustr) | ||||
| 			ParseStatLine(line, cpu, m.config.ExcludeMetrics, out) | ||||
| 		} | ||||
| 	num_cpus_metric, err := lp.New("num_cpus", | ||||
| 		m.nodetags, | ||||
| 		m.meta, | ||||
| 		map[string]interface{}{"value": int(num_cpus)}, | ||||
| 		time.Now(), | ||||
| 	) | ||||
| 	if err == nil { | ||||
| 		output <- num_cpus_metric | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										23
									
								
								collectors/cpustatMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								collectors/cpustatMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
|  | ||||
| ## `cpustat` collector | ||||
| ```json | ||||
|   "cpustat": { | ||||
|     "exclude_metrics": [ | ||||
|       "cpu_idle" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `cpustat` collector reads data from `/proc/stats` and outputs a handful **node** and **hwthread** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `cpu_user` | ||||
| * `cpu_nice` | ||||
| * `cpu_system` | ||||
| * `cpu_idle` | ||||
| * `cpu_iowait` | ||||
| * `cpu_irq` | ||||
| * `cpu_softirq` | ||||
| * `cpu_steal` | ||||
| * `cpu_guest` | ||||
| * `cpu_guest_nice` | ||||
| @@ -9,7 +9,13 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| <<<<<<< HEAD | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| ======= | ||||
| 	ccmetric "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influx "github.com/influxdata/line-protocol" | ||||
| >>>>>>> develop | ||||
| ) | ||||
|  | ||||
| const CUSTOMCMDPATH = `/home/unrz139/Work/cc-metric-collector/collectors/custom` | ||||
| @@ -21,17 +27,18 @@ type CustomCmdCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type CustomCmdCollector struct { | ||||
| 	MetricCollector | ||||
| 	handler  *lp.MetricHandler | ||||
| 	parser   *lp.Parser | ||||
| 	metricCollector | ||||
| 	handler  *influx.MetricHandler | ||||
| 	parser   *influx.Parser | ||||
| 	config   CustomCmdCollectorConfig | ||||
| 	commands []string | ||||
| 	files    []string | ||||
| } | ||||
|  | ||||
| func (m *CustomCmdCollector) Init(config []byte) error { | ||||
| func (m *CustomCmdCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "CustomCmdCollector" | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Custom"} | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| @@ -61,8 +68,8 @@ func (m *CustomCmdCollector) Init(config []byte) error { | ||||
| 	if len(m.files) == 0 && len(m.commands) == 0 { | ||||
| 		return errors.New("No metrics to collect") | ||||
| 	} | ||||
| 	m.handler = lp.NewMetricHandler() | ||||
| 	m.parser = lp.NewParser(m.handler) | ||||
| 	m.handler = influx.NewMetricHandler() | ||||
| 	m.parser = influx.NewParser(m.handler) | ||||
| 	m.parser.SetTimeFunc(DefaultTime) | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| @@ -72,7 +79,7 @@ var DefaultTime = func() time.Time { | ||||
| 	return time.Unix(42, 0) | ||||
| } | ||||
|  | ||||
| func (m *CustomCmdCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *CustomCmdCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -95,9 +102,10 @@ func (m *CustomCmdCollector) Read(interval time.Duration, out *[]lp.MutableMetri | ||||
| 			if skip { | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err := lp.New(c.Name(), Tags2Map(c), Fields2Map(c), c.Time()) | ||||
|  | ||||
| 			y := ccmetric.FromInfluxMetric(c) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| @@ -117,9 +125,9 @@ func (m *CustomCmdCollector) Read(interval time.Duration, out *[]lp.MutableMetri | ||||
| 			if skip { | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err := lp.New(f.Name(), Tags2Map(f), Fields2Map(f), f.Time()) | ||||
| 			y := ccmetric.FromInfluxMetric(f) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										20
									
								
								collectors/customCmdMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								collectors/customCmdMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
|  | ||||
| ## `customcmd` collector | ||||
|  | ||||
| ```json | ||||
|   "customcmd": { | ||||
|     "exclude_metrics": [ | ||||
|       "mymetric" | ||||
|     ], | ||||
|     "files" : [ | ||||
|       "/var/run/myapp.metrics" | ||||
|     ], | ||||
|     "commands" : [ | ||||
|       "/usr/local/bin/getmetrics.pl" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `customcmd` collector reads data from files and the output of executed commands. The files and commands can output multiple metrics (separated by newline) but the have to be in the [InfluxDB line protocol](https://docs.influxdata.com/influxdb/cloud/reference/syntax/line-protocol/). If a metric is not parsable, it is skipped. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
|  | ||||
| @@ -1,113 +1,111 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"io/ioutil" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
|  | ||||
| 	//	"log" | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"strconv" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const DISKSTATFILE = `/proc/diskstats` | ||||
| const DISKSTAT_SYSFSPATH = `/sys/block` | ||||
| //	"log" | ||||
|  | ||||
| const MOUNTFILE = `/proc/self/mounts` | ||||
|  | ||||
| type DiskstatCollectorConfig struct { | ||||
| 	ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| } | ||||
|  | ||||
| type DiskstatCollector struct { | ||||
| 	MetricCollector | ||||
| 	matches map[int]string | ||||
| 	config  DiskstatCollectorConfig | ||||
| 	metricCollector | ||||
| 	//matches map[string]int | ||||
| 	config IOstatCollectorConfig | ||||
| 	//devices map[string]IOstatCollectorEntry | ||||
| } | ||||
|  | ||||
| func (m *DiskstatCollector) Init(config []byte) error { | ||||
| 	var err error | ||||
| func (m *DiskstatCollector) Init(config json.RawMessage) error { | ||||
| 	m.name = "DiskstatCollector" | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Disk"} | ||||
| 	m.setup() | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	// https://www.kernel.org/doc/html/latest/admin-guide/iostats.html | ||||
| 	matches := map[int]string{ | ||||
| 		3:  "reads", | ||||
| 		4:  "reads_merged", | ||||
| 		5:  "read_sectors", | ||||
| 		6:  "read_ms", | ||||
| 		7:  "writes", | ||||
| 		8:  "writes_merged", | ||||
| 		9:  "writes_sectors", | ||||
| 		10: "writes_ms", | ||||
| 		11: "ioops", | ||||
| 		12: "ioops_ms", | ||||
| 		13: "ioops_weighted_ms", | ||||
| 		14: "discards", | ||||
| 		15: "discards_merged", | ||||
| 		16: "discards_sectors", | ||||
| 		17: "discards_ms", | ||||
| 		18: "flushes", | ||||
| 		19: "flushes_ms", | ||||
| 	file, err := os.Open(string(MOUNTFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	m.matches = make(map[int]string) | ||||
| 	for k, v := range matches { | ||||
| 		_, skip := stringArrayContains(m.config.ExcludeMetrics, v) | ||||
| 		if !skip { | ||||
| 			m.matches[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	if len(m.matches) == 0 { | ||||
| 		return errors.New("No metrics to collect") | ||||
| 	} | ||||
| 	_, err = ioutil.ReadFile(string(DISKSTATFILE)) | ||||
| 	if err == nil { | ||||
| 		m.init = true | ||||
| 	} | ||||
| 	return err | ||||
| 	defer file.Close() | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *DiskstatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| 	var lines []string | ||||
| func (m *DiskstatCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	buffer, err := ioutil.ReadFile(string(DISKSTATFILE)) | ||||
| 	file, err := os.Open(string(MOUNTFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	lines = strings.Split(string(buffer), "\n") | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	for _, line := range lines { | ||||
| 	part_max_used := uint64(0) | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		if len(line) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		f := strings.Fields(line) | ||||
| 		if strings.Contains(f[2], "loop") { | ||||
| 		if !strings.HasPrefix(line, "/dev") { | ||||
| 			continue | ||||
| 		} | ||||
| 		tags := map[string]string{ | ||||
| 			"device": f[2], | ||||
| 			"type":   "node", | ||||
| 		linefields := strings.Fields(line) | ||||
| 		if strings.Contains(linefields[0], "loop") { | ||||
| 			continue | ||||
| 		} | ||||
| 		for idx, name := range m.matches { | ||||
| 			if idx < len(f) { | ||||
| 				x, err := strconv.ParseInt(f[idx], 0, 64) | ||||
| 				if err == nil { | ||||
| 					y, err := lp.New(name, tags, map[string]interface{}{"value": int(x)}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						*out = append(*out, y) | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		if strings.Contains(linefields[1], "boot") { | ||||
| 			continue | ||||
| 		} | ||||
| 		path := strings.Replace(linefields[1], `\040`, " ", -1) | ||||
| 		stat := syscall.Statfs_t{} | ||||
| 		err := syscall.Statfs(path, &stat) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 		tags := map[string]string{"type": "node", "device": linefields[0]} | ||||
| 		total := (stat.Blocks * uint64(stat.Bsize)) / uint64(1000000000) | ||||
| 		y, err := lp.New("disk_total", tags, m.meta, map[string]interface{}{"value": total}, time.Now()) | ||||
| 		if err == nil { | ||||
| 			y.AddMeta("unit", "GBytes") | ||||
| 			output <- y | ||||
| 		} | ||||
| 		free := (stat.Bfree * uint64(stat.Bsize)) / uint64(1000000000) | ||||
| 		y, err = lp.New("disk_free", tags, m.meta, map[string]interface{}{"value": free}, time.Now()) | ||||
| 		if err == nil { | ||||
| 			y.AddMeta("unit", "GBytes") | ||||
| 			output <- y | ||||
| 		} | ||||
| 		perc := (100 * (total - free)) / total | ||||
| 		if perc > part_max_used { | ||||
| 			part_max_used = perc | ||||
| 		} | ||||
| 	} | ||||
| 	y, err := lp.New("part_max_used", map[string]string{"type": "node"}, m.meta, map[string]interface{}{"value": part_max_used}, time.Now()) | ||||
| 	if err == nil { | ||||
| 		y.AddMeta("unit", "percent") | ||||
| 		output <- y | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										21
									
								
								collectors/diskstatMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								collectors/diskstatMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
|  | ||||
| ## `diskstat` collector | ||||
|  | ||||
| ```json | ||||
|   "diskstat": { | ||||
|     "exclude_metrics": [ | ||||
|       "disk_total" | ||||
|     ], | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `diskstat` collector reads data from `/proc/self/mounts` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics per device (with `device` tag): | ||||
| * `disk_total` (unit `GBytes`) | ||||
| * `disk_free` (unit `GBytes`) | ||||
|  | ||||
| Global metrics: | ||||
| * `part_max_used` (unit `percent`) | ||||
|  | ||||
|  | ||||
| @@ -7,24 +7,32 @@ import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"os/user" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type GpfsCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	tags   map[string]string | ||||
| 	config struct { | ||||
| 		Mmpmon string `json:"mmpmon"` | ||||
| 		Mmpmon            string   `json:"mmpmon_path,omitempty"` | ||||
| 		ExcludeFilesystem []string `json:"exclude_filesystem,omitempty"` | ||||
| 	} | ||||
| 	skipFS map[string]struct{} | ||||
| } | ||||
|  | ||||
| func (m *GpfsCollector) Init(config []byte) error { | ||||
| func (m *GpfsCollector) Init(config json.RawMessage) error { | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	var err error | ||||
| 	m.name = "GpfsCollector" | ||||
| 	m.setup() | ||||
| @@ -40,27 +48,40 @@ func (m *GpfsCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "GPFS", | ||||
| 	} | ||||
| 	m.tags = map[string]string{ | ||||
| 		"type":       "node", | ||||
| 		"filesystem": "", | ||||
| 	} | ||||
| 	m.skipFS = make(map[string]struct{}) | ||||
| 	for _, fs := range m.config.ExcludeFilesystem { | ||||
| 		m.skipFS[fs] = struct{}{} | ||||
| 	} | ||||
|  | ||||
| 	// GPFS / IBM Spectrum Scale file system statistics can only be queried by user root | ||||
| 	user, err := user.Current() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("GpfsCollector.Init(): Failed to get current user: %v", err) | ||||
| 		return fmt.Errorf("Failed to get current user: %v", err) | ||||
| 	} | ||||
| 	if user.Uid != "0" { | ||||
| 		return fmt.Errorf("GpfsCollector.Init(): GPFS file system statistics can only be queried by user root") | ||||
| 		return fmt.Errorf("GPFS file system statistics can only be queried by user root") | ||||
| 	} | ||||
|  | ||||
| 	// Check if mmpmon is in executable search path | ||||
| 	_, err = exec.LookPath(m.config.Mmpmon) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("GpfsCollector.Init(): Failed to find mmpmon binary '%s': %v", m.config.Mmpmon, err) | ||||
| 		return fmt.Errorf("Failed to find mmpmon binary '%s': %v", m.config.Mmpmon, err) | ||||
| 	} | ||||
|  | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *GpfsCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *GpfsCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	// Check if already initialized | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -77,12 +98,15 @@ func (m *GpfsCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| 	cmd.Stderr = cmdStderr | ||||
| 	err := cmd.Run() | ||||
| 	if err != nil { | ||||
| 		fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to execute command \"%s\": %s\n", cmd.String(), err.Error()) | ||||
| 		fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): command exit code: \"%d\"\n", cmd.ProcessState.ExitCode()) | ||||
| 		data, _ := ioutil.ReadAll(cmdStderr) | ||||
| 		fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): command stderr: \"%s\"\n", string(data)) | ||||
| 		data, _ = ioutil.ReadAll(cmdStdout) | ||||
| 		fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): command stdout: \"%s\"\n", string(data)) | ||||
| 		dataStdErr, _ := ioutil.ReadAll(cmdStderr) | ||||
| 		dataStdOut, _ := ioutil.ReadAll(cmdStdout) | ||||
| 		cclog.ComponentError( | ||||
| 			m.name, | ||||
| 			fmt.Sprintf("Read(): Failed to execute command \"%s\": %v\n", cmd.String(), err), | ||||
| 			fmt.Sprintf("Read(): command exit code: \"%d\"\n", cmd.ProcessState.ExitCode()), | ||||
| 			fmt.Sprintf("Read(): command stderr: \"%s\"\n", string(dataStdErr)), | ||||
| 			fmt.Sprintf("Read(): command stdout: \"%s\"\n", string(dataStdOut)), | ||||
| 		) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| @@ -90,194 +114,163 @@ func (m *GpfsCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| 	scanner := bufio.NewScanner(cmdStdout) | ||||
| 	for scanner.Scan() { | ||||
| 		lineSplit := strings.Fields(scanner.Text()) | ||||
| 		if lineSplit[0] == "_fs_io_s_" { | ||||
| 			key_value := make(map[string]string) | ||||
| 			for i := 1; i < len(lineSplit); i += 2 { | ||||
| 				key_value[lineSplit[i]] = lineSplit[i+1] | ||||
| 			} | ||||
|  | ||||
| 			// Ignore keys: | ||||
| 			// _n_:  node IP address, | ||||
| 			// _nn_: node name, | ||||
| 			// _cl_: cluster name, | ||||
| 			// _d_:  number of disks | ||||
| 		// Only process lines starting with _fs_io_s_ | ||||
| 		if lineSplit[0] != "_fs_io_s_" { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 			filesystem, ok := key_value["_fs_"] | ||||
| 			if !ok { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to get filesystem name.\n") | ||||
| 				continue | ||||
| 			} | ||||
| 		key_value := make(map[string]string) | ||||
| 		for i := 1; i < len(lineSplit); i += 2 { | ||||
| 			key_value[lineSplit[i]] = lineSplit[i+1] | ||||
| 		} | ||||
|  | ||||
| 			tagList := map[string]string{ | ||||
| 				"type":       "node", | ||||
| 				"filesystem": filesystem, | ||||
| 			} | ||||
| 		// Ignore keys: | ||||
| 		// _n_:  node IP address, | ||||
| 		// _nn_: node name, | ||||
| 		// _cl_: cluster name, | ||||
| 		// _d_:  number of disks | ||||
|  | ||||
| 			// return code | ||||
| 			rc, err := strconv.Atoi(key_value["_rc_"]) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert return code: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			if rc != 0 { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Filesystem %s not ok.", filesystem) | ||||
| 				continue | ||||
| 			} | ||||
| 		filesystem, ok := key_value["_fs_"] | ||||
| 		if !ok { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				"Read(): Failed to get filesystem name.") | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 			/* requires go 1.17 | ||||
| 			// unix epoch in microseconds | ||||
| 			timestampInt, err := strconv.ParseInt(key_value["_t_"]+key_value["_tu_"], 10, 64) | ||||
| 			timestamp := time.UnixMicro(timestampInt) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, | ||||
| 					"GpfsCollector.Read(): Failed to convert time stamp '%s': %s\n", | ||||
| 					key_value["_t_"]+key_value["_tu_"], err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			*/ | ||||
| 			timestamp := time.Now() | ||||
| 		// Skip excluded filesystems | ||||
| 		if _, skip := m.skipFS[filesystem]; skip { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 			// bytes read | ||||
| 			bytesRead, err := strconv.ParseInt(key_value["_br_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, | ||||
| 					"GpfsCollector.Read(): Failed to convert bytes read '%s': %s\n", | ||||
| 					key_value["_br_"], err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err := lp.New( | ||||
| 				"gpfs_bytes_read", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": bytesRead, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		m.tags["filesystem"] = filesystem | ||||
|  | ||||
| 			// bytes written | ||||
| 			bytesWritten, err := strconv.ParseInt(key_value["_bw_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, | ||||
| 					"GpfsCollector.Read(): Failed to convert bytes written '%s': %s\n", | ||||
| 					key_value["_bw_"], err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_bytes_written", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": bytesWritten, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// return code | ||||
| 		rc, err := strconv.Atoi(key_value["_rc_"]) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert return code '%s' to int: %v", key_value["_rc_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if rc != 0 { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Filesystem '%s' is not ok.", filesystem)) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 			// number of opens | ||||
| 			numOpens, err := strconv.ParseInt(key_value["_oc_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, | ||||
| 					"GpfsCollector.Read(): Failed to convert number of opens '%s': %s\n", | ||||
| 					key_value["_oc_"], err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_opens", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numOpens, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		sec, err := strconv.ParseInt(key_value["_t_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert seconds '%s' to int64: %v", key_value["_t_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		msec, err := strconv.ParseInt(key_value["_tu_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert micro seconds '%s' to int64: %v", key_value["_tu_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		timestamp := time.Unix(sec, msec*1000) | ||||
|  | ||||
| 			// number of closes | ||||
| 			numCloses, err := strconv.ParseInt(key_value["_cc_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert number of closes: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_closes", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numCloses, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// bytes read | ||||
| 		bytesRead, err := strconv.ParseInt(key_value["_br_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert bytes read '%s' to int64: %v", key_value["_br_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_bytes_read", m.tags, m.meta, map[string]interface{}{"value": bytesRead}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 			// number of reads | ||||
| 			numReads, err := strconv.ParseInt(key_value["_rdc_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert number of reads: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_reads", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numReads, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// bytes written | ||||
| 		bytesWritten, err := strconv.ParseInt(key_value["_bw_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert bytes written '%s' to int64: %v", key_value["_bw_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_bytes_written", m.tags, m.meta, map[string]interface{}{"value": bytesWritten}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 			// number of writes | ||||
| 			numWrites, err := strconv.ParseInt(key_value["_wc_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert number of writes: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_writes", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numWrites, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// number of opens | ||||
| 		numOpens, err := strconv.ParseInt(key_value["_oc_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of opens '%s' to int64: %v", key_value["_oc_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_opens", m.tags, m.meta, map[string]interface{}{"value": numOpens}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 			// number of read directories | ||||
| 			numReaddirs, err := strconv.ParseInt(key_value["_dir_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert number of read directories: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_readdirs", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numReaddirs, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// number of closes | ||||
| 		numCloses, err := strconv.ParseInt(key_value["_cc_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of closes: '%s' to int64: %v", key_value["_cc_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_closes", m.tags, m.meta, map[string]interface{}{"value": numCloses}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 			// Number of inode updates | ||||
| 			numInodeUpdates, err := strconv.ParseInt(key_value["_iu_"], 10, 64) | ||||
| 			if err != nil { | ||||
| 				fmt.Fprintf(os.Stderr, "GpfsCollector.Read(): Failed to convert Number of inode updates: %s\n", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err = lp.New( | ||||
| 				"gpfs_num_inode_updates", | ||||
| 				tagList, | ||||
| 				map[string]interface{}{ | ||||
| 					"value": numInodeUpdates, | ||||
| 				}, | ||||
| 				timestamp) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		// number of reads | ||||
| 		numReads, err := strconv.ParseInt(key_value["_rdc_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of reads: '%s' to int64: %v", key_value["_rdc_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_reads", m.tags, m.meta, map[string]interface{}{"value": numReads}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 		// number of writes | ||||
| 		numWrites, err := strconv.ParseInt(key_value["_wc_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of writes: '%s' to int64: %v", key_value["_wc_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_writes", m.tags, m.meta, map[string]interface{}{"value": numWrites}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 		// number of read directories | ||||
| 		numReaddirs, err := strconv.ParseInt(key_value["_dir_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of read directories: '%s' to int64: %v", key_value["_dir_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_readdirs", m.tags, m.meta, map[string]interface{}{"value": numReaddirs}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 		// Number of inode updates | ||||
| 		numInodeUpdates, err := strconv.ParseInt(key_value["_iu_"], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert number of inode updates: '%s' to int: %v", key_value["_iu_"], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if y, err := lp.New("gpfs_num_inode_updates", m.tags, m.meta, map[string]interface{}{"value": numInodeUpdates}, timestamp); err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										30
									
								
								collectors/gpfsMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								collectors/gpfsMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| ## `gpfs` collector | ||||
|  | ||||
| ```json | ||||
|   "ibstat": { | ||||
|     "mmpmon_path": "/path/to/mmpmon", | ||||
|     "exclude_filesystem": [ | ||||
|       "fs1" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `gpfs` collector uses the `mmpmon` command to read performance metrics for | ||||
| GPFS / IBM Spectrum Scale filesystems. | ||||
|  | ||||
| The reported filesystems can be filtered with the `exclude_filesystem` option | ||||
| in the configuration. | ||||
|  | ||||
| The path to the `mmpmon` command can be configured with the `mmpmon_path` option | ||||
| in the configuration. | ||||
|  | ||||
| Metrics: | ||||
| * `bytes_read` | ||||
| * `gpfs_bytes_written` | ||||
| * `gpfs_num_opens` | ||||
| * `gpfs_num_closes` | ||||
| * `gpfs_num_reads` | ||||
| * `gpfs_num_readdirs` | ||||
| * `gpfs_num_inode_updates` | ||||
|  | ||||
| The collector adds a `filesystem` tag to all metrics | ||||
| @@ -3,282 +3,168 @@ package collectors | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os/exec" | ||||
| 	"os" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	"golang.org/x/sys/unix" | ||||
|  | ||||
| 	//	"os" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	IBBASEPATH = `/sys/class/infiniband/` | ||||
| 	PERFQUERY  = `/usr/sbin/perfquery` | ||||
| ) | ||||
| const IB_BASEPATH = `/sys/class/infiniband/` | ||||
|  | ||||
| type InfinibandCollectorConfig struct { | ||||
| 	ExcludeDevices []string `json:"exclude_devices,omitempty"` | ||||
| 	PerfQueryPath  string   `json:"perfquery_path"` | ||||
| type InfinibandCollectorInfo struct { | ||||
| 	LID              string            // IB local Identifier (LID) | ||||
| 	device           string            // IB device | ||||
| 	port             string            // IB device port | ||||
| 	portCounterFiles map[string]string // mapping counter name -> sysfs file | ||||
| 	tagSet           map[string]string // corresponding tag list | ||||
| } | ||||
|  | ||||
| type InfinibandCollector struct { | ||||
| 	MetricCollector | ||||
| 	tags          map[string]string | ||||
| 	lids          map[string]map[string]string | ||||
| 	config        InfinibandCollectorConfig | ||||
| 	use_perfquery bool | ||||
| 	metricCollector | ||||
| 	config struct { | ||||
| 		ExcludeDevices []string `json:"exclude_devices,omitempty"` // IB device to exclude e.g. mlx5_0 | ||||
| 	} | ||||
| 	info []*InfinibandCollectorInfo | ||||
| } | ||||
|  | ||||
| func (m *InfinibandCollector) Help() { | ||||
| 	fmt.Println("This collector includes all devices that can be found below ", IBBASEPATH) | ||||
| 	fmt.Println("and where any of the ports provides a 'lid' file (glob ", IBBASEPATH, "/<dev>/ports/<port>/lid).") | ||||
| 	fmt.Println("The devices can be filtered with the 'exclude_devices' option in the configuration.") | ||||
| 	fmt.Println("For each found LIDs the collector calls the 'perfquery' command") | ||||
| 	fmt.Println("The path to the 'perfquery' command can be configured with the 'perfquery_path' option") | ||||
| 	fmt.Println("in the configuration") | ||||
| 	fmt.Println("") | ||||
| 	fmt.Println("Full configuration object:") | ||||
| 	fmt.Println("\"ibstat\" : {") | ||||
| 	fmt.Println("  \"perfquery_path\" : \"path/to/perfquery\"  # if omitted, it searches in $PATH") | ||||
| 	fmt.Println("  \"exclude_devices\" : [\"dev1\"]") | ||||
| 	fmt.Println("}") | ||||
| 	fmt.Println("") | ||||
| 	fmt.Println("Metrics:") | ||||
| 	fmt.Println("- ib_recv") | ||||
| 	fmt.Println("- ib_xmit") | ||||
| 	fmt.Println("- ib_recv_pkts") | ||||
| 	fmt.Println("- ib_xmit_pkts") | ||||
| } | ||||
| // Init initializes the Infiniband collector by walking through files below IB_BASEPATH | ||||
| func (m *InfinibandCollector) Init(config json.RawMessage) error { | ||||
|  | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| func (m *InfinibandCollector) Init(config []byte) error { | ||||
| 	var err error | ||||
| 	m.name = "InfinibandCollector" | ||||
| 	m.use_perfquery = false | ||||
| 	m.setup() | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "Network", | ||||
| 	} | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(m.config.PerfQueryPath) == 0 { | ||||
| 		path, err := exec.LookPath("perfquery") | ||||
| 		if err == nil { | ||||
| 			m.config.PerfQueryPath = path | ||||
| 		} | ||||
| 	} | ||||
| 	m.lids = make(map[string]map[string]string) | ||||
| 	p := fmt.Sprintf("%s/*/ports/*/lid", string(IBBASEPATH)) | ||||
| 	files, err := filepath.Glob(p) | ||||
| 	for _, f := range files { | ||||
| 		lid, err := ioutil.ReadFile(f) | ||||
| 		if err == nil { | ||||
| 			plist := strings.Split(strings.Replace(f, string(IBBASEPATH), "", -1), "/") | ||||
| 			skip := false | ||||
| 			for _, d := range m.config.ExcludeDevices { | ||||
| 				if d == plist[0] { | ||||
| 					skip = true | ||||
| 				} | ||||
| 			} | ||||
| 			if !skip { | ||||
| 				m.lids[plist[0]] = make(map[string]string) | ||||
| 				m.lids[plist[0]][plist[2]] = string(lid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, ports := range m.lids { | ||||
| 		for port, lid := range ports { | ||||
| 			args := fmt.Sprintf("-r %s %s 0xf000", lid, port) | ||||
| 			command := exec.Command(m.config.PerfQueryPath, args) | ||||
| 			command.Wait() | ||||
| 			_, err := command.Output() | ||||
| 			if err == nil { | ||||
| 				m.use_perfquery = true | ||||
| 			} | ||||
| 			break | ||||
| 		} | ||||
| 		break | ||||
| 	} | ||||
|  | ||||
| 	if len(m.lids) > 0 { | ||||
| 		m.init = true | ||||
| 	} else { | ||||
| 		err = errors.New("No usable devices") | ||||
| 	} | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DoPerfQuery(cmd string, dev string, lid string, port string, tags map[string]string, out *[]lp.MutableMetric) error { | ||||
|  | ||||
| 	args := fmt.Sprintf("-r %s %s 0xf000", lid, port) | ||||
| 	command := exec.Command(cmd, args) | ||||
| 	command.Wait() | ||||
| 	stdout, err := command.Output() | ||||
| 	// Loop for all InfiniBand directories | ||||
| 	globPattern := filepath.Join(IB_BASEPATH, "*", "ports", "*") | ||||
| 	ibDirs, err := filepath.Glob(globPattern) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return err | ||||
| 		return fmt.Errorf("Unable to glob files with pattern %s: %v", globPattern, err) | ||||
| 	} | ||||
| 	if ibDirs == nil { | ||||
| 		return fmt.Errorf("Unable to find any directories with pattern %s", globPattern) | ||||
| 	} | ||||
| 	ll := strings.Split(string(stdout), "\n") | ||||
|  | ||||
| 	for _, line := range ll { | ||||
| 		if strings.HasPrefix(line, "PortRcvData") || strings.HasPrefix(line, "RcvData") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_recv", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 	for _, path := range ibDirs { | ||||
|  | ||||
| 		// Skip, when no LID is assigned | ||||
| 		line, err := ioutil.ReadFile(filepath.Join(path, "lid")) | ||||
| 		if err != nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		LID := strings.TrimSpace(string(line)) | ||||
| 		if LID == "0x0" { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// Get device and port component | ||||
| 		pathSplit := strings.Split(path, string(os.PathSeparator)) | ||||
| 		device := pathSplit[4] | ||||
| 		port := pathSplit[6] | ||||
|  | ||||
| 		// Skip excluded devices | ||||
| 		skip := false | ||||
| 		for _, excludedDevice := range m.config.ExcludeDevices { | ||||
| 			if excludedDevice == device { | ||||
| 				skip = true | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortXmitData") || strings.HasPrefix(line, "XmtData") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_xmit", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortRcvPkts") || strings.HasPrefix(line, "RcvPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_recv_pkts", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortXmitPkts") || strings.HasPrefix(line, "XmtPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_xmit_pkts", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 		if skip { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// Check access to counter files | ||||
| 		countersDir := filepath.Join(path, "counters") | ||||
| 		portCounterFiles := map[string]string{ | ||||
| 			"ib_recv":      filepath.Join(countersDir, "port_rcv_data"), | ||||
| 			"ib_xmit":      filepath.Join(countersDir, "port_xmit_data"), | ||||
| 			"ib_recv_pkts": filepath.Join(countersDir, "port_rcv_packets"), | ||||
| 			"ib_xmit_pkts": filepath.Join(countersDir, "port_xmit_packets"), | ||||
| 		} | ||||
| 		for _, counterFile := range portCounterFiles { | ||||
| 			err := unix.Access(counterFile, unix.R_OK) | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("Unable to access %s: %v", counterFile, err) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		m.info = append(m.info, | ||||
| 			&InfinibandCollectorInfo{ | ||||
| 				LID:              LID, | ||||
| 				device:           device, | ||||
| 				port:             port, | ||||
| 				portCounterFiles: portCounterFiles, | ||||
| 				tagSet: map[string]string{ | ||||
| 					"type":   "node", | ||||
| 					"device": device, | ||||
| 					"port":   port, | ||||
| 					"lid":    LID, | ||||
| 				}, | ||||
| 			}) | ||||
| 	} | ||||
|  | ||||
| 	if len(m.info) == 0 { | ||||
| 		return fmt.Errorf("Found no IB devices") | ||||
| 	} | ||||
|  | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func DoSysfsRead(dev string, lid string, port string, tags map[string]string, out *[]lp.MutableMetric) error { | ||||
| 	path := fmt.Sprintf("%s/%s/ports/%s/counters/", string(IBBASEPATH), dev, port) | ||||
| 	buffer, err := ioutil.ReadFile(fmt.Sprintf("%s/port_rcv_data", path)) | ||||
| 	if err == nil { | ||||
| 		data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		v, err := strconv.ParseFloat(data, 64) | ||||
| 		if err == nil { | ||||
| 			y, err := lp.New("ib_recv", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_xmit_data", path)) | ||||
| 	if err == nil { | ||||
| 		data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		v, err := strconv.ParseFloat(data, 64) | ||||
| 		if err == nil { | ||||
| 			y, err := lp.New("ib_xmit", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_rcv_packets", path)) | ||||
| 	if err == nil { | ||||
| 		data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		v, err := strconv.ParseFloat(data, 64) | ||||
| 		if err == nil { | ||||
| 			y, err := lp.New("ib_recv_pkts", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_xmit_packets", path)) | ||||
| 	if err == nil { | ||||
| 		data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		v, err := strconv.ParseFloat(data, 64) | ||||
| 		if err == nil { | ||||
| 			y, err := lp.New("ib_xmit_pkts", tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| // Read reads Infiniband counter files below IB_BASEPATH | ||||
| func (m *InfinibandCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
|  | ||||
| func (m *InfinibandCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
|  | ||||
| 	if m.init { | ||||
| 		for dev, ports := range m.lids { | ||||
| 			for port, lid := range ports { | ||||
| 				tags := map[string]string{"type": "node", "device": dev, "port": port} | ||||
| 				if m.use_perfquery { | ||||
| 					DoPerfQuery(m.config.PerfQueryPath, dev, lid, port, tags, out) | ||||
| 				} else { | ||||
| 					DoSysfsRead(dev, lid, port, tags, out) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	// Check if already initialized | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	//	buffer, err := ioutil.ReadFile(string(LIDFILE)) | ||||
| 	now := time.Now() | ||||
| 	for _, info := range m.info { | ||||
| 		for counterName, counterFile := range info.portCounterFiles { | ||||
| 			line, err := ioutil.ReadFile(counterFile) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError( | ||||
| 					m.name, | ||||
| 					fmt.Sprintf("Read(): Failed to read from file '%s': %v", counterFile, err)) | ||||
| 				continue | ||||
| 			} | ||||
| 			data := strings.TrimSpace(string(line)) | ||||
| 			v, err := strconv.ParseInt(data, 10, 64) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError( | ||||
| 					m.name, | ||||
| 					fmt.Sprintf("Read(): Failed to convert Infininiband metrice %s='%s' to int64: %v", counterName, data, err)) | ||||
| 				continue | ||||
| 			} | ||||
| 			if y, err := lp.New(counterName, info.tagSet, m.meta, map[string]interface{}{"value": v}, now); err == nil { | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 	//	if err != nil { | ||||
| 	//		log.Print(err) | ||||
| 	//		return | ||||
| 	//	} | ||||
|  | ||||
| 	//	args := fmt.Sprintf("-r %s 1 0xf000", string(buffer)) | ||||
|  | ||||
| 	//	command := exec.Command(PERFQUERY, args) | ||||
| 	//	command.Wait() | ||||
| 	//	stdout, err := command.Output() | ||||
| 	//	if err != nil { | ||||
| 	//		log.Print(err) | ||||
| 	//		return | ||||
| 	//	} | ||||
|  | ||||
| 	//	ll := strings.Split(string(stdout), "\n") | ||||
|  | ||||
| 	//	for _, line := range ll { | ||||
| 	//		if strings.HasPrefix(line, "PortRcvData") || strings.HasPrefix(line, "RcvData") { | ||||
| 	//			lv := strings.Fields(line) | ||||
| 	//			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 	//			if err == nil { | ||||
| 	//				y, err := lp.New("ib_recv", m.tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 	//				if err == nil { | ||||
| 	//					*out = append(*out, y) | ||||
| 	//				} | ||||
| 	//			} | ||||
| 	//		} | ||||
| 	//		if strings.HasPrefix(line, "PortXmitData") || strings.HasPrefix(line, "XmtData") { | ||||
| 	//			lv := strings.Fields(line) | ||||
| 	//			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 	//			if err == nil { | ||||
| 	//				y, err := lp.New("ib_xmit", m.tags, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 	//				if err == nil { | ||||
| 	//					*out = append(*out, y) | ||||
| 	//				} | ||||
| 	//			} | ||||
| 	//		} | ||||
| 	//	} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *InfinibandCollector) Close() { | ||||
|   | ||||
							
								
								
									
										26
									
								
								collectors/infinibandMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								collectors/infinibandMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
|  | ||||
| ## `ibstat` collector | ||||
|  | ||||
| ```json | ||||
|   "ibstat": { | ||||
|     "exclude_devices": [ | ||||
|       "mlx4" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `ibstat` collector includes all Infiniband devices that can be | ||||
| found below `/sys/class/infiniband/` and where any of the ports provides a | ||||
| LID file (`/sys/class/infiniband/<dev>/ports/<port>/lid`) | ||||
|  | ||||
| The devices can be filtered with the `exclude_devices` option in the configuration. | ||||
|  | ||||
| For each found LID the collector reads data through the sysfs files below `/sys/class/infiniband/<device>`. | ||||
|  | ||||
| Metrics: | ||||
| * `ib_recv` | ||||
| * `ib_xmit` | ||||
| * `ib_recv_pkts` | ||||
| * `ib_xmit_pkts` | ||||
|  | ||||
| The collector adds a `device` tag to all metrics | ||||
							
								
								
									
										232
									
								
								collectors/infinibandPerfQueryMetric.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										232
									
								
								collectors/infinibandPerfQueryMetric.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,232 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os/exec" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
|  | ||||
| 	//	"os" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const PERFQUERY = `/usr/sbin/perfquery` | ||||
|  | ||||
| type InfinibandPerfQueryCollector struct { | ||||
| 	metricCollector | ||||
| 	tags   map[string]string | ||||
| 	lids   map[string]map[string]string | ||||
| 	config struct { | ||||
| 		ExcludeDevices []string `json:"exclude_devices,omitempty"` | ||||
| 		PerfQueryPath  string   `json:"perfquery_path"` | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *InfinibandPerfQueryCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "InfinibandCollectorPerfQuery" | ||||
| 	m.setup() | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Network"} | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(m.config.PerfQueryPath) == 0 { | ||||
| 		path, err := exec.LookPath("perfquery") | ||||
| 		if err == nil { | ||||
| 			m.config.PerfQueryPath = path | ||||
| 		} | ||||
| 	} | ||||
| 	m.lids = make(map[string]map[string]string) | ||||
| 	p := fmt.Sprintf("%s/*/ports/*/lid", string(IB_BASEPATH)) | ||||
| 	files, err := filepath.Glob(p) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	for _, f := range files { | ||||
| 		lid, err := ioutil.ReadFile(f) | ||||
| 		if err == nil { | ||||
| 			plist := strings.Split(strings.Replace(f, string(IB_BASEPATH), "", -1), "/") | ||||
| 			skip := false | ||||
| 			for _, d := range m.config.ExcludeDevices { | ||||
| 				if d == plist[0] { | ||||
| 					skip = true | ||||
| 				} | ||||
| 			} | ||||
| 			if !skip { | ||||
| 				m.lids[plist[0]] = make(map[string]string) | ||||
| 				m.lids[plist[0]][plist[2]] = string(lid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, ports := range m.lids { | ||||
| 		for port, lid := range ports { | ||||
| 			args := fmt.Sprintf("-r %s %s 0xf000", lid, port) | ||||
| 			command := exec.Command(m.config.PerfQueryPath, args) | ||||
| 			command.Wait() | ||||
| 			_, err := command.Output() | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("Failed to execute %s: %v", m.config.PerfQueryPath, err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if len(m.lids) == 0 { | ||||
| 		return errors.New("No usable IB devices") | ||||
| 	} | ||||
|  | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *InfinibandPerfQueryCollector) doPerfQuery(cmd string, dev string, lid string, port string, tags map[string]string, output chan lp.CCMetric) error { | ||||
|  | ||||
| 	args := fmt.Sprintf("-r %s %s 0xf000", lid, port) | ||||
| 	command := exec.Command(cmd, args) | ||||
| 	command.Wait() | ||||
| 	stdout, err := command.Output() | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return err | ||||
| 	} | ||||
| 	ll := strings.Split(string(stdout), "\n") | ||||
|  | ||||
| 	for _, line := range ll { | ||||
| 		if strings.HasPrefix(line, "PortRcvData") || strings.HasPrefix(line, "RcvData") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_recv", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortXmitData") || strings.HasPrefix(line, "XmtData") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_xmit", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortRcvPkts") || strings.HasPrefix(line, "RcvPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_recv_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortXmitPkts") || strings.HasPrefix(line, "XmtPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_xmit_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortRcvPkts") || strings.HasPrefix(line, "RcvPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_recv_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if strings.HasPrefix(line, "PortXmitPkts") || strings.HasPrefix(line, "XmtPkts") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			v, err := strconv.ParseFloat(lv[1], 64) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New("ib_xmit_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *InfinibandPerfQueryCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
|  | ||||
| 	if m.init { | ||||
| 		for dev, ports := range m.lids { | ||||
| 			for port, lid := range ports { | ||||
| 				tags := map[string]string{ | ||||
| 					"type":   "node", | ||||
| 					"device": dev, | ||||
| 					"port":   port, | ||||
| 					"lid":    lid} | ||||
| 				path := fmt.Sprintf("%s/%s/ports/%s/counters/", string(IB_BASEPATH), dev, port) | ||||
| 				buffer, err := ioutil.ReadFile(fmt.Sprintf("%s/port_rcv_data", path)) | ||||
| 				if err == nil { | ||||
| 					data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 					v, err := strconv.ParseFloat(data, 64) | ||||
| 					if err == nil { | ||||
| 						y, err := lp.New("ib_recv", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_xmit_data", path)) | ||||
| 				if err == nil { | ||||
| 					data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 					v, err := strconv.ParseFloat(data, 64) | ||||
| 					if err == nil { | ||||
| 						y, err := lp.New("ib_xmit", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_rcv_packets", path)) | ||||
| 				if err == nil { | ||||
| 					data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 					v, err := strconv.ParseFloat(data, 64) | ||||
| 					if err == nil { | ||||
| 						y, err := lp.New("ib_recv_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				buffer, err = ioutil.ReadFile(fmt.Sprintf("%s/port_xmit_packets", path)) | ||||
| 				if err == nil { | ||||
| 					data := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 					v, err := strconv.ParseFloat(data, 64) | ||||
| 					if err == nil { | ||||
| 						y, err := lp.New("ib_xmit_pkts", tags, m.meta, map[string]interface{}{"value": float64(v)}, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *InfinibandPerfQueryCollector) Close() { | ||||
| 	m.init = false | ||||
| } | ||||
							
								
								
									
										28
									
								
								collectors/infinibandPerfQueryMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								collectors/infinibandPerfQueryMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
|  | ||||
| ## `ibstat_perfquery` collector | ||||
|  | ||||
| ```json | ||||
|   "ibstat_perfquery": { | ||||
|     "perfquery_path": "/path/to/perfquery", | ||||
|     "exclude_devices": [ | ||||
|       "mlx4" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `ibstat_perfquery` collector includes all Infiniband devices that can be | ||||
| found below `/sys/class/infiniband/` and where any of the ports provides a | ||||
| LID file (`/sys/class/infiniband/<dev>/ports/<port>/lid`) | ||||
|  | ||||
| The devices can be filtered with the `exclude_devices` option in the configuration. | ||||
|  | ||||
| For each found LID the collector calls the `perfquery` command. The path to the | ||||
| `perfquery` command can be configured with the `perfquery_path` option in the configuration | ||||
|  | ||||
| Metrics: | ||||
| * `ib_recv` | ||||
| * `ib_xmit` | ||||
| * `ib_recv_pkts` | ||||
| * `ib_xmit_pkts` | ||||
|  | ||||
| The collector adds a `device` tag to all metrics | ||||
							
								
								
									
										155
									
								
								collectors/iostatMetric.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								collectors/iostatMetric.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,155 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"os" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
|  | ||||
| 	//	"log" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const IOSTATFILE = `/proc/diskstats` | ||||
| const IOSTAT_SYSFSPATH = `/sys/block` | ||||
|  | ||||
| type IOstatCollectorConfig struct { | ||||
| 	ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| } | ||||
|  | ||||
| type IOstatCollectorEntry struct { | ||||
| 	lastValues map[string]int64 | ||||
| 	tags       map[string]string | ||||
| } | ||||
|  | ||||
| type IOstatCollector struct { | ||||
| 	metricCollector | ||||
| 	matches map[string]int | ||||
| 	config  IOstatCollectorConfig | ||||
| 	devices map[string]IOstatCollectorEntry | ||||
| } | ||||
|  | ||||
| func (m *IOstatCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "IOstatCollector" | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Disk"} | ||||
| 	m.setup() | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	// https://www.kernel.org/doc/html/latest/admin-guide/iostats.html | ||||
| 	matches := map[string]int{ | ||||
| 		"io_reads":             3, | ||||
| 		"io_reads_merged":      4, | ||||
| 		"io_read_sectors":      5, | ||||
| 		"io_read_ms":           6, | ||||
| 		"io_writes":            7, | ||||
| 		"io_writes_merged":     8, | ||||
| 		"io_writes_sectors":    9, | ||||
| 		"io_writes_ms":         10, | ||||
| 		"io_ioops":             11, | ||||
| 		"io_ioops_ms":          12, | ||||
| 		"io_ioops_weighted_ms": 13, | ||||
| 		"io_discards":          14, | ||||
| 		"io_discards_merged":   15, | ||||
| 		"io_discards_sectors":  16, | ||||
| 		"io_discards_ms":       17, | ||||
| 		"io_flushes":           18, | ||||
| 		"io_flushes_ms":        19, | ||||
| 	} | ||||
| 	m.devices = make(map[string]IOstatCollectorEntry) | ||||
| 	m.matches = make(map[string]int) | ||||
| 	for k, v := range matches { | ||||
| 		if _, skip := stringArrayContains(m.config.ExcludeMetrics, k); !skip { | ||||
| 			m.matches[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	if len(m.matches) == 0 { | ||||
| 		return errors.New("no metrics to collect") | ||||
| 	} | ||||
| 	file, err := os.Open(string(IOSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		linefields := strings.Fields(line) | ||||
| 		device := linefields[2] | ||||
| 		if strings.Contains(device, "loop") { | ||||
| 			continue | ||||
| 		} | ||||
| 		values := make(map[string]int64) | ||||
| 		for m := range m.matches { | ||||
| 			values[m] = 0 | ||||
| 		} | ||||
| 		m.devices[device] = IOstatCollectorEntry{ | ||||
| 			tags: map[string]string{ | ||||
| 				"device": linefields[2], | ||||
| 				"type":   "node", | ||||
| 			}, | ||||
| 			lastValues: values, | ||||
| 		} | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (m *IOstatCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	file, err := os.Open(string(IOSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		if len(line) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		linefields := strings.Fields(line) | ||||
| 		device := linefields[2] | ||||
| 		if strings.Contains(device, "loop") { | ||||
| 			continue | ||||
| 		} | ||||
| 		if _, ok := m.devices[device]; !ok { | ||||
| 			continue | ||||
| 		} | ||||
| 		entry := m.devices[device] | ||||
| 		for name, idx := range m.matches { | ||||
| 			if idx < len(linefields) { | ||||
| 				x, err := strconv.ParseInt(linefields[idx], 0, 64) | ||||
| 				if err == nil { | ||||
| 					diff := x - entry.lastValues[name] | ||||
| 					y, err := lp.New(name, entry.tags, m.meta, map[string]interface{}{"value": int(diff)}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 				entry.lastValues[name] = x | ||||
| 			} | ||||
| 		} | ||||
| 		m.devices[device] = entry | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *IOstatCollector) Close() { | ||||
| 	m.init = false | ||||
| } | ||||
							
								
								
									
										34
									
								
								collectors/iostatMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								collectors/iostatMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
|  | ||||
| ## `iostat` collector | ||||
|  | ||||
| ```json | ||||
|   "iostat": { | ||||
|     "exclude_metrics": [ | ||||
|       "read_ms" | ||||
|     ], | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `iostat` collector reads data from `/proc/diskstats` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `io_reads` | ||||
| * `io_reads_merged` | ||||
| * `io_read_sectors` | ||||
| * `io_read_ms` | ||||
| * `io_writes` | ||||
| * `io_writes_merged` | ||||
| * `io_writes_sectors` | ||||
| * `io_writes_ms` | ||||
| * `io_ioops` | ||||
| * `io_ioops_ms` | ||||
| * `io_ioops_weighted_ms` | ||||
| * `io_discards` | ||||
| * `io_discards_merged` | ||||
| * `io_discards_sectors` | ||||
| * `io_discards_ms` | ||||
| * `io_flushes` | ||||
| * `io_flushes_ms` | ||||
|  | ||||
| The device name is added as tag `device`. For more details, see https://www.kernel.org/doc/html/latest/admin-guide/iostats.html | ||||
|  | ||||
| @@ -10,11 +10,11 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const IPMITOOL_PATH = `/usr/bin/ipmitool` | ||||
| const IPMISENSORS_PATH = `/usr/sbin/ipmi-sensors` | ||||
| const IPMITOOL_PATH = `ipmitool` | ||||
| const IPMISENSORS_PATH = `ipmi-sensors` | ||||
|  | ||||
| type IpmiCollectorConfig struct { | ||||
| 	ExcludeDevices  []string `json:"exclude_devices"` | ||||
| @@ -23,37 +23,44 @@ type IpmiCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type IpmiCollector struct { | ||||
| 	MetricCollector | ||||
| 	tags    map[string]string | ||||
| 	matches map[string]string | ||||
| 	config  IpmiCollectorConfig | ||||
| 	metricCollector | ||||
| 	//tags        map[string]string | ||||
| 	//matches     map[string]string | ||||
| 	config      IpmiCollectorConfig | ||||
| 	ipmitool    string | ||||
| 	ipmisensors string | ||||
| } | ||||
|  | ||||
| func (m *IpmiCollector) Init(config []byte) error { | ||||
| func (m *IpmiCollector) Init(config json.RawMessage) error { | ||||
| 	m.name = "IpmiCollector" | ||||
| 	m.setup() | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "IPMI"} | ||||
| 	m.config.IpmitoolPath = string(IPMITOOL_PATH) | ||||
| 	m.config.IpmisensorsPath = string(IPMISENSORS_PATH) | ||||
| 	m.ipmitool = "" | ||||
| 	m.ipmisensors = "" | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	_, err1 := os.Stat(m.config.IpmitoolPath) | ||||
| 	_, err2 := os.Stat(m.config.IpmisensorsPath) | ||||
| 	if err1 != nil { | ||||
| 		m.config.IpmitoolPath = "" | ||||
| 	p, err := exec.LookPath(m.config.IpmitoolPath) | ||||
| 	if err == nil { | ||||
| 		m.ipmitool = p | ||||
| 	} | ||||
| 	if err2 != nil { | ||||
| 		m.config.IpmisensorsPath = "" | ||||
| 	p, err = exec.LookPath(m.config.IpmisensorsPath) | ||||
| 	if err == nil { | ||||
| 		m.ipmisensors = p | ||||
| 	} | ||||
| 	if err1 != nil && err2 != nil { | ||||
| 	if len(m.ipmitool) == 0 && len(m.ipmisensors) == 0 { | ||||
| 		return errors.New("No IPMI reader found") | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func ReadIpmiTool(cmd string, out *[]lp.MutableMetric) { | ||||
| func (m *IpmiCollector) readIpmiTool(cmd string, output chan lp.CCMetric) { | ||||
| 	command := exec.Command(cmd, "sensor") | ||||
| 	command.Wait() | ||||
| 	stdout, err := command.Output() | ||||
| @@ -74,24 +81,25 @@ func ReadIpmiTool(cmd string, out *[]lp.MutableMetric) { | ||||
| 			name := strings.ToLower(strings.Replace(strings.Trim(lv[0], " "), " ", "_", -1)) | ||||
| 			unit := strings.Trim(lv[2], " ") | ||||
| 			if unit == "Volts" { | ||||
| 				unit = "V" | ||||
| 				unit = "Volts" | ||||
| 			} else if unit == "degrees C" { | ||||
| 				unit = "C" | ||||
| 				unit = "degC" | ||||
| 			} else if unit == "degrees F" { | ||||
| 				unit = "F" | ||||
| 				unit = "degF" | ||||
| 			} else if unit == "Watts" { | ||||
| 				unit = "W" | ||||
| 				unit = "Watts" | ||||
| 			} | ||||
|  | ||||
| 			y, err := lp.New(name, map[string]string{"unit": unit, "type": "node"}, map[string]interface{}{"value": v}, time.Now()) | ||||
| 			y, err := lp.New(name, map[string]string{"type": "node"}, m.meta, map[string]interface{}{"value": v}, time.Now()) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 				y.AddMeta("unit", unit) | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ReadIpmiSensors(cmd string, out *[]lp.MutableMetric) { | ||||
| func (m *IpmiCollector) readIpmiSensors(cmd string, output chan lp.CCMetric) { | ||||
|  | ||||
| 	command := exec.Command(cmd, "--comma-separated-output", "--sdr-cache-recreate") | ||||
| 	command.Wait() | ||||
| @@ -109,25 +117,28 @@ func ReadIpmiSensors(cmd string, out *[]lp.MutableMetric) { | ||||
| 			v, err := strconv.ParseFloat(lv[3], 64) | ||||
| 			if err == nil { | ||||
| 				name := strings.ToLower(strings.Replace(lv[1], " ", "_", -1)) | ||||
| 				y, err := lp.New(name, map[string]string{"unit": lv[4], "type": "node"}, map[string]interface{}{"value": v}, time.Now()) | ||||
| 				y, err := lp.New(name, map[string]string{"type": "node"}, m.meta, map[string]interface{}{"value": v}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 					if len(lv) > 4 { | ||||
| 						y.AddMeta("unit", lv[4]) | ||||
| 					} | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *IpmiCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *IpmiCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if len(m.config.IpmitoolPath) > 0 { | ||||
| 		_, err := os.Stat(m.config.IpmitoolPath) | ||||
| 		if err == nil { | ||||
| 			ReadIpmiTool(m.config.IpmitoolPath, out) | ||||
| 			m.readIpmiTool(m.config.IpmitoolPath, output) | ||||
| 		} | ||||
| 	} else if len(m.config.IpmisensorsPath) > 0 { | ||||
| 		_, err := os.Stat(m.config.IpmisensorsPath) | ||||
| 		if err == nil { | ||||
| 			ReadIpmiSensors(m.config.IpmisensorsPath, out) | ||||
| 			m.readIpmiSensors(m.config.IpmisensorsPath, output) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										16
									
								
								collectors/ipmiMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								collectors/ipmiMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
|  | ||||
| ## `ipmistat` collector | ||||
|  | ||||
| ```json | ||||
|   "ipmistat": { | ||||
|     "ipmitool_path": "/path/to/ipmitool", | ||||
|     "ipmisensors_path": "/path/to/ipmi-sensors", | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `ipmistat` collector reads data from `ipmitool` (`ipmitool sensor`) or `ipmi-sensors` (`ipmi-sensors --sdr-cache-recreate --comma-separated-output`).  | ||||
|  | ||||
| The metrics depend on the output of the underlying tools but contain temperature, power and energy metrics. | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -2,7 +2,7 @@ package collectors | ||||
|  | ||||
| /* | ||||
| #cgo CFLAGS: -I./likwid | ||||
| #cgo LDFLAGS: -L./likwid -llikwid -llikwid-hwloc -lm | ||||
| #cgo LDFLAGS: -L./likwid -llikwid -llikwid-hwloc -lm -Wl,--unresolved-symbols=ignore-in-object-files | ||||
| #include <stdlib.h> | ||||
| #include <likwid.h> | ||||
| */ | ||||
| @@ -13,55 +13,111 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"math" | ||||
| 	"os" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| 	"unsafe" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	"gopkg.in/Knetic/govaluate.v2" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	topo "github.com/ClusterCockpit/cc-metric-collector/internal/ccTopology" | ||||
| 	agg "github.com/ClusterCockpit/cc-metric-collector/internal/metricAggregator" | ||||
| 	"github.com/NVIDIA/go-nvml/pkg/dl" | ||||
| ) | ||||
|  | ||||
| type MetricScope string | ||||
|  | ||||
| const ( | ||||
| 	METRIC_SCOPE_HWTHREAD = iota | ||||
| 	METRIC_SCOPE_CORE | ||||
| 	METRIC_SCOPE_LLC | ||||
| 	METRIC_SCOPE_NUMA | ||||
| 	METRIC_SCOPE_DIE | ||||
| 	METRIC_SCOPE_SOCKET | ||||
| 	METRIC_SCOPE_NODE | ||||
| ) | ||||
|  | ||||
| func (ms MetricScope) String() string { | ||||
| 	return string(ms) | ||||
| } | ||||
|  | ||||
| func (ms MetricScope) Likwid() string { | ||||
| 	LikwidDomains := map[string]string{ | ||||
| 		"cpu":        "", | ||||
| 		"core":       "", | ||||
| 		"llc":        "C", | ||||
| 		"numadomain": "M", | ||||
| 		"die":        "D", | ||||
| 		"socket":     "S", | ||||
| 		"node":       "N", | ||||
| 	} | ||||
| 	return LikwidDomains[string(ms)] | ||||
| } | ||||
|  | ||||
| func (ms MetricScope) Granularity() int { | ||||
| 	for i, g := range GetAllMetricScopes() { | ||||
| 		if ms == g { | ||||
| 			return i | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| func GetAllMetricScopes() []MetricScope { | ||||
| 	return []MetricScope{"cpu" /*, "core", "llc", "numadomain", "die",*/, "socket", "node"} | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	LIKWID_LIB_NAME     = "liblikwid.so" | ||||
| 	LIKWID_LIB_DL_FLAGS = dl.RTLD_LAZY | dl.RTLD_GLOBAL | ||||
| ) | ||||
|  | ||||
| type LikwidCollectorMetricConfig struct { | ||||
| 	Name         string `json:"name"` | ||||
| 	Calc         string `json:"calc"` | ||||
| 	Socket_scope bool   `json:"socket_scope"` | ||||
| 	Publish      bool   `json:"publish"` | ||||
| 	Name string `json:"name"` // Name of the metric | ||||
| 	Calc string `json:"calc"` // Calculation for the metric using | ||||
| 	//Aggr        string      `json:"aggregation"` // if scope unequal to LIKWID metric scope, the values are combined (sum, min, max, mean or avg, median) | ||||
| 	Scope       MetricScope `json:"scope"` // scope for calculation. subscopes are aggregated using the 'aggregation' function | ||||
| 	Publish     bool        `json:"publish"` | ||||
| 	granulatity MetricScope | ||||
| } | ||||
|  | ||||
| type LikwidCollectorEventsetConfig struct { | ||||
| 	Events  map[string]string             `json:"events"` | ||||
| 	Metrics []LikwidCollectorMetricConfig `json:"metrics"` | ||||
| 	Events      map[string]string `json:"events"` | ||||
| 	granulatity map[string]MetricScope | ||||
| 	Metrics     []LikwidCollectorMetricConfig `json:"metrics"` | ||||
| } | ||||
|  | ||||
| type LikwidCollectorConfig struct { | ||||
| 	Eventsets      []LikwidCollectorEventsetConfig `json:"eventsets"` | ||||
| 	Metrics        []LikwidCollectorMetricConfig   `json:"globalmetrics"` | ||||
| 	ExcludeMetrics []string                        `json:"exclude_metrics"` | ||||
| 	ForceOverwrite bool                            `json:"force_overwrite"` | ||||
| 	Metrics        []LikwidCollectorMetricConfig   `json:"globalmetrics,omitempty"` | ||||
| 	ForceOverwrite bool                            `json:"force_overwrite,omitempty"` | ||||
| 	InvalidToZero  bool                            `json:"invalid_to_zero,omitempty"` | ||||
| } | ||||
|  | ||||
| type LikwidCollector struct { | ||||
| 	MetricCollector | ||||
| 	cpulist   []C.int | ||||
| 	sock2tid  map[int]int | ||||
| 	metrics   map[C.int]map[string]int | ||||
| 	groups    []C.int | ||||
| 	config    LikwidCollectorConfig | ||||
| 	results   map[int]map[int]map[string]interface{} | ||||
| 	mresults  map[int]map[int]map[string]float64 | ||||
| 	gmresults map[int]map[string]float64 | ||||
| 	basefreq  float64 | ||||
| 	metricCollector | ||||
| 	cpulist       []C.int | ||||
| 	cpu2tid       map[int]int | ||||
| 	sock2tid      map[int]int | ||||
| 	scopeRespTids map[MetricScope]map[int]int | ||||
| 	metrics       map[C.int]map[string]int | ||||
| 	groups        []C.int | ||||
| 	config        LikwidCollectorConfig | ||||
| 	results       map[int]map[int]map[string]interface{} | ||||
| 	mresults      map[int]map[int]map[string]float64 | ||||
| 	gmresults     map[int]map[string]float64 | ||||
| 	basefreq      float64 | ||||
| 	running       bool | ||||
| } | ||||
|  | ||||
| type LikwidMetric struct { | ||||
| 	name         string | ||||
| 	search       string | ||||
| 	socket_scope bool | ||||
| 	group_idx    int | ||||
| 	name      string | ||||
| 	search    string | ||||
| 	scope     MetricScope | ||||
| 	group_idx int | ||||
| } | ||||
|  | ||||
| func eventsToEventStr(events map[string]string) string { | ||||
| @@ -72,12 +128,27 @@ func eventsToEventStr(events map[string]string) string { | ||||
| 	return strings.Join(elist, ",") | ||||
| } | ||||
|  | ||||
| func getGranularity(counter, event string) MetricScope { | ||||
| 	if strings.HasPrefix(counter, "PMC") || strings.HasPrefix(counter, "FIXC") { | ||||
| 		return "cpu" | ||||
| 	} else if strings.Contains(counter, "BOX") || strings.Contains(counter, "DEV") { | ||||
| 		return "socket" | ||||
| 	} else if strings.HasPrefix(counter, "PWR") { | ||||
| 		if event == "RAPL_CORE_ENERGY" { | ||||
| 			return "cpu" | ||||
| 		} else { | ||||
| 			return "socket" | ||||
| 		} | ||||
| 	} | ||||
| 	return "unknown" | ||||
| } | ||||
|  | ||||
| func getBaseFreq() float64 { | ||||
| 	var freq float64 = math.NaN() | ||||
| 	C.power_init(0) | ||||
| 	info := C.get_powerInfo() | ||||
| 	if float64(info.baseFrequency) != 0 { | ||||
| 		freq = float64(info.baseFrequency) | ||||
| 		freq = float64(info.baseFrequency) * 1e3 | ||||
| 	} else { | ||||
| 		buffer, err := ioutil.ReadFile("/sys/devices/system/cpu/cpu0/cpufreq/bios_limit") | ||||
| 		if err == nil { | ||||
| @@ -91,21 +162,102 @@ func getBaseFreq() float64 { | ||||
| 	return freq | ||||
| } | ||||
|  | ||||
| func getSocketCpus() map[C.int]int { | ||||
| 	slist := SocketList() | ||||
| 	var cpu C.int | ||||
| 	outmap := make(map[C.int]int) | ||||
| 	for _, s := range slist { | ||||
| 		t := C.CString(fmt.Sprintf("S%d", s)) | ||||
| 		clen := C.cpustr_to_cpulist(t, &cpu, 1) | ||||
| 		if int(clen) == 1 { | ||||
| 			outmap[cpu] = s | ||||
| func (m *LikwidCollector) initGranularity() { | ||||
| 	splitRegex := regexp.MustCompile("[+-/*()]") | ||||
| 	for _, evset := range m.config.Eventsets { | ||||
| 		evset.granulatity = make(map[string]MetricScope) | ||||
| 		for counter, event := range evset.Events { | ||||
| 			gran := getGranularity(counter, event) | ||||
| 			if gran.Granularity() >= 0 { | ||||
| 				evset.granulatity[counter] = gran | ||||
| 			} | ||||
| 		} | ||||
| 		for i, metric := range evset.Metrics { | ||||
| 			s := splitRegex.Split(metric.Calc, -1) | ||||
| 			gran := MetricScope("cpu") | ||||
| 			evset.Metrics[i].granulatity = gran | ||||
| 			for _, x := range s { | ||||
| 				if _, ok := evset.Events[x]; ok { | ||||
| 					if evset.granulatity[x].Granularity() > gran.Granularity() { | ||||
| 						gran = evset.granulatity[x] | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			evset.Metrics[i].granulatity = gran | ||||
| 		} | ||||
| 	} | ||||
| 	return outmap | ||||
| 	for i, metric := range m.config.Metrics { | ||||
| 		s := splitRegex.Split(metric.Calc, -1) | ||||
| 		gran := MetricScope("cpu") | ||||
| 		m.config.Metrics[i].granulatity = gran | ||||
| 		for _, x := range s { | ||||
| 			for _, evset := range m.config.Eventsets { | ||||
| 				for _, m := range evset.Metrics { | ||||
| 					if m.Name == x && m.granulatity.Granularity() > gran.Granularity() { | ||||
| 						gran = m.granulatity | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		m.config.Metrics[i].granulatity = gran | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *LikwidCollector) Init(config []byte) error { | ||||
| type TopoResolveFunc func(cpuid int) int | ||||
|  | ||||
| func (m *LikwidCollector) getResponsiblities() map[MetricScope]map[int]int { | ||||
| 	get_cpus := func(scope MetricScope) map[int]int { | ||||
| 		var slist []int | ||||
| 		var cpu C.int | ||||
| 		var input func(index int) string | ||||
| 		switch scope { | ||||
| 		case "node": | ||||
| 			slist = []int{0} | ||||
| 			input = func(index int) string { return "N:0" } | ||||
| 		case "socket": | ||||
| 			input = func(index int) string { return fmt.Sprintf("%s%d:0", scope.Likwid(), index) } | ||||
| 			slist = topo.SocketList() | ||||
| 		// case "numadomain": | ||||
| 		// 	input = func(index int) string { return fmt.Sprintf("%s%d:0", scope.Likwid(), index) } | ||||
| 		// 	slist = topo.NumaNodeList() | ||||
| 		// 	cclog.Debug(scope, " ", input(0), " ", slist) | ||||
| 		// case "die": | ||||
| 		// 	input = func(index int) string { return fmt.Sprintf("%s%d:0", scope.Likwid(), index) } | ||||
| 		// 	slist = topo.DieList() | ||||
| 		// case "llc": | ||||
| 		// 	input = fmt.Sprintf("%s%d:0", scope.Likwid(), s) | ||||
| 		// 	slist = topo.LLCacheList() | ||||
| 		case "cpu": | ||||
| 			input = func(index int) string { return fmt.Sprintf("%d", index) } | ||||
| 			slist = topo.CpuList() | ||||
| 		case "hwthread": | ||||
| 			input = func(index int) string { return fmt.Sprintf("%d", index) } | ||||
| 			slist = topo.CpuList() | ||||
| 		} | ||||
| 		outmap := make(map[int]int) | ||||
| 		for _, s := range slist { | ||||
| 			t := C.CString(input(s)) | ||||
| 			clen := C.cpustr_to_cpulist(t, &cpu, 1) | ||||
| 			if int(clen) == 1 { | ||||
| 				outmap[s] = m.cpu2tid[int(cpu)] | ||||
| 			} else { | ||||
| 				cclog.Error(fmt.Sprintf("Cannot determine responsible CPU for %s", input(s))) | ||||
| 				outmap[s] = -1 | ||||
| 			} | ||||
| 			C.free(unsafe.Pointer(t)) | ||||
| 		} | ||||
| 		return outmap | ||||
| 	} | ||||
|  | ||||
| 	scopes := GetAllMetricScopes() | ||||
| 	complete := make(map[MetricScope]map[int]int) | ||||
| 	for _, s := range scopes { | ||||
| 		complete[s] = get_cpus(s) | ||||
| 	} | ||||
| 	return complete | ||||
| } | ||||
|  | ||||
| func (m *LikwidCollector) Init(config json.RawMessage) error { | ||||
| 	var ret C.int | ||||
| 	m.name = "LikwidCollector" | ||||
| 	if len(config) > 0 { | ||||
| @@ -114,36 +266,78 @@ func (m *LikwidCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	lib := dl.New(LIKWID_LIB_NAME, LIKWID_LIB_DL_FLAGS) | ||||
| 	if lib == nil { | ||||
| 		return fmt.Errorf("error instantiating DynamicLibrary for %s", LIKWID_LIB_NAME) | ||||
| 	} | ||||
| 	if m.config.ForceOverwrite { | ||||
| 		cclog.ComponentDebug(m.name, "Set LIKWID_FORCE=1") | ||||
| 		os.Setenv("LIKWID_FORCE", "1") | ||||
| 	} | ||||
| 	m.setup() | ||||
| 	cpulist := CpuList() | ||||
| 	m.cpulist = make([]C.int, len(cpulist)) | ||||
| 	slist := getSocketCpus() | ||||
|  | ||||
| 	m.sock2tid = make(map[int]int) | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "PerfCounter"} | ||||
| 	cclog.ComponentDebug(m.name, "Get cpulist and init maps and lists") | ||||
| 	cpulist := topo.CpuList() | ||||
| 	m.cpulist = make([]C.int, len(cpulist)) | ||||
| 	m.cpu2tid = make(map[int]int) | ||||
| 	for i, c := range cpulist { | ||||
| 		m.cpulist[i] = C.int(c) | ||||
| 		if sid, found := slist[m.cpulist[i]]; found { | ||||
| 			m.sock2tid[sid] = i | ||||
| 		} | ||||
| 		m.cpu2tid[c] = i | ||||
|  | ||||
| 	} | ||||
| 	m.results = make(map[int]map[int]map[string]interface{}) | ||||
| 	m.mresults = make(map[int]map[int]map[string]float64) | ||||
| 	m.gmresults = make(map[int]map[string]float64) | ||||
| 	cclog.ComponentDebug(m.name, "initialize LIKWID topology") | ||||
| 	ret = C.topology_init() | ||||
| 	if ret != 0 { | ||||
| 		return errors.New("Failed to initialize LIKWID topology") | ||||
| 	} | ||||
| 	if m.config.ForceOverwrite { | ||||
| 		os.Setenv("LIKWID_FORCE", "1") | ||||
| 		err := errors.New("failed to initialize LIKWID topology") | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Determine which counter works at which level. PMC*: cpu, *BOX*: socket, ... | ||||
| 	m.initGranularity() | ||||
| 	// Generate map for MetricScope -> scope_id (like socket id) -> responsible id (offset in cpulist) | ||||
| 	m.scopeRespTids = m.getResponsiblities() | ||||
|  | ||||
| 	cclog.ComponentDebug(m.name, "initialize LIKWID perfmon module") | ||||
| 	ret = C.perfmon_init(C.int(len(m.cpulist)), &m.cpulist[0]) | ||||
| 	if ret != 0 { | ||||
| 		C.topology_finalize() | ||||
| 		return errors.New("Failed to initialize LIKWID topology") | ||||
| 		err := errors.New("failed to initialize LIKWID topology") | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// This is for the global metrics computation test | ||||
| 	globalParams := make(map[string]interface{}) | ||||
| 	globalParams["time"] = float64(1.0) | ||||
| 	globalParams["inverseClock"] = float64(1.0) | ||||
| 	// While adding the events, we test the metrics whether they can be computed at all | ||||
| 	for i, evset := range m.config.Eventsets { | ||||
| 		estr := eventsToEventStr(evset.Events) | ||||
| 		// Generate parameter list for the metric computing test | ||||
| 		params := make(map[string]interface{}) | ||||
| 		params["time"] = float64(1.0) | ||||
| 		params["inverseClock"] = float64(1.0) | ||||
| 		for counter := range evset.Events { | ||||
| 			params[counter] = float64(1.0) | ||||
| 		} | ||||
| 		for _, metric := range evset.Metrics { | ||||
| 			// Try to evaluate the metric | ||||
| 			_, err := agg.EvalFloat64Condition(metric.Calc, params) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError(m.name, "Calculation for metric", metric.Name, "failed:", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			// If the metric is not in the parameter list for the global metrics, add it | ||||
| 			if _, ok := globalParams[metric.Name]; !ok { | ||||
| 				globalParams[metric.Name] = float64(1.0) | ||||
| 			} | ||||
| 		} | ||||
| 		// Now we add the list of events to likwid | ||||
| 		cstr := C.CString(estr) | ||||
| 		gid := C.perfmon_addEventSet(cstr) | ||||
| 		if gid >= 0 { | ||||
| @@ -155,153 +349,208 @@ func (m *LikwidCollector) Init(config []byte) error { | ||||
| 		for tid := range m.cpulist { | ||||
| 			m.results[i][tid] = make(map[string]interface{}) | ||||
| 			m.mresults[i][tid] = make(map[string]float64) | ||||
| 			m.gmresults[tid] = make(map[string]float64) | ||||
| 			if i == 0 { | ||||
| 				m.gmresults[tid] = make(map[string]float64) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	for _, metric := range m.config.Metrics { | ||||
| 		// Try to evaluate the global metric | ||||
| 		_, err := agg.EvalFloat64Condition(metric.Calc, globalParams) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError(m.name, "Calculation for metric", metric.Name, "failed:", err.Error()) | ||||
| 			continue | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// If no event set could be added, shut down LikwidCollector | ||||
| 	if len(m.groups) == 0 { | ||||
| 		C.perfmon_finalize() | ||||
| 		C.topology_finalize() | ||||
| 		return errors.New("No LIKWID performance group initialized") | ||||
| 		err := errors.New("no LIKWID performance group initialized") | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	m.basefreq = getBaseFreq() | ||||
| 	cclog.ComponentDebug(m.name, "BaseFreq", m.basefreq) | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *LikwidCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| // take a measurement for 'interval' seconds of event set index 'group' | ||||
| func (m *LikwidCollector) takeMeasurement(group int, interval time.Duration) error { | ||||
| 	var ret C.int | ||||
| 	gid := m.groups[group] | ||||
| 	ret = C.perfmon_setupCounters(gid) | ||||
| 	if ret != 0 { | ||||
| 		gctr := C.GoString(C.perfmon_getGroupName(gid)) | ||||
| 		err := fmt.Errorf("failed to setup performance group %d (%s)", gid, gctr) | ||||
| 		return err | ||||
| 	} | ||||
| 	ret = C.perfmon_startCounters() | ||||
| 	if ret != 0 { | ||||
| 		gctr := C.GoString(C.perfmon_getGroupName(gid)) | ||||
| 		err := fmt.Errorf("failed to start performance group %d (%s)", gid, gctr) | ||||
| 		return err | ||||
| 	} | ||||
| 	m.running = true | ||||
| 	time.Sleep(interval) | ||||
| 	m.running = false | ||||
| 	ret = C.perfmon_stopCounters() | ||||
| 	if ret != 0 { | ||||
| 		gctr := C.GoString(C.perfmon_getGroupName(gid)) | ||||
| 		err := fmt.Errorf("failed to stop performance group %d (%s)", gid, gctr) | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Get all measurement results for an event set, derive the metric values out of the measurement results and send it | ||||
| func (m *LikwidCollector) calcEventsetMetrics(group int, interval time.Duration, output chan lp.CCMetric) error { | ||||
| 	var eidx C.int | ||||
| 	evset := m.config.Eventsets[group] | ||||
| 	gid := m.groups[group] | ||||
| 	invClock := float64(1.0 / m.basefreq) | ||||
|  | ||||
| 	// Go over events and get the results | ||||
| 	for eidx = 0; int(eidx) < len(evset.Events); eidx++ { | ||||
| 		ctr := C.perfmon_getCounterName(gid, eidx) | ||||
| 		ev := C.perfmon_getEventName(gid, eidx) | ||||
| 		gctr := C.GoString(ctr) | ||||
| 		gev := C.GoString(ev) | ||||
| 		// MetricScope for the counter (and if needed the event) | ||||
| 		scope := getGranularity(gctr, gev) | ||||
| 		// Get the map scope-id -> tids | ||||
| 		// This way we read less counters like only the responsible hardware thread for a socket | ||||
| 		scopemap := m.scopeRespTids[scope] | ||||
| 		for _, tid := range scopemap { | ||||
| 			if tid >= 0 { | ||||
| 				m.results[group][tid]["time"] = interval.Seconds() | ||||
| 				m.results[group][tid]["inverseClock"] = invClock | ||||
| 				res := C.perfmon_getLastResult(gid, eidx, C.int(tid)) | ||||
| 				m.results[group][tid][gctr] = float64(res) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Go over the event set metrics, derive the value out of the event:counter values and send it | ||||
| 	for _, metric := range evset.Metrics { | ||||
| 		// The metric scope is determined in the Init() function | ||||
| 		// Get the map scope-id -> tids | ||||
| 		scopemap := m.scopeRespTids[metric.Scope] | ||||
| 		for domain, tid := range scopemap { | ||||
| 			if tid >= 0 { | ||||
| 				value, err := agg.EvalFloat64Condition(metric.Calc, m.results[group][tid]) | ||||
| 				if err != nil { | ||||
| 					cclog.ComponentError(m.name, "Calculation for metric", metric.Name, "failed:", err.Error()) | ||||
| 					continue | ||||
| 				} | ||||
| 				m.mresults[group][tid][metric.Name] = value | ||||
| 				if m.config.InvalidToZero && math.IsNaN(value) { | ||||
| 					value = 0.0 | ||||
| 				} | ||||
| 				if m.config.InvalidToZero && math.IsInf(value, 0) { | ||||
| 					value = 0.0 | ||||
| 				} | ||||
| 				// Now we have the result, send it with the proper tags | ||||
| 				if !math.IsNaN(value) { | ||||
| 					if metric.Publish { | ||||
| 						tags := map[string]string{"type": metric.Scope.String()} | ||||
| 						if metric.Scope != "node" { | ||||
| 							tags["type-id"] = fmt.Sprintf("%d", domain) | ||||
| 						} | ||||
| 						fields := map[string]interface{}{"value": value} | ||||
| 						y, err := lp.New(metric.Name, tags, m.meta, fields, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Go over the global metrics, derive the value out of the event sets' metric values and send it | ||||
| func (m *LikwidCollector) calcGlobalMetrics(interval time.Duration, output chan lp.CCMetric) error { | ||||
| 	for _, metric := range m.config.Metrics { | ||||
| 		scopemap := m.scopeRespTids[metric.Scope] | ||||
| 		for domain, tid := range scopemap { | ||||
| 			if tid >= 0 { | ||||
| 				// Here we generate parameter list | ||||
| 				params := make(map[string]interface{}) | ||||
| 				for j := range m.groups { | ||||
| 					for mname, mres := range m.mresults[j][tid] { | ||||
| 						params[mname] = mres | ||||
| 					} | ||||
| 				} | ||||
| 				// Evaluate the metric | ||||
| 				value, err := agg.EvalFloat64Condition(metric.Calc, params) | ||||
| 				if err != nil { | ||||
| 					cclog.ComponentError(m.name, "Calculation for metric", metric.Name, "failed:", err.Error()) | ||||
| 					continue | ||||
| 				} | ||||
| 				m.gmresults[tid][metric.Name] = value | ||||
| 				if m.config.InvalidToZero && math.IsNaN(value) { | ||||
| 					value = 0.0 | ||||
| 				} | ||||
| 				if m.config.InvalidToZero && math.IsInf(value, 0) { | ||||
| 					value = 0.0 | ||||
| 				} | ||||
| 				// Now we have the result, send it with the proper tags | ||||
| 				if !math.IsNaN(value) { | ||||
| 					if metric.Publish { | ||||
| 						tags := map[string]string{"type": metric.Scope.String()} | ||||
| 						if metric.Scope != "node" { | ||||
| 							tags["type-id"] = fmt.Sprintf("%d", domain) | ||||
| 						} | ||||
| 						fields := map[string]interface{}{"value": value} | ||||
| 						y, err := lp.New(metric.Name, tags, m.meta, fields, time.Now()) | ||||
| 						if err == nil { | ||||
| 							output <- y | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // main read function taking multiple measurement rounds, each 'interval' seconds long | ||||
| func (m *LikwidCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	var ret C.int | ||||
|  | ||||
| 	for i, gid := range m.groups { | ||||
| 		evset := m.config.Eventsets[i] | ||||
| 		ret = C.perfmon_setupCounters(gid) | ||||
| 		if ret != 0 { | ||||
| 			log.Print("Failed to setup performance group ", C.perfmon_getGroupName(gid)) | ||||
| 			continue | ||||
| 		} | ||||
| 		ret = C.perfmon_startCounters() | ||||
| 		if ret != 0 { | ||||
| 			log.Print("Failed to start performance group ", C.perfmon_getGroupName(gid)) | ||||
| 			continue | ||||
| 		} | ||||
| 		time.Sleep(interval) | ||||
| 		ret = C.perfmon_stopCounters() | ||||
| 		if ret != 0 { | ||||
| 			log.Print("Failed to stop performance group ", C.perfmon_getGroupName(gid)) | ||||
| 			continue | ||||
| 		} | ||||
| 		var eidx C.int | ||||
| 		for tid := range m.cpulist { | ||||
| 			for eidx = 0; int(eidx) < len(evset.Events); eidx++ { | ||||
| 				ctr := C.perfmon_getCounterName(gid, eidx) | ||||
| 				gctr := C.GoString(ctr) | ||||
| 				res := C.perfmon_getLastResult(gid, eidx, C.int(tid)) | ||||
| 				m.results[i][tid][gctr] = float64(res) | ||||
| 			} | ||||
| 			m.results[i][tid]["time"] = interval.Seconds() | ||||
| 			m.results[i][tid]["inverseClock"] = float64(1.0 / m.basefreq) | ||||
| 			for _, metric := range evset.Metrics { | ||||
| 				expression, err := govaluate.NewEvaluableExpression(metric.Calc) | ||||
| 				if err != nil { | ||||
| 					log.Print(err.Error()) | ||||
| 					continue | ||||
| 				} | ||||
| 				result, err := expression.Evaluate(m.results[i][tid]) | ||||
| 				if err != nil { | ||||
| 					log.Print(err.Error()) | ||||
| 					continue | ||||
| 				} | ||||
| 				m.mresults[i][tid][metric.Name] = float64(result.(float64)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, metric := range m.config.Metrics { | ||||
| 		for tid := range m.cpulist { | ||||
| 			var params map[string]interface{} | ||||
| 			expression, err := govaluate.NewEvaluableExpression(metric.Calc) | ||||
| 			if err != nil { | ||||
| 				log.Print(err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			params = make(map[string]interface{}) | ||||
| 			for j := range m.groups { | ||||
| 				for mname, mres := range m.mresults[j][tid] { | ||||
| 					params[mname] = mres | ||||
| 				} | ||||
| 			} | ||||
| 			result, err := expression.Evaluate(params) | ||||
| 			if err != nil { | ||||
| 				log.Print(err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			m.gmresults[tid][metric.Name] = float64(result.(float64)) | ||||
| 		} | ||||
| 	} | ||||
| 	for i := range m.groups { | ||||
| 		evset := m.config.Eventsets[i] | ||||
| 		for _, metric := range evset.Metrics { | ||||
| 			_, skip := stringArrayContains(m.config.ExcludeMetrics, metric.Name) | ||||
| 			if metric.Publish && !skip { | ||||
| 				if metric.Socket_scope { | ||||
| 					for sid, tid := range m.sock2tid { | ||||
| 						y, err := lp.New(metric.Name, | ||||
| 							map[string]string{"type": "socket", "type-id": fmt.Sprintf("%d", int(sid))}, | ||||
| 							map[string]interface{}{"value": m.mresults[i][tid][metric.Name]}, | ||||
| 							time.Now()) | ||||
| 						if err == nil { | ||||
| 							*out = append(*out, y) | ||||
| 						} | ||||
| 					} | ||||
| 				} else { | ||||
| 					for tid, cpu := range m.cpulist { | ||||
| 						y, err := lp.New(metric.Name, | ||||
| 							map[string]string{"type": "cpu", "type-id": fmt.Sprintf("%d", int(cpu))}, | ||||
| 							map[string]interface{}{"value": m.mresults[i][tid][metric.Name]}, | ||||
| 							time.Now()) | ||||
| 						if err == nil { | ||||
| 							*out = append(*out, y) | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	for _, metric := range m.config.Metrics { | ||||
| 		_, skip := stringArrayContains(m.config.ExcludeMetrics, metric.Name) | ||||
| 		if metric.Publish && !skip { | ||||
| 			if metric.Socket_scope { | ||||
| 				for sid, tid := range m.sock2tid { | ||||
| 					y, err := lp.New(metric.Name, | ||||
| 						map[string]string{"type": "socket", "type-id": fmt.Sprintf("%d", int(sid))}, | ||||
| 						map[string]interface{}{"value": m.gmresults[tid][metric.Name]}, | ||||
| 						time.Now()) | ||||
| 					if err == nil { | ||||
| 						*out = append(*out, y) | ||||
| 					} | ||||
| 				} | ||||
| 			} else { | ||||
| 				for tid, cpu := range m.cpulist { | ||||
| 					y, err := lp.New(metric.Name, | ||||
| 						map[string]string{"type": "cpu", "type-id": fmt.Sprintf("%d", int(cpu))}, | ||||
| 						map[string]interface{}{"value": m.gmresults[tid][metric.Name]}, | ||||
| 						time.Now()) | ||||
| 					if err == nil { | ||||
| 						*out = append(*out, y) | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		// measure event set 'i' for 'interval' seconds | ||||
| 		err := m.takeMeasurement(i, interval) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError(m.name, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 		// read measurements and derive event set metrics | ||||
| 		m.calcEventsetMetrics(i, interval, output) | ||||
| 	} | ||||
| 	// use the event set metrics to derive the global metrics | ||||
| 	m.calcGlobalMetrics(interval, output) | ||||
| } | ||||
|  | ||||
| func (m *LikwidCollector) Close() { | ||||
| 	if m.init { | ||||
| 		cclog.ComponentDebug(m.name, "Closing ...") | ||||
| 		m.init = false | ||||
| 		if m.running { | ||||
| 			cclog.ComponentDebug(m.name, "Stopping counters") | ||||
| 			C.perfmon_stopCounters() | ||||
| 		} | ||||
| 		cclog.ComponentDebug(m.name, "Finalize LIKWID perfmon module") | ||||
| 		C.perfmon_finalize() | ||||
| 		cclog.ComponentDebug(m.name, "Finalize LIKWID topology module") | ||||
| 		C.topology_finalize() | ||||
| 		cclog.ComponentDebug(m.name, "Closing done") | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										148
									
								
								collectors/likwidMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										148
									
								
								collectors/likwidMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,148 @@ | ||||
|  | ||||
| ## `likwid` collector | ||||
|  | ||||
| The `likwid` collector is probably the most complicated collector. The LIKWID library is included as static library with *direct* access mode. The *direct* access mode is suitable if the daemon is executed by a root user. The static library does not contain the performance groups, so all information needs to be provided in the configuration. | ||||
|  | ||||
| The `likwid` configuration consists of two parts, the "eventsets" and "globalmetrics": | ||||
| - An event set list itself has two parts, the "events" and a set of derivable "metrics". Each of the "events" is a counter:event pair in LIKWID's syntax. The "metrics" are a list of formulas to derive the metric value from the measurements of the "events". Each metric has a name, the formula, a scope and a publish flag. A counter names can be used like variables in the formulas, so `PMC0+PMC1` sums the measurements for the both events configured in the counters `PMC0` and `PMC1`. The scope tells the Collector whether it is a metric for each hardware thread (`cpu`) or each CPU socket (`socket`). The last one is the publishing flag. It tells the collector whether a metric should be sent to the router. | ||||
| - The global metrics are metrics which require data from all event set measurements to be derived. The inputs are the metrics in the event sets. Similar to the metrics in the event sets, the global metrics are defined by a name, a formula, a scope and a publish flag. See event set metrics for details. The only difference is that there is no access to the raw event measurements anymore but only to the metrics. So, the idea is to derive a metric in the "eventsets" section and reuse it in the "globalmetrics" part. If you need a metric only for deriving the global metrics, disable forwarding of the event set metrics. **Be aware** that the combination might be misleading because the "behavior" of a metric changes over time and the multiple measurements might count different computing phases. | ||||
|  | ||||
| Additional options: | ||||
| - `force_overwrite`: Same as setting `LIKWID_FORCE=1`. In case counters are already in-use, LIKWID overwrites their configuration to do its measurements | ||||
| - `invalid_to_zero`: In some cases, the calculations result in `NaN` or `Inf`. With this option, all `NaN` and `Inf` values are replaces with `0.0`. | ||||
|  | ||||
| ### Available metric scopes | ||||
|  | ||||
| Hardware performance counters are scattered all over the system nowadays. A counter coveres a specific part of the system. While there are hardware thread specific counter for CPU cycles, instructions and so on, some others are specific for a whole CPU socket/package. To address that, the collector provides the specification of a 'scope' for each metric. | ||||
|  | ||||
| - `cpu` : One metric per CPU hardware thread with the tags `"type" : "cpu"` and `"type-id" : "$cpu_id"` | ||||
| - `socket` : One metric per CPU socket/package with the tags `"type" : "socket"` and `"type-id" : "$socket_id"` | ||||
|  | ||||
| **Note:** You cannot specify `socket` scope for a metric that is measured at `cpu` scope, so some kind of expert knowledge or lookup work in the [Likwid Wiki](https://github.com/RRZE-HPC/likwid/wiki) is required. Get the scope of each counter from the *Architecture* pages and as soon as one counter in a metric is socket-specific, the whole metric is socket-specific. | ||||
|  | ||||
| As a guideline: | ||||
| - All counters `FIXCx`, `PMCy` and `TMAz` have the scope `cpu` | ||||
| - All counters names containing `BOX` have the scope `socket` | ||||
| - All `PWRx` counters have scope `socket`, except `"PWR1" : "RAPL_CORE_ENERGY"` has `cpu` scope | ||||
| - All `DFCx` counters have scope `socket` | ||||
|  | ||||
|  | ||||
| ### Example configuration | ||||
|  | ||||
|  | ||||
| ```json | ||||
|   "likwid": { | ||||
|     "force_overwrite" : false, | ||||
|     "nan_to_zero" : false, | ||||
|     "eventsets": [ | ||||
|       { | ||||
|         "events": { | ||||
|           "FIXC1": "ACTUAL_CPU_CLOCK", | ||||
|           "FIXC2": "MAX_CPU_CLOCK", | ||||
|           "PMC0": "RETIRED_INSTRUCTIONS", | ||||
|           "PMC1": "CPU_CLOCKS_UNHALTED", | ||||
|           "PMC2": "RETIRED_SSE_AVX_FLOPS_ALL", | ||||
|           "PMC3": "MERGE", | ||||
|           "DFC0": "DRAM_CHANNEL_0", | ||||
|           "DFC1": "DRAM_CHANNEL_1", | ||||
|           "DFC2": "DRAM_CHANNEL_2", | ||||
|           "DFC3": "DRAM_CHANNEL_3" | ||||
|         }, | ||||
|         "metrics": [ | ||||
|           { | ||||
|             "name": "ipc", | ||||
|             "calc": "PMC0/PMC1", | ||||
|             "scope": "cpu", | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "flops_any", | ||||
|             "calc": "0.000001*PMC2/time", | ||||
|             "scope": "cpu", | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "clock_mhz", | ||||
|             "calc": "0.000001*(FIXC1/FIXC2)/inverseClock", | ||||
|             "scope": "cpu", | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "mem1", | ||||
|             "calc": "0.000001*(DFC0+DFC1+DFC2+DFC3)*64.0/time", | ||||
|             "scope": "socket", | ||||
|             "publish": false | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "events": { | ||||
|           "DFC0": "DRAM_CHANNEL_4", | ||||
|           "DFC1": "DRAM_CHANNEL_5", | ||||
|           "DFC2": "DRAM_CHANNEL_6", | ||||
|           "DFC3": "DRAM_CHANNEL_7", | ||||
|           "PWR0": "RAPL_CORE_ENERGY", | ||||
|           "PWR1": "RAPL_PKG_ENERGY" | ||||
|         }, | ||||
|         "metrics": [ | ||||
|           { | ||||
|             "name": "pwr_core", | ||||
|             "calc": "PWR0/time", | ||||
|             "scope": "socket", | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "pwr_pkg", | ||||
|             "calc": "PWR1/time", | ||||
|             "scope": "socket", | ||||
|             "publish": true | ||||
|           }, | ||||
|           { | ||||
|             "name": "mem2", | ||||
|             "calc": "0.000001*(DFC0+DFC1+DFC2+DFC3)*64.0/time", | ||||
|             "scope": "socket", | ||||
|             "publish": false | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "globalmetrics": [ | ||||
|       { | ||||
|         "name": "mem_bw", | ||||
|         "calc": "mem1+mem2", | ||||
|         "scope": "socket", | ||||
|         "publish": true | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| ### How to get the eventsets and metrics from LIKWID | ||||
|  | ||||
| The `likwid` collector reads hardware performance counters at a **cpu** and **socket** level. The configuration looks quite complicated but it is basically copy&paste from [LIKWID's performance groups](https://github.com/RRZE-HPC/likwid/tree/master/groups). The collector made multiple iterations and tried to use the performance groups but it lacked flexibility. The current way of configuration provides most flexibility. | ||||
|  | ||||
| The logic is as following: There are multiple eventsets, each consisting of a list of counters+events and a list of metrics. If you compare a common performance group with the example setting above, there is not much difference: | ||||
| ``` | ||||
| EVENTSET                         ->   "events": { | ||||
| FIXC1 ACTUAL_CPU_CLOCK           ->     "FIXC1": "ACTUAL_CPU_CLOCK", | ||||
| FIXC2 MAX_CPU_CLOCK              ->     "FIXC2": "MAX_CPU_CLOCK", | ||||
| PMC0  RETIRED_INSTRUCTIONS       ->     "PMC0" : "RETIRED_INSTRUCTIONS", | ||||
| PMC1  CPU_CLOCKS_UNHALTED        ->     "PMC1" : "CPU_CLOCKS_UNHALTED", | ||||
| PMC2  RETIRED_SSE_AVX_FLOPS_ALL  ->     "PMC2": "RETIRED_SSE_AVX_FLOPS_ALL", | ||||
| PMC3  MERGE                      ->     "PMC3": "MERGE", | ||||
|                                  ->   } | ||||
| ``` | ||||
|  | ||||
| The metrics are following the same procedure: | ||||
|  | ||||
| ``` | ||||
| METRICS                          ->   "metrics": [ | ||||
| IPC   PMC0/PMC1                  ->     { | ||||
|                                  ->       "name" : "IPC", | ||||
|                                  ->       "calc" : "PMC0/PMC1", | ||||
|                                  ->       "scope": "cpu", | ||||
|                                  ->       "publish": true | ||||
|                                  ->     } | ||||
|                                  ->   ] | ||||
| ``` | ||||
|  | ||||
| @@ -2,29 +2,39 @@ package collectors | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const LOADAVGFILE = `/proc/loadavg` | ||||
|  | ||||
| type LoadavgCollectorConfig struct { | ||||
| 	ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| } | ||||
| // | ||||
| // LoadavgCollector collects: | ||||
| // * load average of last 1, 5 & 15 minutes | ||||
| // * number of processes currently runnable | ||||
| // * total number of processes in system | ||||
| // | ||||
| // See: https://www.kernel.org/doc/html/latest/filesystems/proc.html | ||||
| // | ||||
| const LOADAVGFILE = "/proc/loadavg" | ||||
|  | ||||
| type LoadavgCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	tags         map[string]string | ||||
| 	load_matches []string | ||||
| 	load_skips   []bool | ||||
| 	proc_matches []string | ||||
| 	config       LoadavgCollectorConfig | ||||
| 	proc_skips   []bool | ||||
| 	config       struct { | ||||
| 		ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *LoadavgCollector) Init(config []byte) error { | ||||
| func (m *LoadavgCollector) Init(config json.RawMessage) error { | ||||
| 	m.name = "LoadavgCollector" | ||||
| 	m.setup() | ||||
| 	if len(config) > 0 { | ||||
| @@ -33,45 +43,82 @@ func (m *LoadavgCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "LOAD"} | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| 	m.load_matches = []string{"load_one", "load_five", "load_fifteen"} | ||||
| 	m.proc_matches = []string{"proc_run", "proc_total"} | ||||
| 	m.load_matches = []string{ | ||||
| 		"load_one", | ||||
| 		"load_five", | ||||
| 		"load_fifteen"} | ||||
| 	m.load_skips = make([]bool, len(m.load_matches)) | ||||
| 	m.proc_matches = []string{ | ||||
| 		"proc_run", | ||||
| 		"proc_total"} | ||||
| 	m.proc_skips = make([]bool, len(m.proc_matches)) | ||||
|  | ||||
| 	for i, name := range m.load_matches { | ||||
| 		_, m.load_skips[i] = stringArrayContains(m.config.ExcludeMetrics, name) | ||||
| 	} | ||||
| 	for i, name := range m.proc_matches { | ||||
| 		_, m.proc_skips[i] = stringArrayContains(m.config.ExcludeMetrics, name) | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *LoadavgCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| 	var skip bool | ||||
| func (m *LoadavgCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	buffer, err := ioutil.ReadFile(string(LOADAVGFILE)) | ||||
|  | ||||
| 	buffer, err := ioutil.ReadFile(LOADAVGFILE) | ||||
| 	if err != nil { | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to read file '%s': %v", LOADAVGFILE, err)) | ||||
| 		} | ||||
| 		return | ||||
| 	} | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	// Load metrics | ||||
| 	ls := strings.Split(string(buffer), ` `) | ||||
| 	for i, name := range m.load_matches { | ||||
| 		x, err := strconv.ParseFloat(ls[i], 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert '%s' to float64: %v", ls[i], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if m.load_skips[i] { | ||||
| 			continue | ||||
| 		} | ||||
| 		y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": x}, now) | ||||
| 		if err == nil { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, name) | ||||
| 			y, err := lp.New(name, m.tags, map[string]interface{}{"value": float64(x)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Process metrics | ||||
| 	lv := strings.Split(ls[3], `/`) | ||||
| 	for i, name := range m.proc_matches { | ||||
| 		x, err := strconv.ParseFloat(lv[i], 64) | ||||
| 		if err == nil { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, name) | ||||
| 			y, err := lp.New(name, m.tags, map[string]interface{}{"value": float64(x)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		x, err := strconv.ParseInt(lv[i], 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert '%s' to float64: %v", lv[i], err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		if m.proc_skips[i] { | ||||
| 			continue | ||||
| 		} | ||||
| 		y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": x}, now) | ||||
| 		if err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										19
									
								
								collectors/loadavgMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								collectors/loadavgMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
|  | ||||
| ## `loadavg` collector | ||||
|  | ||||
| ```json | ||||
|   "loadavg": { | ||||
|     "exclude_metrics": [ | ||||
|       "proc_run" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `loadavg` collector reads data from `/proc/loadavg` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
| Metrics: | ||||
| * `load_one` | ||||
| * `load_five` | ||||
| * `load_fifteen` | ||||
| * `proc_run` | ||||
| * `proc_total` | ||||
| @@ -3,31 +3,84 @@ package collectors | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"fmt" | ||||
| 	"os/exec" | ||||
| 	"os/user" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const LUSTREFILE = `/proc/fs/lustre/llite/lnec-XXXXXX/stats` | ||||
| const LUSTRE_SYSFS = `/sys/fs/lustre` | ||||
| const LCTL_CMD = `lctl` | ||||
| const LCTL_OPTION = `get_param` | ||||
|  | ||||
| type LustreCollectorConfig struct { | ||||
| 	Procfiles      []string `json:"procfiles"` | ||||
| 	LCtlCommand    string   `json:"lctl_command"` | ||||
| 	ExcludeMetrics []string `json:"exclude_metrics"` | ||||
| 	SendAllMetrics bool     `json:"send_all_metrics"` | ||||
| } | ||||
|  | ||||
| type LustreCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	tags    map[string]string | ||||
| 	matches map[string]map[string]int | ||||
| 	devices []string | ||||
| 	stats   map[string]map[string]int64 | ||||
| 	config  LustreCollectorConfig | ||||
| 	lctl    string | ||||
| } | ||||
|  | ||||
| func (m *LustreCollector) Init(config []byte) error { | ||||
| func (m *LustreCollector) getDeviceDataCommand(device string) []string { | ||||
| 	statsfile := fmt.Sprintf("llite.%s.stats", device) | ||||
| 	command := exec.Command(m.lctl, LCTL_OPTION, statsfile) | ||||
| 	command.Wait() | ||||
| 	stdout, _ := command.Output() | ||||
| 	return strings.Split(string(stdout), "\n") | ||||
| } | ||||
|  | ||||
| func (m *LustreCollector) getDevices() []string { | ||||
| 	devices := make([]string, 0) | ||||
|  | ||||
| 	// //Version reading devices from sysfs | ||||
| 	// globPattern := filepath.Join(LUSTRE_SYSFS, "llite/*/stats") | ||||
| 	// files, err := filepath.Glob(globPattern) | ||||
| 	// if err != nil { | ||||
| 	// 	return devices | ||||
| 	// } | ||||
| 	// for _, f := range files { | ||||
| 	// 	pathlist := strings.Split(f, "/") | ||||
| 	// 	devices = append(devices, pathlist[4]) | ||||
| 	// } | ||||
|  | ||||
| 	data := m.getDeviceDataCommand("*") | ||||
|  | ||||
| 	for _, line := range data { | ||||
| 		if strings.HasPrefix(line, "llite") { | ||||
| 			linefields := strings.Split(line, ".") | ||||
| 			if len(linefields) > 2 { | ||||
| 				devices = append(devices, linefields[1]) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return devices | ||||
| } | ||||
|  | ||||
| // //Version reading the stats data of a device from sysfs | ||||
| // func (m *LustreCollector) getDeviceDataSysfs(device string) []string { | ||||
| // 	llitedir := filepath.Join(LUSTRE_SYSFS, "llite") | ||||
| // 	devdir := filepath.Join(llitedir, device) | ||||
| // 	statsfile := filepath.Join(devdir, "stats") | ||||
| // 	buffer, err := ioutil.ReadFile(statsfile) | ||||
| // 	if err != nil { | ||||
| // 		return make([]string, 0) | ||||
| // 	} | ||||
| // 	return strings.Split(string(buffer), "\n") | ||||
| // } | ||||
|  | ||||
| func (m *LustreCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "LustreCollector" | ||||
| 	if len(config) > 0 { | ||||
| @@ -38,66 +91,120 @@ func (m *LustreCollector) Init(config []byte) error { | ||||
| 	} | ||||
| 	m.setup() | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| 	m.matches = map[string]map[string]int{"read_bytes": {"read_bytes": 6, "read_requests": 1}, | ||||
| 		"write_bytes":      {"write_bytes": 6, "write_requests": 1}, | ||||
| 		"open":             {"open": 1}, | ||||
| 		"close":            {"close": 1}, | ||||
| 		"setattr":          {"setattr": 1}, | ||||
| 		"getattr":          {"getattr": 1}, | ||||
| 		"statfs":           {"statfs": 1}, | ||||
| 		"inode_permission": {"inode_permission": 1}} | ||||
| 	m.devices = make([]string, 0) | ||||
| 	for _, p := range m.config.Procfiles { | ||||
| 		_, err := ioutil.ReadFile(p) | ||||
| 		if err == nil { | ||||
| 			m.devices = append(m.devices, p) | ||||
| 		} else { | ||||
| 			log.Print(err.Error()) | ||||
| 			continue | ||||
| 		} | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Lustre"} | ||||
| 	defmatches := map[string]map[string]int{ | ||||
| 		"read_bytes":       {"lustre_read_bytes": 6, "lustre_read_requests": 1}, | ||||
| 		"write_bytes":      {"lustre_write_bytes": 6, "lustre_write_requests": 1}, | ||||
| 		"open":             {"lustre_open": 1}, | ||||
| 		"close":            {"lustre_close": 1}, | ||||
| 		"setattr":          {"lustre_setattr": 1}, | ||||
| 		"getattr":          {"lustre_getattr": 1}, | ||||
| 		"statfs":           {"lustre_statfs": 1}, | ||||
| 		"inode_permission": {"lustre_inode_permission": 1}} | ||||
|  | ||||
| 	// Lustre file system statistics can only be queried by user root | ||||
| 	user, err := user.Current() | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, "Failed to get current user:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	if user.Uid != "0" { | ||||
| 		cclog.ComponentError(m.name, "Lustre file system statistics can only be queried by user root:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if len(m.devices) == 0 { | ||||
| 		return errors.New("No metrics to collect") | ||||
| 	m.matches = make(map[string]map[string]int) | ||||
| 	for lineprefix, names := range defmatches { | ||||
| 		for metricname, offset := range names { | ||||
| 			_, skip := stringArrayContains(m.config.ExcludeMetrics, metricname) | ||||
| 			if skip { | ||||
| 				continue | ||||
| 			} | ||||
| 			if _, prefixExist := m.matches[lineprefix]; !prefixExist { | ||||
| 				m.matches[lineprefix] = make(map[string]int) | ||||
| 			} | ||||
| 			if _, metricExist := m.matches[lineprefix][metricname]; !metricExist { | ||||
| 				m.matches[lineprefix][metricname] = offset | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	p, err := exec.LookPath(m.config.LCtlCommand) | ||||
| 	if err != nil { | ||||
| 		p, err = exec.LookPath(LCTL_CMD) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.lctl = p | ||||
|  | ||||
| 	devices := m.getDevices() | ||||
| 	if len(devices) == 0 { | ||||
| 		return errors.New("no metrics to collect") | ||||
| 	} | ||||
| 	m.stats = make(map[string]map[string]int64) | ||||
| 	for _, d := range devices { | ||||
| 		m.stats[d] = make(map[string]int64) | ||||
| 		for _, names := range m.matches { | ||||
| 			for metricname := range names { | ||||
| 				m.stats[d][metricname] = 0 | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *LustreCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *LustreCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	for _, p := range m.devices { | ||||
| 		buffer, err := ioutil.ReadFile(p) | ||||
| 	for device, devData := range m.stats { | ||||
| 		stats := m.getDeviceDataCommand(device) | ||||
| 		processed := []string{} | ||||
|  | ||||
| 		if err != nil { | ||||
| 			log.Print(err) | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		for _, line := range strings.Split(string(buffer), "\n") { | ||||
| 		for _, line := range stats { | ||||
| 			lf := strings.Fields(line) | ||||
| 			if len(lf) > 1 { | ||||
| 				for match, fields := range m.matches { | ||||
| 					if lf[0] == match { | ||||
| 						for name, idx := range fields { | ||||
| 							_, skip := stringArrayContains(m.config.ExcludeMetrics, name) | ||||
| 							if skip { | ||||
| 								continue | ||||
| 				if fields, ok := m.matches[lf[0]]; ok { | ||||
| 					for name, idx := range fields { | ||||
| 						x, err := strconv.ParseInt(lf[idx], 0, 64) | ||||
| 						if err != nil { | ||||
| 							continue | ||||
| 						} | ||||
| 						value := x - devData[name] | ||||
| 						devData[name] = x | ||||
| 						if value < 0 { | ||||
| 							value = 0 | ||||
| 						} | ||||
| 						y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": value}, time.Now()) | ||||
| 						if err == nil { | ||||
| 							y.AddTag("device", device) | ||||
| 							if strings.Contains(name, "byte") { | ||||
| 								y.AddMeta("unit", "Byte") | ||||
| 							} | ||||
| 							x, err := strconv.ParseInt(lf[idx], 0, 64) | ||||
| 							if err == nil { | ||||
| 								y, err := lp.New(name, m.tags, map[string]interface{}{"value": x}, time.Now()) | ||||
| 								if err == nil { | ||||
| 									*out = append(*out, y) | ||||
| 								} | ||||
| 							output <- y | ||||
| 							if m.config.SendAllMetrics { | ||||
| 								processed = append(processed, name) | ||||
| 							} | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		if m.config.SendAllMetrics { | ||||
| 			for name := range devData { | ||||
| 				if _, done := stringArrayContains(processed, name); !done { | ||||
| 					y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": 0}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						y.AddTag("device", device) | ||||
| 						if strings.Contains(name, "byte") { | ||||
| 							y.AddMeta("unit", "Byte") | ||||
| 						} | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										29
									
								
								collectors/lustreMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								collectors/lustreMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
|  | ||||
| ## `lustrestat` collector | ||||
|  | ||||
| ```json | ||||
|   "lustrestat": { | ||||
|     "procfiles" : [ | ||||
|       "/proc/fs/lustre/llite/lnec-XXXXXX/stats" | ||||
|     ], | ||||
|     "exclude_metrics": [ | ||||
|       "setattr", | ||||
|       "getattr" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `lustrestat` collector reads from the procfs stat files for Lustre like `/proc/fs/lustre/llite/lnec-XXXXXX/stats`. | ||||
|  | ||||
| Metrics: | ||||
| * `read_bytes` | ||||
| * `read_requests` | ||||
| * `write_bytes` | ||||
| * `write_requests` | ||||
| * `open` | ||||
| * `close` | ||||
| * `getattr` | ||||
| * `setattr` | ||||
| * `statfs` | ||||
| * `inode_permission` | ||||
|  | ||||
| @@ -10,7 +10,7 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const MEMSTATFILE = `/proc/meminfo` | ||||
| @@ -20,14 +20,14 @@ type MemstatCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type MemstatCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	stats   map[string]int64 | ||||
| 	tags    map[string]string | ||||
| 	matches map[string]string | ||||
| 	config  MemstatCollectorConfig | ||||
| } | ||||
|  | ||||
| func (m *MemstatCollector) Init(config []byte) error { | ||||
| func (m *MemstatCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "MemstatCollector" | ||||
| 	if len(config) > 0 { | ||||
| @@ -36,6 +36,7 @@ func (m *MemstatCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Memory", "unit": "kByte"} | ||||
| 	m.stats = make(map[string]int64) | ||||
| 	m.matches = make(map[string]string) | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| @@ -65,7 +66,7 @@ func (m *MemstatCollector) Init(config []byte) error { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (m *MemstatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *MemstatCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -93,13 +94,13 @@ func (m *MemstatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) | ||||
|  | ||||
| 	for match, name := range m.matches { | ||||
| 		if _, exists := m.stats[match]; !exists { | ||||
| 			err = errors.New(fmt.Sprintf("Parse error for %s : %s", match, name)) | ||||
| 			err = fmt.Errorf("Parse error for %s : %s", match, name) | ||||
| 			log.Print(err) | ||||
| 			continue | ||||
| 		} | ||||
| 		y, err := lp.New(name, m.tags, map[string]interface{}{"value": int(float64(m.stats[match]) * 1.0e-3)}, time.Now()) | ||||
| 		y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": int(float64(m.stats[match]) * 1.0e-3)}, time.Now()) | ||||
| 		if err == nil { | ||||
| 			*out = append(*out, y) | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @@ -108,18 +109,18 @@ func (m *MemstatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) | ||||
| 			if _, cached := m.stats[`Cached`]; cached { | ||||
| 				memUsed := m.stats[`MemTotal`] - (m.stats[`MemFree`] + m.stats[`Buffers`] + m.stats[`Cached`]) | ||||
| 				_, skip := stringArrayContains(m.config.ExcludeMetrics, "mem_used") | ||||
| 				y, err := lp.New("mem_used", m.tags, map[string]interface{}{"value": int(float64(memUsed) * 1.0e-3)}, time.Now()) | ||||
| 				y, err := lp.New("mem_used", m.tags, m.meta, map[string]interface{}{"value": int(float64(memUsed) * 1.0e-3)}, time.Now()) | ||||
| 				if err == nil && !skip { | ||||
| 					*out = append(*out, y) | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	if _, found := m.stats[`MemShared`]; found { | ||||
| 		_, skip := stringArrayContains(m.config.ExcludeMetrics, "mem_shared") | ||||
| 		y, err := lp.New("mem_shared", m.tags, map[string]interface{}{"value": int(float64(m.stats[`MemShared`]) * 1.0e-3)}, time.Now()) | ||||
| 		y, err := lp.New("mem_shared", m.tags, m.meta, map[string]interface{}{"value": int(float64(m.stats[`MemShared`]) * 1.0e-3)}, time.Now()) | ||||
| 		if err == nil && !skip { | ||||
| 			*out = append(*out, y) | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										27
									
								
								collectors/memstatMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								collectors/memstatMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
|  | ||||
| ## `memstat` collector | ||||
|  | ||||
| ```json | ||||
|   "memstat": { | ||||
|     "exclude_metrics": [ | ||||
|       "mem_used" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `memstat` collector reads data from `/proc/meminfo` and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. | ||||
|  | ||||
|  | ||||
| Metrics: | ||||
| * `mem_total` | ||||
| * `mem_sreclaimable` | ||||
| * `mem_slab` | ||||
| * `mem_free` | ||||
| * `mem_buffers` | ||||
| * `mem_cached` | ||||
| * `mem_available` | ||||
| * `mem_shared` | ||||
| * `swap_total` | ||||
| * `swap_free` | ||||
| * `mem_used` = `mem_total` - (`mem_free` + `mem_buffers` + `mem_cached`) | ||||
|  | ||||
| @@ -1,40 +1,48 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type MetricGetter interface { | ||||
| type MetricCollector interface { | ||||
| 	Name() string | ||||
| 	Init(config []byte) error | ||||
| 	Init(config json.RawMessage) error | ||||
| 	Initialized() bool | ||||
| 	Read(time.Duration, *[]lp.MutableMetric) | ||||
| 	Read(duration time.Duration, output chan lp.CCMetric) | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| type MetricCollector struct { | ||||
| type metricCollector struct { | ||||
| 	name string | ||||
| 	init bool | ||||
| 	meta map[string]string | ||||
| } | ||||
|  | ||||
| func (c *MetricCollector) Name() string { | ||||
| // Name() returns the name of the metric collector | ||||
| func (c *metricCollector) Name() string { | ||||
| 	return c.name | ||||
| } | ||||
|  | ||||
| func (c *MetricCollector) setup() error { | ||||
| func (c *metricCollector) setup() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *MetricCollector) Initialized() bool { | ||||
| 	return c.init == true | ||||
| // Initialized() indicates whether the metric collector has been initialized. | ||||
| func (c *metricCollector) Initialized() bool { | ||||
| 	return c.init | ||||
| } | ||||
|  | ||||
| // intArrayContains scans an array of ints if the value str is present in the array | ||||
| // If the specified value is found, the corresponding array index is returned. | ||||
| // The bool value is used to signal success or failure | ||||
| func intArrayContains(array []int, str int) (int, bool) { | ||||
| 	for i, a := range array { | ||||
| 		if a == str { | ||||
| @@ -44,6 +52,9 @@ func intArrayContains(array []int, str int) (int, bool) { | ||||
| 	return -1, false | ||||
| } | ||||
|  | ||||
| // stringArrayContains scans an array of strings if the value str is present in the array | ||||
| // If the specified value is found, the corresponding array index is returned. | ||||
| // The bool value is used to signal success or failure | ||||
| func stringArrayContains(array []string, str string) (int, bool) { | ||||
| 	for i, a := range array { | ||||
| 		if a == str { | ||||
| @@ -103,27 +114,13 @@ func CpuList() []int { | ||||
| 	return cpulist | ||||
| } | ||||
|  | ||||
| func Tags2Map(metric lp.Metric) map[string]string { | ||||
| 	tags := make(map[string]string) | ||||
| 	for _, t := range metric.TagList() { | ||||
| 		tags[t.Key] = t.Value | ||||
| 	} | ||||
| 	return tags | ||||
| } | ||||
|  | ||||
| func Fields2Map(metric lp.Metric) map[string]interface{} { | ||||
| 	fields := make(map[string]interface{}) | ||||
| 	for _, f := range metric.FieldList() { | ||||
| 		fields[f.Key] = f.Value | ||||
| 	} | ||||
| 	return fields | ||||
| } | ||||
|  | ||||
| // RemoveFromStringList removes the string r from the array of strings s | ||||
| // If r is not contained in the array an error is returned | ||||
| func RemoveFromStringList(s []string, r string) ([]string, error) { | ||||
| 	for i, item := range s { | ||||
| 		if r == item { | ||||
| 			return append(s[:i], s[i+1:]...), nil | ||||
| 		} | ||||
| 	} | ||||
| 	return s, errors.New("No such string in list") | ||||
| 	return s, fmt.Errorf("No such string in list") | ||||
| } | ||||
|   | ||||
| @@ -1,86 +1,138 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"errors" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const NETSTATFILE = `/proc/net/dev` | ||||
|  | ||||
| type NetstatCollectorConfig struct { | ||||
| 	ExcludeDevices []string `json:"exclude_devices"` | ||||
| 	IncludeDevices []string `json:"include_devices"` | ||||
| } | ||||
|  | ||||
| type NetstatCollectorMetric struct { | ||||
| 	index     int | ||||
| 	lastValue float64 | ||||
| } | ||||
|  | ||||
| type NetstatCollector struct { | ||||
| 	MetricCollector | ||||
| 	config  NetstatCollectorConfig | ||||
| 	matches map[int]string | ||||
| 	metricCollector | ||||
| 	config        NetstatCollectorConfig | ||||
| 	matches       map[string]map[string]NetstatCollectorMetric | ||||
| 	devtags       map[string]map[string]string | ||||
| 	lastTimestamp time.Time | ||||
| } | ||||
|  | ||||
| func (m *NetstatCollector) Init(config []byte) error { | ||||
| func (m *NetstatCollector) Init(config json.RawMessage) error { | ||||
| 	m.name = "NetstatCollector" | ||||
| 	m.setup() | ||||
| 	m.matches = map[int]string{ | ||||
| 		1:  "bytes_in", | ||||
| 		9:  "bytes_out", | ||||
| 		2:  "pkts_in", | ||||
| 		10: "pkts_out", | ||||
| 	m.lastTimestamp = time.Now() | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "Network"} | ||||
| 	m.devtags = make(map[string]map[string]string) | ||||
| 	nameIndexMap := map[string]int{ | ||||
| 		"net_bytes_in":  1, | ||||
| 		"net_pkts_in":   2, | ||||
| 		"net_bytes_out": 9, | ||||
| 		"net_pkts_out":  10, | ||||
| 	} | ||||
| 	m.matches = make(map[string]map[string]NetstatCollectorMetric) | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			log.Print(err.Error()) | ||||
| 			cclog.ComponentError(m.name, "Error reading config:", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	_, err := ioutil.ReadFile(string(NETSTATFILE)) | ||||
| 	if err == nil { | ||||
| 		m.init = true | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *NetstatCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| 	data, err := ioutil.ReadFile(string(NETSTATFILE)) | ||||
| 	file, err := os.Open(string(NETSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		log.Print(err.Error()) | ||||
| 		return | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer file.Close() | ||||
|  | ||||
| 	lines := strings.Split(string(data), "\n") | ||||
| 	for _, l := range lines { | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		l := scanner.Text() | ||||
| 		if !strings.Contains(l, ":") { | ||||
| 			continue | ||||
| 		} | ||||
| 		f := strings.Fields(l) | ||||
| 		dev := f[0][0 : len(f[0])-1] | ||||
| 		cont := false | ||||
| 		for _, d := range m.config.ExcludeDevices { | ||||
| 			if d == dev { | ||||
| 				cont = true | ||||
| 		dev := strings.Trim(f[0], ": ") | ||||
| 		if _, ok := stringArrayContains(m.config.IncludeDevices, dev); ok { | ||||
| 			m.matches[dev] = make(map[string]NetstatCollectorMetric) | ||||
| 			for name, idx := range nameIndexMap { | ||||
| 				m.matches[dev][name] = NetstatCollectorMetric{ | ||||
| 					index:     idx, | ||||
| 					lastValue: 0, | ||||
| 				} | ||||
| 			} | ||||
| 			m.devtags[dev] = map[string]string{"device": dev, "type": "node"} | ||||
| 		} | ||||
| 		if cont { | ||||
| 	} | ||||
| 	if len(m.devtags) == 0 { | ||||
| 		return errors.New("no devices to collector metrics found") | ||||
| 	} | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *NetstatCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	now := time.Now() | ||||
| 	file, err := os.Open(string(NETSTATFILE)) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError(m.name, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	defer file.Close() | ||||
| 	tdiff := now.Sub(m.lastTimestamp) | ||||
|  | ||||
| 	scanner := bufio.NewScanner(file) | ||||
| 	for scanner.Scan() { | ||||
| 		l := scanner.Text() | ||||
| 		if !strings.Contains(l, ":") { | ||||
| 			continue | ||||
| 		} | ||||
| 		tags := map[string]string{"device": dev, "type": "node"} | ||||
| 		for i, name := range m.matches { | ||||
| 			v, err := strconv.ParseInt(f[i], 10, 0) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New(name, tags, map[string]interface{}{"value": int(float64(v) * 1.0e-3)}, time.Now()) | ||||
| 		f := strings.Fields(l) | ||||
| 		dev := strings.Trim(f[0], ":") | ||||
|  | ||||
| 		if devmetrics, ok := m.matches[dev]; ok { | ||||
| 			for name, data := range devmetrics { | ||||
| 				v, err := strconv.ParseFloat(f[data.index], 64) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 					vdiff := v - data.lastValue | ||||
| 					value := vdiff / tdiff.Seconds() | ||||
| 					if data.lastValue == 0 { | ||||
| 						value = 0 | ||||
| 					} | ||||
| 					data.lastValue = v | ||||
| 					y, err := lp.New(name, m.devtags[dev], m.meta, map[string]interface{}{"value": value}, now) | ||||
| 					if err == nil { | ||||
| 						switch { | ||||
| 						case strings.Contains(name, "byte"): | ||||
| 							y.AddMeta("unit", "bytes/sec") | ||||
| 						case strings.Contains(name, "pkt"): | ||||
| 							y.AddMeta("unit", "packets/sec") | ||||
| 						} | ||||
| 						output <- y | ||||
| 					} | ||||
| 					devmetrics[name] = data | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	m.lastTimestamp = time.Now() | ||||
| } | ||||
|  | ||||
| func (m *NetstatCollector) Close() { | ||||
|   | ||||
							
								
								
									
										21
									
								
								collectors/netstatMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								collectors/netstatMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
|  | ||||
| ## `netstat` collector | ||||
|  | ||||
| ```json | ||||
|   "netstat": { | ||||
|     "include_devices": [ | ||||
|       "eth0" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `netstat` collector reads data from `/proc/net/dev` and outputs a handful **node** metrics. With the `include_devices` list you can specify which network devices should be measured. **Note**: Most other collectors use an _exclude_ list instead of an include list. | ||||
|  | ||||
| Metrics: | ||||
| * `net_bytes_in` (`unit=bytes/sec`) | ||||
| * `net_bytes_out` (`unit=bytes/sec`) | ||||
| * `net_pkts_in` (`unit=packets/sec`) | ||||
| * `net_pkts_out` (`unit=packets/sec`) | ||||
|  | ||||
| The device name is added as tag `device`. | ||||
|  | ||||
							
								
								
									
										39
									
								
								collectors/nfs3Metric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								collectors/nfs3Metric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
|  | ||||
| ## `nfs3stat` collector | ||||
|  | ||||
| ```json | ||||
|   "nfs3stat": { | ||||
|     "nfsstat" : "/path/to/nfsstat", | ||||
|     "exclude_metrics": [ | ||||
|       "nfs3_total" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `nfs3stat` collector reads data from `nfsstat` command and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. There is currently no possibility to get the metrics per mount point. | ||||
|  | ||||
|  | ||||
| Metrics: | ||||
| * `nfs3_total`  | ||||
| * `nfs3_null`  | ||||
| * `nfs3_getattr`  | ||||
| * `nfs3_setattr`  | ||||
| * `nfs3_lookup`  | ||||
| * `nfs3_access`  | ||||
| * `nfs3_readlink`  | ||||
| * `nfs3_read`  | ||||
| * `nfs3_write`  | ||||
| * `nfs3_create`  | ||||
| * `nfs3_mkdir`  | ||||
| * `nfs3_symlink`  | ||||
| * `nfs3_remove`  | ||||
| * `nfs3_rmdir`  | ||||
| * `nfs3_rename`  | ||||
| * `nfs3_link`  | ||||
| * `nfs3_readdir`  | ||||
| * `nfs3_readdirplus`  | ||||
| * `nfs3_fsstat`  | ||||
| * `nfs3_fsinfo`  | ||||
| * `nfs3_pathconf`  | ||||
| * `nfs3_commit`  | ||||
|  | ||||
							
								
								
									
										62
									
								
								collectors/nfs4Metric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								collectors/nfs4Metric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
|  | ||||
| ## `nfs4stat` collector | ||||
|  | ||||
| ```json | ||||
|   "nfs4stat": { | ||||
|     "nfsstat" : "/path/to/nfsstat", | ||||
|     "exclude_metrics": [ | ||||
|       "nfs4_total" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `nfs4stat` collector reads data from `nfsstat` command and outputs a handful **node** metrics. If a metric is not required, it can be excluded from forwarding it to the sink. There is currently no possibility to get the metrics per mount point. | ||||
|  | ||||
|  | ||||
| Metrics: | ||||
| * `nfs4_total`  | ||||
| * `nfs4_null`  | ||||
| * `nfs4_read`  | ||||
| * `nfs4_write`  | ||||
| * `nfs4_commit`  | ||||
| * `nfs4_open`  | ||||
| * `nfs4_open_conf`  | ||||
| * `nfs4_open_noat`  | ||||
| * `nfs4_open_dgrd`  | ||||
| * `nfs4_close`  | ||||
| * `nfs4_setattr`  | ||||
| * `nfs4_fsinfo`  | ||||
| * `nfs4_renew`  | ||||
| * `nfs4_setclntid`  | ||||
| * `nfs4_confirm`  | ||||
| * `nfs4_lock`  | ||||
| * `nfs4_lockt`  | ||||
| * `nfs4_locku`  | ||||
| * `nfs4_access`  | ||||
| * `nfs4_getattr`  | ||||
| * `nfs4_lookup`  | ||||
| * `nfs4_lookup_root`  | ||||
| * `nfs4_remove`  | ||||
| * `nfs4_rename`  | ||||
| * `nfs4_link`  | ||||
| * `nfs4_symlink`  | ||||
| * `nfs4_create`  | ||||
| * `nfs4_pathconf`  | ||||
| * `nfs4_statfs`  | ||||
| * `nfs4_readlink`  | ||||
| * `nfs4_readdir`  | ||||
| * `nfs4_server_caps`  | ||||
| * `nfs4_delegreturn`  | ||||
| * `nfs4_getacl`  | ||||
| * `nfs4_setacl`  | ||||
| * `nfs4_rel_lkowner`  | ||||
| * `nfs4_exchange_id`  | ||||
| * `nfs4_create_session`  | ||||
| * `nfs4_destroy_session`  | ||||
| * `nfs4_sequence`  | ||||
| * `nfs4_get_lease_time`  | ||||
| * `nfs4_reclaim_comp`  | ||||
| * `nfs4_secinfo_no`  | ||||
| * `nfs4_bind_conn_to_ses`  | ||||
|  | ||||
|  | ||||
							
								
								
									
										174
									
								
								collectors/nfsMetric.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								collectors/nfsMetric.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | ||||
| package collectors | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
|  | ||||
| 	//	"os" | ||||
| 	"os/exec" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| // First part contains the code for the general NfsCollector. | ||||
| // Later, the general NfsCollector is more limited to Nfs3- and Nfs4Collector. | ||||
|  | ||||
| const NFSSTAT_EXEC = `nfsstat` | ||||
|  | ||||
| type NfsCollectorData struct { | ||||
| 	current int64 | ||||
| 	last    int64 | ||||
| } | ||||
|  | ||||
| type nfsCollector struct { | ||||
| 	metricCollector | ||||
| 	tags    map[string]string | ||||
| 	version string | ||||
| 	config  struct { | ||||
| 		Nfsstats       string   `json:"nfsstat"` | ||||
| 		ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| 	} | ||||
| 	data map[string]NfsCollectorData | ||||
| } | ||||
|  | ||||
| func (m *nfsCollector) initStats() error { | ||||
| 	cmd := exec.Command(m.config.Nfsstats, `-l`) | ||||
| 	cmd.Wait() | ||||
| 	buffer, err := cmd.Output() | ||||
| 	if err == nil { | ||||
| 		for _, line := range strings.Split(string(buffer), "\n") { | ||||
| 			lf := strings.Fields(line) | ||||
| 			if len(lf) != 5 { | ||||
| 				continue | ||||
| 			} | ||||
| 			if lf[1] == m.version { | ||||
| 				name := strings.Trim(lf[3], ":") | ||||
| 				if _, exist := m.data[name]; !exist { | ||||
| 					value, err := strconv.ParseInt(lf[4], 0, 64) | ||||
| 					if err == nil { | ||||
| 						x := m.data[name] | ||||
| 						x.current = value | ||||
| 						x.last = 0 | ||||
| 						m.data[name] = x | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (m *nfsCollector) updateStats() error { | ||||
| 	cmd := exec.Command(m.config.Nfsstats, `-l`) | ||||
| 	cmd.Wait() | ||||
| 	buffer, err := cmd.Output() | ||||
| 	if err == nil { | ||||
| 		for _, line := range strings.Split(string(buffer), "\n") { | ||||
| 			lf := strings.Fields(line) | ||||
| 			if len(lf) != 5 { | ||||
| 				continue | ||||
| 			} | ||||
| 			if lf[1] == m.version { | ||||
| 				name := strings.Trim(lf[3], ":") | ||||
| 				if _, exist := m.data[name]; exist { | ||||
| 					value, err := strconv.ParseInt(lf[4], 0, 64) | ||||
| 					if err == nil { | ||||
| 						x := m.data[name] | ||||
| 						x.last = x.current | ||||
| 						x.current = value | ||||
| 						m.data[name] = x | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (m *nfsCollector) MainInit(config json.RawMessage) error { | ||||
| 	m.config.Nfsstats = string(NFSSTAT_EXEC) | ||||
| 	// Read JSON configuration | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			log.Print(err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "NFS", | ||||
| 	} | ||||
| 	m.tags = map[string]string{ | ||||
| 		"type": "node", | ||||
| 	} | ||||
| 	// Check if nfsstat is in executable search path | ||||
| 	_, err := exec.LookPath(m.config.Nfsstats) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("NfsCollector.Init(): Failed to find nfsstat binary '%s': %v", m.config.Nfsstats, err) | ||||
| 	} | ||||
| 	m.data = make(map[string]NfsCollectorData) | ||||
| 	m.initStats() | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *nfsCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	timestamp := time.Now() | ||||
|  | ||||
| 	m.updateStats() | ||||
| 	prefix := "" | ||||
| 	switch m.version { | ||||
| 	case "v3": | ||||
| 		prefix = "nfs3" | ||||
| 	case "v4": | ||||
| 		prefix = "nfs4" | ||||
| 	default: | ||||
| 		prefix = "nfs" | ||||
| 	} | ||||
|  | ||||
| 	for name, data := range m.data { | ||||
| 		if _, skip := stringArrayContains(m.config.ExcludeMetrics, name); skip { | ||||
| 			continue | ||||
| 		} | ||||
| 		value := data.current - data.last | ||||
| 		y, err := lp.New(fmt.Sprintf("%s_%s", prefix, name), m.tags, m.meta, map[string]interface{}{"value": value}, timestamp) | ||||
| 		if err == nil { | ||||
| 			y.AddMeta("version", m.version) | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *nfsCollector) Close() { | ||||
| 	m.init = false | ||||
| } | ||||
|  | ||||
| type Nfs3Collector struct { | ||||
| 	nfsCollector | ||||
| } | ||||
|  | ||||
| type Nfs4Collector struct { | ||||
| 	nfsCollector | ||||
| } | ||||
|  | ||||
| func (m *Nfs3Collector) Init(config json.RawMessage) error { | ||||
| 	m.name = "Nfs3Collector" | ||||
| 	m.version = `v3` | ||||
| 	m.setup() | ||||
| 	return m.MainInit(config) | ||||
| } | ||||
|  | ||||
| func (m *Nfs4Collector) Init(config json.RawMessage) error { | ||||
| 	m.name = "Nfs4Collector" | ||||
| 	m.version = `v4` | ||||
| 	m.setup() | ||||
| 	return m.MainInit(config) | ||||
| } | ||||
| @@ -2,15 +2,16 @@ package collectors | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| // | ||||
| @@ -42,11 +43,11 @@ type NUMAStatsCollectorTopolgy struct { | ||||
| } | ||||
|  | ||||
| type NUMAStatsCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	topology []NUMAStatsCollectorTopolgy | ||||
| } | ||||
|  | ||||
| func (m *NUMAStatsCollector) Init(config []byte) error { | ||||
| func (m *NUMAStatsCollector) Init(config json.RawMessage) error { | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| @@ -54,25 +55,29 @@ func (m *NUMAStatsCollector) Init(config []byte) error { | ||||
|  | ||||
| 	m.name = "NUMAStatsCollector" | ||||
| 	m.setup() | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "NUMA", | ||||
| 	} | ||||
|  | ||||
| 	// Loop for all NUMA node directories | ||||
| 	baseDir := "/sys/devices/system/node" | ||||
| 	globPattern := filepath.Join(baseDir, "node[0-9]*") | ||||
| 	base := "/sys/devices/system/node/node" | ||||
| 	globPattern := base + "[0-9]*" | ||||
| 	dirs, err := filepath.Glob(globPattern) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("unable to glob files with pattern %s", globPattern) | ||||
| 		return fmt.Errorf("unable to glob files with pattern '%s'", globPattern) | ||||
| 	} | ||||
| 	if dirs == nil { | ||||
| 		return fmt.Errorf("unable to find any files with pattern %s", globPattern) | ||||
| 		return fmt.Errorf("unable to find any files with pattern '%s'", globPattern) | ||||
| 	} | ||||
| 	m.topology = make([]NUMAStatsCollectorTopolgy, 0, len(dirs)) | ||||
| 	for _, dir := range dirs { | ||||
| 		node := strings.TrimPrefix(dir, "/sys/devices/system/node/node") | ||||
| 		node := strings.TrimPrefix(dir, base) | ||||
| 		file := filepath.Join(dir, "numastat") | ||||
| 		m.topology = append(m.topology, | ||||
| 			NUMAStatsCollectorTopolgy{ | ||||
| 				file:   file, | ||||
| 				tagSet: map[string]string{"domain": node}, | ||||
| 				tagSet: map[string]string{"memoryDomain": node}, | ||||
| 			}) | ||||
| 	} | ||||
|  | ||||
| @@ -80,7 +85,7 @@ func (m *NUMAStatsCollector) Init(config []byte) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *NUMAStatsCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *NUMAStatsCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -92,9 +97,14 @@ func (m *NUMAStatsCollector) Read(interval time.Duration, out *[]lp.MutableMetri | ||||
| 		now := time.Now() | ||||
| 		file, err := os.Open(t.file) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to open file '%s': %v", t.file, err)) | ||||
| 			return | ||||
| 		} | ||||
| 		scanner := bufio.NewScanner(file) | ||||
|  | ||||
| 		// Read line by line | ||||
| 		for scanner.Scan() { | ||||
| 			split := strings.Fields(scanner.Text()) | ||||
| 			if len(split) != 2 { | ||||
| @@ -103,12 +113,20 @@ func (m *NUMAStatsCollector) Read(interval time.Duration, out *[]lp.MutableMetri | ||||
| 			key := split[0] | ||||
| 			value, err := strconv.ParseInt(split[1], 10, 64) | ||||
| 			if err != nil { | ||||
| 				log.Printf("failed to convert %s='%s' to int64: %v", key, split[1], err) | ||||
| 				cclog.ComponentError( | ||||
| 					m.name, | ||||
| 					fmt.Sprintf("Read(): Failed to convert %s='%s' to int64: %v", key, split[1], err)) | ||||
| 				continue | ||||
| 			} | ||||
| 			y, err := lp.New("numastats_"+key, t.tagSet, map[string]interface{}{"value": value}, now) | ||||
| 			y, err := lp.New( | ||||
| 				"numastats_"+key, | ||||
| 				t.tagSet, | ||||
| 				m.meta, | ||||
| 				map[string]interface{}{"value": value}, | ||||
| 				now, | ||||
| 			) | ||||
| 			if err == nil { | ||||
| 				*out = append(*out, y) | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
|   | ||||
							
								
								
									
										15
									
								
								collectors/numastatsMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								collectors/numastatsMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
|  | ||||
| ## `numastat` collector | ||||
| ```json | ||||
|   "numastat": {} | ||||
| ``` | ||||
|  | ||||
| The `numastat` collector reads data from `/sys/devices/system/node/node*/numastat` and outputs a handful **memoryDomain** metrics. See: https://www.kernel.org/doc/html/latest/admin-guide/numastat.html | ||||
|  | ||||
| Metrics: | ||||
| * `numastats_numa_hit`: A process wanted to allocate memory from this node, and succeeded. | ||||
| * `numastats_numa_miss`: A process wanted to allocate memory from another node, but ended up with memory from this node. | ||||
| * `numastats_numa_foreign`: A process wanted to allocate on this node, but ended up with memory from another node. | ||||
| * `numastats_local_node`: A process ran on this node's CPU, and got memory from this node. | ||||
| * `numastats_other_node`: A process ran on a different node's CPU, and got memory from this node. | ||||
| * `numastats_interleave_hit`: Interleaving wanted to allocate from this node and succeeded. | ||||
| @@ -7,19 +7,28 @@ import ( | ||||
| 	"log" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	"github.com/NVIDIA/go-nvml/pkg/nvml" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| ) | ||||
|  | ||||
| type NvidiaCollectorConfig struct { | ||||
| 	ExcludeMetrics []string `json:"exclude_metrics,omitempty"` | ||||
| 	ExcludeDevices []string `json:"exclude_devices,omitempty"` | ||||
| 	AddPciInfoTag  bool     `json:"add_pci_info_tag,omitempty"` | ||||
| } | ||||
|  | ||||
| type NvidiaCollectorDevice struct { | ||||
| 	device         nvml.Device | ||||
| 	excludeMetrics map[string]bool | ||||
| 	tags           map[string]string | ||||
| } | ||||
|  | ||||
| type NvidiaCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	num_gpus int | ||||
| 	config   NvidiaCollectorConfig | ||||
| 	gpus     []NvidiaCollectorDevice | ||||
| } | ||||
|  | ||||
| func (m *NvidiaCollector) CatchPanic() { | ||||
| @@ -29,9 +38,10 @@ func (m *NvidiaCollector) CatchPanic() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (m *NvidiaCollector) Init(config []byte) error { | ||||
| func (m *NvidiaCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "NvidiaCollector" | ||||
| 	m.config.AddPciInfoTag = false | ||||
| 	m.setup() | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| @@ -39,224 +49,415 @@ func (m *NvidiaCollector) Init(config []byte) error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "Nvidia", | ||||
| 	} | ||||
|  | ||||
| 	m.num_gpus = 0 | ||||
| 	defer m.CatchPanic() | ||||
|  | ||||
| 	// Initialize NVIDIA Management Library (NVML) | ||||
| 	ret := nvml.Init() | ||||
| 	if ret != nvml.SUCCESS { | ||||
| 		err = errors.New(nvml.ErrorString(ret)) | ||||
| 		cclog.ComponentError(m.name, "Unable to initialize NVML", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	m.num_gpus, ret = nvml.DeviceGetCount() | ||||
|  | ||||
| 	// Number of NVIDIA GPUs | ||||
| 	num_gpus, ret := nvml.DeviceGetCount() | ||||
| 	if ret != nvml.SUCCESS { | ||||
| 		err = errors.New(nvml.ErrorString(ret)) | ||||
| 		cclog.ComponentError(m.name, "Unable to get device count", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// For all GPUs | ||||
| 	m.gpus = make([]NvidiaCollectorDevice, num_gpus) | ||||
| 	for i := 0; i < num_gpus; i++ { | ||||
| 		g := &m.gpus[i] | ||||
|  | ||||
| 		// Skip excluded devices | ||||
| 		str_i := fmt.Sprintf("%d", i) | ||||
| 		if _, skip := stringArrayContains(m.config.ExcludeDevices, str_i); skip { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// Get device handle | ||||
| 		device, ret := nvml.DeviceGetHandleByIndex(i) | ||||
| 		if ret != nvml.SUCCESS { | ||||
| 			err = errors.New(nvml.ErrorString(ret)) | ||||
| 			cclog.ComponentError(m.name, "Unable to get device at index", i, ":", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 		g.device = device | ||||
|  | ||||
| 		// Add tags | ||||
| 		g.tags = map[string]string{ | ||||
| 			"type":    "accelerator", | ||||
| 			"type-id": str_i, | ||||
| 		} | ||||
|  | ||||
| 		// Add excluded metrics | ||||
| 		g.excludeMetrics = map[string]bool{} | ||||
| 		for _, e := range m.config.ExcludeMetrics { | ||||
| 			g.excludeMetrics[e] = true | ||||
| 		} | ||||
|  | ||||
| 		// Add PCI info as tag | ||||
| 		if m.config.AddPciInfoTag { | ||||
| 			pciInfo, ret := nvml.DeviceGetPciInfo(g.device) | ||||
| 			if ret != nvml.SUCCESS { | ||||
| 				err = errors.New(nvml.ErrorString(ret)) | ||||
| 				cclog.ComponentError(m.name, "Unable to get PCI info for device at index", i, ":", err.Error()) | ||||
| 				return err | ||||
| 			} | ||||
| 			g.tags["pci_identifier"] = fmt.Sprintf( | ||||
| 				"%08X:%02X:%02X.0", | ||||
| 				pciInfo.Domain, | ||||
| 				pciInfo.Bus, | ||||
| 				pciInfo.Device) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *NvidiaCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *NvidiaCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| 	for i := 0; i < m.num_gpus; i++ { | ||||
| 		device, ret := nvml.DeviceGetHandleByIndex(i) | ||||
| 		if ret != nvml.SUCCESS { | ||||
| 			log.Fatalf("Unable to get device at index %d: %v", i, nvml.ErrorString(ret)) | ||||
| 			return | ||||
| 		} | ||||
| 		_, skip := stringArrayContains(m.config.ExcludeDevices, fmt.Sprintf("%d", i)) | ||||
| 		if skip { | ||||
| 			continue | ||||
| 		} | ||||
| 		tags := map[string]string{"type": "accelerator", "type-id": fmt.Sprintf("%d", i)} | ||||
|  | ||||
| 		util, ret := nvml.DeviceGetUtilizationRates(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "util") | ||||
| 			y, err := lp.New("util", tags, map[string]interface{}{"value": float64(util.Gpu)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "mem_util") | ||||
| 			y, err = lp.New("mem_util", tags, map[string]interface{}{"value": float64(util.Memory)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 	for i := range m.gpus { | ||||
| 		device := &m.gpus[i] | ||||
|  | ||||
| 		if !device.excludeMetrics["nv_util"] || !device.excludeMetrics["nv_mem_util"] { | ||||
| 			// Retrieves the current utilization rates for the device's major subsystems. | ||||
| 			// | ||||
| 			// Available utilization rates | ||||
| 			// * Gpu: Percent of time over the past sample period during which one or more kernels was executing on the GPU. | ||||
| 			// * Memory: Percent of time over the past sample period during which global (device) memory was being read or written | ||||
| 			// | ||||
| 			// Note: | ||||
| 			// * During driver initialization when ECC is enabled one can see high GPU and Memory Utilization readings. | ||||
| 			//   This is caused by ECC Memory Scrubbing mechanism that is performed during driver initialization. | ||||
| 			// * On MIG-enabled GPUs, querying device utilization rates is not currently supported. | ||||
| 			util, ret := nvml.DeviceGetUtilizationRates(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				if !device.excludeMetrics["nv_util"] { | ||||
| 					y, err := lp.New("nv_util", device.tags, m.meta, map[string]interface{}{"value": float64(util.Gpu)}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						y.AddMeta("unit", "%") | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 				if !device.excludeMetrics["nv_mem_util"] { | ||||
| 					y, err := lp.New("nv_mem_util", device.tags, m.meta, map[string]interface{}{"value": float64(util.Memory)}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						y.AddMeta("unit", "%") | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		meminfo, ret := nvml.DeviceGetMemoryInfo(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			t := float64(meminfo.Total) / (1024 * 1024) | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "mem_total") | ||||
| 			y, err := lp.New("mem_total", tags, map[string]interface{}{"value": t}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 			f := float64(meminfo.Used) / (1024 * 1024) | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "fb_memory") | ||||
| 			y, err = lp.New("fb_memory", tags, map[string]interface{}{"value": f}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_mem_total"] || !device.excludeMetrics["nv_fb_memory"] { | ||||
| 			// Retrieves the amount of used, free and total memory available on the device, in bytes. | ||||
| 			// | ||||
| 			// Enabling ECC reduces the amount of total available memory, due to the extra required parity bits. | ||||
| 			// | ||||
| 			// The reported amount of used memory is equal to the sum of memory allocated by all active channels on the device. | ||||
| 			// | ||||
| 			// Available memory info: | ||||
| 			// * Free: Unallocated FB memory (in bytes). | ||||
| 			// * Total: Total installed FB memory (in bytes). | ||||
| 			// * Used: Allocated FB memory (in bytes). Note that the driver/GPU always sets aside a small amount of memory for bookkeeping. | ||||
| 			// | ||||
| 			// Note: | ||||
| 			// In MIG mode, if device handle is provided, the API returns aggregate information, only if the caller has appropriate privileges. | ||||
| 			// Per-instance information can be queried by using specific MIG device handles. | ||||
| 			meminfo, ret := nvml.DeviceGetMemoryInfo(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				if !device.excludeMetrics["nv_mem_total"] { | ||||
| 					t := float64(meminfo.Total) / (1024 * 1024) | ||||
| 					y, err := lp.New("nv_mem_total", device.tags, m.meta, map[string]interface{}{"value": t}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						y.AddMeta("unit", "MByte") | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if !device.excludeMetrics["nv_fb_memory"] { | ||||
| 					f := float64(meminfo.Used) / (1024 * 1024) | ||||
| 					y, err := lp.New("nv_fb_memory", device.tags, m.meta, map[string]interface{}{"value": f}, time.Now()) | ||||
| 					if err == nil { | ||||
| 						y.AddMeta("unit", "MByte") | ||||
| 						output <- y | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		temp, ret := nvml.DeviceGetTemperature(device, nvml.TEMPERATURE_GPU) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "temp") | ||||
| 			y, err := lp.New("temp", tags, map[string]interface{}{"value": float64(temp)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_temp"] { | ||||
| 			// Retrieves the current temperature readings for the device, in degrees C. | ||||
| 			// | ||||
| 			// Available temperature sensors: | ||||
| 			// * TEMPERATURE_GPU: Temperature sensor for the GPU die. | ||||
| 			// * NVML_TEMPERATURE_COUNT | ||||
| 			temp, ret := nvml.DeviceGetTemperature(device.device, nvml.TEMPERATURE_GPU) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_temp", device.tags, m.meta, map[string]interface{}{"value": float64(temp)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "degC") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		fan, ret := nvml.DeviceGetFanSpeed(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "fan") | ||||
| 			y, err := lp.New("fan", tags, map[string]interface{}{"value": float64(fan)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_fan"] { | ||||
| 			// Retrieves the intended operating speed of the device's fan. | ||||
| 			// | ||||
| 			// Note: The reported speed is the intended fan speed. | ||||
| 			// If the fan is physically blocked and unable to spin, the output will not match the actual fan speed. | ||||
| 			// | ||||
| 			// For all discrete products with dedicated fans. | ||||
| 			// | ||||
| 			// The fan speed is expressed as a percentage of the product's maximum noise tolerance fan speed. | ||||
| 			// This value may exceed 100% in certain cases. | ||||
| 			fan, ret := nvml.DeviceGetFanSpeed(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_fan", device.tags, m.meta, map[string]interface{}{"value": float64(fan)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "%") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		_, ecc_pend, ret := nvml.DeviceGetEccMode(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			var y lp.MutableMetric | ||||
| 			var err error | ||||
| 			switch ecc_pend { | ||||
| 			case nvml.FEATURE_DISABLED: | ||||
| 				y, err = lp.New("ecc_mode", tags, map[string]interface{}{"value": string("OFF")}, time.Now()) | ||||
| 			case nvml.FEATURE_ENABLED: | ||||
| 				y, err = lp.New("ecc_mode", tags, map[string]interface{}{"value": string("ON")}, time.Now()) | ||||
| 			default: | ||||
| 				y, err = lp.New("ecc_mode", tags, map[string]interface{}{"value": string("UNKNOWN")}, time.Now()) | ||||
| 			} | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "ecc_mode") | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 			} | ||||
| 		} else if ret == nvml.ERROR_NOT_SUPPORTED { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "ecc_mode") | ||||
| 			y, err := lp.New("ecc_mode", tags, map[string]interface{}{"value": string("N/A")}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_ecc_mode"] { | ||||
| 			// Retrieves the current and pending ECC modes for the device. | ||||
| 			// | ||||
| 			// For Fermi or newer fully supported devices. Only applicable to devices with ECC. | ||||
| 			// Requires NVML_INFOROM_ECC version 1.0 or higher. | ||||
| 			// | ||||
| 			// Changing ECC modes requires a reboot. | ||||
| 			// The "pending" ECC mode refers to the target mode following the next reboot. | ||||
| 			_, ecc_pend, ret := nvml.DeviceGetEccMode(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				var y lp.CCMetric | ||||
| 				var err error | ||||
| 				switch ecc_pend { | ||||
| 				case nvml.FEATURE_DISABLED: | ||||
| 					y, err = lp.New("nv_ecc_mode", device.tags, m.meta, map[string]interface{}{"value": "OFF"}, time.Now()) | ||||
| 				case nvml.FEATURE_ENABLED: | ||||
| 					y, err = lp.New("nv_ecc_mode", device.tags, m.meta, map[string]interface{}{"value": "ON"}, time.Now()) | ||||
| 				default: | ||||
| 					y, err = lp.New("nv_ecc_mode", device.tags, m.meta, map[string]interface{}{"value": "UNKNOWN"}, time.Now()) | ||||
| 				} | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} else if ret == nvml.ERROR_NOT_SUPPORTED { | ||||
| 				y, err := lp.New("nv_ecc_mode", device.tags, m.meta, map[string]interface{}{"value": "N/A"}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		pstate, ret := nvml.DeviceGetPerformanceState(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "perf_state") | ||||
| 			y, err := lp.New("perf_state", tags, map[string]interface{}{"value": fmt.Sprintf("P%d", int(pstate))}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_perf_state"] { | ||||
| 			// Retrieves the current performance state for the device. | ||||
| 			// | ||||
| 			// Allowed PStates: | ||||
| 			//  0: Maximum Performance. | ||||
| 			// .. | ||||
| 			// 15: Minimum Performance. | ||||
| 			// 32: Unknown performance state. | ||||
| 			pState, ret := nvml.DeviceGetPerformanceState(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_perf_state", device.tags, m.meta, map[string]interface{}{"value": fmt.Sprintf("P%d", int(pState))}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		power, ret := nvml.DeviceGetPowerUsage(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "power_usage_report") | ||||
| 			y, err := lp.New("power_usage_report", tags, map[string]interface{}{"value": float64(power) / 1000}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_power_usage_report"] { | ||||
| 			// Retrieves power usage for this GPU in milliwatts and its associated circuitry (e.g. memory) | ||||
| 			// | ||||
| 			// On Fermi and Kepler GPUs the reading is accurate to within +/- 5% of current power draw. | ||||
| 			// | ||||
| 			// It is only available if power management mode is supported | ||||
| 			power, ret := nvml.DeviceGetPowerUsage(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_power_usage_report", device.tags, m.meta, map[string]interface{}{"value": float64(power) / 1000}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "watts") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		gclk, ret := nvml.DeviceGetClockInfo(device, nvml.CLOCK_GRAPHICS) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "graphics_clock_report") | ||||
| 			y, err := lp.New("graphics_clock_report", tags, map[string]interface{}{"value": float64(gclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		// Retrieves the current clock speeds for the device. | ||||
| 		// | ||||
| 		// Available clock information: | ||||
| 		// * CLOCK_GRAPHICS: Graphics clock domain. | ||||
| 		// * CLOCK_SM: Streaming Multiprocessor clock domain. | ||||
| 		// * CLOCK_MEM: Memory clock domain. | ||||
| 		if !device.excludeMetrics["nv_graphics_clock_report"] { | ||||
| 			graphicsClock, ret := nvml.DeviceGetClockInfo(device.device, nvml.CLOCK_GRAPHICS) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_graphics_clock_report", device.tags, m.meta, map[string]interface{}{"value": float64(graphicsClock)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		smclk, ret := nvml.DeviceGetClockInfo(device, nvml.CLOCK_SM) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "sm_clock_report") | ||||
| 			y, err := lp.New("sm_clock_report", tags, map[string]interface{}{"value": float64(smclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_sm_clock_report"] { | ||||
| 			smCock, ret := nvml.DeviceGetClockInfo(device.device, nvml.CLOCK_SM) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_sm_clock_report", device.tags, m.meta, map[string]interface{}{"value": float64(smCock)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		memclk, ret := nvml.DeviceGetClockInfo(device, nvml.CLOCK_MEM) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "mem_clock_report") | ||||
| 			y, err := lp.New("mem_clock_report", tags, map[string]interface{}{"value": float64(memclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_mem_clock_report"] { | ||||
| 			memClock, ret := nvml.DeviceGetClockInfo(device.device, nvml.CLOCK_MEM) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_mem_clock_report", device.tags, m.meta, map[string]interface{}{"value": float64(memClock)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		max_gclk, ret := nvml.DeviceGetMaxClockInfo(device, nvml.CLOCK_GRAPHICS) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "max_graphics_clock") | ||||
| 			y, err := lp.New("max_graphics_clock", tags, map[string]interface{}{"value": float64(max_gclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		// Retrieves the maximum clock speeds for the device. | ||||
| 		// | ||||
| 		// Available clock information: | ||||
| 		// * CLOCK_GRAPHICS: Graphics clock domain. | ||||
| 		// * CLOCK_SM:       Streaming multiprocessor clock domain. | ||||
| 		// * CLOCK_MEM:      Memory clock domain. | ||||
| 		// * CLOCK_VIDEO:    Video encoder/decoder clock domain. | ||||
| 		// * CLOCK_COUNT:    Count of clock types. | ||||
| 		// | ||||
| 		// Note: | ||||
| 		/// On GPUs from Fermi family current P0 clocks (reported by nvmlDeviceGetClockInfo) can differ from max clocks by few MHz. | ||||
| 		if !device.excludeMetrics["nv_max_graphics_clock"] { | ||||
| 			max_gclk, ret := nvml.DeviceGetMaxClockInfo(device.device, nvml.CLOCK_GRAPHICS) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_max_graphics_clock", device.tags, m.meta, map[string]interface{}{"value": float64(max_gclk)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		max_smclk, ret := nvml.DeviceGetClockInfo(device, nvml.CLOCK_SM) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "max_sm_clock") | ||||
| 			y, err := lp.New("max_sm_clock", tags, map[string]interface{}{"value": float64(max_smclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_max_sm_clock"] { | ||||
| 			maxSmClock, ret := nvml.DeviceGetClockInfo(device.device, nvml.CLOCK_SM) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_max_sm_clock", device.tags, m.meta, map[string]interface{}{"value": float64(maxSmClock)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		max_memclk, ret := nvml.DeviceGetClockInfo(device, nvml.CLOCK_MEM) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "max_mem_clock") | ||||
| 			y, err := lp.New("max_mem_clock", tags, map[string]interface{}{"value": float64(max_memclk)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_max_mem_clock"] { | ||||
| 			maxMemClock, ret := nvml.DeviceGetClockInfo(device.device, nvml.CLOCK_MEM) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_max_mem_clock", device.tags, m.meta, map[string]interface{}{"value": float64(maxMemClock)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "MHz") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		ecc_db, ret := nvml.DeviceGetTotalEccErrors(device, 1, 1) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "ecc_db_error") | ||||
| 			y, err := lp.New("ecc_db_error", tags, map[string]interface{}{"value": float64(ecc_db)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_ecc_db_error"] { | ||||
| 			// Retrieves the total ECC error counts for the device. | ||||
| 			// | ||||
| 			// For Fermi or newer fully supported devices. | ||||
| 			// Only applicable to devices with ECC. | ||||
| 			// Requires NVML_INFOROM_ECC version 1.0 or higher. | ||||
| 			// Requires ECC Mode to be enabled. | ||||
| 			// | ||||
| 			// The total error count is the sum of errors across each of the separate memory systems, | ||||
| 			// i.e. the total set of errors across the entire device. | ||||
| 			ecc_db, ret := nvml.DeviceGetTotalEccErrors(device.device, nvml.MEMORY_ERROR_TYPE_UNCORRECTED, nvml.AGGREGATE_ECC) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_ecc_db_error", device.tags, m.meta, map[string]interface{}{"value": float64(ecc_db)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		ecc_sb, ret := nvml.DeviceGetTotalEccErrors(device, 0, 1) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "ecc_sb_error") | ||||
| 			y, err := lp.New("ecc_sb_error", tags, map[string]interface{}{"value": float64(ecc_sb)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_ecc_sb_error"] { | ||||
| 			ecc_sb, ret := nvml.DeviceGetTotalEccErrors(device.device, nvml.MEMORY_ERROR_TYPE_CORRECTED, nvml.AGGREGATE_ECC) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_ecc_sb_error", device.tags, m.meta, map[string]interface{}{"value": float64(ecc_sb)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		pwr_limit, ret := nvml.DeviceGetPowerManagementLimit(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "power_man_limit") | ||||
| 			y, err := lp.New("power_man_limit", tags, map[string]interface{}{"value": float64(pwr_limit)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_power_man_limit"] { | ||||
| 			// Retrieves the power management limit associated with this device. | ||||
| 			// | ||||
| 			// For Fermi or newer fully supported devices. | ||||
| 			// | ||||
| 			// The power limit defines the upper boundary for the card's power draw. | ||||
| 			// If the card's total power draw reaches this limit the power management algorithm kicks in. | ||||
| 			pwr_limit, ret := nvml.DeviceGetPowerManagementLimit(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_power_man_limit", device.tags, m.meta, map[string]interface{}{"value": float64(pwr_limit) / 1000}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "watts") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		enc_util, _, ret := nvml.DeviceGetEncoderUtilization(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "encoder_util") | ||||
| 			y, err := lp.New("encoder_util", tags, map[string]interface{}{"value": float64(enc_util)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_encoder_util"] { | ||||
| 			// Retrieves the current utilization and sampling size in microseconds for the Encoder | ||||
| 			// | ||||
| 			// For Kepler or newer fully supported devices. | ||||
| 			// | ||||
| 			// Note: On MIG-enabled GPUs, querying encoder utilization is not currently supported. | ||||
| 			enc_util, _, ret := nvml.DeviceGetEncoderUtilization(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_encoder_util", device.tags, m.meta, map[string]interface{}{"value": float64(enc_util)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "%") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		dec_util, _, ret := nvml.DeviceGetDecoderUtilization(device) | ||||
| 		if ret == nvml.SUCCESS { | ||||
| 			_, skip = stringArrayContains(m.config.ExcludeMetrics, "decoder_util") | ||||
| 			y, err := lp.New("decoder_util", tags, map[string]interface{}{"value": float64(dec_util)}, time.Now()) | ||||
| 			if err == nil && !skip { | ||||
| 				*out = append(*out, y) | ||||
| 		if !device.excludeMetrics["nv_decoder_util"] { | ||||
| 			// Retrieves the current utilization and sampling size in microseconds for the Decoder | ||||
| 			// | ||||
| 			// For Kepler or newer fully supported devices. | ||||
| 			// | ||||
| 			// Note: On MIG-enabled GPUs, querying decoder utilization is not currently supported. | ||||
| 			dec_util, _, ret := nvml.DeviceGetDecoderUtilization(device.device) | ||||
| 			if ret == nvml.SUCCESS { | ||||
| 				y, err := lp.New("nv_decoder_util", device.tags, m.meta, map[string]interface{}{"value": float64(dec_util)}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					y.AddMeta("unit", "%") | ||||
| 					output <- y | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										40
									
								
								collectors/nvidiaMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								collectors/nvidiaMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
|  | ||||
| ## `nvidia` collector | ||||
|  | ||||
| ```json | ||||
|   "nvidia": { | ||||
|     "exclude_devices" : [ | ||||
|       "0","1" | ||||
|     ], | ||||
|     "exclude_metrics": [ | ||||
|       "nv_fb_memory", | ||||
|       "nv_fan" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| Metrics: | ||||
| * `nv_util` | ||||
| * `nv_mem_util` | ||||
| * `nv_mem_total` | ||||
| * `nv_fb_memory` | ||||
| * `nv_temp` | ||||
| * `nv_fan` | ||||
| * `nv_ecc_mode` | ||||
| * `nv_perf_state` | ||||
| * `nv_power_usage_report` | ||||
| * `nv_graphics_clock_report` | ||||
| * `nv_sm_clock_report` | ||||
| * `nv_mem_clock_report` | ||||
| * `nv_max_graphics_clock` | ||||
| * `nv_max_sm_clock` | ||||
| * `nv_max_mem_clock` | ||||
| * `nv_ecc_db_error` | ||||
| * `nv_ecc_sb_error` | ||||
| * `nv_power_man_limit` | ||||
| * `nv_encoder_util` | ||||
| * `nv_decoder_util` | ||||
|  | ||||
| It uses a separate `type` in the metrics. The output metric looks like this: | ||||
| `<name>,type=accelerator,type-id=<nvidia-gpu-id> value=<metric value> <timestamp>` | ||||
|  | ||||
| @@ -4,104 +4,227 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const HWMON_PATH = `/sys/class/hwmon` | ||||
| // See: https://www.kernel.org/doc/html/latest/hwmon/sysfs-interface.html | ||||
| // /sys/class/hwmon/hwmon*/name -> coretemp | ||||
| // /sys/class/hwmon/hwmon*/temp*_label -> Core 0 | ||||
| // /sys/class/hwmon/hwmon*/temp*_input -> 27800 = 27.8°C | ||||
| // /sys/class/hwmon/hwmon*/temp*_max -> 86000 = 86.0°C | ||||
| // /sys/class/hwmon/hwmon*/temp*_crit -> 100000 = 100.0°C | ||||
|  | ||||
| type TempCollectorConfig struct { | ||||
| 	ExcludeMetrics []string                     `json:"exclude_metrics"` | ||||
| 	TagOverride    map[string]map[string]string `json:"tag_override"` | ||||
| type TempCollectorSensor struct { | ||||
| 	name         string | ||||
| 	label        string | ||||
| 	metricName   string // Default: name_label | ||||
| 	file         string | ||||
| 	maxTempName  string | ||||
| 	maxTemp      int64 | ||||
| 	critTempName string | ||||
| 	critTemp     int64 | ||||
| 	tags         map[string]string | ||||
| } | ||||
|  | ||||
| type TempCollector struct { | ||||
| 	MetricCollector | ||||
| 	config TempCollectorConfig | ||||
| 	metricCollector | ||||
| 	config struct { | ||||
| 		ExcludeMetrics     []string                     `json:"exclude_metrics"` | ||||
| 		TagOverride        map[string]map[string]string `json:"tag_override"` | ||||
| 		ReportMaxTemp      bool                         `json:"report_max_temperature"` | ||||
| 		ReportCriticalTemp bool                         `json:"report_critical_temperature"` | ||||
| 	} | ||||
| 	sensors []*TempCollectorSensor | ||||
| } | ||||
|  | ||||
| func (m *TempCollector) Init(config []byte) error { | ||||
| func (m *TempCollector) Init(config json.RawMessage) error { | ||||
| 	// Check if already initialized | ||||
| 	if m.init { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	m.name = "TempCollector" | ||||
| 	m.setup() | ||||
| 	m.init = true | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	m.meta = map[string]string{ | ||||
| 		"source": m.name, | ||||
| 		"group":  "IPMI", | ||||
| 		"unit":   "degC", | ||||
| 	} | ||||
|  | ||||
| 	m.sensors = make([]*TempCollectorSensor, 0) | ||||
|  | ||||
| 	// Find all temperature sensor files | ||||
| 	globPattern := filepath.Join("/sys/class/hwmon", "*", "temp*_input") | ||||
| 	inputFiles, err := filepath.Glob(globPattern) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("Unable to glob files with pattern '%s': %v", globPattern, err) | ||||
| 	} | ||||
| 	if inputFiles == nil { | ||||
| 		return fmt.Errorf("Unable to find any files with pattern '%s'", globPattern) | ||||
| 	} | ||||
|  | ||||
| 	// Get sensor name for each temperature sensor file | ||||
| 	for _, file := range inputFiles { | ||||
| 		sensor := new(TempCollectorSensor) | ||||
|  | ||||
| 		// sensor name | ||||
| 		nameFile := filepath.Join(filepath.Dir(file), "name") | ||||
| 		name, err := ioutil.ReadFile(nameFile) | ||||
| 		if err == nil { | ||||
| 			sensor.name = strings.TrimSpace(string(name)) | ||||
| 		} | ||||
|  | ||||
| 		// sensor label | ||||
| 		labelFile := strings.TrimSuffix(file, "_input") + "_label" | ||||
| 		label, err := ioutil.ReadFile(labelFile) | ||||
| 		if err == nil { | ||||
| 			sensor.label = strings.TrimSpace(string(label)) | ||||
| 		} | ||||
|  | ||||
| 		// sensor metric name | ||||
| 		switch { | ||||
| 		case len(sensor.name) == 0 && len(sensor.label) == 0: | ||||
| 			continue | ||||
| 		case sensor.name == "coretemp" && strings.HasPrefix(sensor.label, "Core ") || | ||||
| 			sensor.name == "coretemp" && strings.HasPrefix(sensor.label, "Package id "): | ||||
| 			sensor.metricName = "temp_" + sensor.label | ||||
| 		case len(sensor.name) != 0 && len(sensor.label) != 0: | ||||
| 			sensor.metricName = sensor.name + "_" + sensor.label | ||||
| 		case len(sensor.name) != 0: | ||||
| 			sensor.metricName = sensor.name | ||||
| 		case len(sensor.label) != 0: | ||||
| 			sensor.metricName = sensor.label | ||||
| 		} | ||||
| 		sensor.metricName = strings.ToLower(sensor.metricName) | ||||
| 		sensor.metricName = strings.Replace(sensor.metricName, " ", "_", -1) | ||||
| 		// Add temperature prefix, if required | ||||
| 		if !strings.Contains(sensor.metricName, "temp") { | ||||
| 			sensor.metricName = "temp_" + sensor.metricName | ||||
| 		} | ||||
|  | ||||
| 		// Sensor file | ||||
| 		sensor.file = file | ||||
|  | ||||
| 		// Sensor tags | ||||
| 		sensor.tags = map[string]string{ | ||||
| 			"type": "node", | ||||
| 		} | ||||
|  | ||||
| 		// Apply tag override configuration | ||||
| 		for key, newtags := range m.config.TagOverride { | ||||
| 			if strings.Contains(sensor.file, key) { | ||||
| 				sensor.tags = newtags | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// max temperature | ||||
| 		if m.config.ReportMaxTemp { | ||||
| 			maxTempFile := strings.TrimSuffix(file, "_input") + "_max" | ||||
| 			if buffer, err := ioutil.ReadFile(maxTempFile); err == nil { | ||||
| 				if x, err := strconv.ParseInt(strings.TrimSpace(string(buffer)), 10, 64); err == nil { | ||||
| 					sensor.maxTempName = strings.Replace(sensor.metricName, "temp", "max_temp", 1) | ||||
| 					sensor.maxTemp = x / 1000 | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// critical temperature | ||||
| 		if m.config.ReportCriticalTemp { | ||||
| 			criticalTempFile := strings.TrimSuffix(file, "_input") + "_crit" | ||||
| 			if buffer, err := ioutil.ReadFile(criticalTempFile); err == nil { | ||||
| 				if x, err := strconv.ParseInt(strings.TrimSpace(string(buffer)), 10, 64); err == nil { | ||||
| 					sensor.critTempName = strings.Replace(sensor.metricName, "temp", "crit_temp", 1) | ||||
| 					sensor.critTemp = x / 1000 | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		m.sensors = append(m.sensors, sensor) | ||||
| 	} | ||||
|  | ||||
| 	// Empty sensors map | ||||
| 	if len(m.sensors) == 0 { | ||||
| 		return fmt.Errorf("No temperature sensors found") | ||||
| 	} | ||||
|  | ||||
| 	// Finished initialization | ||||
| 	m.init = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func get_hwmon_sensors() (map[string]map[string]string, error) { | ||||
| 	var folders []string | ||||
| 	var sensors map[string]map[string]string | ||||
| 	sensors = make(map[string]map[string]string) | ||||
| 	err := filepath.Walk(HWMON_PATH, func(p string, info os.FileInfo, err error) error { | ||||
| 		if info.IsDir() { | ||||
| 			return nil | ||||
| 		} | ||||
| 		folders = append(folders, p) | ||||
| 		return nil | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return sensors, err | ||||
| 	} | ||||
| func (m *TempCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
|  | ||||
| 	for _, f := range folders { | ||||
| 		sensors[f] = make(map[string]string) | ||||
| 		myp := fmt.Sprintf("%s/", f) | ||||
| 		err := filepath.Walk(myp, func(path string, info os.FileInfo, err error) error { | ||||
| 			dir, fname := filepath.Split(path) | ||||
| 			if strings.Contains(fname, "temp") && strings.Contains(fname, "_input") { | ||||
| 				namefile := fmt.Sprintf("%s/%s", dir, strings.Replace(fname, "_input", "_label", -1)) | ||||
| 				name, ierr := ioutil.ReadFile(namefile) | ||||
| 				if ierr == nil { | ||||
| 					sensors[f][strings.Replace(string(name), "\n", "", -1)] = path | ||||
| 				} | ||||
| 			} | ||||
| 			return nil | ||||
| 		}) | ||||
| 	for _, sensor := range m.sensors { | ||||
| 		// Read sensor file | ||||
| 		buffer, err := ioutil.ReadFile(sensor.file) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to read file '%s': %v", sensor.file, err)) | ||||
| 			continue | ||||
| 		} | ||||
| 	} | ||||
| 	return sensors, nil | ||||
| } | ||||
| 		x, err := strconv.ParseInt(strings.TrimSpace(string(buffer)), 10, 64) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError( | ||||
| 				m.name, | ||||
| 				fmt.Sprintf("Read(): Failed to convert temperature '%s' to int64: %v", buffer, err)) | ||||
| 			continue | ||||
| 		} | ||||
| 		x /= 1000 | ||||
| 		y, err := lp.New( | ||||
| 			sensor.metricName, | ||||
| 			sensor.tags, | ||||
| 			m.meta, | ||||
| 			map[string]interface{}{"value": x}, | ||||
| 			time.Now(), | ||||
| 		) | ||||
| 		if err == nil { | ||||
| 			output <- y | ||||
| 		} | ||||
|  | ||||
| func (m *TempCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
|  | ||||
| 	sensors, err := get_hwmon_sensors() | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	for _, files := range sensors { | ||||
| 		for name, file := range files { | ||||
| 			tags := map[string]string{"type": "node"} | ||||
| 			for key, newtags := range m.config.TagOverride { | ||||
| 				if strings.Contains(file, key) { | ||||
| 					tags = newtags | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
| 			buffer, err := ioutil.ReadFile(string(file)) | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			x, err := strconv.ParseInt(strings.Replace(string(buffer), "\n", "", -1), 0, 64) | ||||
| 		// max temperature | ||||
| 		if m.config.ReportMaxTemp && sensor.maxTemp != 0 { | ||||
| 			y, err := lp.New( | ||||
| 				sensor.maxTempName, | ||||
| 				sensor.tags, | ||||
| 				m.meta, | ||||
| 				map[string]interface{}{"value": sensor.maxTemp}, | ||||
| 				time.Now(), | ||||
| 			) | ||||
| 			if err == nil { | ||||
| 				y, err := lp.New(strings.ToLower(name), tags, map[string]interface{}{"value": float64(x) / 1000}, time.Now()) | ||||
| 				if err == nil { | ||||
| 					*out = append(*out, y) | ||||
| 				} | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// critical temperature | ||||
| 		if m.config.ReportCriticalTemp && sensor.critTemp != 0 { | ||||
| 			y, err := lp.New( | ||||
| 				sensor.critTempName, | ||||
| 				sensor.tags, | ||||
| 				m.meta, | ||||
| 				map[string]interface{}{"value": sensor.critTemp}, | ||||
| 				time.Now(), | ||||
| 			) | ||||
| 			if err == nil { | ||||
| 				output <- y | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| func (m *TempCollector) Close() { | ||||
|   | ||||
							
								
								
									
										22
									
								
								collectors/tempMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								collectors/tempMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
|  | ||||
| ## `tempstat` collector | ||||
|  | ||||
| ```json | ||||
|   "tempstat": { | ||||
|     "tag_override" : { | ||||
|         "<device like hwmon1>" : { | ||||
|             "type" : "socket", | ||||
|             "type-id" : "0" | ||||
|         } | ||||
|     }, | ||||
|     "exclude_metrics": [ | ||||
|       "metric1", | ||||
|       "metric2" | ||||
|     ] | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `tempstat` collector reads the data from `/sys/class/hwmon/<device>/tempX_{input,label}` | ||||
|  | ||||
| Metrics: | ||||
| * `temp_*`: The metric name is taken from the `label` files. | ||||
| @@ -9,7 +9,7 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const MAX_NUM_PROCS = 10 | ||||
| @@ -20,15 +20,16 @@ type TopProcsCollectorConfig struct { | ||||
| } | ||||
|  | ||||
| type TopProcsCollector struct { | ||||
| 	MetricCollector | ||||
| 	metricCollector | ||||
| 	tags   map[string]string | ||||
| 	config TopProcsCollectorConfig | ||||
| } | ||||
|  | ||||
| func (m *TopProcsCollector) Init(config []byte) error { | ||||
| func (m *TopProcsCollector) Init(config json.RawMessage) error { | ||||
| 	var err error | ||||
| 	m.name = "TopProcsCollector" | ||||
| 	m.tags = map[string]string{"type": "node"} | ||||
| 	m.meta = map[string]string{"source": m.name, "group": "TopProcs"} | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &m.config) | ||||
| 		if err != nil { | ||||
| @@ -51,7 +52,7 @@ func (m *TopProcsCollector) Init(config []byte) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *TopProcsCollector) Read(interval time.Duration, out *[]lp.MutableMetric) { | ||||
| func (m *TopProcsCollector) Read(interval time.Duration, output chan lp.CCMetric) { | ||||
| 	if !m.init { | ||||
| 		return | ||||
| 	} | ||||
| @@ -66,9 +67,9 @@ func (m *TopProcsCollector) Read(interval time.Duration, out *[]lp.MutableMetric | ||||
| 	lines := strings.Split(string(stdout), "\n") | ||||
| 	for i := 1; i < m.config.Num_procs+1; i++ { | ||||
| 		name := fmt.Sprintf("topproc%d", i) | ||||
| 		y, err := lp.New(name, m.tags, map[string]interface{}{"value": string(lines[i])}, time.Now()) | ||||
| 		y, err := lp.New(name, m.tags, m.meta, map[string]interface{}{"value": string(lines[i])}, time.Now()) | ||||
| 		if err == nil { | ||||
| 			*out = append(*out, y) | ||||
| 			output <- y | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										15
									
								
								collectors/topprocsMetric.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								collectors/topprocsMetric.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
|  | ||||
| ## `topprocs` collector | ||||
|  | ||||
| ```json | ||||
|   "topprocs": { | ||||
|     "num_procs": 5 | ||||
|   } | ||||
| ``` | ||||
|  | ||||
| The `topprocs` collector reads the TopX processes (sorted by CPU utilization, `ps -Ao comm --sort=-pcpu`).  | ||||
|  | ||||
| In contrast to most other collectors, the metric value is a `string`. | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										40
									
								
								config.json
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								config.json
									
									
									
									
									
								
							| @@ -1,36 +1,8 @@ | ||||
| { | ||||
|   "sink": { | ||||
|     "user": "testuser", | ||||
|     "password": "testpass", | ||||
|     "host": "127.0.0.1", | ||||
|     "port": "9090", | ||||
|     "database": "testdb", | ||||
|     "organization": "testorg", | ||||
|     "type": "stdout" | ||||
|   }, | ||||
|   "interval": 3, | ||||
|   "duration": 1, | ||||
|   "collectors": [ | ||||
|     "tempstat" | ||||
|   ], | ||||
|   "default_tags": { | ||||
|     "cluster": "testcluster" | ||||
|   }, | ||||
|   "receiver": { | ||||
|     "type": "none" | ||||
|   }, | ||||
|   "collect_config": { | ||||
|     "tempstat": { | ||||
|       "tag_override": { | ||||
|         "hwmon0" : { | ||||
|             "type" : "socket", | ||||
|             "type-id" : "0" | ||||
|         }, | ||||
|         "hwmon1" : { | ||||
|             "type" : "socket", | ||||
|             "type-id" : "1" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   "sinks": "sinks.json", | ||||
|   "collectors" : "collectors.json", | ||||
|   "receivers" : "receivers.json", | ||||
|   "router" : "router.json", | ||||
|   "interval": 10, | ||||
|   "duration": 1 | ||||
| } | ||||
|   | ||||
							
								
								
									
										19
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								go.mod
									
									
									
									
									
								
							| @@ -3,10 +3,17 @@ module github.com/ClusterCockpit/cc-metric-collector | ||||
| go 1.16 | ||||
|  | ||||
| require ( | ||||
| 	github.com/NVIDIA/go-nvml v0.11.1-0 // indirect | ||||
| 	github.com/influxdata/influxdb-client-go/v2 v2.2.2 | ||||
| 	github.com/influxdata/line-protocol v0.0.0-20210311194329-9aa0e372d097 | ||||
| 	github.com/nats-io/nats.go v1.10.0 | ||||
| 	github.com/nats-io/nkeys v0.1.4 // indirect | ||||
| 	github.com/prometheus/client_golang v1.10.0 // indirect | ||||
| 	github.com/NVIDIA/go-nvml v0.11.1-0 | ||||
| 	github.com/influxdata/influxdb-client-go/v2 v2.7.0 | ||||
| 	github.com/influxdata/line-protocol v0.0.0-20210922203350-b1ad95c89adf | ||||
| 	github.com/nats-io/nats.go v1.13.1-0.20211122170419-d7c1d78a50fc | ||||
| 	golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 | ||||
| 	gopkg.in/Knetic/govaluate.v2 v2.3.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/PaesslerAG/gval v1.1.2 | ||||
| 	github.com/golang/protobuf v1.5.2 // indirect | ||||
| 	github.com/nats-io/nats-server/v2 v2.7.0 // indirect | ||||
| 	google.golang.org/protobuf v1.27.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										462
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										462
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,447 +1,145 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible/go.mod h1:r7JcOSlj0wfOMncg0iLm8Leh48TZaKVeNIfJntJ2wa0= | ||||
| github.com/NVIDIA/go-nvml v0.11.1-0 h1:XHSz3zZKC4NCP2ja1rI7++DXFhA+uDhdYa3MykCTGHY= | ||||
| github.com/NVIDIA/go-nvml v0.11.1-0/go.mod h1:hy7HYeQy335x6nEss0Ne3PYqleRa6Ct+VKD9RQ4nyFs= | ||||
| github.com/Shopify/sarama v1.19.0/go.mod h1:FVkBWblsNy7DGZRfXLU0O9RCGt5g3g3yEuWXgklEdEo= | ||||
| github.com/Shopify/toxiproxy v2.1.4+incompatible/go.mod h1:OXgGpZ6Cli1/URJOF1DMxUHB2q5Ap20/P/eIdh4G0pI= | ||||
| github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= | ||||
| github.com/afex/hystrix-go v0.0.0-20180502004556-fa1af6a1f4f5/go.mod h1:SkGFH1ia65gfNATL8TAiHDNxPzPdmEL5uirI2Uyuz6c= | ||||
| github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= | ||||
| github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= | ||||
| github.com/apache/thrift v0.13.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= | ||||
| github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= | ||||
| github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= | ||||
| github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| github.com/aryann/difflib v0.0.0-20170710044230-e206f873d14a/go.mod h1:DAHtR1m6lCRdSC2Tm3DSWRPvIPr6xNKyeHdqDQSQT+A= | ||||
| github.com/aws/aws-lambda-go v1.13.3/go.mod h1:4UKl9IzQMoD+QF79YdCuzCwp8VbmG4VAQwij/eHl5CU= | ||||
| github.com/aws/aws-sdk-go v1.27.0/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= | ||||
| github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= | ||||
| github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= | ||||
| github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= | ||||
| github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= | ||||
| github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= | ||||
| github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= | ||||
| github.com/casbin/casbin/v2 v2.1.2/go.mod h1:YcPU1XXisHhLzuxH9coDNf2FbKpjGlbCg3n9yuLkIJQ= | ||||
| github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/clbanning/x2j v0.0.0-20191024224557-825249438eec/go.mod h1:jMjuTZXRI4dUb/I5gc9Hdhagfvm9+RyrPryS/auMzxE= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/cockroachdb/datadriven v0.0.0-20190809214429-80d97fb3cbaa/go.mod h1:zn76sxSg3SzpJ0PPJaLDCu+Bu0Lg3sKTORVIj19EIF8= | ||||
| github.com/codahale/hdrhistogram v0.0.0-20161010025455-3a0bb77429bd/go.mod h1:sE/e/2PUdi/liOCUjSTXgM1o87ZssimdTWN964YiIeI= | ||||
| github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | ||||
| github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= | ||||
| github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= | ||||
| github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= | ||||
| github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= | ||||
| github.com/PaesslerAG/gval v1.1.2 h1:EROKxV4/fAKWb0Qoj7NOxmHZA7gcpjOV9XgiRZMRCUU= | ||||
| github.com/PaesslerAG/gval v1.1.2/go.mod h1:Fa8gfkCmUsELXgayr8sfL/sw+VzCVoa03dcOcR/if2w= | ||||
| github.com/PaesslerAG/jsonpath v0.1.0/go.mod h1:4BzmtoM/PI8fPO4aQGIusjGxGir2BzcV0grWtFzq1Y8= | ||||
| github.com/cyberdelia/templates v0.0.0-20141128023046-ca7fffd4298c/go.mod h1:GyV+0YP4qX0UQ7r2MoYZ+AvYDp12OF5yg4q8rGnyNh4= | ||||
| 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/deepmap/oapi-codegen v1.3.13 h1:9HKGCsdJqE4dnrQ8VerFS0/1ZOJPmAhN+g8xgp8y3K4= | ||||
| github.com/deepmap/oapi-codegen v1.3.13/go.mod h1:WAmG5dWY8/PYHt4vKxlt90NsbHMAOCiteYKZMiIRfOo= | ||||
| github.com/deepmap/oapi-codegen v1.8.2 h1:SegyeYGcdi0jLLrpbCMoJxnUUn8GBXHsvr4rbzjuhfU= | ||||
| github.com/deepmap/oapi-codegen v1.8.2/go.mod h1:YLgSKSDv/bZQB7N4ws6luhozi3cEdRktEqrX88CvjIw= | ||||
| github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= | ||||
| github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= | ||||
| github.com/eapache/go-resiliency v1.1.0/go.mod h1:kFI+JgMyC7bLPUVY133qvEBtVayf5mFgVsvEsIPBvNs= | ||||
| github.com/eapache/go-xerial-snappy v0.0.0-20180814174437-776d5712da21/go.mod h1:+020luEh2TKB4/GOp8oxxtq0Daoen/Cii55CzbTV6DU= | ||||
| github.com/eapache/queue v1.1.0/go.mod h1:6eCeP0CKFpHLu8blIFXhExK/dRa7WDZfr6jVFPTqq+I= | ||||
| github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaBNrHW5M= | ||||
| github.com/envoyproxy/go-control-plane v0.6.9/go.mod h1:SBwIajubJHhxtWwsL9s8ss4safvEdbitLhGGK48rN6g= | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= | ||||
| github.com/franela/goblin v0.0.0-20200105215937-c9ffbefa60db/go.mod h1:7dvUGVsVBjqR7JHJk0brhHOZYGmfBYOrK0ZhYMEtBr4= | ||||
| github.com/franela/goreq v0.0.0-20171204163338-bcd34c9993f8/go.mod h1:ZhphrRTfi2rbfLwlschooIH4+wKKDR4Pdxhh+TRoA20= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/getkin/kin-openapi v0.13.0/go.mod h1:WGRs2ZMM1Q8LR1QBEwUxC6RJEfaBcD0s+pcEVXFuAjw= | ||||
| github.com/getkin/kin-openapi v0.61.0/go.mod h1:7Yn5whZr5kJi6t+kShccXS8ae1APpYTW6yheSwk8Yi4= | ||||
| github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/go-chi/chi v4.0.2+incompatible/go.mod h1:eB3wogJHnLi3x/kFX2A+IbTBlXxmMeXJVKy9tTv1XzQ= | ||||
| github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/kit v0.10.0/go.mod h1:xUsJbQ/Fp4kEt7AFgCuvyX4a71u8h9jB8tj/ORgOZ7o= | ||||
| github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= | ||||
| github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | ||||
| github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/gogo/googleapis v1.1.0/go.mod h1:gf4bu3Q80BeJ6H1S1vYPm8/ELATdvryBaNFGgqEef3s= | ||||
| github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/gogo/protobuf v1.2.0/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| 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.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/go-chi/chi/v5 v5.0.0/go.mod h1:BBug9lr0cqtdAhsu6R4AAdvufI0/XBzAQSsUqJpoZOs= | ||||
| github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= | ||||
| github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3 h1:JjCZWpVbqXDqFVmTfYWEVTMIYrL/NPdPSCHPJ0T/raM= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= | ||||
| github.com/golangci/lint-1 v0.0.0-20181222135242-d2cdd8c08219/go.mod h1:/X8TswGSh1pIozq4ZwCfxS0WA5JGXguxk94ar/4c87Y= | ||||
| github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= | ||||
| github.com/google/uuid v1.0.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= | ||||
| github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51q0aT7Yg= | ||||
| github.com/gorilla/mux v1.6.2/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= | ||||
| github.com/gorilla/mux v1.7.3/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= | ||||
| github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.0.1-0.20190118093823-f849b5445de4/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= | ||||
| github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= | ||||
| github.com/hashicorp/consul/api v1.3.0/go.mod h1:MmDNSzIMUjNpY/mQ398R4bk2FnqQLoPndWW5VkKPlCE= | ||||
| github.com/hashicorp/consul/sdk v0.3.0/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= | ||||
| github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||
| github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= | ||||
| github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= | ||||
| github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= | ||||
| github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= | ||||
| github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= | ||||
| github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= | ||||
| github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-version v1.2.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= | ||||
| github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90= | ||||
| github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= | ||||
| github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ= | ||||
| github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= | ||||
| github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/hudl/fargo v1.3.0/go.mod h1:y3CKSmjA+wD2gak7sUSXTAoopbhU08POFhmITJgmKTg= | ||||
| github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= | ||||
| github.com/influxdata/influxdb-client-go v1.4.0 h1:+KavOkwhLClHFfYcJMHHnTL5CZQhXJzOm5IKHI9BqJk= | ||||
| github.com/influxdata/influxdb-client-go/v2 v2.2.2 h1:O0CGIuIwQafvAxttAJ/VqMKfbWWn2Mt8rbOmaM2Zj4w= | ||||
| github.com/influxdata/influxdb-client-go/v2 v2.2.2/go.mod h1:fa/d1lAdUHxuc1jedx30ZfNG573oQTQmUni3N6pcW+0= | ||||
| github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= | ||||
| github.com/influxdata/influxdb-client-go/v2 v2.7.0 h1:QgP5mlBE9sGnzplpnf96pr+p7uqlIlL4W2GAP3n+XZg= | ||||
| github.com/influxdata/influxdb-client-go/v2 v2.7.0/go.mod h1:Y/0W1+TZir7ypoQZYd2IrnVOKB3Tq6oegAQeSVN/+EU= | ||||
| github.com/influxdata/line-protocol v0.0.0-20200327222509-2487e7298839/go.mod h1:xaLFMmpvUxqXtVkUJfg9QmT88cDaCJ3ZKgdZ78oO8Qo= | ||||
| github.com/influxdata/line-protocol v0.0.0-20210311194329-9aa0e372d097 h1:vilfsDSy7TDxedi9gyBkMvAirat/oRcL0lFdJBf6tdM= | ||||
| github.com/influxdata/line-protocol v0.0.0-20210311194329-9aa0e372d097/go.mod h1:xaLFMmpvUxqXtVkUJfg9QmT88cDaCJ3ZKgdZ78oO8Qo= | ||||
| github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= | ||||
| github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= | ||||
| github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= | ||||
| github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||
| github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | ||||
| github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= | ||||
| github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= | ||||
| github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= | ||||
| github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= | ||||
| github.com/influxdata/line-protocol v0.0.0-20210922203350-b1ad95c89adf h1:7JTmneyiNEwVBOHSjoMxiWAqB992atOeepeFYegn5RU= | ||||
| github.com/influxdata/line-protocol v0.0.0-20210922203350-b1ad95c89adf/go.mod h1:xaLFMmpvUxqXtVkUJfg9QmT88cDaCJ3ZKgdZ78oO8Qo= | ||||
| github.com/klauspost/compress v1.13.4 h1:0zhec2I8zGnjWcKyLl6i3gPqKANCCn5e9xmviEEeX6s= | ||||
| github.com/klauspost/compress v1.13.4/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= | ||||
| github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/labstack/echo/v4 v4.1.11/go.mod h1:i541M3Fj6f76NZtHSj7TXnyM8n2gaodfvfxNnFqi74g= | ||||
| github.com/labstack/echo/v4 v4.2.1/go.mod h1:AA49e0DZ8kk5jTOOCKNuPR6oTnBS0dYiM4FW1e6jwpg= | ||||
| github.com/labstack/gommon v0.3.0/go.mod h1:MULnywXg0yavhxWKc+lOruYdAhDwPK9wf0OL7NoOu+k= | ||||
| github.com/lightstep/lightstep-tracer-common/golang/gogo v0.0.0-20190605223551-bc2310a04743/go.mod h1:qklhhLq1aX+mtWk9cPHPzaBjWImj5ULL6C7HFJtXQMM= | ||||
| github.com/lightstep/lightstep-tracer-go v0.18.1/go.mod h1:jlF1pusYV4pidLvZ+XD0UBX0ZE6WURAspgAczcDHrL4= | ||||
| github.com/lyft/protoc-gen-validate v0.0.13/go.mod h1:XbGvPuh87YZc5TdIa2/I4pLk0QoUACkjt2znoq26NVQ= | ||||
| github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= | ||||
| github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= | ||||
| github.com/matryer/moq v0.0.0-20190312154309-6cfb0558e1bd/go.mod h1:9ELz6aaclSIGnZBoaSLZ3NAl1VTufbOrXBPvtcy6WiQ= | ||||
| github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= | ||||
| github.com/mattn/go-colorable v0.1.2/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= | ||||
| github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= | ||||
| github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||
| github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||
| github.com/mattn/go-colorable v0.1.7/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= | ||||
| github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= | ||||
| github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= | ||||
| github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ= | ||||
| github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= | ||||
| github.com/mattn/go-runewidth v0.0.2/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | ||||
| github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= | ||||
| github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= | ||||
| github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= | ||||
| github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= | ||||
| github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= | ||||
| github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/nats-io/jwt v0.3.0/go.mod h1:fRYCDE99xlTsqUzISS1Bi75UBJ6ljOJQOAAu5VglpSg= | ||||
| github.com/nats-io/jwt v0.3.2 h1:+RB5hMpXUUA2dfxuhBTEkMOrYmM+gKIZYS1KjSostMI= | ||||
| github.com/nats-io/jwt v0.3.2/go.mod h1:/euKqTS1ZD+zzjYrY7pseZrTtWQSjujC7xjPc8wL6eU= | ||||
| github.com/nats-io/nats-server/v2 v2.1.2/go.mod h1:Afk+wRZqkMQs/p45uXdrVLuab3gwv3Z8C4HTBu8GD/k= | ||||
| github.com/nats-io/nats.go v1.9.1/go.mod h1:ZjDU1L/7fJ09jvUSRVBR2e7+RnLiiIQyqyzEE/Zbp4w= | ||||
| github.com/nats-io/nats.go v1.10.0 h1:L8qnKaofSfNFbXg0C5F71LdjPRnmQwSsA4ukmkt1TvY= | ||||
| github.com/nats-io/nats.go v1.10.0/go.mod h1:AjGArbfyR50+afOUotNX2Xs5SYHf+CoOa5HH1eEl2HE= | ||||
| github.com/nats-io/nkeys v0.1.0/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= | ||||
| github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= | ||||
| github.com/nats-io/nkeys v0.1.4 h1:aEsHIssIk6ETN5m2/MD8Y4B2X7FfXrBAUdkyRvbVYzA= | ||||
| github.com/nats-io/nkeys v0.1.4/go.mod h1:XdZpAbhgyyODYqjTawOnIOI7VlbKSarI9Gfy1tqEu/s= | ||||
| github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||
| github.com/minio/highwayhash v1.0.1 h1:dZ6IIu8Z14VlC0VpfKofAhCy74wu/Qb5gcn52yWoz/0= | ||||
| github.com/minio/highwayhash v1.0.1/go.mod h1:BQskDq+xkJ12lmlUUi7U0M5Swg3EWR+dLTk+kldvVxY= | ||||
| github.com/nats-io/jwt/v2 v2.2.1-0.20220113022732-58e87895b296 h1:vU9tpM3apjYlLLeY23zRWJ9Zktr5jp+mloR942LEOpY= | ||||
| github.com/nats-io/jwt/v2 v2.2.1-0.20220113022732-58e87895b296/go.mod h1:0tqz9Hlu6bCBFLWAASKhE5vUA4c24L9KPUUgvwumE/k= | ||||
| github.com/nats-io/nats-server/v2 v2.7.0 h1:UpqcAM93FI7AHlCyI2FD5QcV3QuHNCauQF2LBVU0238= | ||||
| github.com/nats-io/nats-server/v2 v2.7.0/go.mod h1:cjxtMhZsZovK1XS2iiapCduR8HuqB/YpFamL0qntIcw= | ||||
| github.com/nats-io/nats.go v1.13.1-0.20211122170419-d7c1d78a50fc h1:SHr4MUUZJ/fAC0uSm2OzWOJYsHpapmR86mpw7q1qPXU= | ||||
| github.com/nats-io/nats.go v1.13.1-0.20211122170419-d7c1d78a50fc/go.mod h1:BPko4oXsySz4aSWeFgOHLZs3G4Jq4ZAyE6/zMCxRT6w= | ||||
| github.com/nats-io/nkeys v0.3.0 h1:cgM5tL53EvYRU+2YLXIK0G2mJtK12Ft9oeooSZMA2G8= | ||||
| github.com/nats-io/nkeys v0.3.0/go.mod h1:gvUNGjVcM2IPr5rCsRsC6Wb3Hr2CQAm08dsxtV6A5y4= | ||||
| github.com/nats-io/nuid v1.0.1 h1:5iA8DT8V7q8WK2EScv2padNa/rTESc1KdnPw4TC2paw= | ||||
| github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= | ||||
| github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs= | ||||
| github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA= | ||||
| github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | ||||
| github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= | ||||
| github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis= | ||||
| github.com/opentracing/basictracer-go v1.0.0/go.mod h1:QfBfYuafItcjQuMwinw9GhYKwFXS9KnPs5lxoYwgW74= | ||||
| github.com/opentracing/opentracing-go v1.0.2/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= | ||||
| github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= | ||||
| github.com/openzipkin-contrib/zipkin-go-opentracing v0.4.5/go.mod h1:/wsWhb9smxSfWAKL3wpBW7V8scJMt8N8gnaMCS9E/cA= | ||||
| github.com/openzipkin/zipkin-go v0.1.6/go.mod h1:QgAqvLzwWbR/WpD4A3cGpPtJrZXNIiJc5AZX7/PBEpw= | ||||
| github.com/openzipkin/zipkin-go v0.2.1/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= | ||||
| github.com/openzipkin/zipkin-go v0.2.2/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= | ||||
| github.com/pact-foundation/pact-go v1.0.4/go.mod h1:uExwJY4kCzNPcHRj+hCR/HBbOOIwwtUjcrb0b5/5kLM= | ||||
| github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= | ||||
| github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k= | ||||
| github.com/performancecopilot/speed v3.0.0+incompatible/go.mod h1:/CLtqpZ5gBg1M9iaPbIdPPGyKcA8hKdoy6hAWba7Yac= | ||||
| github.com/pierrec/lz4 v1.0.2-0.20190131084431-473cd7ce01a1/go.mod h1:3/3N9NVKO0jef7pBehbT1qWhCMrIgbYNnFAZCqQ5LRc= | ||||
| github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY= | ||||
| github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/profile v1.2.1/go.mod h1:hJw3o1OdXxsrSjjVksARp5W95eeEaEfptyVZyv6JUPA= | ||||
| 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/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= | ||||
| github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= | ||||
| github.com/prometheus/client_golang v0.9.3-0.20190127221311-3c4408c8b829/go.mod h1:p2iRAGwDERtqlqzRXnrOVns+ignqQo//hLXqYxZYVNs= | ||||
| github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.3.0/go.mod h1:hJaj2vgQTGQmVCsAACORcieXFeDPbaTKGT+JTgUa3og= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.10.0 h1:/o0BDeWzLWXNZ+4q5gXltUvaMpJqckTa+jTNoB+z4cg= | ||||
| github.com/prometheus/client_golang v1.10.0/go.mod h1:WJM3cc3yu7XKBKa/I8WeZm+V3eltZnBwfENSU7mdogU= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.1.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= | ||||
| github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/common v0.2.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| github.com/prometheus/common v0.7.0/go.mod h1:DjGbpBbp5NYNiECxcL/VnbXCCaQpKd3tt26CguLLsqA= | ||||
| github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= | ||||
| github.com/prometheus/common v0.18.0 h1:WCVKW7aL6LEe1uryfI9dnEc2ZqNB1Fn0ok930v0iL1Y= | ||||
| github.com/prometheus/common v0.18.0/go.mod h1:U+gB1OBLb1lF3O42bTCL+FK18tX9Oar16Clt/msog/s= | ||||
| github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= | ||||
| github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= | ||||
| github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= | ||||
| github.com/prometheus/procfs v0.6.0 h1:mxy4L2jP6qMonqmq+aTtOx1ifVWUgG/TAmntgbh3xv4= | ||||
| github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||
| github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= | ||||
| github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= | ||||
| github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | ||||
| github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | ||||
| github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= | ||||
| github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= | ||||
| github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= | ||||
| github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= | ||||
| github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||
| github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= | ||||
| github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= | ||||
| github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= | ||||
| github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= | ||||
| github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= | ||||
| github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY= | ||||
| github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= | ||||
| github.com/spf13/pflag v1.0.1/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | ||||
| github.com/streadway/amqp v0.0.0-20190404075320-75d898a42a94/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= | ||||
| github.com/streadway/amqp v0.0.0-20190827072141-edfb9018d271/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= | ||||
| github.com/streadway/handy v0.0.0-20190108123426-d5acb3125c2a/go.mod h1:qNTQ5P5JnDBl6z3cMAg/SywNDC5ABu5ApDIw6lUbRmI= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.5.1 h1:nOGnQDM7FYENwehXlg/kFVnos3rEvtKTjRvOWSzb6H4= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= | ||||
| github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | ||||
| github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= | ||||
| github.com/valyala/fasttemplate v1.0.1/go.mod h1:UQGH1tvbgY+Nz5t2n7tXsz52dQxojPUpymEIMZ47gx8= | ||||
| github.com/valyala/fasttemplate v1.1.0/go.mod h1:UQGH1tvbgY+Nz5t2n7tXsz52dQxojPUpymEIMZ47gx8= | ||||
| github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= | ||||
| go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= | ||||
| go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg= | ||||
| go.opencensus.io v0.20.1/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= | ||||
| go.opencensus.io v0.20.2/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= | ||||
| go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= | ||||
| go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= | ||||
| go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= | ||||
| go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= | ||||
| go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= | ||||
| go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= | ||||
| go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= | ||||
| go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20191112222119-e1110fd1c708/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59 h1:3zb4D3T4G8jdExgVU/95+vQXfpEPiMdCaZgmGVxjNHM= | ||||
| golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | ||||
| golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= | ||||
| golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= | ||||
| golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190125091013-d26f9f9a57f3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= | ||||
| golang.org/x/crypto v0.0.0-20210314154223-e6e6c4f2bb5b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/crypto v0.0.0-20220112180741-5e0467b6c7ce h1:Roh6XWxHFKrPgC/EQhVubSAGQ6Ozk6IdxHSzt1mR0EI= | ||||
| golang.org/x/crypto v0.0.0-20220112180741-5e0467b6c7ce/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= | ||||
| golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190813141303-74dc4d7220e7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20191112182307-2180aed22343 h1:00ohfJ4K98s3m6BGUoBd8nyfp4Yl0GoIKvw5abItTjI= | ||||
| golang.org/x/net v0.0.0-20191112182307-2180aed22343/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344 h1:vGXIOMxbNfDTk/aXCmfdLgkrSV+Z2tcbze+pEc3v5W4= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2 h1:CIJ76btIcR3eFI5EgSo6k1qKw9KJexJuRLI9G7Hp5wE= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181122145206-62eef0e2fa9b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190130150945-aca44879d564/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191115151921-52ab43148777/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191220142924-d4481acd189f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200826173525-f9321e4c35a6/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210309074719-68d13333faf2 h1:46ULzRKLh1CwgRq2dC5SlBzEqqNCi8rreOZnNrbqcIY= | ||||
| golang.org/x/sys v0.0.0-20210309074719-68d13333faf2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220111092808-5a964db01320/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 h1:XfKQ4OlFl8okEOr5UvAqFRVj8pY/4yfcXrddB8qAbU0= | ||||
| golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| 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/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20201208040808-7e3f01d25324/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20211116232009-f0f3c7e86c11 h1:GZokNIeuVkl3aZHJchRrr13WCsols02MLUcz1U9is6M= | ||||
| golang.org/x/time v0.0.0-20211116232009-f0f3c7e86c11/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20191029041327-9cc4af7d6b2c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20191029190741-b9c20aec41a5/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20200103221440-774c71fcf114/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.2.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190530194941-fb225487d101/go.mod h1:z3L6/3dTEVtUr6QSP8miRzeRqwQOioJ9I66odjN4I7s= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.20.0/go.mod h1:chYK+tFQF0nDUGJgXMSgLCQk3phJEuONr2DCgLDdAQM= | ||||
| google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= | ||||
| google.golang.org/grpc v1.21.0/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= | ||||
| google.golang.org/grpc v1.22.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.23.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.23.0 h1:4MY060fB1DLGMB/7MBTLnwQUY6+F09GEiz6SsrNqyzM= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| 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.27.1 h1:SnqbnDw1V7RiZcXPx5MEeqPv2s79L9i7BJUlG/+RurQ= | ||||
| google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/Knetic/govaluate.v2 v2.3.0 h1:naJVc9CZlWA8rC8f5mvECJD7jreTrn7FvGXjBthkHJQ= | ||||
| gopkg.in/Knetic/govaluate.v2 v2.3.0/go.mod h1:NW0gr10J8s7aNghEg6uhdxiEaBvc0+8VgJjVViHUKp4= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= | ||||
| gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= | ||||
| gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | ||||
| gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= | ||||
| sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o= | ||||
| sourcegraph.com/sourcegraph/appdash v0.0.0-20190731080439-ebfcffb1b5c0/go.mod h1:hI742Nqp5OhwiqlzhgfbWU4mW4yO10fP+LoT9WOswdU= | ||||
|   | ||||
							
								
								
									
										113
									
								
								internal/ccLogger/cclogger.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								internal/ccLogger/cclogger.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| package cclogger | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	globalDebug             = false | ||||
| 	stdout                  = os.Stdout | ||||
| 	stderr                  = os.Stderr | ||||
| 	debugLog    *log.Logger = nil | ||||
| 	infoLog     *log.Logger = nil | ||||
| 	errorLog    *log.Logger = nil | ||||
| 	warnLog     *log.Logger = nil | ||||
| 	defaultLog  *log.Logger = nil | ||||
| ) | ||||
|  | ||||
| func initLogger() { | ||||
| 	if debugLog == nil { | ||||
| 		debugLog = log.New(stderr, "DEBUG ", log.LstdFlags) | ||||
| 	} | ||||
| 	if infoLog == nil { | ||||
| 		infoLog = log.New(stdout, "INFO ", log.LstdFlags) | ||||
| 	} | ||||
| 	if errorLog == nil { | ||||
| 		errorLog = log.New(stderr, "ERROR ", log.LstdFlags) | ||||
| 	} | ||||
| 	if warnLog == nil { | ||||
| 		warnLog = log.New(stderr, "WARN ", log.LstdFlags) | ||||
| 	} | ||||
| 	if defaultLog == nil { | ||||
| 		defaultLog = log.New(stdout, "", log.LstdFlags) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Print(e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	defaultLog.Print(e...) | ||||
| } | ||||
|  | ||||
| func ComponentPrint(component string, e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	defaultLog.Print(fmt.Sprintf("[%s] ", component), e) | ||||
| } | ||||
|  | ||||
| func Info(e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	infoLog.Print(e...) | ||||
| } | ||||
|  | ||||
| func ComponentInfo(component string, e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	infoLog.Print(fmt.Sprintf("[%s] ", component), e) | ||||
| } | ||||
|  | ||||
| func Debug(e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	if globalDebug { | ||||
| 		debugLog.Print(e...) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ComponentDebug(component string, e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	if globalDebug && debugLog != nil { | ||||
| 		//CCComponentPrint(debugLog, component,  e) | ||||
| 		debugLog.Print(fmt.Sprintf("[%s] ", component), e) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Error(e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	_, fn, line, _ := runtime.Caller(1) | ||||
| 	errorLog.Print(fmt.Sprintf("[%s:%d] ", fn, line), e) | ||||
| } | ||||
|  | ||||
| func ComponentError(component string, e ...interface{}) { | ||||
| 	initLogger() | ||||
| 	_, fn, line, _ := runtime.Caller(1) | ||||
| 	errorLog.Print(fmt.Sprintf("[%s|%s:%d] ", component, fn, line), e) | ||||
| } | ||||
|  | ||||
| func SetDebug() { | ||||
| 	globalDebug = true | ||||
| 	initLogger() | ||||
| } | ||||
|  | ||||
| func SetOutput(filename string) { | ||||
| 	if filename == "stderr" { | ||||
| 		if stderr != os.Stderr && stderr != os.Stdout { | ||||
| 			stderr.Close() | ||||
| 		} | ||||
| 		stderr = os.Stderr | ||||
| 	} else if filename == "stdout" { | ||||
| 		if stderr != os.Stderr && stderr != os.Stdout { | ||||
| 			stderr.Close() | ||||
| 		} | ||||
| 		stderr = os.Stdout | ||||
| 	} else { | ||||
| 		file, err := os.OpenFile(filename, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0600) | ||||
| 		if err == nil { | ||||
| 			defer file.Close() | ||||
| 			stderr = file | ||||
| 		} | ||||
| 	} | ||||
| 	debugLog = nil | ||||
| 	errorLog = nil | ||||
| 	warnLog = nil | ||||
| 	initLogger() | ||||
| } | ||||
							
								
								
									
										32
									
								
								internal/ccMetric/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								internal/ccMetric/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| # ClusterCockpit metrics | ||||
|  | ||||
| As described in the [ClusterCockpit specifications](https://github.com/ClusterCockpit/cc-specifications), the whole ClusterCockpit stack uses metrics in the InfluxDB line protocol format. This is also the input and output format for the ClusterCockpit Metric Collector but internally it uses an extended format while processing, named CCMetric. | ||||
|  | ||||
| It is basically a copy of the [InfluxDB line protocol](https://github.com/influxdata/line-protocol) `MutableMetric` interface with one extension. Besides the tags and fields, it contains a list of meta information (re-using the `Tag` structure of the original protocol): | ||||
|  | ||||
| ```golang | ||||
| type ccMetric struct { | ||||
|     name   string            // same as | ||||
|     tags   []*influx.Tag     // original | ||||
|     fields []*influx.Field   // Influx | ||||
|     tm     time.Time         // line-protocol | ||||
|     meta   []*influx.Tag | ||||
| } | ||||
|  | ||||
| type CCMetric interface { | ||||
|     influx.MutableMetric        // the same functions as defined by influx.MutableMetric | ||||
|     RemoveTag(key string)       // this is not published by the original influx.MutableMetric | ||||
|     Meta() map[string]string | ||||
|     MetaList() []*inlux.Tag | ||||
|     AddMeta(key, value string) | ||||
|     HasMeta(key string) bool | ||||
|     GetMeta(key string) (string, bool) | ||||
|     RemoveMeta(key string) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The `CCMetric` interface provides the same functions as the `MutableMetric` like `{Add, Remove, Has}{Tag, Field}` and additionally provides `{Add, Remove, Has}Meta`. | ||||
|  | ||||
| The InfluxDB protocol creates a new metric with `influx.New(name, tags, fields, time)` while CCMetric uses `ccMetric.New(name, tags, meta, fields, time)` where `tags` and `meta` are both of type `map[string]string`. | ||||
|  | ||||
| You can copy a CCMetric with `FromMetric(other CCMetric) CCMetric`. If you get an `influx.Metric` from a function, like the line protocol parser, you can use `FromInfluxMetric(other influx.Metric) CCMetric` to get a CCMetric out of it (see `NatsReceiver` for an example). | ||||
							
								
								
									
										371
									
								
								internal/ccMetric/ccMetric.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										371
									
								
								internal/ccMetric/ccMetric.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,371 @@ | ||||
| package ccmetric | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	influxdb2 "github.com/influxdata/influxdb-client-go/v2" | ||||
| 	write "github.com/influxdata/influxdb-client-go/v2/api/write" | ||||
| 	lp "github.com/influxdata/line-protocol" // MIT license | ||||
| ) | ||||
|  | ||||
| // Most functions are derived from github.com/influxdata/line-protocol/metric.go | ||||
| // The metric type is extended with an extra meta information list re-using the Tag | ||||
| // type. | ||||
| // | ||||
| // See: https://docs.influxdata.com/influxdb/latest/reference/syntax/line-protocol/ | ||||
| type ccMetric struct { | ||||
| 	name   string                 // Measurement name | ||||
| 	meta   map[string]string      // map of meta data tags | ||||
| 	tags   map[string]string      // map of of tags | ||||
| 	fields map[string]interface{} // map of of fields | ||||
| 	tm     time.Time              // timestamp | ||||
| } | ||||
|  | ||||
| // ccMetric access functions | ||||
| type CCMetric interface { | ||||
| 	ToPoint(metaAsTags bool) *write.Point  // Generate influxDB point for data type ccMetric | ||||
| 	ToLineProtocol(metaAsTags bool) string // Generate influxDB line protocol for data type ccMetric | ||||
|  | ||||
| 	Name() string        // Get metric name | ||||
| 	SetName(name string) // Set metric name | ||||
|  | ||||
| 	Time() time.Time     // Get timestamp | ||||
| 	SetTime(t time.Time) // Set timestamp | ||||
|  | ||||
| 	Tags() map[string]string                   // Map of tags | ||||
| 	AddTag(key, value string)                  // Add a tag | ||||
| 	GetTag(key string) (value string, ok bool) // Get a tag by its key | ||||
| 	HasTag(key string) (ok bool)               // Check if a tag key is present | ||||
| 	RemoveTag(key string)                      // Remove a tag by its key | ||||
|  | ||||
| 	Meta() map[string]string                    // Map of meta data tags | ||||
| 	AddMeta(key, value string)                  // Add a meta data tag | ||||
| 	GetMeta(key string) (value string, ok bool) // Get a meta data tab addressed by its key | ||||
| 	HasMeta(key string) (ok bool)               // Check if a meta data key is present | ||||
| 	RemoveMeta(key string)                      // Remove a meta data tag by its key | ||||
|  | ||||
| 	Fields() map[string]interface{}                   // Map of fields | ||||
| 	AddField(key string, value interface{})           // Add a field | ||||
| 	GetField(key string) (value interface{}, ok bool) // Get a field addressed by its key | ||||
| 	HasField(key string) (ok bool)                    // Check if a field key is present | ||||
| 	RemoveField(key string)                           // Remove a field addressed by its key | ||||
| } | ||||
|  | ||||
| // String implements the stringer interface for data type ccMetric | ||||
| func (m *ccMetric) String() string { | ||||
| 	return fmt.Sprintf( | ||||
| 		"Name: %s, Tags: %+v, Meta: %+v, fields: %+v, Timestamp: %d", | ||||
| 		m.name, m.tags, m.meta, m.fields, m.tm.UnixNano(), | ||||
| 	) | ||||
| } | ||||
|  | ||||
| // ToLineProtocol generates influxDB line protocol for data type ccMetric | ||||
| func (m *ccMetric) ToPoint(metaAsTags bool) (p *write.Point) { | ||||
|  | ||||
| 	if !metaAsTags { | ||||
| 		p = influxdb2.NewPoint(m.name, m.tags, m.fields, m.tm) | ||||
| 	} else { | ||||
| 		tags := make(map[string]string, len(m.tags)+len(m.meta)) | ||||
| 		for key, value := range m.tags { | ||||
| 			tags[key] = value | ||||
| 		} | ||||
| 		for key, value := range m.meta { | ||||
| 			tags[key] = value | ||||
| 		} | ||||
| 		p = influxdb2.NewPoint(m.name, tags, m.fields, m.tm) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
| // ToLineProtocol generates influxDB line protocol for data type ccMetric | ||||
| func (m *ccMetric) ToLineProtocol(metaAsTags bool) string { | ||||
|  | ||||
| 	return write.PointToLineProtocol( | ||||
| 		m.ToPoint(metaAsTags), | ||||
| 		time.Nanosecond, | ||||
| 	) | ||||
| } | ||||
|  | ||||
| // Name returns the measurement name | ||||
| func (m *ccMetric) Name() string { | ||||
| 	return m.name | ||||
| } | ||||
|  | ||||
| // SetName sets the measurement name | ||||
| func (m *ccMetric) SetName(name string) { | ||||
| 	m.name = name | ||||
| } | ||||
|  | ||||
| // Time returns timestamp | ||||
| func (m *ccMetric) Time() time.Time { | ||||
| 	return m.tm | ||||
| } | ||||
|  | ||||
| // SetTime sets the timestamp | ||||
| func (m *ccMetric) SetTime(t time.Time) { | ||||
| 	m.tm = t | ||||
| } | ||||
|  | ||||
| // Tags returns the the list of tags as key-value-mapping | ||||
| func (m *ccMetric) Tags() map[string]string { | ||||
| 	return m.tags | ||||
| } | ||||
|  | ||||
| // AddTag adds a tag (consisting of key and value) to the map of tags | ||||
| func (m *ccMetric) AddTag(key, value string) { | ||||
| 	m.tags[key] = value | ||||
| } | ||||
|  | ||||
| // GetTag returns the tag with tag's key equal to <key> | ||||
| func (m *ccMetric) GetTag(key string) (string, bool) { | ||||
| 	value, ok := m.tags[key] | ||||
| 	return value, ok | ||||
| } | ||||
|  | ||||
| // HasTag checks if a tag with key equal to <key> is present in the list of tags | ||||
| func (m *ccMetric) HasTag(key string) bool { | ||||
| 	_, ok := m.tags[key] | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| // RemoveTag removes the tag with tag's key equal to <key> | ||||
| func (m *ccMetric) RemoveTag(key string) { | ||||
| 	delete(m.tags, key) | ||||
| } | ||||
|  | ||||
| // Meta returns the meta data tags as key-value mapping | ||||
| func (m *ccMetric) Meta() map[string]string { | ||||
| 	return m.meta | ||||
| } | ||||
|  | ||||
| // AddMeta adds a meta data tag (consisting of key and value) to the map of meta data tags | ||||
| func (m *ccMetric) AddMeta(key, value string) { | ||||
| 	m.meta[key] = value | ||||
| } | ||||
|  | ||||
| // GetMeta returns the meta data tag with meta data's key equal to <key> | ||||
| func (m *ccMetric) GetMeta(key string) (string, bool) { | ||||
| 	value, ok := m.meta[key] | ||||
| 	return value, ok | ||||
| } | ||||
|  | ||||
| // HasMeta checks if a meta data tag with meta data's key equal to <key> is present in the map of meta data tags | ||||
| func (m *ccMetric) HasMeta(key string) bool { | ||||
| 	_, ok := m.meta[key] | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| // RemoveMeta removes the meta data tag with tag's key equal to <key> | ||||
| func (m *ccMetric) RemoveMeta(key string) { | ||||
| 	delete(m.meta, key) | ||||
| } | ||||
|  | ||||
| // Fields returns the list of fields as key-value-mapping | ||||
| func (m *ccMetric) Fields() map[string]interface{} { | ||||
| 	return m.fields | ||||
| } | ||||
|  | ||||
| // AddField adds a field (consisting of key and value) to the map of fields | ||||
| func (m *ccMetric) AddField(key string, value interface{}) { | ||||
| 	m.fields[key] = value | ||||
| } | ||||
|  | ||||
| // GetField returns the field with field's key equal to <key> | ||||
| func (m *ccMetric) GetField(key string) (interface{}, bool) { | ||||
| 	v, ok := m.fields[key] | ||||
| 	return v, ok | ||||
| } | ||||
|  | ||||
| // HasField checks if a field with field's key equal to <key> is present in the map of fields | ||||
| func (m *ccMetric) HasField(key string) bool { | ||||
| 	_, ok := m.fields[key] | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| // RemoveField removes the field with field's key equal to <key> | ||||
| // from the map of fields | ||||
| func (m *ccMetric) RemoveField(key string) { | ||||
| 	delete(m.fields, key) | ||||
| } | ||||
|  | ||||
| // New creates a new measurement point | ||||
| func New( | ||||
| 	name string, | ||||
| 	tags map[string]string, | ||||
| 	meta map[string]string, | ||||
| 	fields map[string]interface{}, | ||||
| 	tm time.Time, | ||||
| ) (CCMetric, error) { | ||||
| 	m := &ccMetric{ | ||||
| 		name:   name, | ||||
| 		tags:   make(map[string]string, len(tags)), | ||||
| 		meta:   make(map[string]string, len(meta)), | ||||
| 		fields: make(map[string]interface{}, len(fields)), | ||||
| 		tm:     tm, | ||||
| 	} | ||||
|  | ||||
| 	// deep copy tags, meta data tags and fields | ||||
| 	for k, v := range tags { | ||||
| 		m.tags[k] = v | ||||
| 	} | ||||
| 	for k, v := range meta { | ||||
| 		m.meta[k] = v | ||||
| 	} | ||||
| 	for k, v := range fields { | ||||
| 		v := convertField(v) | ||||
| 		if v == nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		m.fields[k] = v | ||||
| 	} | ||||
|  | ||||
| 	return m, nil | ||||
| } | ||||
|  | ||||
| // FromMetric copies the metric <other> | ||||
| func FromMetric(other ccMetric) CCMetric { | ||||
| 	m := &ccMetric{ | ||||
| 		name:   other.Name(), | ||||
| 		tags:   make(map[string]string, len(other.tags)), | ||||
| 		meta:   make(map[string]string, len(other.meta)), | ||||
| 		fields: make(map[string]interface{}, len(other.fields)), | ||||
| 		tm:     other.Time(), | ||||
| 	} | ||||
|  | ||||
| 	// deep copy tags, meta data tags and fields | ||||
| 	for key, value := range other.tags { | ||||
| 		m.tags[key] = value | ||||
| 	} | ||||
| 	for key, value := range other.meta { | ||||
| 		m.meta[key] = value | ||||
| 	} | ||||
| 	for key, value := range other.fields { | ||||
| 		m.fields[key] = value | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // FromInfluxMetric copies the influxDB line protocol metric <other> | ||||
| func FromInfluxMetric(other lp.Metric) CCMetric { | ||||
| 	m := &ccMetric{ | ||||
| 		name:   other.Name(), | ||||
| 		tags:   make(map[string]string), | ||||
| 		meta:   make(map[string]string), | ||||
| 		fields: make(map[string]interface{}), | ||||
| 		tm:     other.Time(), | ||||
| 	} | ||||
|  | ||||
| 	// deep copy tags and fields | ||||
| 	for _, otherTag := range other.TagList() { | ||||
| 		m.tags[otherTag.Key] = otherTag.Value | ||||
| 	} | ||||
| 	for _, otherField := range other.FieldList() { | ||||
| 		m.fields[otherField.Key] = otherField.Value | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| // convertField converts data types of fields by the following schemata: | ||||
| //                         *float32, *float64,                      float32, float64 -> float64 | ||||
| //  *int,  *int8,  *int16,   *int32,   *int64,  int,  int8,  int16,   int32,   int64 ->   int64 | ||||
| // *uint, *uint8, *uint16,  *uint32,  *uint64, uint, uint8, uint16,  uint32,  uint64 ->  uint64 | ||||
| // *[]byte, *string,                           []byte, string                        -> string | ||||
| // *bool,                                      bool                                  -> bool | ||||
| func convertField(v interface{}) interface{} { | ||||
| 	switch v := v.(type) { | ||||
| 	case float64: | ||||
| 		return v | ||||
| 	case int64: | ||||
| 		return v | ||||
| 	case string: | ||||
| 		return v | ||||
| 	case bool: | ||||
| 		return v | ||||
| 	case int: | ||||
| 		return int64(v) | ||||
| 	case uint: | ||||
| 		return uint64(v) | ||||
| 	case uint64: | ||||
| 		return uint64(v) | ||||
| 	case []byte: | ||||
| 		return string(v) | ||||
| 	case int32: | ||||
| 		return int64(v) | ||||
| 	case int16: | ||||
| 		return int64(v) | ||||
| 	case int8: | ||||
| 		return int64(v) | ||||
| 	case uint32: | ||||
| 		return uint64(v) | ||||
| 	case uint16: | ||||
| 		return uint64(v) | ||||
| 	case uint8: | ||||
| 		return uint64(v) | ||||
| 	case float32: | ||||
| 		return float64(v) | ||||
| 	case *float64: | ||||
| 		if v != nil { | ||||
| 			return *v | ||||
| 		} | ||||
| 	case *int64: | ||||
| 		if v != nil { | ||||
| 			return *v | ||||
| 		} | ||||
| 	case *string: | ||||
| 		if v != nil { | ||||
| 			return *v | ||||
| 		} | ||||
| 	case *bool: | ||||
| 		if v != nil { | ||||
| 			return *v | ||||
| 		} | ||||
| 	case *int: | ||||
| 		if v != nil { | ||||
| 			return int64(*v) | ||||
| 		} | ||||
| 	case *uint: | ||||
| 		if v != nil { | ||||
| 			return uint64(*v) | ||||
| 		} | ||||
| 	case *uint64: | ||||
| 		if v != nil { | ||||
| 			return uint64(*v) | ||||
| 		} | ||||
| 	case *[]byte: | ||||
| 		if v != nil { | ||||
| 			return string(*v) | ||||
| 		} | ||||
| 	case *int32: | ||||
| 		if v != nil { | ||||
| 			return int64(*v) | ||||
| 		} | ||||
| 	case *int16: | ||||
| 		if v != nil { | ||||
| 			return int64(*v) | ||||
| 		} | ||||
| 	case *int8: | ||||
| 		if v != nil { | ||||
| 			return int64(*v) | ||||
| 		} | ||||
| 	case *uint32: | ||||
| 		if v != nil { | ||||
| 			return uint64(*v) | ||||
| 		} | ||||
| 	case *uint16: | ||||
| 		if v != nil { | ||||
| 			return uint64(*v) | ||||
| 		} | ||||
| 	case *uint8: | ||||
| 		if v != nil { | ||||
| 			return uint64(*v) | ||||
| 		} | ||||
| 	case *float32: | ||||
| 		if v != nil { | ||||
| 			return float64(*v) | ||||
| 		} | ||||
| 	default: | ||||
| 		return nil | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										422
									
								
								internal/ccTopology/ccTopology.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										422
									
								
								internal/ccTopology/ccTopology.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,422 @@ | ||||
| package ccTopology | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	cclogger "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| ) | ||||
|  | ||||
| const SYSFS_NUMABASE = `/sys/devices/system/node` | ||||
| const SYSFS_CPUBASE = `/sys/devices/system/cpu` | ||||
| const PROCFS_CPUINFO = `/proc/cpuinfo` | ||||
|  | ||||
| // intArrayContains scans an array of ints if the value str is present in the array | ||||
| // If the specified value is found, the corresponding array index is returned. | ||||
| // The bool value is used to signal success or failure | ||||
| func intArrayContains(array []int, str int) (int, bool) { | ||||
| 	for i, a := range array { | ||||
| 		if a == str { | ||||
| 			return i, true | ||||
| 		} | ||||
| 	} | ||||
| 	return -1, false | ||||
| } | ||||
|  | ||||
| func fileToInt(path string) int { | ||||
| 	buffer, err := ioutil.ReadFile(path) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		cclogger.ComponentError("ccTopology", "Reading", path, ":", err.Error()) | ||||
| 		return -1 | ||||
| 	} | ||||
| 	sbuffer := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 	var id int64 | ||||
| 	//_, err = fmt.Scanf("%d", sbuffer, &id) | ||||
| 	id, err = strconv.ParseInt(sbuffer, 10, 32) | ||||
| 	if err != nil { | ||||
| 		cclogger.ComponentError("ccTopology", "Parsing", path, ":", sbuffer, err.Error()) | ||||
| 		return -1 | ||||
| 	} | ||||
| 	return int(id) | ||||
| } | ||||
|  | ||||
| func SocketList() []int { | ||||
| 	buffer, err := ioutil.ReadFile(string(PROCFS_CPUINFO)) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return nil | ||||
| 	} | ||||
| 	ll := strings.Split(string(buffer), "\n") | ||||
| 	var packs []int | ||||
| 	for _, line := range ll { | ||||
| 		if strings.HasPrefix(line, "physical id") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			id, err := strconv.ParseInt(lv[3], 10, 32) | ||||
| 			if err != nil { | ||||
| 				log.Print(err) | ||||
| 				return packs | ||||
| 			} | ||||
| 			_, found := intArrayContains(packs, int(id)) | ||||
| 			if !found { | ||||
| 				packs = append(packs, int(id)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return packs | ||||
| } | ||||
|  | ||||
| func CpuList() []int { | ||||
| 	buffer, err := ioutil.ReadFile(string(PROCFS_CPUINFO)) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return nil | ||||
| 	} | ||||
| 	ll := strings.Split(string(buffer), "\n") | ||||
| 	cpulist := make([]int, 0) | ||||
| 	for _, line := range ll { | ||||
| 		if strings.HasPrefix(line, "processor") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			id, err := strconv.ParseInt(lv[2], 10, 32) | ||||
| 			if err != nil { | ||||
| 				log.Print(err) | ||||
| 				return cpulist | ||||
| 			} | ||||
| 			_, found := intArrayContains(cpulist, int(id)) | ||||
| 			if !found { | ||||
| 				cpulist = append(cpulist, int(id)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist | ||||
| } | ||||
|  | ||||
| func CoreList() []int { | ||||
| 	buffer, err := ioutil.ReadFile(string(PROCFS_CPUINFO)) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return nil | ||||
| 	} | ||||
| 	ll := strings.Split(string(buffer), "\n") | ||||
| 	corelist := make([]int, 0) | ||||
| 	for _, line := range ll { | ||||
| 		if strings.HasPrefix(line, "core id") { | ||||
| 			lv := strings.Fields(line) | ||||
| 			id, err := strconv.ParseInt(lv[3], 10, 32) | ||||
| 			if err != nil { | ||||
| 				log.Print(err) | ||||
| 				return corelist | ||||
| 			} | ||||
| 			_, found := intArrayContains(corelist, int(id)) | ||||
| 			if !found { | ||||
| 				corelist = append(corelist, int(id)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return corelist | ||||
| } | ||||
|  | ||||
| func NumaNodeList() []int { | ||||
| 	numaList := make([]int, 0) | ||||
| 	globPath := filepath.Join(string(SYSFS_NUMABASE), "node*") | ||||
| 	regexPath := filepath.Join(string(SYSFS_NUMABASE), "node(\\d+)") | ||||
| 	regex := regexp.MustCompile(regexPath) | ||||
| 	files, err := filepath.Glob(globPath) | ||||
| 	if err != nil { | ||||
| 		cclogger.ComponentError("CCTopology", "NumaNodeList", err.Error()) | ||||
| 	} | ||||
| 	for _, f := range files { | ||||
| 		if !regex.MatchString(f) { | ||||
| 			continue | ||||
| 		} | ||||
| 		finfo, err := os.Lstat(f) | ||||
| 		if err != nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		if !finfo.IsDir() { | ||||
| 			continue | ||||
| 		} | ||||
| 		matches := regex.FindStringSubmatch(f) | ||||
| 		if len(matches) == 2 { | ||||
| 			id, err := strconv.Atoi(matches[1]) | ||||
| 			if err == nil { | ||||
| 				if _, found := intArrayContains(numaList, id); !found { | ||||
| 					numaList = append(numaList, id) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
| 	return numaList | ||||
| } | ||||
|  | ||||
| func DieList() []int { | ||||
| 	cpulist := CpuList() | ||||
| 	dielist := make([]int, 0) | ||||
| 	for _, c := range cpulist { | ||||
| 		diepath := filepath.Join(string(SYSFS_CPUBASE), fmt.Sprintf("cpu%d", c), "topology/die_id") | ||||
| 		dieid := fileToInt(diepath) | ||||
| 		if dieid > 0 { | ||||
| 			_, found := intArrayContains(dielist, int(dieid)) | ||||
| 			if !found { | ||||
| 				dielist = append(dielist, int(dieid)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return dielist | ||||
| } | ||||
|  | ||||
| type CpuEntry struct { | ||||
| 	Cpuid      int | ||||
| 	SMT        int | ||||
| 	Core       int | ||||
| 	Socket     int | ||||
| 	Numadomain int | ||||
| 	Die        int | ||||
| } | ||||
|  | ||||
| func CpuData() []CpuEntry { | ||||
|  | ||||
| 	fileToInt := func(path string) int { | ||||
| 		buffer, err := ioutil.ReadFile(path) | ||||
| 		if err != nil { | ||||
| 			log.Print(err) | ||||
| 			//cclogger.ComponentError("ccTopology", "Reading", path, ":", err.Error()) | ||||
| 			return -1 | ||||
| 		} | ||||
| 		sbuffer := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		var id int64 | ||||
| 		//_, err = fmt.Scanf("%d", sbuffer, &id) | ||||
| 		id, err = strconv.ParseInt(sbuffer, 10, 32) | ||||
| 		if err != nil { | ||||
| 			cclogger.ComponentError("ccTopology", "Parsing", path, ":", sbuffer, err.Error()) | ||||
| 			return -1 | ||||
| 		} | ||||
| 		return int(id) | ||||
| 	} | ||||
| 	getCore := func(basepath string) int { | ||||
| 		return fileToInt(fmt.Sprintf("%s/core_id", basepath)) | ||||
| 	} | ||||
|  | ||||
| 	getSocket := func(basepath string) int { | ||||
| 		return fileToInt(fmt.Sprintf("%s/physical_package_id", basepath)) | ||||
| 	} | ||||
|  | ||||
| 	getDie := func(basepath string) int { | ||||
| 		return fileToInt(fmt.Sprintf("%s/die_id", basepath)) | ||||
| 	} | ||||
|  | ||||
| 	getSMT := func(cpuid int, basepath string) int { | ||||
| 		buffer, err := ioutil.ReadFile(fmt.Sprintf("%s/thread_siblings_list", basepath)) | ||||
| 		if err != nil { | ||||
| 			cclogger.ComponentError("CCTopology", "CpuData:getSMT", err.Error()) | ||||
| 		} | ||||
| 		threadlist := make([]int, 0) | ||||
| 		sbuffer := strings.Replace(string(buffer), "\n", "", -1) | ||||
| 		for _, x := range strings.Split(sbuffer, ",") { | ||||
| 			id, err := strconv.ParseInt(x, 10, 32) | ||||
| 			if err != nil { | ||||
| 				cclogger.ComponentError("CCTopology", "CpuData:getSMT", err.Error()) | ||||
| 			} | ||||
| 			threadlist = append(threadlist, int(id)) | ||||
| 		} | ||||
| 		for i, x := range threadlist { | ||||
| 			if x == cpuid { | ||||
| 				return i | ||||
| 			} | ||||
| 		} | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	getNumaDomain := func(basepath string) int { | ||||
| 		globPath := filepath.Join(basepath, "node*") | ||||
| 		regexPath := filepath.Join(basepath, "node(\\d+)") | ||||
| 		regex := regexp.MustCompile(regexPath) | ||||
| 		files, err := filepath.Glob(globPath) | ||||
| 		if err != nil { | ||||
| 			cclogger.ComponentError("CCTopology", "CpuData:getNumaDomain", err.Error()) | ||||
| 		} | ||||
| 		for _, f := range files { | ||||
| 			finfo, err := os.Lstat(f) | ||||
| 			if err == nil && finfo.IsDir() { | ||||
| 				matches := regex.FindStringSubmatch(f) | ||||
| 				if len(matches) == 2 { | ||||
| 					id, err := strconv.Atoi(matches[1]) | ||||
| 					if err == nil { | ||||
| 						return id | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		return 0 | ||||
| 	} | ||||
|  | ||||
| 	clist := make([]CpuEntry, 0) | ||||
| 	for _, c := range CpuList() { | ||||
| 		clist = append(clist, CpuEntry{Cpuid: c}) | ||||
| 	} | ||||
| 	for _, centry := range clist { | ||||
| 		centry.Socket = -1 | ||||
| 		centry.Numadomain = -1 | ||||
| 		centry.Die = -1 | ||||
| 		centry.Core = -1 | ||||
| 		// Set base directory for topology lookup | ||||
| 		cpustr := fmt.Sprintf("cpu%d", centry.Cpuid) | ||||
| 		base := filepath.Join("/sys/devices/system/cpu", cpustr) | ||||
| 		topoBase := filepath.Join(base, "topology") | ||||
|  | ||||
| 		// Lookup CPU core id | ||||
| 		centry.Core = getCore(topoBase) | ||||
|  | ||||
| 		// Lookup CPU socket id | ||||
| 		centry.Socket = getSocket(topoBase) | ||||
|  | ||||
| 		// Lookup CPU die id | ||||
| 		centry.Die = getDie(topoBase) | ||||
| 		if centry.Die < 0 { | ||||
| 			centry.Die = centry.Socket | ||||
| 		} | ||||
|  | ||||
| 		// Lookup SMT thread id | ||||
| 		centry.SMT = getSMT(centry.Cpuid, topoBase) | ||||
|  | ||||
| 		// Lookup NUMA domain id | ||||
| 		centry.Numadomain = getNumaDomain(base) | ||||
|  | ||||
| 	} | ||||
| 	return clist | ||||
| } | ||||
|  | ||||
| type CpuInformation struct { | ||||
| 	NumHWthreads   int | ||||
| 	SMTWidth       int | ||||
| 	NumSockets     int | ||||
| 	NumDies        int | ||||
| 	NumCores       int | ||||
| 	NumNumaDomains int | ||||
| } | ||||
|  | ||||
| func CpuInfo() CpuInformation { | ||||
| 	var c CpuInformation | ||||
|  | ||||
| 	smtList := make([]int, 0) | ||||
| 	numaList := make([]int, 0) | ||||
| 	dieList := make([]int, 0) | ||||
| 	socketList := make([]int, 0) | ||||
| 	coreList := make([]int, 0) | ||||
| 	cdata := CpuData() | ||||
| 	for _, d := range cdata { | ||||
| 		if _, ok := intArrayContains(smtList, d.SMT); !ok { | ||||
| 			smtList = append(smtList, d.SMT) | ||||
| 		} | ||||
| 		if _, ok := intArrayContains(numaList, d.Numadomain); !ok { | ||||
| 			numaList = append(numaList, d.Numadomain) | ||||
| 		} | ||||
| 		if _, ok := intArrayContains(dieList, d.Die); !ok { | ||||
| 			dieList = append(dieList, d.Die) | ||||
| 		} | ||||
| 		if _, ok := intArrayContains(socketList, d.Socket); !ok { | ||||
| 			socketList = append(socketList, d.Socket) | ||||
| 		} | ||||
| 		if _, ok := intArrayContains(coreList, d.Core); !ok { | ||||
| 			coreList = append(coreList, d.Core) | ||||
| 		} | ||||
| 	} | ||||
| 	c.NumNumaDomains = len(numaList) | ||||
| 	c.SMTWidth = len(smtList) | ||||
| 	c.NumDies = len(dieList) | ||||
| 	c.NumCores = len(coreList) | ||||
| 	c.NumSockets = len(socketList) | ||||
| 	c.NumHWthreads = len(cdata) | ||||
| 	return c | ||||
| } | ||||
|  | ||||
| func GetCpuSocket(cpuid int) int { | ||||
| 	cdata := CpuData() | ||||
| 	for _, d := range cdata { | ||||
| 		if d.Cpuid == cpuid { | ||||
| 			return d.Socket | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| func GetCpuNumaDomain(cpuid int) int { | ||||
| 	cdata := CpuData() | ||||
| 	for _, d := range cdata { | ||||
| 		if d.Cpuid == cpuid { | ||||
| 			return d.Numadomain | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| func GetCpuDie(cpuid int) int { | ||||
| 	cdata := CpuData() | ||||
| 	for _, d := range cdata { | ||||
| 		if d.Cpuid == cpuid { | ||||
| 			return d.Die | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| func GetCpuCore(cpuid int) int { | ||||
| 	cdata := CpuData() | ||||
| 	for _, d := range cdata { | ||||
| 		if d.Cpuid == cpuid { | ||||
| 			return d.Core | ||||
| 		} | ||||
| 	} | ||||
| 	return -1 | ||||
| } | ||||
|  | ||||
| func GetSocketCpus(socket int) []int { | ||||
| 	all := CpuData() | ||||
| 	cpulist := make([]int, 0) | ||||
| 	for _, d := range all { | ||||
| 		if d.Socket == socket { | ||||
| 			cpulist = append(cpulist, d.Cpuid) | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist | ||||
| } | ||||
|  | ||||
| func GetNumaDomainCpus(domain int) []int { | ||||
| 	all := CpuData() | ||||
| 	cpulist := make([]int, 0) | ||||
| 	for _, d := range all { | ||||
| 		if d.Numadomain == domain { | ||||
| 			cpulist = append(cpulist, d.Cpuid) | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist | ||||
| } | ||||
|  | ||||
| func GetDieCpus(die int) []int { | ||||
| 	all := CpuData() | ||||
| 	cpulist := make([]int, 0) | ||||
| 	for _, d := range all { | ||||
| 		if d.Die == die { | ||||
| 			cpulist = append(cpulist, d.Cpuid) | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist | ||||
| } | ||||
|  | ||||
| func GetCoreCpus(core int) []int { | ||||
| 	all := CpuData() | ||||
| 	cpulist := make([]int, 0) | ||||
| 	for _, d := range all { | ||||
| 		if d.Core == core { | ||||
| 			cpulist = append(cpulist, d.Cpuid) | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist | ||||
| } | ||||
							
								
								
									
										38
									
								
								internal/metricAggregator/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								internal/metricAggregator/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| # The MetricAggregator | ||||
|  | ||||
| In some cases, further combination of metrics or raw values is required. For that strings like `foo + 1` with runtime dependent `foo` need to be evaluated. The MetricAggregator relies on the [`gval`](https://github.com/PaesslerAG/gval) Golang package to perform all expression evaluation. The `gval` package provides the basic arithmetic operations but the MetricAggregator defines additional ones. | ||||
|  | ||||
| **Note**: To get an impression which expressions can be handled by `gval`, see its [README](https://github.com/PaesslerAG/gval/blob/master/README.md) | ||||
|  | ||||
| ## Simple expression evaluation | ||||
|  | ||||
| For simple expression evaluation, the MetricAggregator provides two function for different use-cases: | ||||
| - `EvalBoolCondition(expression string, params map[string]interface{}`: Used by the MetricRouter to match metrics like `metric.Name() == 'mymetric'` | ||||
| - `EvalFloat64Condition(expression string, params map[string]interface{})`: Used by the MetricRouter and LikwidCollector to derive new values like `(PMC0+PMC1)/PMC3` | ||||
|  | ||||
| ## MetricAggregator extensions for `gval` | ||||
|  | ||||
| The MetricAggregator provides these functions additional to the `Full` language in `gval`: | ||||
| - `sum(array)`: Sum up values in an array like `sum(values)` | ||||
| - `min(array)`: Get the minimum value in an array like `min(values)` | ||||
| - `avg(array)`: Get the mean value in an array like `avg(values)` | ||||
| - `mean(array)`: Get the mean value in an array like `mean(values)` | ||||
| - `max(array)`: Get the maximum value in an array like `max(values)` | ||||
| - `len(array)`: Get the length of an array like `len(values)` | ||||
| - `median(array)`: Get the median value in an array like `mean(values)` | ||||
| - `in`: Check existence in an array like `0 in getCpuList()` to check whether there is an entry `0`. Also substring matching works like `temp in metric.Name()` | ||||
| - `match`: Regular-expression matching like `match('temp_cores_%d+', metric.Name())`. **Note** all `\` in an regex has to be replaced with `%` | ||||
| - `getCpuCore(cpuid)`: For a CPU id, the the corresponding CPU core id like `getCpuCore(0)` | ||||
| - `getCpuSocket(cpuid)`: For a CPU id, the the corresponding CPU socket id | ||||
| - `getCpuNuma(cpuid)`: For a CPU id, the the corresponding NUMA domain id | ||||
| - `getCpuDie(cpuid)`: For a CPU id, the the corresponding CPU die id | ||||
| - `getSockCpuList(sockid)`: For a given CPU socket id, the list of CPU ids is returned like the CPUs on socket 1 `getSockCpuList(1)` | ||||
| - `getNumaCpuList(numaid)`: For a given NUMA node id, the list of CPU ids is returned | ||||
| - `getDieCpuList(dieid)`: For a given CPU die id, the list of CPU ids is returned | ||||
| - `getCoreCpuList(coreid)`: For a given CPU core id, the list of CPU ids is returned | ||||
| - `getCpuList`: Get the list of all CPUs | ||||
|  | ||||
| ## Limitations | ||||
|  | ||||
| - Since the metrics are written in JSON files which do not allow `""` without proper escaping inside of JSON strings, you have to use `''` for strings. | ||||
| - Since `\` is interpreted by JSON as escape character, it cannot be used in metrics. But it is required to write regular expressions. So instead of `/`, use `%` and the MetricAggregator replaces them after reading the JSON file. | ||||
							
								
								
									
										348
									
								
								internal/metricAggregator/metricAggregator.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										348
									
								
								internal/metricAggregator/metricAggregator.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,348 @@ | ||||
| package metricAggregator | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	topo "github.com/ClusterCockpit/cc-metric-collector/internal/ccTopology" | ||||
|  | ||||
| 	"github.com/PaesslerAG/gval" | ||||
| ) | ||||
|  | ||||
| type MetricAggregatorIntervalConfig struct { | ||||
| 	Name      string            `json:"name"`     // Metric name for the new metric | ||||
| 	Function  string            `json:"function"` // Function to apply on the metric | ||||
| 	Condition string            `json:"if"`       // Condition for applying function | ||||
| 	Tags      map[string]string `json:"tags"`     // Tags for the new metric | ||||
| 	Meta      map[string]string `json:"meta"`     // Meta information for the new metric | ||||
| 	gvalCond  gval.Evaluable | ||||
| 	gvalFunc  gval.Evaluable | ||||
| } | ||||
|  | ||||
| type metricAggregator struct { | ||||
| 	functions []*MetricAggregatorIntervalConfig | ||||
| 	constants map[string]interface{} | ||||
| 	language  gval.Language | ||||
| 	output    chan lp.CCMetric | ||||
| } | ||||
|  | ||||
| type MetricAggregator interface { | ||||
| 	AddAggregation(name, function, condition string, tags, meta map[string]string) error | ||||
| 	DeleteAggregation(name string) error | ||||
| 	Init(output chan lp.CCMetric) error | ||||
| 	Eval(starttime time.Time, endtime time.Time, metrics []lp.CCMetric) | ||||
| } | ||||
|  | ||||
| var metricCacheLanguage = gval.NewLanguage( | ||||
| 	gval.Base(), | ||||
| 	gval.Function("sum", sumfunc), | ||||
| 	gval.Function("min", minfunc), | ||||
| 	gval.Function("avg", avgfunc), | ||||
| 	gval.Function("mean", avgfunc), | ||||
| 	gval.Function("max", maxfunc), | ||||
| 	gval.Function("len", lenfunc), | ||||
| 	gval.Function("median", medianfunc), | ||||
| 	gval.InfixOperator("in", infunc), | ||||
| 	gval.Function("match", matchfunc), | ||||
| 	gval.Function("getCpuCore", getCpuCoreFunc), | ||||
| 	gval.Function("getCpuSocket", getCpuSocketFunc), | ||||
| 	gval.Function("getCpuNuma", getCpuNumaDomainFunc), | ||||
| 	gval.Function("getCpuDie", getCpuDieFunc), | ||||
| 	gval.Function("getSockCpuList", getCpuListOfSocketFunc), | ||||
| 	gval.Function("getNumaCpuList", getCpuListOfNumaDomainFunc), | ||||
| 	gval.Function("getDieCpuList", getCpuListOfDieFunc), | ||||
| 	gval.Function("getCoreCpuList", getCpuListOfCoreFunc), | ||||
| 	gval.Function("getCpuList", getCpuListOfNode), | ||||
| 	gval.Function("getCpuListOfType", getCpuListOfType), | ||||
| ) | ||||
| var language gval.Language = gval.NewLanguage( | ||||
| 	gval.Full(), | ||||
| 	metricCacheLanguage, | ||||
| ) | ||||
| var evaluables = struct { | ||||
| 	mapping map[string]gval.Evaluable | ||||
| 	mutex   sync.Mutex | ||||
| }{ | ||||
| 	mapping: make(map[string]gval.Evaluable), | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) Init(output chan lp.CCMetric) error { | ||||
| 	c.output = output | ||||
| 	c.functions = make([]*MetricAggregatorIntervalConfig, 0) | ||||
| 	c.constants = make(map[string]interface{}) | ||||
|  | ||||
| 	// add constants like hostname, numSockets, ... to constants list | ||||
| 	// Set hostname | ||||
| 	hostname, err := os.Hostname() | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	// Drop domain part of host name | ||||
| 	c.constants["hostname"] = strings.SplitN(hostname, `.`, 2)[0] | ||||
| 	cinfo := topo.CpuInfo() | ||||
| 	c.constants["numHWThreads"] = cinfo.NumHWthreads | ||||
| 	c.constants["numSockets"] = cinfo.NumSockets | ||||
| 	c.constants["numNumaDomains"] = cinfo.NumNumaDomains | ||||
| 	c.constants["numDies"] = cinfo.NumDies | ||||
| 	c.constants["smtWidth"] = cinfo.SMTWidth | ||||
|  | ||||
| 	c.language = gval.NewLanguage( | ||||
| 		gval.Full(), | ||||
| 		metricCacheLanguage, | ||||
| 	) | ||||
|  | ||||
| 	// Example aggregation function | ||||
| 	// var f metricCacheFunctionConfig | ||||
| 	// f.Name = "temp_cores_avg" | ||||
| 	// //f.Condition = `"temp_core_" in name` | ||||
| 	// f.Condition = `match("temp_core_%d+", metric.Name())` | ||||
| 	// f.Function = `avg(values)` | ||||
| 	// f.Tags = map[string]string{"type": "node"} | ||||
| 	// f.Meta = map[string]string{"group": "IPMI", "unit": "degC", "source": "TempCollector"} | ||||
| 	// c.functions = append(c.functions, &f) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) Eval(starttime time.Time, endtime time.Time, metrics []lp.CCMetric) { | ||||
| 	vars := make(map[string]interface{}) | ||||
| 	for k, v := range c.constants { | ||||
| 		vars[k] = v | ||||
| 	} | ||||
| 	vars["starttime"] = starttime | ||||
| 	vars["endtime"] = endtime | ||||
| 	for _, f := range c.functions { | ||||
| 		cclog.ComponentDebug("MetricCache", "COLLECT", f.Name, "COND", f.Condition) | ||||
| 		values := make([]float64, 0) | ||||
| 		matches := make([]lp.CCMetric, 0) | ||||
| 		for _, m := range metrics { | ||||
| 			vars["metric"] = m | ||||
| 			//value, err := gval.Evaluate(f.Condition, vars, c.language) | ||||
| 			value, err := f.gvalCond.EvalBool(context.Background(), vars) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError("MetricCache", "COLLECT", f.Name, "COND", f.Condition, ":", err.Error()) | ||||
| 				continue | ||||
| 			} | ||||
| 			if value { | ||||
| 				v, valid := m.GetField("value") | ||||
| 				if valid { | ||||
| 					switch x := v.(type) { | ||||
| 					case float64: | ||||
| 						values = append(values, x) | ||||
| 					case float32: | ||||
| 					case int: | ||||
| 					case int64: | ||||
| 						values = append(values, float64(x)) | ||||
| 					case bool: | ||||
| 						if x { | ||||
| 							values = append(values, float64(1.0)) | ||||
| 						} else { | ||||
| 							values = append(values, float64(0.0)) | ||||
| 						} | ||||
| 					default: | ||||
| 						cclog.ComponentError("MetricCache", "COLLECT ADD VALUE", v, "FAILED") | ||||
| 					} | ||||
| 				} | ||||
| 				matches = append(matches, m) | ||||
| 			} | ||||
| 		} | ||||
| 		delete(vars, "metric") | ||||
| 		cclog.ComponentDebug("MetricCache", "EVALUATE", f.Name, "METRICS", len(values), "CALC", f.Function) | ||||
| 		vars["values"] = values | ||||
| 		vars["metrics"] = matches | ||||
| 		if len(values) > 0 { | ||||
| 			value, err := gval.Evaluate(f.Function, vars, c.language) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError("MetricCache", "EVALUATE", f.Name, "METRICS", len(values), "CALC", f.Function, ":", err.Error()) | ||||
| 				break | ||||
| 			} | ||||
|  | ||||
| 			copy_tags := func(tags map[string]string, metrics []lp.CCMetric) map[string]string { | ||||
| 				out := make(map[string]string) | ||||
| 				for key, value := range tags { | ||||
| 					switch value { | ||||
| 					case "<copy>": | ||||
| 						for _, m := range metrics { | ||||
| 							v, err := m.GetTag(key) | ||||
| 							if err { | ||||
| 								out[key] = v | ||||
| 							} | ||||
| 						} | ||||
| 					default: | ||||
| 						out[key] = value | ||||
| 					} | ||||
| 				} | ||||
| 				return out | ||||
| 			} | ||||
| 			copy_meta := func(meta map[string]string, metrics []lp.CCMetric) map[string]string { | ||||
| 				out := make(map[string]string) | ||||
| 				for key, value := range meta { | ||||
| 					switch value { | ||||
| 					case "<copy>": | ||||
| 						for _, m := range metrics { | ||||
| 							v, err := m.GetMeta(key) | ||||
| 							if err { | ||||
| 								out[key] = v | ||||
| 							} | ||||
| 						} | ||||
| 					default: | ||||
| 						out[key] = value | ||||
| 					} | ||||
| 				} | ||||
| 				return out | ||||
| 			} | ||||
| 			tags := copy_tags(f.Tags, matches) | ||||
| 			meta := copy_meta(f.Meta, matches) | ||||
|  | ||||
| 			var m lp.CCMetric | ||||
| 			switch t := value.(type) { | ||||
| 			case float64: | ||||
| 				m, err = lp.New(f.Name, tags, meta, map[string]interface{}{"value": t}, starttime) | ||||
| 			case float32: | ||||
| 				m, err = lp.New(f.Name, tags, meta, map[string]interface{}{"value": t}, starttime) | ||||
| 			case int: | ||||
| 				m, err = lp.New(f.Name, tags, meta, map[string]interface{}{"value": t}, starttime) | ||||
| 			case int64: | ||||
| 				m, err = lp.New(f.Name, tags, meta, map[string]interface{}{"value": t}, starttime) | ||||
| 			case string: | ||||
| 				m, err = lp.New(f.Name, tags, meta, map[string]interface{}{"value": t}, starttime) | ||||
| 			default: | ||||
| 				cclog.ComponentError("MetricCache", "Gval returned invalid type", t, "skipping metric", f.Name) | ||||
| 			} | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError("MetricCache", "Cannot create metric from Gval result", value, ":", err.Error()) | ||||
| 			} | ||||
| 			cclog.ComponentDebug("MetricCache", "SEND", m) | ||||
| 			select { | ||||
| 			case c.output <- m: | ||||
| 			default: | ||||
| 			} | ||||
|  | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) AddAggregation(name, function, condition string, tags, meta map[string]string) error { | ||||
| 	// Since "" cannot be used inside of JSON strings, we use '' and replace them here because gval does not like '' | ||||
| 	// but wants "" | ||||
| 	newfunc := strings.ReplaceAll(function, "'", "\"") | ||||
| 	newcond := strings.ReplaceAll(condition, "'", "\"") | ||||
| 	gvalCond, err := gval.Full(metricCacheLanguage).NewEvaluable(newcond) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("MetricAggregator", "Cannot add aggregation, invalid if condition", newcond, ":", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	gvalFunc, err := gval.Full(metricCacheLanguage).NewEvaluable(newfunc) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("MetricAggregator", "Cannot add aggregation, invalid function condition", newfunc, ":", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	for _, agg := range c.functions { | ||||
| 		if agg.Name == name { | ||||
| 			agg.Name = name | ||||
| 			agg.Condition = newcond | ||||
| 			agg.Function = newfunc | ||||
| 			agg.Tags = tags | ||||
| 			agg.Meta = meta | ||||
| 			agg.gvalCond = gvalCond | ||||
| 			agg.gvalFunc = gvalFunc | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
| 	agg := &MetricAggregatorIntervalConfig{ | ||||
| 		Name:      name, | ||||
| 		Condition: newcond, | ||||
| 		gvalCond:  gvalCond, | ||||
| 		Function:  newfunc, | ||||
| 		gvalFunc:  gvalFunc, | ||||
| 		Tags:      tags, | ||||
| 		Meta:      meta, | ||||
| 	} | ||||
| 	c.functions = append(c.functions, agg) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) DeleteAggregation(name string) error { | ||||
| 	for i, agg := range c.functions { | ||||
| 		if agg.Name == name { | ||||
| 			copy(c.functions[i:], c.functions[i+1:]) | ||||
| 			c.functions[len(c.functions)-1] = nil | ||||
| 			c.functions = c.functions[:len(c.functions)-1] | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
| 	return fmt.Errorf("no aggregation for metric name %s", name) | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) AddConstant(name string, value interface{}) { | ||||
| 	c.constants[name] = value | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) DelConstant(name string) { | ||||
| 	delete(c.constants, name) | ||||
| } | ||||
|  | ||||
| func (c *metricAggregator) AddFunction(name string, function func(args ...interface{}) (interface{}, error)) { | ||||
| 	c.language = gval.NewLanguage(c.language, gval.Function(name, function)) | ||||
| } | ||||
|  | ||||
| func EvalBoolCondition(condition string, params map[string]interface{}) (bool, error) { | ||||
| 	evaluables.mutex.Lock() | ||||
| 	evaluable, ok := evaluables.mapping[condition] | ||||
| 	evaluables.mutex.Unlock() | ||||
| 	if !ok { | ||||
| 		newcond := | ||||
| 			strings.ReplaceAll( | ||||
| 				strings.ReplaceAll( | ||||
| 					condition, "'", "\""), "%", "\\") | ||||
| 		var err error | ||||
| 		evaluable, err = language.NewEvaluable(newcond) | ||||
| 		if err != nil { | ||||
| 			return false, err | ||||
| 		} | ||||
| 		evaluables.mutex.Lock() | ||||
| 		evaluables.mapping[condition] = evaluable | ||||
| 		evaluables.mutex.Unlock() | ||||
| 	} | ||||
| 	value, err := evaluable.EvalBool(context.Background(), params) | ||||
| 	return value, err | ||||
| } | ||||
|  | ||||
| func EvalFloat64Condition(condition string, params map[string]interface{}) (float64, error) { | ||||
| 	evaluables.mutex.Lock() | ||||
| 	evaluable, ok := evaluables.mapping[condition] | ||||
| 	evaluables.mutex.Unlock() | ||||
| 	if !ok { | ||||
| 		newcond := | ||||
| 			strings.ReplaceAll( | ||||
| 				strings.ReplaceAll( | ||||
| 					condition, "'", "\""), "%", "\\") | ||||
| 		var err error | ||||
| 		evaluable, err = language.NewEvaluable(newcond) | ||||
| 		if err != nil { | ||||
| 			return math.NaN(), err | ||||
| 		} | ||||
| 		evaluables.mutex.Lock() | ||||
| 		evaluables.mapping[condition] = evaluable | ||||
| 		evaluables.mutex.Unlock() | ||||
| 	} | ||||
| 	value, err := evaluable.EvalFloat64(context.Background(), params) | ||||
| 	return value, err | ||||
| } | ||||
|  | ||||
| func NewAggregator(output chan lp.CCMetric) (MetricAggregator, error) { | ||||
| 	a := new(metricAggregator) | ||||
| 	err := a.Init(output) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return a, err | ||||
| } | ||||
							
								
								
									
										376
									
								
								internal/metricAggregator/metricAggregatorFunctions.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										376
									
								
								internal/metricAggregator/metricAggregatorFunctions.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,376 @@ | ||||
| package metricAggregator | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"regexp" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	topo "github.com/ClusterCockpit/cc-metric-collector/internal/ccTopology" | ||||
| ) | ||||
|  | ||||
| /* | ||||
|  * Arithmetic functions on value arrays | ||||
|  */ | ||||
|  | ||||
| // Sum up values | ||||
| func sumfunc(args ...interface{}) (interface{}, error) { | ||||
| 	s := 0.0 | ||||
| 	values, ok := args[0].([]float64) | ||||
| 	if ok { | ||||
| 		cclog.ComponentDebug("MetricCache", "SUM FUNC START") | ||||
| 		for _, x := range values { | ||||
| 			s += x | ||||
| 		} | ||||
| 		cclog.ComponentDebug("MetricCache", "SUM FUNC END", s) | ||||
| 	} else { | ||||
| 		cclog.ComponentDebug("MetricCache", "SUM FUNC CAST FAILED") | ||||
| 	} | ||||
| 	return s, nil | ||||
| } | ||||
|  | ||||
| // Get the minimum value | ||||
| func minfunc(args ...interface{}) (interface{}, error) { | ||||
| 	var err error = nil | ||||
| 	switch values := args[0].(type) { | ||||
| 	case []float64: | ||||
| 		var s float64 = math.MaxFloat64 | ||||
| 		for _, x := range values { | ||||
| 			if x < s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 		return s, nil | ||||
| 	case []float32: | ||||
| 		var s float32 = math.MaxFloat32 | ||||
| 		for _, x := range values { | ||||
| 			if x < s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 		return s, nil | ||||
| 	case []int: | ||||
| 		var s int = int(math.MaxInt32) | ||||
| 		for _, x := range values { | ||||
| 			if x < s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 		return s, nil | ||||
| 	case []int64: | ||||
| 		var s int64 = math.MaxInt64 | ||||
| 		for _, x := range values { | ||||
| 			if x < s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 		return s, nil | ||||
| 	case []int32: | ||||
| 		var s int32 = math.MaxInt32 | ||||
| 		for _, x := range values { | ||||
| 			if x < s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 		return s, nil | ||||
| 	default: | ||||
| 		err = errors.New("function 'min' only on list of values (float64, float32, int, int32, int64)") | ||||
| 	} | ||||
|  | ||||
| 	return 0.0, err | ||||
| } | ||||
|  | ||||
| // Get the average or mean value | ||||
| func avgfunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch values := args[0].(type) { | ||||
| 	case []float64: | ||||
| 		var s float64 = 0 | ||||
| 		for _, x := range values { | ||||
| 			s += x | ||||
| 		} | ||||
| 		return s / float64(len(values)), nil | ||||
| 	case []float32: | ||||
| 		var s float32 = 0 | ||||
| 		for _, x := range values { | ||||
| 			s += x | ||||
| 		} | ||||
| 		return s / float32(len(values)), nil | ||||
| 	case []int: | ||||
| 		var s int = 0 | ||||
| 		for _, x := range values { | ||||
| 			s += x | ||||
| 		} | ||||
| 		return s / len(values), nil | ||||
| 	case []int64: | ||||
| 		var s int64 = 0 | ||||
| 		for _, x := range values { | ||||
| 			s += x | ||||
| 		} | ||||
| 		return s / int64(len(values)), nil | ||||
| 	} | ||||
| 	return 0.0, nil | ||||
| } | ||||
|  | ||||
| // Get the maximum value | ||||
| func maxfunc(args ...interface{}) (interface{}, error) { | ||||
| 	s := 0.0 | ||||
| 	values, ok := args[0].([]float64) | ||||
| 	if ok { | ||||
| 		for _, x := range values { | ||||
| 			if x > s { | ||||
| 				s = x | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return s, nil | ||||
| } | ||||
|  | ||||
| // Get the median value | ||||
| func medianfunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch values := args[0].(type) { | ||||
| 	case []float64: | ||||
| 		sort.Float64s(values) | ||||
| 		return values[len(values)/2], nil | ||||
| 	// case []float32: | ||||
| 	// 	sort.Float64s(values) | ||||
| 	// 	return values[len(values)/2], nil | ||||
| 	case []int: | ||||
| 		sort.Ints(values) | ||||
| 		return values[len(values)/2], nil | ||||
|  | ||||
| 		// case []int64: | ||||
| 		// 	sort.Ints(values) | ||||
| 		// 	return values[len(values)/2], nil | ||||
| 		// case []int32: | ||||
| 		// 	sort.Ints(values) | ||||
| 		// 	return values[len(values)/2], nil | ||||
| 	} | ||||
| 	return 0.0, errors.New("function 'median()' only on lists of type float64 and int") | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Get number of values in list. Returns always an int | ||||
|  */ | ||||
|  | ||||
| func lenfunc(args ...interface{}) (interface{}, error) { | ||||
| 	var err error = nil | ||||
| 	var length int = 0 | ||||
| 	switch values := args[0].(type) { | ||||
| 	case []float64: | ||||
| 		length = len(values) | ||||
| 	case []float32: | ||||
| 		length = len(values) | ||||
| 	case []int: | ||||
| 		length = len(values) | ||||
| 	case []int64: | ||||
| 		length = len(values) | ||||
| 	case []int32: | ||||
| 		length = len(values) | ||||
| 	case float64: | ||||
| 		err = errors.New("function 'len' can only be applied on arrays and strings") | ||||
| 	case float32: | ||||
| 		err = errors.New("function 'len' can only be applied on arrays and strings") | ||||
| 	case int: | ||||
| 		err = errors.New("function 'len' can only be applied on arrays and strings") | ||||
| 	case int64: | ||||
| 		err = errors.New("function 'len' can only be applied on arrays and strings") | ||||
| 	case string: | ||||
| 		length = len(values) | ||||
| 	} | ||||
| 	return length, err | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Check if a values is in a list | ||||
|  * In constrast to most of the other functions, this one is an infix operator for | ||||
|  * - substring matching: `"abc" in "abcdef"` -> true | ||||
|  * - substring matching with int casting: `3 in "abd3"` -> true | ||||
|  * - search for an int in an int list: `3 in getCpuList()` -> true (if you have more than 4 CPU hardware threads) | ||||
|  */ | ||||
|  | ||||
| func infunc(a interface{}, b interface{}) (interface{}, error) { | ||||
| 	switch match := a.(type) { | ||||
| 	case string: | ||||
| 		switch total := b.(type) { | ||||
| 		case string: | ||||
| 			return strings.Contains(total, match), nil | ||||
| 		} | ||||
| 	case int: | ||||
| 		switch total := b.(type) { | ||||
| 		case []int: | ||||
| 			for _, x := range total { | ||||
| 				if x == match { | ||||
| 					return true, nil | ||||
| 				} | ||||
| 			} | ||||
| 		case string: | ||||
| 			smatch := fmt.Sprintf("%d", match) | ||||
| 			return strings.Contains(total, smatch), nil | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
| 	return false, nil | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Regex matching of strings (metric name, tag keys, tag values, meta keys, meta values) | ||||
|  * Since we cannot use \ inside JSON strings without escaping, we use % instead for the | ||||
|  * format keys \d = %d, \w = %d, ... Not sure how to fix this | ||||
|  */ | ||||
|  | ||||
| func matchfunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch match := args[0].(type) { | ||||
| 	case string: | ||||
| 		switch total := args[1].(type) { | ||||
| 		case string: | ||||
| 			smatch := strings.Replace(match, "%", "\\", -1) | ||||
| 			regex, err := regexp.Compile(smatch) | ||||
| 			if err != nil { | ||||
| 				return false, err | ||||
| 			} | ||||
| 			s := regex.Find([]byte(total)) | ||||
| 			return s != nil, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return false, nil | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * System topology getter functions | ||||
|  */ | ||||
|  | ||||
| // for a given cpuid, it returns the core id | ||||
| func getCpuCoreFunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch cpuid := args[0].(type) { | ||||
| 	case int: | ||||
| 		return topo.GetCpuCore(cpuid), nil | ||||
| 	} | ||||
| 	return -1, errors.New("function 'getCpuCore' accepts only an 'int' cpuid") | ||||
| } | ||||
|  | ||||
| // for a given cpuid, it returns the socket id | ||||
| func getCpuSocketFunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch cpuid := args[0].(type) { | ||||
| 	case int: | ||||
| 		return topo.GetCpuSocket(cpuid), nil | ||||
| 	} | ||||
| 	return -1, errors.New("function 'getCpuCore' accepts only an 'int' cpuid") | ||||
| } | ||||
|  | ||||
| // for a given cpuid, it returns the id of the NUMA node | ||||
| func getCpuNumaDomainFunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch cpuid := args[0].(type) { | ||||
| 	case int: | ||||
| 		return topo.GetCpuNumaDomain(cpuid), nil | ||||
| 	} | ||||
| 	return -1, errors.New("function 'getCpuNuma' accepts only an 'int' cpuid") | ||||
| } | ||||
|  | ||||
| // for a given cpuid, it returns the id of the CPU die | ||||
| func getCpuDieFunc(args ...interface{}) (interface{}, error) { | ||||
| 	switch cpuid := args[0].(type) { | ||||
| 	case int: | ||||
| 		return topo.GetCpuDie(cpuid), nil | ||||
| 	} | ||||
| 	return -1, errors.New("function 'getCpuDie' accepts only an 'int' cpuid") | ||||
| } | ||||
|  | ||||
| // for a given core id, it returns the list of cpuids | ||||
| func getCpuListOfCoreFunc(args ...interface{}) (interface{}, error) { | ||||
| 	cpulist := make([]int, 0) | ||||
| 	switch in := args[0].(type) { | ||||
| 	case int: | ||||
| 		for _, c := range topo.CpuData() { | ||||
| 			if c.Core == in { | ||||
| 				cpulist = append(cpulist, c.Cpuid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist, nil | ||||
| } | ||||
|  | ||||
| // for a given socket id, it returns the list of cpuids | ||||
| func getCpuListOfSocketFunc(args ...interface{}) (interface{}, error) { | ||||
| 	cpulist := make([]int, 0) | ||||
| 	switch in := args[0].(type) { | ||||
| 	case int: | ||||
| 		for _, c := range topo.CpuData() { | ||||
| 			if c.Socket == in { | ||||
| 				cpulist = append(cpulist, c.Cpuid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist, nil | ||||
| } | ||||
|  | ||||
| // for a given id of a NUMA domain, it returns the list of cpuids | ||||
| func getCpuListOfNumaDomainFunc(args ...interface{}) (interface{}, error) { | ||||
| 	cpulist := make([]int, 0) | ||||
| 	switch in := args[0].(type) { | ||||
| 	case int: | ||||
| 		for _, c := range topo.CpuData() { | ||||
| 			if c.Numadomain == in { | ||||
| 				cpulist = append(cpulist, c.Cpuid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist, nil | ||||
| } | ||||
|  | ||||
| // for a given CPU die id, it returns the list of cpuids | ||||
| func getCpuListOfDieFunc(args ...interface{}) (interface{}, error) { | ||||
| 	cpulist := make([]int, 0) | ||||
| 	switch in := args[0].(type) { | ||||
| 	case int: | ||||
| 		for _, c := range topo.CpuData() { | ||||
| 			if c.Die == in { | ||||
| 				cpulist = append(cpulist, c.Cpuid) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist, nil | ||||
| } | ||||
|  | ||||
| // wrapper function to get a list of all cpuids of the node | ||||
| func getCpuListOfNode(args ...interface{}) (interface{}, error) { | ||||
| 	return topo.CpuList(), nil | ||||
| } | ||||
|  | ||||
| // helper function to get the cpuid list for a CCMetric type tag set (type and type-id) | ||||
| // since there is no access to the metric data in the function, is should be called like | ||||
| // `getCpuListOfType()` | ||||
| func getCpuListOfType(args ...interface{}) (interface{}, error) { | ||||
| 	cpulist := make([]int, 0) | ||||
| 	switch typ := args[0].(type) { | ||||
| 	case string: | ||||
| 		switch typ { | ||||
| 		case "node": | ||||
| 			return topo.CpuList(), nil | ||||
| 		case "socket": | ||||
| 			return getCpuListOfSocketFunc(args[1]) | ||||
| 		case "numadomain": | ||||
| 			return getCpuListOfNumaDomainFunc(args[1]) | ||||
| 		case "core": | ||||
| 			return getCpuListOfCoreFunc(args[1]) | ||||
| 		case "cpu": | ||||
| 			var cpu int | ||||
|  | ||||
| 			switch id := args[1].(type) { | ||||
| 			case string: | ||||
| 				_, err := fmt.Scanf(id, "%d", &cpu) | ||||
| 				if err == nil { | ||||
| 					cpulist = append(cpulist, cpu) | ||||
| 				} | ||||
| 			case int: | ||||
| 				cpulist = append(cpulist, id) | ||||
| 			case int64: | ||||
| 				cpulist = append(cpulist, int(id)) | ||||
| 			} | ||||
|  | ||||
| 		} | ||||
| 	} | ||||
| 	return cpulist, errors.New("no valid args type and type-id") | ||||
| } | ||||
							
								
								
									
										217
									
								
								internal/metricRouter/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								internal/metricRouter/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,217 @@ | ||||
| # CC Metric Router | ||||
|  | ||||
| The CCMetric router sits in between the collectors and the sinks and can be used to add and remove tags to/from traversing [CCMetrics](../ccMetric/README.md). | ||||
|  | ||||
| # Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "num_cache_intervals" : 1, | ||||
|     "interval_timestamp" : true, | ||||
|     "add_tags" : [ | ||||
|         { | ||||
|             "key" : "cluster", | ||||
|             "value" : "testcluster", | ||||
|             "if" : "*" | ||||
|         }, | ||||
|         { | ||||
|             "key" : "test", | ||||
|             "value" : "testing", | ||||
|             "if" : "name == 'temp_package_id_0'" | ||||
|         } | ||||
|     ], | ||||
|     "delete_tags" : [ | ||||
|         { | ||||
|             "key" : "unit", | ||||
|             "value" : "*", | ||||
|             "if" : "*" | ||||
|         } | ||||
|     ], | ||||
|     "interval_aggregates" : [ | ||||
|         { | ||||
|             "name" : "temp_cores_avg", | ||||
|             "if" : "match('temp_core_%d+', metric.Name())", | ||||
|             "function" : "avg(values)", | ||||
|             "tags" : { | ||||
|                 "type" : "node" | ||||
|             }, | ||||
|             "meta" : { | ||||
|                 "group": "IPMI", | ||||
|                 "unit": "degC", | ||||
|                 "source": "TempCollector" | ||||
|             } | ||||
|         } | ||||
|     ], | ||||
|     "drop_metrics" : [ | ||||
|         "not_interesting_metric_at_all" | ||||
|     ], | ||||
|     "drop_metrics_if" : [ | ||||
|         "match('temp_core_%d+', metric.Name())" | ||||
|     ], | ||||
|     "rename_metrics" : { | ||||
|         "metric_12345" : "mymetric" | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| There are three main options `add_tags`, `delete_tags` and `interval_timestamp`. `add_tags` and `delete_tags` are lists consisting of dicts with `key`, `value` and `if`. The `value` can be omitted in the `delete_tags` part as it only uses the `key` for removal. The `interval_timestamp` setting means that a unique timestamp is applied to all metrics traversing the router during an interval. | ||||
| # The `interval_timestamp` option | ||||
|  | ||||
| The collectors' `Read()` functions are not called simultaneously and therefore the metrics gathered in an interval can have different timestamps. If you want to avoid that and have a common timestamp (the beginning of the interval), set this option to `true` and the MetricRouter sets the time. | ||||
|  | ||||
| # The `num_cache_intervals` option | ||||
|  | ||||
| If the MetricRouter should buffer metrics of intervals in a MetricCache, this option specifies the number of past intervals that should be kept. If `num_cache_intervals = 0`, the cache is disabled. With `num_cache_intervals = 1`, only the metrics of the last interval are buffered. | ||||
|  | ||||
| A `num_cache_intervals > 0` is required to use the `interval_aggregates` option. | ||||
|  | ||||
| # The `rename_metrics` option | ||||
|  | ||||
| In the ClusterCockpit world we specified a set of standard metrics. Since some collectors determine the metric names based on files, execuables and libraries, they might change from system to system (or installation to installtion, OS to OS, ...). In order to get the common names, you can rename incoming metrics before sending them to the sink. If the metric name matches the `oldname`, it is changed to `newname` | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "oldname" : "newname", | ||||
|   "clock_mhz" : "clock" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| # Conditional manipulation of tags (`add_tags` and `del_tags`) | ||||
|  | ||||
| Common config format: | ||||
| ```json | ||||
| { | ||||
|     "key" : "test", | ||||
|     "value" : "testing", | ||||
|     "if" : "name == 'temp_package_id_0'" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## The `del_tags` option | ||||
|  | ||||
| The collectors are free to add whatever `key=value` pair to the metric tags (although the usage of tags should be minimized). If you want to delete a tag afterwards, you can do that. When the `if` condition matches on a metric, the `key` is removed from the metric's tags. | ||||
|  | ||||
| If you want to remove a tag for all metrics, use the condition wildcard `*`. The `value` field can be omitted in the `del_tags` case. | ||||
|  | ||||
| Never delete tags: | ||||
| - `hostname` | ||||
| - `type` | ||||
| - `type-id` | ||||
|  | ||||
| ## The `add_tags` option | ||||
|  | ||||
| In some cases, metrics should be tagged or an existing tag changed based on some condition. This can be done in the `add_tags` section. When the `if` condition evaluates to `true`, the tag `key` is added or gets changed to the new `value`. | ||||
|  | ||||
| If the CCMetric name is equal to `temp_package_id_0`, it adds an additional tag `test=testing` to the metric. | ||||
|  | ||||
| For this metric, a more useful example would be: | ||||
|  | ||||
| ```json | ||||
| [ | ||||
|   { | ||||
|     "key" : "type", | ||||
|     "value" : "socket", | ||||
|     "if" : "name == 'temp_package_id_0'" | ||||
|   }, | ||||
|   { | ||||
|     "key" : "type-id", | ||||
|     "value" : "0", | ||||
|     "if" : "name == 'temp_package_id_0'" | ||||
|   }, | ||||
| ] | ||||
| ``` | ||||
|  | ||||
| The metric `temp_package_id_0` corresponds to the tempature of the first CPU socket (=package). With the above configuration, the tags would reflect that because commonly the [TempCollector](../../collectors/tempMetric.md) submits only `node` metrics. | ||||
|  | ||||
| In order to match all metrics, you can use `*`, so in order to add a flag per default. This is useful to attached system-specific tags like `cluster=testcluster`: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "key" : "cluster", | ||||
|     "value" : "testcluster", | ||||
|     "if" : "*" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| # Dropping metrics | ||||
|  | ||||
| In some cases, you want to drop a metric and don't get it forwarded to the sinks. There are two options based on the required specification: | ||||
| - Based only on the metric name -> `drop_metrics` section | ||||
| - An evaluable condition with more overhead -> `drop_metrics_if` section | ||||
|  | ||||
| ## The `drop_metrics` section | ||||
|  | ||||
| The argument is a list of metric names. No futher checks are performed, only a comparison of the metric name | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "drop_metrics" : [ | ||||
|       "drop_metric_1", | ||||
|       "drop_metric_2" | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The example drops all metrics with the name `drop_metric_1` and `drop_metric_2`. | ||||
|  | ||||
| ## The `drop_metrics_if` section | ||||
|  | ||||
| This option takes a list of evaluable conditions and performs them one after the other on **all** metrics incoming from the collectors and the metric cache (aka `interval_aggregates`). | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "drop_metrics_if" : [ | ||||
|       "match('drop_metric_%d+', name)", | ||||
|       "match('cpu', type) && type-id == 0" | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| The first line is comparable with the example in `drop_metrics`, it drops all metrics starting with `drop_metric_` and ending with a number. The second line drops all metrics of the first hardware thread (**not** recommended) | ||||
|  | ||||
|  | ||||
| # Aggregate metric values of the current interval with the `interval_aggregates` option | ||||
|  | ||||
| **Note:** `interval_aggregates` works only if `num_cache_intervals` > 0 | ||||
|  | ||||
| In some cases, you need to derive new metrics based on the metrics arriving during an interval. This can be done in the `interval_aggregates` section. The logic is similar to the other metric manipulation and filtering options. A cache stores all metrics that arrive during an interval. At the beginning of the *next* interval, the list of metrics is submitted to the MetricAggregator. It derives new metrics and submits them back to the MetricRouter, so they are sent in the next interval but have the timestamp of the previous interval beginning. | ||||
|  | ||||
| ```json | ||||
| "interval_aggregates" : [ | ||||
|   { | ||||
|     "name" : "new_metric_name", | ||||
|     "if" : "match('sub_metric_%d+', metric.Name())", | ||||
|     "function" : "avg(values)", | ||||
|     "tags" : { | ||||
|       "key" : "value", | ||||
|       "type" : "node" | ||||
|     }, | ||||
|     "meta" : { | ||||
|       "key" : "value", | ||||
|       "group": "IPMI", | ||||
|       "unit": "<copy>", | ||||
|     } | ||||
|   } | ||||
| ] | ||||
| ``` | ||||
|  | ||||
| The above configuration, collects all metric values for metrics evaluating `if` to `true`. Afterwards it calculates the average `avg` of the `values` (list of all metrics' field `value`) and creates a new CCMetric with the name `new_metric_name` and adds the tags in `tags` and the meta information in `meta`. The special value `<copy>` searches the input metrics and copies the value of the first match of `key` to the new CCMetric. | ||||
|  | ||||
| If you are not interested in the input metrics `sub_metric_%d+` at all, you can add the same condition used here to the `drop_metrics_if` section to drop them. | ||||
|  | ||||
| Use cases for `interval_aggregates`: | ||||
| - Combine multiple metrics of the a collector to a new one like the [MemstatCollector](../../collectors/memstatMetric.md) does it for `mem_used`)): | ||||
| ```json | ||||
|   { | ||||
|     "name" : "mem_used", | ||||
|     "if" : "source == 'MemstatCollector'", | ||||
|     "function" : "sum(mem_total) - (sum(mem_free) + sum(mem_buffers) + sum(mem_cached))", | ||||
|     "tags" : { | ||||
|       "type" : "node" | ||||
|     }, | ||||
|     "meta" : { | ||||
|       "group": "<copy>", | ||||
|       "unit": "<copy>", | ||||
|       "source": "<copy>" | ||||
|     } | ||||
|   } | ||||
| ``` | ||||
							
								
								
									
										192
									
								
								internal/metricRouter/metricCache.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								internal/metricRouter/metricCache.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | ||||
| package metricRouter | ||||
|  | ||||
| import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	agg "github.com/ClusterCockpit/cc-metric-collector/internal/metricAggregator" | ||||
| 	mct "github.com/ClusterCockpit/cc-metric-collector/internal/multiChanTicker" | ||||
| ) | ||||
|  | ||||
| type metricCachePeriod struct { | ||||
| 	startstamp  time.Time | ||||
| 	stopstamp   time.Time | ||||
| 	numMetrics  int | ||||
| 	sizeMetrics int | ||||
| 	metrics     []lp.CCMetric | ||||
| } | ||||
|  | ||||
| // Metric cache data structure | ||||
| type metricCache struct { | ||||
| 	numPeriods int | ||||
| 	curPeriod  int | ||||
| 	lock       sync.Mutex | ||||
| 	intervals  []*metricCachePeriod | ||||
| 	wg         *sync.WaitGroup | ||||
| 	ticker     mct.MultiChanTicker | ||||
| 	tickchan   chan time.Time | ||||
| 	done       chan bool | ||||
| 	output     chan lp.CCMetric | ||||
| 	aggEngine  agg.MetricAggregator | ||||
| } | ||||
|  | ||||
| type MetricCache interface { | ||||
| 	Init(output chan lp.CCMetric, ticker mct.MultiChanTicker, wg *sync.WaitGroup, numPeriods int) error | ||||
| 	Start() | ||||
| 	Add(metric lp.CCMetric) | ||||
| 	GetPeriod(index int) (time.Time, time.Time, []lp.CCMetric) | ||||
| 	AddAggregation(name, function, condition string, tags, meta map[string]string) error | ||||
| 	DeleteAggregation(name string) error | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| func (c *metricCache) Init(output chan lp.CCMetric, ticker mct.MultiChanTicker, wg *sync.WaitGroup, numPeriods int) error { | ||||
| 	var err error = nil | ||||
| 	c.done = make(chan bool) | ||||
| 	c.wg = wg | ||||
| 	c.ticker = ticker | ||||
| 	c.numPeriods = numPeriods | ||||
| 	c.output = output | ||||
| 	c.intervals = make([]*metricCachePeriod, 0) | ||||
| 	for i := 0; i < c.numPeriods+1; i++ { | ||||
| 		p := new(metricCachePeriod) | ||||
| 		p.numMetrics = 0 | ||||
| 		p.sizeMetrics = 0 | ||||
| 		p.metrics = make([]lp.CCMetric, 0) | ||||
| 		c.intervals = append(c.intervals, p) | ||||
| 	} | ||||
|  | ||||
| 	// Create a new aggregation engine. No separate goroutine at the moment | ||||
| 	// The code is executed by the MetricCache goroutine | ||||
| 	c.aggEngine, err = agg.NewAggregator(c.output) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("MetricCache", "Cannot create aggregator") | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Start starts the metric cache | ||||
| func (c *metricCache) Start() { | ||||
|  | ||||
| 	c.tickchan = make(chan time.Time) | ||||
| 	c.ticker.AddChannel(c.tickchan) | ||||
| 	// Router cache is done | ||||
| 	done := func() { | ||||
| 		cclog.ComponentDebug("MetricCache", "DONE") | ||||
| 		close(c.done) | ||||
| 	} | ||||
|  | ||||
| 	// Rotate cache interval | ||||
| 	rotate := func(timestamp time.Time) int { | ||||
| 		oldPeriod := c.curPeriod | ||||
| 		c.curPeriod = oldPeriod + 1 | ||||
| 		if c.curPeriod >= c.numPeriods { | ||||
| 			c.curPeriod = 0 | ||||
| 		} | ||||
| 		c.intervals[oldPeriod].numMetrics = 0 | ||||
| 		c.intervals[oldPeriod].stopstamp = timestamp | ||||
| 		c.intervals[c.curPeriod].startstamp = timestamp | ||||
| 		c.intervals[c.curPeriod].stopstamp = timestamp | ||||
| 		return oldPeriod | ||||
| 	} | ||||
|  | ||||
| 	c.wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer c.wg.Done() | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-c.done: | ||||
| 				done() | ||||
| 				return | ||||
| 			case tick := <-c.tickchan: | ||||
| 				c.lock.Lock() | ||||
| 				old := rotate(tick) | ||||
| 				// Get the last period and evaluate aggregation metrics | ||||
| 				starttime, endtime, metrics := c.GetPeriod(old) | ||||
| 				c.lock.Unlock() | ||||
| 				if len(metrics) > 0 { | ||||
| 					c.aggEngine.Eval(starttime, endtime, metrics) | ||||
| 				} else { | ||||
| 					// This message is also printed in the first interval after startup | ||||
| 					cclog.ComponentDebug("MetricCache", "EMPTY INTERVAL?") | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 	cclog.ComponentDebug("MetricCache", "START") | ||||
| } | ||||
|  | ||||
| // Add a metric to the cache. The interval is defined by the global timer (rotate() in Start()) | ||||
| // The intervals list is used as round-robin buffer and the metric list grows dynamically and | ||||
| // to avoid reallocations | ||||
| func (c *metricCache) Add(metric lp.CCMetric) { | ||||
| 	if c.curPeriod >= 0 && c.curPeriod < c.numPeriods { | ||||
| 		c.lock.Lock() | ||||
| 		p := c.intervals[c.curPeriod] | ||||
| 		if p.numMetrics < p.sizeMetrics { | ||||
| 			p.metrics[p.numMetrics] = metric | ||||
| 			p.numMetrics = p.numMetrics + 1 | ||||
| 			p.stopstamp = metric.Time() | ||||
| 		} else { | ||||
| 			p.metrics = append(p.metrics, metric) | ||||
| 			p.numMetrics = p.numMetrics + 1 | ||||
| 			p.sizeMetrics = p.sizeMetrics + 1 | ||||
| 			p.stopstamp = metric.Time() | ||||
| 		} | ||||
| 		c.lock.Unlock() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *metricCache) AddAggregation(name, function, condition string, tags, meta map[string]string) error { | ||||
| 	return c.aggEngine.AddAggregation(name, function, condition, tags, meta) | ||||
| } | ||||
|  | ||||
| func (c *metricCache) DeleteAggregation(name string) error { | ||||
| 	return c.aggEngine.DeleteAggregation(name) | ||||
| } | ||||
|  | ||||
| // Get all metrics of a interval. The index is the difference to the current interval, so index=0 | ||||
| // is the current one, index=1 the last interval and so on. Returns and empty array if a wrong index | ||||
| // is given (negative index, index larger than configured number of total intervals, ...) | ||||
| func (c *metricCache) GetPeriod(index int) (time.Time, time.Time, []lp.CCMetric) { | ||||
| 	var start time.Time = time.Now() | ||||
| 	var stop time.Time = time.Now() | ||||
| 	var metrics []lp.CCMetric | ||||
| 	if index >= 0 && index < c.numPeriods { | ||||
| 		pindex := c.curPeriod - index | ||||
| 		if pindex < 0 { | ||||
| 			pindex = c.numPeriods - pindex | ||||
| 		} | ||||
| 		if pindex >= 0 && pindex < c.numPeriods { | ||||
| 			start = c.intervals[pindex].startstamp | ||||
| 			stop = c.intervals[pindex].stopstamp | ||||
| 			metrics = c.intervals[pindex].metrics | ||||
| 			//return c.intervals[pindex].startstamp, c.intervals[pindex].stopstamp, c.intervals[pindex].metrics | ||||
| 		} else { | ||||
| 			metrics = make([]lp.CCMetric, 0) | ||||
| 		} | ||||
| 	} else { | ||||
| 		metrics = make([]lp.CCMetric, 0) | ||||
| 	} | ||||
| 	return start, stop, metrics | ||||
| } | ||||
|  | ||||
| // Close finishes / stops the metric cache | ||||
| func (c *metricCache) Close() { | ||||
| 	cclog.ComponentDebug("MetricCache", "CLOSE") | ||||
| 	c.done <- true | ||||
| } | ||||
|  | ||||
| func NewCache(output chan lp.CCMetric, ticker mct.MultiChanTicker, wg *sync.WaitGroup, numPeriods int) (MetricCache, error) { | ||||
| 	c := new(metricCache) | ||||
| 	err := c.Init(output, ticker, wg, numPeriods) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return c, err | ||||
| } | ||||
							
								
								
									
										379
									
								
								internal/metricRouter/metricRouter.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								internal/metricRouter/metricRouter.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,379 @@ | ||||
| package metricRouter | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	agg "github.com/ClusterCockpit/cc-metric-collector/internal/metricAggregator" | ||||
| 	mct "github.com/ClusterCockpit/cc-metric-collector/internal/multiChanTicker" | ||||
| ) | ||||
|  | ||||
| const ROUTER_MAX_FORWARD = 50 | ||||
|  | ||||
| // Metric router tag configuration | ||||
| type metricRouterTagConfig struct { | ||||
| 	Key       string `json:"key"`   // Tag name | ||||
| 	Value     string `json:"value"` // Tag value | ||||
| 	Condition string `json:"if"`    // Condition for adding or removing corresponding tag | ||||
| } | ||||
|  | ||||
| // Metric router configuration | ||||
| type metricRouterConfig struct { | ||||
| 	AddTags           []metricRouterTagConfig              `json:"add_tags"`            // List of tags that are added when the condition is met | ||||
| 	DelTags           []metricRouterTagConfig              `json:"delete_tags"`         // List of tags that are removed when the condition is met | ||||
| 	IntervalAgg       []agg.MetricAggregatorIntervalConfig `json:"interval_aggregates"` // List of aggregation function processed at the end of an interval | ||||
| 	DropMetrics       []string                             `json:"drop_metrics"`        // List of metric names to drop. For fine-grained dropping use drop_metrics_if | ||||
| 	DropMetricsIf     []string                             `json:"drop_metrics_if"`     // List of evaluatable terms to drop metrics | ||||
| 	RenameMetrics     map[string]string                    `json:"rename_metrics"`      // Map to rename metric name from key to value | ||||
| 	IntervalStamp     bool                                 `json:"interval_timestamp"`  // Update timestamp periodically by ticker each interval? | ||||
| 	NumCacheIntervals int                                  `json:"num_cache_intervals"` // Number of intervals of cached metrics for evaluation | ||||
| 	dropMetrics       map[string]bool                      // Internal map for O(1) lookup | ||||
| } | ||||
|  | ||||
| // Metric router data structure | ||||
| type metricRouter struct { | ||||
| 	hostname    string              // Hostname used in tags | ||||
| 	coll_input  chan lp.CCMetric    // Input channel from CollectorManager | ||||
| 	recv_input  chan lp.CCMetric    // Input channel from ReceiveManager | ||||
| 	cache_input chan lp.CCMetric    // Input channel from MetricCache | ||||
| 	outputs     []chan lp.CCMetric  // List of all output channels | ||||
| 	done        chan bool           // channel to finish / stop metric router | ||||
| 	wg          *sync.WaitGroup     // wait group for all goroutines in cc-metric-collector | ||||
| 	timestamp   time.Time           // timestamp periodically updated by ticker each interval | ||||
| 	timerdone   chan bool           // channel to finish / stop timestamp updater | ||||
| 	ticker      mct.MultiChanTicker // periodically ticking once each interval | ||||
| 	config      metricRouterConfig  // json encoded config for metric router | ||||
| 	cache       MetricCache         // pointer to MetricCache | ||||
| 	cachewg     sync.WaitGroup      // wait group for MetricCache | ||||
| 	maxForward  int                 // number of metrics to forward maximally in one iteration | ||||
| } | ||||
|  | ||||
| // MetricRouter access functions | ||||
| type MetricRouter interface { | ||||
| 	Init(ticker mct.MultiChanTicker, wg *sync.WaitGroup, routerConfigFile string) error | ||||
| 	AddCollectorInput(input chan lp.CCMetric) | ||||
| 	AddReceiverInput(input chan lp.CCMetric) | ||||
| 	AddOutput(output chan lp.CCMetric) | ||||
| 	Start() | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| // Init initializes a metric router by setting up: | ||||
| // * input and output channels | ||||
| // * done channel | ||||
| // * wait group synchronization (from variable wg) | ||||
| // * ticker (from variable ticker) | ||||
| // * configuration (read from config file in variable routerConfigFile) | ||||
| func (r *metricRouter) Init(ticker mct.MultiChanTicker, wg *sync.WaitGroup, routerConfigFile string) error { | ||||
| 	r.outputs = make([]chan lp.CCMetric, 0) | ||||
| 	r.done = make(chan bool) | ||||
| 	r.cache_input = make(chan lp.CCMetric) | ||||
| 	r.wg = wg | ||||
| 	r.ticker = ticker | ||||
| 	r.maxForward = ROUTER_MAX_FORWARD | ||||
|  | ||||
| 	// Set hostname | ||||
| 	hostname, err := os.Hostname() | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	// Drop domain part of host name | ||||
| 	r.hostname = strings.SplitN(hostname, `.`, 2)[0] | ||||
|  | ||||
| 	// Read metric router config file | ||||
| 	configFile, err := os.Open(routerConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("MetricRouter", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer configFile.Close() | ||||
| 	jsonParser := json.NewDecoder(configFile) | ||||
| 	err = jsonParser.Decode(&r.config) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("MetricRouter", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	if r.config.NumCacheIntervals > 0 { | ||||
| 		r.cache, err = NewCache(r.cache_input, r.ticker, &r.cachewg, r.config.NumCacheIntervals) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError("MetricRouter", "MetricCache initialization failed:", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 		for _, agg := range r.config.IntervalAgg { | ||||
| 			r.cache.AddAggregation(agg.Name, agg.Function, agg.Condition, agg.Tags, agg.Meta) | ||||
| 		} | ||||
| 	} | ||||
| 	r.config.dropMetrics = make(map[string]bool) | ||||
| 	for _, mname := range r.config.DropMetrics { | ||||
| 		r.config.dropMetrics[mname] = true | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // StartTimer starts a timer which updates timestamp periodically | ||||
| func (r *metricRouter) StartTimer() { | ||||
| 	m := make(chan time.Time) | ||||
| 	r.ticker.AddChannel(m) | ||||
| 	r.timerdone = make(chan bool) | ||||
|  | ||||
| 	r.wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer r.wg.Done() | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-r.timerdone: | ||||
| 				close(r.timerdone) | ||||
| 				cclog.ComponentDebug("MetricRouter", "TIMER DONE") | ||||
| 				return | ||||
| 			case t := <-m: | ||||
| 				r.timestamp = t | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 	cclog.ComponentDebug("MetricRouter", "TIMER START") | ||||
| } | ||||
|  | ||||
| func getParamMap(point lp.CCMetric) map[string]interface{} { | ||||
| 	params := make(map[string]interface{}) | ||||
| 	params["metric"] = point | ||||
| 	params["name"] = point.Name() | ||||
| 	for key, value := range point.Tags() { | ||||
| 		params[key] = value | ||||
| 	} | ||||
| 	for key, value := range point.Meta() { | ||||
| 		params[key] = value | ||||
| 	} | ||||
| 	for key, value := range point.Fields() { | ||||
| 		params[key] = value | ||||
| 	} | ||||
| 	params["timestamp"] = point.Time() | ||||
| 	return params | ||||
| } | ||||
|  | ||||
| // DoAddTags adds a tag when condition is fullfiled | ||||
| func (r *metricRouter) DoAddTags(point lp.CCMetric) { | ||||
| 	var conditionMatches bool | ||||
| 	for _, m := range r.config.AddTags { | ||||
| 		if m.Condition == "*" { | ||||
| 			// Condition is always matched | ||||
| 			conditionMatches = true | ||||
| 		} else { | ||||
| 			// Evaluate condition | ||||
| 			var err error | ||||
| 			conditionMatches, err = agg.EvalBoolCondition(m.Condition, getParamMap(point)) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError("MetricRouter", err.Error()) | ||||
| 				conditionMatches = false | ||||
| 			} | ||||
| 		} | ||||
| 		if conditionMatches { | ||||
| 			point.AddTag(m.Key, m.Value) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // DoDelTags removes a tag when condition is fullfiled | ||||
| func (r *metricRouter) DoDelTags(point lp.CCMetric) { | ||||
| 	var conditionMatches bool | ||||
| 	for _, m := range r.config.DelTags { | ||||
| 		if m.Condition == "*" { | ||||
| 			// Condition is always matched | ||||
| 			conditionMatches = true | ||||
| 		} else { | ||||
| 			// Evaluate condition | ||||
| 			var err error | ||||
| 			conditionMatches, err = agg.EvalBoolCondition(m.Condition, getParamMap(point)) | ||||
| 			if err != nil { | ||||
| 				cclog.ComponentError("MetricRouter", err.Error()) | ||||
| 				conditionMatches = false | ||||
| 			} | ||||
| 		} | ||||
| 		if conditionMatches { | ||||
| 			point.RemoveTag(m.Key) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Conditional test whether a metric should be dropped | ||||
| func (r *metricRouter) dropMetric(point lp.CCMetric) bool { | ||||
| 	// Simple drop check | ||||
| 	if conditionMatches, ok := r.config.dropMetrics[point.Name()]; ok { | ||||
| 		return conditionMatches | ||||
| 	} | ||||
|  | ||||
| 	// Checking the dropping conditions | ||||
| 	for _, m := range r.config.DropMetricsIf { | ||||
| 		conditionMatches, err := agg.EvalBoolCondition(m, getParamMap(point)) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError("MetricRouter", err.Error()) | ||||
| 			conditionMatches = false | ||||
| 		} | ||||
| 		if conditionMatches { | ||||
| 			return conditionMatches | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// No dropping condition met | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // Start starts the metric router | ||||
| func (r *metricRouter) Start() { | ||||
| 	// start timer if configured | ||||
| 	r.timestamp = time.Now() | ||||
| 	if r.config.IntervalStamp { | ||||
| 		r.StartTimer() | ||||
| 	} | ||||
|  | ||||
| 	// Router manager is done | ||||
| 	done := func() { | ||||
| 		close(r.done) | ||||
| 		cclog.ComponentDebug("MetricRouter", "DONE") | ||||
| 	} | ||||
|  | ||||
| 	// Forward takes a received metric, adds or deletes tags | ||||
| 	// and forwards it to the output channels | ||||
| 	forward := func(point lp.CCMetric) { | ||||
| 		cclog.ComponentDebug("MetricRouter", "FORWARD", point) | ||||
| 		r.DoAddTags(point) | ||||
| 		r.DoDelTags(point) | ||||
| 		if new, ok := r.config.RenameMetrics[point.Name()]; ok { | ||||
| 			point.SetName(new) | ||||
| 		} | ||||
| 		r.DoAddTags(point) | ||||
| 		r.DoDelTags(point) | ||||
|  | ||||
| 		for _, o := range r.outputs { | ||||
| 			o <- point | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Foward message received from collector channel | ||||
| 	coll_forward := func(p lp.CCMetric) { | ||||
| 		// receive from metric collector | ||||
| 		p.AddTag("hostname", r.hostname) | ||||
| 		if r.config.IntervalStamp { | ||||
| 			p.SetTime(r.timestamp) | ||||
| 		} | ||||
| 		if !r.dropMetric(p) { | ||||
| 			forward(p) | ||||
| 		} | ||||
| 		// even if the metric is dropped, it is stored in the cache for | ||||
| 		// aggregations | ||||
| 		if r.config.NumCacheIntervals > 0 { | ||||
| 			r.cache.Add(p) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Forward message received from receivers channel | ||||
| 	recv_forward := func(p lp.CCMetric) { | ||||
| 		// receive from receive manager | ||||
| 		if r.config.IntervalStamp { | ||||
| 			p.SetTime(r.timestamp) | ||||
| 		} | ||||
| 		if !r.dropMetric(p) { | ||||
| 			forward(p) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Forward message received from cache channel | ||||
| 	cache_forward := func(p lp.CCMetric) { | ||||
| 		// receive from metric collector | ||||
| 		if !r.dropMetric(p) { | ||||
| 			p.AddTag("hostname", r.hostname) | ||||
| 			forward(p) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Start Metric Cache | ||||
| 	if r.config.NumCacheIntervals > 0 { | ||||
| 		r.cache.Start() | ||||
| 	} | ||||
|  | ||||
| 	r.wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer r.wg.Done() | ||||
|  | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-r.done: | ||||
| 				done() | ||||
| 				return | ||||
|  | ||||
| 			case p := <-r.coll_input: | ||||
| 				coll_forward(p) | ||||
| 				for i := 0; len(r.coll_input) > 0 && i < r.maxForward; i++ { | ||||
| 					coll_forward(<-r.coll_input) | ||||
| 				} | ||||
|  | ||||
| 			case p := <-r.recv_input: | ||||
| 				recv_forward(p) | ||||
| 				for i := 0; len(r.recv_input) > 0 && i < r.maxForward; i++ { | ||||
| 					recv_forward(<-r.recv_input) | ||||
| 				} | ||||
|  | ||||
| 			case p := <-r.cache_input: | ||||
| 				cache_forward(p) | ||||
| 				for i := 0; len(r.cache_input) > 0 && i < r.maxForward; i++ { | ||||
| 					cache_forward(<-r.cache_input) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 	cclog.ComponentDebug("MetricRouter", "STARTED") | ||||
| } | ||||
|  | ||||
| // AddCollectorInput adds a channel between metric collector and metric router | ||||
| func (r *metricRouter) AddCollectorInput(input chan lp.CCMetric) { | ||||
| 	r.coll_input = input | ||||
| } | ||||
|  | ||||
| // AddReceiverInput adds a channel between metric receiver and metric router | ||||
| func (r *metricRouter) AddReceiverInput(input chan lp.CCMetric) { | ||||
| 	r.recv_input = input | ||||
| } | ||||
|  | ||||
| // AddOutput adds a output channel to the metric router | ||||
| func (r *metricRouter) AddOutput(output chan lp.CCMetric) { | ||||
| 	r.outputs = append(r.outputs, output) | ||||
| } | ||||
|  | ||||
| // Close finishes / stops the metric router | ||||
| func (r *metricRouter) Close() { | ||||
| 	cclog.ComponentDebug("MetricRouter", "CLOSE") | ||||
| 	r.done <- true | ||||
| 	// wait for close of channel r.done | ||||
| 	<-r.done | ||||
|  | ||||
| 	// stop timer | ||||
| 	if r.config.IntervalStamp { | ||||
| 		cclog.ComponentDebug("MetricRouter", "TIMER CLOSE") | ||||
| 		r.timerdone <- true | ||||
| 		// wait for close of channel r.timerdone | ||||
| 		<-r.timerdone | ||||
| 	} | ||||
|  | ||||
| 	// stop metric cache | ||||
| 	if r.config.NumCacheIntervals > 0 { | ||||
| 		cclog.ComponentDebug("MetricRouter", "CACHE CLOSE") | ||||
| 		r.cache.Close() | ||||
| 		r.cachewg.Wait() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // New creates a new initialized metric router | ||||
| func New(ticker mct.MultiChanTicker, wg *sync.WaitGroup, routerConfigFile string) (MetricRouter, error) { | ||||
| 	r := new(metricRouter) | ||||
| 	err := r.Init(ticker, wg, routerConfigFile) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return r, err | ||||
| } | ||||
							
								
								
									
										37
									
								
								internal/multiChanTicker/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								internal/multiChanTicker/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| # MultiChanTicker | ||||
|  | ||||
| The idea of this ticker is to multiply the output channels. The original Golang `time.Ticker` provides only a single output channel, so the signal can only be received by a single other class. This ticker allows to add multiple channels which get all notified about the time tick. | ||||
|  | ||||
| ```golang | ||||
| type MultiChanTicker interface { | ||||
| 	Init(duration time.Duration) | ||||
| 	AddChannel(chan time.Time) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The MultiChanTicker is created similarly to the common `time.Ticker`: | ||||
|  | ||||
| ```golang | ||||
| NewTicker(duration time.Duration) MultiChanTicker | ||||
| ``` | ||||
|  | ||||
| Afterwards, you can add channels: | ||||
|  | ||||
| ```golang | ||||
| t := MultiChanTicker(duration) | ||||
| c1 := make(chan time.Time) | ||||
| c2 := make(chan time.Time) | ||||
| t.AddChannel(c1) | ||||
| t.AddChannel(c2) | ||||
|  | ||||
| for { | ||||
|     select { | ||||
|     case t1 := <- c1: | ||||
|         log.Print(t1) | ||||
|     case t2 := <- c2: | ||||
|         log.Print(t2) | ||||
|     } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The result should be the same `time.Time` output in both channels, notified "simultaneously". | ||||
							
								
								
									
										64
									
								
								internal/multiChanTicker/multiChanTicker.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								internal/multiChanTicker/multiChanTicker.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| package multiChanTicker | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| ) | ||||
|  | ||||
| type multiChanTicker struct { | ||||
| 	ticker   *time.Ticker | ||||
| 	channels []chan time.Time | ||||
| 	done     chan bool | ||||
| } | ||||
|  | ||||
| type MultiChanTicker interface { | ||||
| 	Init(duration time.Duration) | ||||
| 	AddChannel(chan time.Time) | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| func (t *multiChanTicker) Init(duration time.Duration) { | ||||
| 	t.ticker = time.NewTicker(duration) | ||||
| 	t.done = make(chan bool) | ||||
| 	go func() { | ||||
| 		done := func() { | ||||
| 			close(t.done) | ||||
| 			cclog.ComponentDebug("MultiChanTicker", "DONE") | ||||
| 		} | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-t.done: | ||||
| 				done() | ||||
| 				return | ||||
| 			case ts := <-t.ticker.C: | ||||
| 				cclog.ComponentDebug("MultiChanTicker", "Tick", ts) | ||||
| 				for _, c := range t.channels { | ||||
| 					select { | ||||
| 					case <-t.done: | ||||
| 						done() | ||||
| 						return | ||||
| 					case c <- ts: | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| } | ||||
|  | ||||
| func (t *multiChanTicker) AddChannel(channel chan time.Time) { | ||||
| 	t.channels = append(t.channels, channel) | ||||
| } | ||||
|  | ||||
| func (t *multiChanTicker) Close() { | ||||
| 	cclog.ComponentDebug("MultiChanTicker", "CLOSE") | ||||
| 	t.done <- true | ||||
| 	// wait for close of channel t.done | ||||
| 	<-t.done | ||||
| } | ||||
|  | ||||
| func NewTicker(duration time.Duration) MultiChanTicker { | ||||
| 	t := &multiChanTicker{} | ||||
| 	t.Init(duration) | ||||
| 	return t | ||||
| } | ||||
| @@ -3,318 +3,286 @@ package main | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"flag" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| 	"syscall" | ||||
|  | ||||
| 	"github.com/ClusterCockpit/cc-metric-collector/collectors" | ||||
| 	"github.com/ClusterCockpit/cc-metric-collector/receivers" | ||||
| 	"github.com/ClusterCockpit/cc-metric-collector/sinks" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
|  | ||||
| 	//	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	mr "github.com/ClusterCockpit/cc-metric-collector/internal/metricRouter" | ||||
| 	mct "github.com/ClusterCockpit/cc-metric-collector/internal/multiChanTicker" | ||||
| ) | ||||
|  | ||||
| // List of provided collectors. Which collector should be run can be | ||||
| // configured at 'collectors' list  in 'config.json'. | ||||
| var Collectors = map[string]collectors.MetricGetter{ | ||||
| 	"likwid":          &collectors.LikwidCollector{}, | ||||
| 	"loadavg":         &collectors.LoadavgCollector{}, | ||||
| 	"memstat":         &collectors.MemstatCollector{}, | ||||
| 	"netstat":         &collectors.NetstatCollector{}, | ||||
| 	"ibstat":          &collectors.InfinibandCollector{}, | ||||
| 	"lustrestat":      &collectors.LustreCollector{}, | ||||
| 	"cpustat":         &collectors.CpustatCollector{}, | ||||
| 	"topprocs":        &collectors.TopProcsCollector{}, | ||||
| 	"nvidia":          &collectors.NvidiaCollector{}, | ||||
| 	"customcmd":       &collectors.CustomCmdCollector{}, | ||||
| 	"diskstat":        &collectors.DiskstatCollector{}, | ||||
| 	"tempstat":        &collectors.TempCollector{}, | ||||
| 	"ipmistat":        &collectors.IpmiCollector{}, | ||||
| 	"gpfs":            new(collectors.GpfsCollector), | ||||
| 	"cpufreq":         new(collectors.CPUFreqCollector), | ||||
| 	"cpufreq_cpuinfo": new(collectors.CPUFreqCpuInfoCollector), | ||||
| 	"numastats":       new(collectors.NUMAStatsCollector), | ||||
| type CentralConfigFile struct { | ||||
| 	Interval            int    `json:"interval"` | ||||
| 	Duration            int    `json:"duration"` | ||||
| 	CollectorConfigFile string `json:"collectors"` | ||||
| 	RouterConfigFile    string `json:"router"` | ||||
| 	SinkConfigFile      string `json:"sinks"` | ||||
| 	ReceiverConfigFile  string `json:"receivers,omitempty"` | ||||
| } | ||||
|  | ||||
| var Sinks = map[string]sinks.SinkFuncs{ | ||||
| 	"influxdb": &sinks.InfluxSink{}, | ||||
| 	"stdout":   &sinks.StdoutSink{}, | ||||
| 	"nats":     &sinks.NatsSink{}, | ||||
| 	"http":     &sinks.HttpSink{}, | ||||
| } | ||||
|  | ||||
| var Receivers = map[string]receivers.ReceiverFuncs{ | ||||
| 	"nats": &receivers.NatsReceiver{}, | ||||
| } | ||||
|  | ||||
| // Structure of the configuration file | ||||
| type GlobalConfig struct { | ||||
| 	Sink           sinks.SinkConfig           `json:"sink"` | ||||
| 	Interval       int                        `json:"interval"` | ||||
| 	Duration       int                        `json:"duration"` | ||||
| 	Collectors     []string                   `json:"collectors"` | ||||
| 	Receiver       receivers.ReceiverConfig   `json:"receiver"` | ||||
| 	DefTags        map[string]string          `json:"default_tags"` | ||||
| 	CollectConfigs map[string]json.RawMessage `json:"collect_config"` | ||||
| } | ||||
|  | ||||
| // Load JSON configuration file | ||||
| func LoadConfiguration(file string, config *GlobalConfig) error { | ||||
| func LoadCentralConfiguration(file string, config *CentralConfigFile) error { | ||||
| 	configFile, err := os.Open(file) | ||||
| 	defer configFile.Close() | ||||
| 	if err != nil { | ||||
| 		fmt.Println(err.Error()) | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer configFile.Close() | ||||
| 	jsonParser := json.NewDecoder(configFile) | ||||
| 	err = jsonParser.Decode(config) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| type RuntimeConfig struct { | ||||
| 	Interval   time.Duration | ||||
| 	Duration   time.Duration | ||||
| 	CliArgs    map[string]string | ||||
| 	ConfigFile CentralConfigFile | ||||
|  | ||||
| 	MetricRouter    mr.MetricRouter | ||||
| 	CollectManager  collectors.CollectorManager | ||||
| 	SinkManager     sinks.SinkManager | ||||
| 	ReceiveManager  receivers.ReceiveManager | ||||
| 	MultiChanTicker mct.MultiChanTicker | ||||
|  | ||||
| 	Channels []chan lp.CCMetric | ||||
| 	Sync     sync.WaitGroup | ||||
| } | ||||
|  | ||||
| //// Structure of the configuration file | ||||
| //type GlobalConfig struct { | ||||
| //	Sink           sinks.SinkConfig           `json:"sink"` | ||||
| //	Interval       int                        `json:"interval"` | ||||
| //	Duration       int                        `json:"duration"` | ||||
| //	Collectors     []string                   `json:"collectors"` | ||||
| //	Receiver       receivers.ReceiverConfig   `json:"receiver"` | ||||
| //	DefTags        map[string]string          `json:"default_tags"` | ||||
| //	CollectConfigs map[string]json.RawMessage `json:"collect_config"` | ||||
| //} | ||||
|  | ||||
| //// Load JSON configuration file | ||||
| //func LoadConfiguration(file string, config *GlobalConfig) error { | ||||
| //	configFile, err := os.Open(file) | ||||
| //	defer configFile.Close() | ||||
| //	if err != nil { | ||||
| //		fmt.Println(err.Error()) | ||||
| //		return err | ||||
| //	} | ||||
| //	jsonParser := json.NewDecoder(configFile) | ||||
| //	err = jsonParser.Decode(config) | ||||
| //	return err | ||||
| //} | ||||
|  | ||||
| func ReadCli() map[string]string { | ||||
| 	var m map[string]string | ||||
| 	cfg := flag.String("config", "./config.json", "Path to configuration file") | ||||
| 	logfile := flag.String("log", "stderr", "Path for logfile") | ||||
| 	pidfile := flag.String("pidfile", "/var/run/cc-metric-collector.pid", "Path for PID file") | ||||
| 	once := flag.Bool("once", false, "Run all collectors only once") | ||||
| 	debug := flag.Bool("debug", false, "Activate debug output") | ||||
| 	flag.Parse() | ||||
| 	m = make(map[string]string) | ||||
| 	m["configfile"] = *cfg | ||||
| 	m["logfile"] = *logfile | ||||
| 	m["pidfile"] = *pidfile | ||||
| 	if *once { | ||||
| 		m["once"] = "true" | ||||
| 	} else { | ||||
| 		m["once"] = "false" | ||||
| 	} | ||||
| 	if *debug { | ||||
| 		m["debug"] = "true" | ||||
| 		cclog.SetDebug() | ||||
| 	} else { | ||||
| 		m["debug"] = "false" | ||||
| 	} | ||||
| 	return m | ||||
| } | ||||
|  | ||||
| func SetLogging(logfile string) error { | ||||
| 	var file *os.File | ||||
| //func SetLogging(logfile string) error { | ||||
| //	var file *os.File | ||||
| //	var err error | ||||
| //	if logfile != "stderr" { | ||||
| //		file, err = os.OpenFile(logfile, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0600) | ||||
| //		if err != nil { | ||||
| //			log.Fatal(err) | ||||
| //			return err | ||||
| //		} | ||||
| //	} else { | ||||
| //		file = os.Stderr | ||||
| //	} | ||||
| //	log.SetOutput(file) | ||||
| //	return nil | ||||
| //} | ||||
|  | ||||
| // General shutdownHandler function that gets executed in case of interrupt or graceful shutdownHandler | ||||
| func shutdownHandler(config *RuntimeConfig, shutdownSignal chan os.Signal) { | ||||
| 	defer config.Sync.Done() | ||||
|  | ||||
| 	<-shutdownSignal | ||||
| 	// Remove shutdown handler | ||||
| 	// every additional interrupt signal will stop without cleaning up | ||||
| 	signal.Stop(shutdownSignal) | ||||
|  | ||||
| 	cclog.Info("Shutdown...") | ||||
|  | ||||
| 	cclog.Debug("Shutdown Ticker...") | ||||
| 	config.MultiChanTicker.Close() | ||||
|  | ||||
| 	if config.CollectManager != nil { | ||||
| 		cclog.Debug("Shutdown CollectManager...") | ||||
| 		config.CollectManager.Close() | ||||
| 	} | ||||
| 	if config.ReceiveManager != nil { | ||||
| 		cclog.Debug("Shutdown ReceiveManager...") | ||||
| 		config.ReceiveManager.Close() | ||||
| 	} | ||||
| 	if config.MetricRouter != nil { | ||||
| 		cclog.Debug("Shutdown Router...") | ||||
| 		config.MetricRouter.Close() | ||||
| 	} | ||||
| 	if config.SinkManager != nil { | ||||
| 		cclog.Debug("Shutdown SinkManager...") | ||||
| 		config.SinkManager.Close() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func mainFunc() int { | ||||
| 	var err error | ||||
| 	if logfile != "stderr" { | ||||
| 		file, err = os.OpenFile(logfile, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0600) | ||||
| 		if err != nil { | ||||
| 			log.Fatal(err) | ||||
| 			return err | ||||
| 		} | ||||
| 	} else { | ||||
| 		file = os.Stderr | ||||
| 	} | ||||
| 	log.SetOutput(file) | ||||
| 	return nil | ||||
| } | ||||
| 	use_recv := false | ||||
|  | ||||
| func CreatePidfile(pidfile string) error { | ||||
| 	file, err := os.OpenFile(pidfile, os.O_CREATE|os.O_RDWR, 0600) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return err | ||||
| 	} | ||||
| 	file.Write([]byte(fmt.Sprintf("%d", os.Getpid()))) | ||||
| 	file.Close() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func RemovePidfile(pidfile string) error { | ||||
| 	info, err := os.Stat(pidfile) | ||||
| 	if !os.IsNotExist(err) && !info.IsDir() { | ||||
| 		os.Remove(pidfile) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // General shutdown function that gets executed in case of interrupt or graceful shutdown | ||||
| func shutdown(wg *sync.WaitGroup, collectors []string, sink sinks.SinkFuncs, recv receivers.ReceiverFuncs, pidfile string) { | ||||
| 	log.Print("Shutdown...") | ||||
| 	for _, c := range collectors { | ||||
| 		col := Collectors[c] | ||||
| 		log.Print("Stop ", col.Name()) | ||||
| 		col.Close() | ||||
| 	} | ||||
| 	time.Sleep(1 * time.Second) | ||||
| 	if recv != nil { | ||||
| 		recv.Close() | ||||
| 	} | ||||
| 	sink.Close() | ||||
| 	RemovePidfile(pidfile) | ||||
| 	wg.Done() | ||||
| } | ||||
|  | ||||
| // Register an interrupt handler for Ctrl+C and similar. At signal, | ||||
| // all collectors are closed | ||||
| func prepare_shutdown(wg *sync.WaitGroup, config *GlobalConfig, sink sinks.SinkFuncs, recv receivers.ReceiverFuncs, pidfile string) { | ||||
| 	sigs := make(chan os.Signal, 1) | ||||
| 	signal.Notify(sigs, os.Interrupt) | ||||
|  | ||||
| 	go func(wg *sync.WaitGroup) { | ||||
| 		<-sigs | ||||
| 		log.Print("Shutdown...") | ||||
| 		shutdown(wg, config.Collectors, sink, recv, pidfile) | ||||
| 	}(wg) | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| 	var config GlobalConfig | ||||
| 	var wg sync.WaitGroup | ||||
| 	var recv receivers.ReceiverFuncs = nil | ||||
| 	var use_recv bool | ||||
| 	use_recv = false | ||||
| 	wg.Add(1) | ||||
| 	host, err := os.Hostname() | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return | ||||
| 	} | ||||
| 	// Drop domain part of host name | ||||
| 	host = strings.SplitN(host, `.`, 2)[0] | ||||
| 	clicfg := ReadCli() | ||||
| 	err = CreatePidfile(clicfg["pidfile"]) | ||||
| 	err = SetLogging(clicfg["logfile"]) | ||||
| 	if err != nil { | ||||
| 		log.Print("Error setting up logging system to ", clicfg["logfile"], " on ", host) | ||||
| 		return | ||||
| 	// Initialize runtime configuration | ||||
| 	rcfg := RuntimeConfig{ | ||||
| 		MetricRouter:   nil, | ||||
| 		CollectManager: nil, | ||||
| 		SinkManager:    nil, | ||||
| 		ReceiveManager: nil, | ||||
| 		CliArgs:        ReadCli(), | ||||
| 	} | ||||
|  | ||||
| 	// Load and check configuration | ||||
| 	err = LoadConfiguration(clicfg["configfile"], &config) | ||||
| 	err = LoadCentralConfiguration(rcfg.CliArgs["configfile"], &rcfg.ConfigFile) | ||||
| 	if err != nil { | ||||
| 		log.Print("Error reading configuration file ", clicfg["configfile"]) | ||||
| 		log.Print(err.Error()) | ||||
| 		return | ||||
| 		cclog.Error("Error reading configuration file ", rcfg.CliArgs["configfile"], ": ", err.Error()) | ||||
| 		return 1 | ||||
| 	} | ||||
| 	if config.Interval <= 0 || time.Duration(config.Interval)*time.Second <= 0 { | ||||
| 		log.Print("Configuration value 'interval' must be greater than zero") | ||||
| 		return | ||||
| 	if rcfg.ConfigFile.Interval <= 0 || time.Duration(rcfg.ConfigFile.Interval)*time.Second <= 0 { | ||||
| 		cclog.Error("Configuration value 'interval' must be greater than zero") | ||||
| 		return 1 | ||||
| 	} | ||||
| 	if config.Duration <= 0 { | ||||
| 		log.Print("Configuration value 'duration' must be greater than zero") | ||||
| 		return | ||||
| 	rcfg.Interval = time.Duration(rcfg.ConfigFile.Interval) * time.Second | ||||
| 	if rcfg.ConfigFile.Duration <= 0 || time.Duration(rcfg.ConfigFile.Duration)*time.Second <= 0 { | ||||
| 		cclog.Error("Configuration value 'duration' must be greater than zero") | ||||
| 		return 1 | ||||
| 	} | ||||
| 	if len(config.Collectors) == 0 { | ||||
| 		var keys []string | ||||
| 		for k := range Collectors { | ||||
| 			keys = append(keys, k) | ||||
| 		} | ||||
| 		log.Print("Configuration value 'collectors' does not contain any collector. Available: ", strings.Join(keys, ", ")) | ||||
| 		return | ||||
| 	} | ||||
| 	for _, name := range config.Collectors { | ||||
| 		if _, found := Collectors[name]; !found { | ||||
| 			log.Print("Invalid collector '", name, "' in configuration") | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	if _, found := Sinks[config.Sink.Type]; !found { | ||||
| 		log.Print("Invalid sink type '", config.Sink.Type, "' in configuration") | ||||
| 		return | ||||
| 	} | ||||
| 	// Setup sink | ||||
| 	sink := Sinks[config.Sink.Type] | ||||
| 	err = sink.Init(config.Sink) | ||||
| 	if err != nil { | ||||
| 		log.Print(err) | ||||
| 		return | ||||
| 	} | ||||
| 	// Setup receiver | ||||
| 	if len(config.Receiver.Type) > 0 && config.Receiver.Type != "none" { | ||||
| 		if _, found := Receivers[config.Receiver.Type]; !found { | ||||
| 			log.Print("Invalid receiver type '", config.Receiver.Type, "' in configuration") | ||||
| 			return | ||||
| 		} else { | ||||
| 			recv = Receivers[config.Receiver.Type] | ||||
| 			err = recv.Init(config.Receiver, sink) | ||||
| 			if err == nil { | ||||
| 				use_recv = true | ||||
| 			} else { | ||||
| 				log.Print(err) | ||||
| 			} | ||||
| 		} | ||||
| 	rcfg.Duration = time.Duration(rcfg.ConfigFile.Duration) * time.Second | ||||
|  | ||||
| 	if len(rcfg.ConfigFile.RouterConfigFile) == 0 { | ||||
| 		cclog.Error("Metric router configuration file must be set") | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Register interrupt handler | ||||
| 	prepare_shutdown(&wg, &config, sink, recv, clicfg["pidfile"]) | ||||
| 	if len(rcfg.ConfigFile.SinkConfigFile) == 0 { | ||||
| 		cclog.Error("Sink configuration file must be set") | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Initialize all collectors | ||||
| 	tmp := make([]string, 0) | ||||
| 	for _, c := range config.Collectors { | ||||
| 		col := Collectors[c] | ||||
| 		conf, found := config.CollectConfigs[c] | ||||
| 		if !found { | ||||
| 			conf = json.RawMessage("") | ||||
| 		} | ||||
| 		err = col.Init([]byte(conf)) | ||||
| 	if len(rcfg.ConfigFile.CollectorConfigFile) == 0 { | ||||
| 		cclog.Error("Metric collector configuration file must be set") | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Set log file | ||||
| 	if logfile := rcfg.CliArgs["logfile"]; logfile != "stderr" { | ||||
| 		cclog.SetOutput(logfile) | ||||
| 	} | ||||
|  | ||||
| 	// Creat new multi channel ticker | ||||
| 	rcfg.MultiChanTicker = mct.NewTicker(rcfg.Interval) | ||||
|  | ||||
| 	// Create new metric router | ||||
| 	rcfg.MetricRouter, err = mr.New(rcfg.MultiChanTicker, &rcfg.Sync, rcfg.ConfigFile.RouterConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Create new sink | ||||
| 	rcfg.SinkManager, err = sinks.New(&rcfg.Sync, rcfg.ConfigFile.SinkConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Connect metric router to sink manager | ||||
| 	RouterToSinksChannel := make(chan lp.CCMetric, 200) | ||||
| 	rcfg.SinkManager.AddInput(RouterToSinksChannel) | ||||
| 	rcfg.MetricRouter.AddOutput(RouterToSinksChannel) | ||||
|  | ||||
| 	// Create new collector manager | ||||
| 	rcfg.CollectManager, err = collectors.New(rcfg.MultiChanTicker, rcfg.Duration, &rcfg.Sync, rcfg.ConfigFile.CollectorConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.Error(err.Error()) | ||||
| 		return 1 | ||||
| 	} | ||||
|  | ||||
| 	// Connect collector manager to metric router | ||||
| 	CollectToRouterChannel := make(chan lp.CCMetric, 200) | ||||
| 	rcfg.CollectManager.AddOutput(CollectToRouterChannel) | ||||
| 	rcfg.MetricRouter.AddCollectorInput(CollectToRouterChannel) | ||||
|  | ||||
| 	// Create new receive manager | ||||
| 	if len(rcfg.ConfigFile.ReceiverConfigFile) > 0 { | ||||
| 		rcfg.ReceiveManager, err = receivers.New(&rcfg.Sync, rcfg.ConfigFile.ReceiverConfigFile) | ||||
| 		if err != nil { | ||||
| 			log.Print("SKIP ", col.Name(), " (", err.Error(), ")") | ||||
| 		} else if !col.Initialized() { | ||||
| 			log.Print("SKIP ", col.Name(), " (Not initialized)") | ||||
| 		} else { | ||||
| 			log.Print("Start ", col.Name()) | ||||
| 			tmp = append(tmp, c) | ||||
| 			cclog.Error(err.Error()) | ||||
| 			return 1 | ||||
| 		} | ||||
|  | ||||
| 		// Connect receive manager to metric router | ||||
| 		ReceiveToRouterChannel := make(chan lp.CCMetric, 200) | ||||
| 		rcfg.ReceiveManager.AddOutput(ReceiveToRouterChannel) | ||||
| 		rcfg.MetricRouter.AddReceiverInput(ReceiveToRouterChannel) | ||||
| 		use_recv = true | ||||
| 	} | ||||
| 	config.Collectors = tmp | ||||
| 	config.DefTags["hostname"] = host | ||||
|  | ||||
| 	// Setup up ticker loop | ||||
| 	if clicfg["once"] != "true" { | ||||
| 		log.Print("Running loop every ", time.Duration(config.Interval)*time.Second) | ||||
| 	} else { | ||||
| 		log.Print("Running loop only once") | ||||
| 	} | ||||
| 	ticker := time.NewTicker(time.Duration(config.Interval) * time.Second) | ||||
| 	done := make(chan bool) | ||||
| 	// Create shutdown handler | ||||
| 	shutdownSignal := make(chan os.Signal, 1) | ||||
| 	signal.Notify(shutdownSignal, os.Interrupt) | ||||
| 	signal.Notify(shutdownSignal, syscall.SIGTERM) | ||||
| 	rcfg.Sync.Add(1) | ||||
| 	go shutdownHandler(&rcfg, shutdownSignal) | ||||
|  | ||||
| 	// Storage for all node metrics | ||||
| 	tmpPoints := make([]lp.MutableMetric, 0) | ||||
| 	// Start the managers | ||||
| 	rcfg.MetricRouter.Start() | ||||
| 	rcfg.SinkManager.Start() | ||||
| 	rcfg.CollectManager.Start() | ||||
|  | ||||
| 	// Start receiver | ||||
| 	if use_recv { | ||||
| 		recv.Start() | ||||
| 		rcfg.ReceiveManager.Start() | ||||
| 	} | ||||
|  | ||||
| 	go func() { | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-done: | ||||
| 				return | ||||
| 			case t := <-ticker.C: | ||||
| 	// Wait until one tick has passed. This is a workaround | ||||
| 	if rcfg.CliArgs["once"] == "true" { | ||||
| 		x := 1.2 * float64(rcfg.ConfigFile.Interval) | ||||
| 		time.Sleep(time.Duration(int(x)) * time.Second) | ||||
| 		shutdownSignal <- os.Interrupt | ||||
| 	} | ||||
|  | ||||
| 				// Read all collectors are sort the results in the right | ||||
| 				// storage locations | ||||
| 				for _, c := range config.Collectors { | ||||
| 					col := Collectors[c] | ||||
| 					col.Read(time.Duration(config.Duration)*time.Second, &tmpPoints) | ||||
| 	// Wait that all goroutines finish | ||||
| 	rcfg.Sync.Wait() | ||||
|  | ||||
| 					for { | ||||
| 						if len(tmpPoints) == 0 { | ||||
| 							break | ||||
| 						} | ||||
| 						p := tmpPoints[0] | ||||
| 						for k, v := range config.DefTags { | ||||
| 							p.AddTag(k, v) | ||||
| 							p.SetTime(t) | ||||
| 						} | ||||
| 						sink.Write(p) | ||||
| 						tmpPoints = tmpPoints[1:] | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if err := sink.Flush(); err != nil { | ||||
| 					log.Printf("sink error: %s\n", err) | ||||
| 				} | ||||
| 				if clicfg["once"] == "true" { | ||||
| 					shutdown(&wg, config.Collectors, sink, recv, clicfg["pidfile"]) | ||||
| 					return | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	// Wait until receiving an interrupt | ||||
| 	wg.Wait() | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| 	exitCode := mainFunc() | ||||
| 	os.Exit(exitCode) | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								receivers.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								receivers.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| { | ||||
|     "natsrecv" : { | ||||
|         "type": "nats", | ||||
|         "address": "nats://my-url", | ||||
|         "port" : "4222", | ||||
|         "database": "testcluster" | ||||
|     } | ||||
| } | ||||
| @@ -1,35 +1,44 @@ | ||||
| This folder contains the receivers for the cc-metric-collector. | ||||
| # CCMetric receivers | ||||
|  | ||||
| # `metricReceiver.go` | ||||
| The base class/configuration is located in `metricReceiver.go`. | ||||
| This folder contains the ReceiveManager and receiver implementations for the cc-metric-collector. | ||||
|  | ||||
| # Receivers | ||||
| * `natsReceiver.go`: Receives metrics from the Nats transport system in Influx line protocol encoding. The database name is used as subscription subject for the NATS messages. It uses https://github.com/nats-io/nats.go | ||||
| # Configuration | ||||
|  | ||||
| # Installation | ||||
| Nothing to do, all receivers are pure Go code | ||||
|  | ||||
| # Receiver configuration | ||||
| The configuration file for the receivers is a list of configurations. The `type` field in each specifies which receiver to initialize. | ||||
|  | ||||
| ```json | ||||
|   "receiver": { | ||||
| [ | ||||
|   { | ||||
|     "type": "nats", | ||||
|     "address": "nats://my-url" | ||||
|     "address": "nats://my-url", | ||||
|     "port" : "4222", | ||||
|     "database": "testcluster" | ||||
|   }, | ||||
|   } | ||||
| ] | ||||
| ``` | ||||
|  | ||||
| ## `nats` | ||||
|  | ||||
| The receiver connects to `address` and `port` and subscribes itself for all messages with topic `database`. The default port is `4222`. | ||||
| ## Type `nats` | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "type": "nats", | ||||
|   "address": "<nats-URI or hostname>", | ||||
|   "port" : "<portnumber>", | ||||
|   "database": "<subscribe topic>" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The `nats` receiver subscribes to the topic `database` and listens on `address` and `port` for metrics in the InfluxDB line protocol. | ||||
|  | ||||
| # Contributing own receivers | ||||
| A receiver contains three functions and is derived from the type `Receiver` (in `metricReceiver.go`): | ||||
| * `Init(config ReceiverConfig) error` | ||||
| * `Start() error` | ||||
| * `Close()` | ||||
| * `Name() string` | ||||
| * `SetSink(sink chan ccMetric.CCMetric)` | ||||
|  | ||||
| The data structures should be set up in `Init()` like opening a file or server connection. The `Start()` function should either start a go routine or issue some other asynchronous mechanism for receiving metrics. The `Close()` function should tear down anything created in `Init()`. | ||||
|  | ||||
| Finally, the receiver needs to be registered in the `metric-collector.go`. There is a list of receivers called `Receivers` which is a map (string -> pointer to receiver). Add a new entry with a descriptive name and the new receiver. | ||||
| Finally, the receiver needs to be registered in the `receiveManager.go`. There is a list of receivers called `AvailableReceivers` which is a map (`receiver_type_string` -> `pointer to Receiver interface`). Add a new entry with a descriptive name and the new receiver. | ||||
|   | ||||
| @@ -2,44 +2,41 @@ package receivers | ||||
|  | ||||
| import ( | ||||
| 	//	"time" | ||||
| 	s "github.com/ClusterCockpit/cc-metric-collector/sinks" | ||||
| 	influx "github.com/influxdata/line-protocol" | ||||
| 	"encoding/json" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type defaultReceiverConfig struct { | ||||
| 	Type string `json:"type"` | ||||
| } | ||||
|  | ||||
| type ReceiverConfig struct { | ||||
| 	Addr     string `json:"address"` | ||||
| 	Port     string `json:"port"` | ||||
| 	Database string `json:"database"` | ||||
| 	Type     string `json:"type"` | ||||
| 	Addr         string `json:"address"` | ||||
| 	Port         string `json:"port"` | ||||
| 	Database     string `json:"database"` | ||||
| 	Organization string `json:"organization,omitempty"` | ||||
| 	Type         string `json:"type"` | ||||
| } | ||||
|  | ||||
| type Receiver struct { | ||||
| 	name         string | ||||
| 	addr         string | ||||
| 	port         string | ||||
| 	database     string | ||||
| 	organization string | ||||
| 	sink         s.SinkFuncs | ||||
| type receiver struct { | ||||
| 	typename string | ||||
| 	name     string | ||||
| 	sink     chan lp.CCMetric | ||||
| } | ||||
|  | ||||
| type ReceiverFuncs interface { | ||||
| 	Init(config ReceiverConfig, sink s.SinkFuncs) error | ||||
| type Receiver interface { | ||||
| 	Init(name string, config json.RawMessage) error | ||||
| 	Start() | ||||
| 	Close() | ||||
| 	Name() string | ||||
| 	SetSink(sink chan lp.CCMetric) | ||||
| } | ||||
|  | ||||
| func Tags2Map(metric influx.Metric) map[string]string { | ||||
| 	tags := make(map[string]string) | ||||
| 	for _, t := range metric.TagList() { | ||||
| 		tags[t.Key] = t.Value | ||||
| 	} | ||||
| 	return tags | ||||
| func (r *receiver) Name() string { | ||||
| 	return r.name | ||||
| } | ||||
|  | ||||
| func Fields2Map(metric influx.Metric) map[string]interface{} { | ||||
| 	fields := make(map[string]interface{}) | ||||
| 	for _, f := range metric.FieldList() { | ||||
| 		fields[f.Key] = f.Value | ||||
| 	} | ||||
| 	return fields | ||||
| func (r *receiver) SetSink(sink chan lp.CCMetric) { | ||||
| 	r.sink = sink | ||||
| } | ||||
|   | ||||
| @@ -1,67 +1,85 @@ | ||||
| package receivers | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	s "github.com/ClusterCockpit/cc-metric-collector/sinks" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| 	"log" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influx "github.com/influxdata/line-protocol" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| ) | ||||
|  | ||||
| type NatsReceiverConfig struct { | ||||
| 	Type    string `json:"type"` | ||||
| 	Addr    string `json:"address"` | ||||
| 	Port    string `json:"port"` | ||||
| 	Subject string `json:"subject"` | ||||
| } | ||||
|  | ||||
| type NatsReceiver struct { | ||||
| 	Receiver | ||||
| 	receiver | ||||
| 	nc      *nats.Conn | ||||
| 	handler *lp.MetricHandler | ||||
| 	parser  *lp.Parser | ||||
| 	handler *influx.MetricHandler | ||||
| 	parser  *influx.Parser | ||||
| 	meta    map[string]string | ||||
| 	config  NatsReceiverConfig | ||||
| } | ||||
|  | ||||
| var DefaultTime = func() time.Time { | ||||
| 	return time.Unix(42, 0) | ||||
| } | ||||
|  | ||||
| func (r *NatsReceiver) Init(config ReceiverConfig, sink s.SinkFuncs) error { | ||||
| 	if len(config.Addr) == 0 || | ||||
| 		len(config.Port) == 0 || | ||||
| 		len(config.Database) == 0 { | ||||
| 		return errors.New("Not all configuration variables set required by NatsReceiver") | ||||
| func (r *NatsReceiver) Init(name string, config json.RawMessage) error { | ||||
| 	r.typename = "NatsReceiver" | ||||
| 	r.name = name | ||||
| 	r.config.Addr = nats.DefaultURL | ||||
| 	r.config.Port = "4222" | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &r.config) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError(r.name, "Error reading config:", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	r.addr = config.Addr | ||||
| 	if len(r.addr) == 0 { | ||||
| 		r.addr = nats.DefaultURL | ||||
| 	if len(r.config.Addr) == 0 || | ||||
| 		len(r.config.Port) == 0 || | ||||
| 		len(r.config.Subject) == 0 { | ||||
| 		return errors.New("not all configuration variables set required by NatsReceiver") | ||||
| 	} | ||||
| 	r.port = config.Port | ||||
| 	if len(r.port) == 0 { | ||||
| 		r.port = "4222" | ||||
| 	} | ||||
| 	log.Print("Init NATS Receiver") | ||||
| 	nc, err := nats.Connect(r.addr) | ||||
| 	r.meta = map[string]string{"source": r.name} | ||||
| 	uri := fmt.Sprintf("%s:%s", r.config.Addr, r.config.Port) | ||||
| 	cclog.ComponentDebug(r.name, "INIT", uri, "Subject", r.config.Subject) | ||||
| 	nc, err := nats.Connect(uri) | ||||
| 	if err == nil { | ||||
| 		r.database = config.Database | ||||
| 		r.sink = sink | ||||
| 		r.nc = nc | ||||
| 	} else { | ||||
| 		log.Print(err) | ||||
| 		r.nc = nil | ||||
| 		return err | ||||
| 	} | ||||
| 	r.handler = lp.NewMetricHandler() | ||||
| 	r.parser = lp.NewParser(r.handler) | ||||
| 	r.handler = influx.NewMetricHandler() | ||||
| 	r.parser = influx.NewParser(r.handler) | ||||
| 	r.parser.SetTimeFunc(DefaultTime) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (r *NatsReceiver) Start() { | ||||
| 	log.Print("Start NATS Receiver") | ||||
| 	r.nc.Subscribe(r.database, r._NatsReceive) | ||||
| 	cclog.ComponentDebug(r.name, "START") | ||||
| 	r.nc.Subscribe(r.config.Subject, r._NatsReceive) | ||||
| } | ||||
|  | ||||
| func (r *NatsReceiver) _NatsReceive(m *nats.Msg) { | ||||
| 	metrics, err := r.parser.Parse(m.Data) | ||||
| 	if err == nil { | ||||
| 		for _, m := range metrics { | ||||
| 			y, err := lp.New(m.Name(), Tags2Map(m), Fields2Map(m), m.Time()) | ||||
| 			if err == nil { | ||||
| 				r.sink.Write(y) | ||||
| 			y := lp.FromInfluxMetric(m) | ||||
| 			for k, v := range r.meta { | ||||
| 				y.AddMeta(k, v) | ||||
| 			} | ||||
| 			if r.sink != nil { | ||||
| 				r.sink <- y | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| @@ -69,7 +87,7 @@ func (r *NatsReceiver) _NatsReceive(m *nats.Msg) { | ||||
|  | ||||
| func (r *NatsReceiver) Close() { | ||||
| 	if r.nc != nil { | ||||
| 		log.Print("Close NATS Receiver") | ||||
| 		cclog.ComponentDebug(r.name, "CLOSE") | ||||
| 		r.nc.Close() | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										113
									
								
								receivers/receiveManager.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								receivers/receiveManager.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| package receivers | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"sync" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| var AvailableReceivers = map[string]Receiver{ | ||||
| 	"nats": &NatsReceiver{}, | ||||
| } | ||||
|  | ||||
| type receiveManager struct { | ||||
| 	inputs []Receiver | ||||
| 	output chan lp.CCMetric | ||||
| 	done   chan bool | ||||
| 	wg     *sync.WaitGroup | ||||
| 	config []json.RawMessage | ||||
| } | ||||
|  | ||||
| type ReceiveManager interface { | ||||
| 	Init(wg *sync.WaitGroup, receiverConfigFile string) error | ||||
| 	AddInput(name string, rawConfig json.RawMessage) error | ||||
| 	AddOutput(output chan lp.CCMetric) | ||||
| 	Start() | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| func (rm *receiveManager) Init(wg *sync.WaitGroup, receiverConfigFile string) error { | ||||
| 	rm.inputs = make([]Receiver, 0) | ||||
| 	rm.output = nil | ||||
| 	rm.done = make(chan bool) | ||||
| 	rm.wg = wg | ||||
| 	rm.config = make([]json.RawMessage, 0) | ||||
| 	configFile, err := os.Open(receiverConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("ReceiveManager", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer configFile.Close() | ||||
| 	jsonParser := json.NewDecoder(configFile) | ||||
| 	var rawConfigs map[string]json.RawMessage | ||||
| 	err = jsonParser.Decode(&rawConfigs) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("ReceiveManager", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	for name, raw := range rawConfigs { | ||||
| 		rm.AddInput(name, raw) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (rm *receiveManager) Start() { | ||||
| 	rm.wg.Add(1) | ||||
|  | ||||
| 	for _, r := range rm.inputs { | ||||
| 		cclog.ComponentDebug("ReceiveManager", "START", r.Name()) | ||||
| 		r.Start() | ||||
| 	} | ||||
| 	cclog.ComponentDebug("ReceiveManager", "STARTED") | ||||
| } | ||||
|  | ||||
| func (rm *receiveManager) AddInput(name string, rawConfig json.RawMessage) error { | ||||
| 	var config defaultReceiverConfig | ||||
| 	err := json.Unmarshal(rawConfig, &config) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("ReceiveManager", "SKIP", config.Type, "JSON config error:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	if _, found := AvailableReceivers[config.Type]; !found { | ||||
| 		cclog.ComponentError("ReceiveManager", "SKIP", config.Type, "unknown receiver:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	r := AvailableReceivers[config.Type] | ||||
| 	err = r.Init(name, rawConfig) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("ReceiveManager", "SKIP", r.Name(), "initialization failed:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	rm.inputs = append(rm.inputs, r) | ||||
| 	rm.config = append(rm.config, rawConfig) | ||||
| 	cclog.ComponentDebug("ReceiveManager", "ADD RECEIVER", r.Name()) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (rm *receiveManager) AddOutput(output chan lp.CCMetric) { | ||||
| 	rm.output = output | ||||
| 	for _, r := range rm.inputs { | ||||
| 		r.SetSink(rm.output) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (rm *receiveManager) Close() { | ||||
| 	for _, r := range rm.inputs { | ||||
| 		cclog.ComponentDebug("ReceiveManager", "CLOSE", r.Name()) | ||||
| 		r.Close() | ||||
| 	} | ||||
| 	rm.wg.Done() | ||||
| 	cclog.ComponentDebug("ReceiveManager", "CLOSE") | ||||
| } | ||||
|  | ||||
| func New(wg *sync.WaitGroup, receiverConfigFile string) (ReceiveManager, error) { | ||||
| 	r := &receiveManager{} | ||||
| 	err := r.Init(wg, receiverConfigFile) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return r, err | ||||
| } | ||||
							
								
								
									
										22
									
								
								router.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								router.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| { | ||||
|     "add_tags" : [ | ||||
|         { | ||||
|             "key" : "cluster", | ||||
|             "value" : "testcluster", | ||||
|             "if" : "*" | ||||
|         }, | ||||
|         { | ||||
|             "key" : "test", | ||||
|             "value" : "testing", | ||||
|             "if" : "name == 'temp_package_id_0'" | ||||
|         } | ||||
|     ], | ||||
|     "delete_tags" : [ | ||||
|         { | ||||
|             "key" : "unit", | ||||
|             "value" : "*", | ||||
|             "if" : "*" | ||||
|         } | ||||
|     ], | ||||
|     "interval_timestamp" : true | ||||
| } | ||||
| @@ -15,6 +15,3 @@ CONF_DIR=/etc/cc-metric-collector | ||||
| CONF_FILE=/etc/cc-metric-collector/cc-metric-collector.json | ||||
|  | ||||
| RESTART_ON_UPGRADE=true | ||||
|  | ||||
| # Only used on systemd systems | ||||
| PID_FILE_DIR=/var/run | ||||
|   | ||||
| @@ -14,11 +14,7 @@ Restart=on-failure | ||||
| WorkingDirectory=/tmp | ||||
| RuntimeDirectory=cc-metric-collector | ||||
| RuntimeDirectoryMode=0750 | ||||
| ExecStart=/usr/sbin/cc-metric-collector                                             \ | ||||
|                             --config=${CONF_FILE}                                   \ | ||||
|                             --pidfile=${PID_FILE_DIR}/cc-metric-collector.pid | ||||
|  | ||||
|  | ||||
| ExecStart=/usr/sbin/cc-metric-collector --config=${CONF_FILE} | ||||
| LimitNOFILE=10000 | ||||
| TimeoutStopSec=20 | ||||
| UMask=0027 | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| Name:           cc-metric-collector | ||||
| Version:        0.1 | ||||
| Version:        0.2 | ||||
| Release:        1%{?dist} | ||||
| Summary:        Metric collection daemon from the ClusterCockpit suite | ||||
|  | ||||
| @@ -26,10 +26,14 @@ make | ||||
|  | ||||
|  | ||||
| %install | ||||
| install -Dpm 0755 %{name} %{buildroot}%{_sbindir}/%{name} | ||||
| install -Dpm 0750 %{name} %{buildroot}%{_sbindir}/%{name} | ||||
| install -Dpm 0600 config.json %{buildroot}%{_sysconfdir}/%{name}/%{name}.json | ||||
| install -Dpm 644 scripts/%{name}.service %{buildroot}%{_unitdir}/%{name}.service | ||||
| install -Dpm 600 scripts/%{name}.config %{buildroot}%{_sysconfdir}/default/%{name} | ||||
| install -Dpm 0600 collectors.json %{buildroot}%{_sysconfdir}/%{name}/collectors.json | ||||
| install -Dpm 0600 sinks.json %{buildroot}%{_sysconfdir}/%{name}/sinks.json | ||||
| install -Dpm 0600 receivers.json %{buildroot}%{_sysconfdir}/%{name}/receivers.json | ||||
| install -Dpm 0600 router.json %{buildroot}%{_sysconfdir}/%{name}/router.json | ||||
| install -Dpm 0644 scripts/%{name}.service %{buildroot}%{_unitdir}/%{name}.service | ||||
| install -Dpm 0600 scripts/%{name}.config %{buildroot}%{_sysconfdir}/default/%{name} | ||||
|  | ||||
|  | ||||
| %check | ||||
| @@ -46,9 +50,15 @@ install -Dpm 600 scripts/%{name}.config %{buildroot}%{_sysconfdir}/default/%{nam | ||||
| %{_sbindir}/%{name} | ||||
| %{_unitdir}/%{name}.service | ||||
| %{_sysconfdir}/default/%{name} | ||||
| %config(noreplace) %{_sysconfdir}/%{name}/%{name}.json | ||||
|  | ||||
| %attr(0600,root,root) %config(noreplace) %{_sysconfdir}/%{name}/%{name}.json | ||||
| %attr(0600,root,root) %config(noreplace) %{_sysconfdir}/%{name}/collectors.json | ||||
| %attr(0600,root,root) %config(noreplace) %{_sysconfdir}/%{name}/sinks.json | ||||
| %attr(0600,root,root) %config(noreplace) %{_sysconfdir}/%{name}/receivers.json | ||||
| %attr(0600,root,root) %config(noreplace) %{_sysconfdir}/%{name}/router.json | ||||
|  | ||||
| %changelog | ||||
| * Mon Feb 14 2022 Thomas Gruber - 0.2 | ||||
| - Add component specific configuration files | ||||
| - Add %attr to config files | ||||
| * Mon Nov 22 2021 Thomas Gruber - 0.1 | ||||
| - Initial spec file | ||||
|   | ||||
							
								
								
									
										83
									
								
								scripts/likwid_perfgroup_to_cc_config.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										83
									
								
								scripts/likwid_perfgroup_to_cc_config.py
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| import os, os.path, sys, getopt, re, json | ||||
|  | ||||
| def which(cmd): | ||||
|     ospath = os.environ.get("PATH", "") | ||||
|     for p in ospath.split(":"): | ||||
|         testcmd = os.path.join(p, cmd) | ||||
|         if os.access(testcmd, os.X_OK): | ||||
|             return testcmd | ||||
|     return None | ||||
|  | ||||
| def group_to_json(groupfile): | ||||
|     gdata = [] | ||||
|     with open(groupfile, "r") as fp: | ||||
|         gdata = fp.read().strip().split("\n") | ||||
|     events = {} | ||||
|     metrics = [] | ||||
|     parse_events = False | ||||
|     parse_metrics = False | ||||
|     for line in gdata: | ||||
|         if line == "EVENTSET": | ||||
|             parse_events = True | ||||
|             parse_metrics = False | ||||
|             continue | ||||
|         if line == "METRICS": | ||||
|             parse_events = False | ||||
|             parse_metrics = True | ||||
|             continue | ||||
|         if len(line) == 0 or line.startswith("SHORT") or line == "LONG": | ||||
|             parse_events = False | ||||
|             parse_metrics = False | ||||
|             continue | ||||
|         if parse_events: | ||||
|             m = re.match("([\w\d]+)\s+([\w\d_]+)", line) | ||||
|             if m: | ||||
|                 events[m.group(1)] = m.group(2) | ||||
|         if parse_metrics: | ||||
|             llist = re.split("\s+", line) | ||||
|             calc = llist[-1] | ||||
|             metric = " ".join(llist[:-1]) | ||||
|             scope = "hwthread" | ||||
|             if "BOX" in calc: | ||||
|                 scope = "socket" | ||||
|             if "PWR" in calc: | ||||
|                 scope = "socket" | ||||
|  | ||||
|             m = {"name" : metric, "calc": calc, "scope" : scope, "publish" : True} | ||||
|             metrics.append(m) | ||||
|     return {"events" : events, "metrics" : metrics} | ||||
|  | ||||
| if len(sys.argv) != 3: | ||||
|     print("Usage: $0 <likwid-arch> <group-name>") | ||||
|     sys.exit(1) | ||||
|  | ||||
|  | ||||
| arch = sys.argv[1] | ||||
| group = sys.argv[2] | ||||
|  | ||||
| ltopo = which("likwid-topology") | ||||
| if not ltopo: | ||||
|     print("Cannot find LIKWID installation. Please add LIKWID bin folder to your PATH.") | ||||
|     sys.exit(1) | ||||
|  | ||||
| bindir = os.path.dirname(ltopo) | ||||
|  | ||||
| groupdir = os.path.normpath(os.path.join(bindir, "../share/likwid/perfgroups")) | ||||
| if not os.path.exists(groupdir): | ||||
|     print("Cannot find LIKWID performance groups in default install location") | ||||
|     sys.exit(1) | ||||
|  | ||||
| archdir = os.path.join(groupdir, arch) | ||||
| if not os.path.exists(archdir): | ||||
|     print("Cannot find LIKWID performance groups for architecture {}".format(arch)) | ||||
|     sys.exit(1) | ||||
|  | ||||
| groupfile = os.path.join(archdir, "{}.txt".format(group)) | ||||
| if not os.path.exists(groupfile): | ||||
|     print("Cannot find LIKWID performance group {} for architecture {}".format(group, arch)) | ||||
|     sys.exit(1) | ||||
|  | ||||
| gdata = group_to_json(groupfile) | ||||
| print(json.dumps(gdata, sort_keys=True, indent=2)) | ||||
							
								
								
									
										6
									
								
								sinks.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								sinks.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| { | ||||
|   "mystdout" : { | ||||
|     "type" : "stdout", | ||||
|     "meta_as_tags" : true | ||||
|   } | ||||
| } | ||||
							
								
								
									
										14
									
								
								sinks/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								sinks/Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
|  | ||||
| all: libganglia.so | ||||
|  | ||||
| libganglia.so: | ||||
| 	@find /usr ! -readable -prune -o -type d ! -executable -prune -o -name "$@*" -print0 | \ | ||||
| 	    xargs --null --no-run-if-empty --replace \ | ||||
| 	        ln --symbolic --verbose --force '{}' "$@" | ||||
| 	@if [[ ! -e "$@" ]]; then touch "$@"; fi | ||||
|  | ||||
|  | ||||
| clean: | ||||
| 	rm -f libganglia.so | ||||
|  | ||||
| .PHONY: clean | ||||
							
								
								
									
										125
									
								
								sinks/README.md
									
									
									
									
									
								
							
							
						
						
									
										125
									
								
								sinks/README.md
									
									
									
									
									
								
							| @@ -1,65 +1,94 @@ | ||||
| This folder contains the sinks for the cc-metric-collector. | ||||
| # CCMetric sinks | ||||
|  | ||||
| # `metricSink.go` | ||||
| The base class/configuration is located in `metricSink.go`. | ||||
| This folder contains the SinkManager and sink implementations for the cc-metric-collector. | ||||
|  | ||||
| # Sinks | ||||
| * `stdoutSink.go`: Writes all metrics to `stdout` in InfluxDB line protocol. The sink does not use https://github.com/influxdata/line-protocol to reduce the executed code for debugging | ||||
| * `influxSink.go`: Writes all metrics to an InfluxDB database instance using a blocking writer. It uses https://github.com/influxdata/influxdb-client-go . Configuration for the server, port, ssl, password, database name and organisation are in the global configuration file. The 'password' is used for the token and the 'database' for the bucket. It uses the v2 API of Influx. | ||||
| * `natsSink.go`: Sends all metrics to an NATS server using the InfluxDB line protocol as encoding. It uses https://github.com/nats-io/nats.go . Configuration for the server, port, user, password and database name are in the global configuration file. The database name is used as subject for the NATS messages. | ||||
| * `httpSink.go`: Sends all metrics to an HTTP endpoint `http://<host>:<port>/<database>` using a POST request. The body of the request will consist of lines in the InfluxDB line protocol. In case password is specified, that password is used as a JWT in the 'Authorization' header. | ||||
| # Available sinks: | ||||
| - [`stdout`](./stdoutSink.md): Print all metrics to `stdout`, `stderr` or a file | ||||
| - [`http`](./httpSink.md): Send metrics to an HTTP server as POST requests | ||||
| - [`influxdb`](./influxSink.md): Send metrics to an [InfluxDB](https://www.influxdata.com/products/influxdb/) database | ||||
| - [`nats`](./natsSink.md): Publish metrics to the [NATS](https://nats.io/) network overlay system | ||||
| - [`ganglia`](./gangliaSink.md): Publish metrics in the [Ganglia Monitoring System](http://ganglia.info/) using the `gmetric` CLI tool | ||||
| - [`libganglia`](./libgangliaSink.md): Publish metrics in the [Ganglia Monitoring System](http://ganglia.info/) directly using `libganglia.so` | ||||
|  | ||||
| # Installation | ||||
| Nothing to do, all sinks are pure Go code | ||||
| # Configuration | ||||
|  | ||||
| # Sink configuration | ||||
| The configuration file for the sinks is a list of configurations. The `type` field in each specifies which sink to initialize. | ||||
|  | ||||
| ```json | ||||
|   "sink": { | ||||
|     "user": "testuser", | ||||
|     "password": "testpass", | ||||
|     "host": "127.0.0.1", | ||||
|     "port": "9090", | ||||
|     "database": "testdb", | ||||
|     "organization": "testorg", | ||||
|     "ssl": false | ||||
|     "type": "stdout" | ||||
| [ | ||||
|   "mystdout" : { | ||||
|     "type" : "stdout", | ||||
|     "meta_as_tags" : false | ||||
|   }, | ||||
|   "metricstore" : { | ||||
|     "type" : "http", | ||||
|     "host" : "localhost", | ||||
|     "port" : "4123", | ||||
|     "database" : "ccmetric", | ||||
|     "password" : "<jwt token>" | ||||
|   } | ||||
| ] | ||||
| ``` | ||||
|  | ||||
| ## `stdout` | ||||
| When configuring `type = stdout`, all metrics are printed to stdout. No further configuration is required or touched, so you can leave your other-sink-config in there and just change the `type` for debugging purposes | ||||
|  | ||||
| ## `influxdb` | ||||
| The InfluxDB sink uses blocking write operations to write to an InfluxDB database using the v2 API. It uses the following configuration options: | ||||
| * `host`: Hostname of the database instance | ||||
| * `port`: Portnumber (as string) of the database | ||||
| * `database`: Name of the database, called 'bucket' in InfluxDB v2 | ||||
| * `organization`: The InfluxDB v2 API uses organizations to separate database instances running on the same host | ||||
| * `ssl`: Boolean to activate SSL/TLS | ||||
| * `user`: Although the v2 API uses API keys instead of username and password, this field can be used if the sink should authentificate with `username:password`. If you want to use an API key, leave this field empty. | ||||
| * `password`: API key for the InfluxDB v2 API or password if `user` is set | ||||
|  | ||||
| ## `nats` | ||||
| * `host`: Hostname of the NATS server | ||||
| * `port`: Portnumber (as string) of the NATS server | ||||
| * `user`: Username for authentification in the NATS transport system | ||||
| * `password`: Password for authentification in the NATS transport system | ||||
|  | ||||
| ## `http` | ||||
| * `host`: Hostname of the HTTP server | ||||
| * `port`: Portnumber (as string) of the HTTP server | ||||
| * `database`: Endpoint to write to. HTTP POST requests are performed on `http://<host>:<port>/<database>` | ||||
| * `password`: JSON Web token used for authentification | ||||
|  | ||||
|  | ||||
| # Contributing own sinks | ||||
| A sink contains three functions and is derived from the type `Sink` (in `metricSink.go`): | ||||
| * `Init(config SinkConfig) error` | ||||
| * `Write(measurement string, tags map[string]string, fields map[string]interface{}, t time.Time) error` | ||||
| A sink contains four functions and is derived from the type `sink`: | ||||
| * `Init(config json.RawMessage) error` | ||||
| * `Write(point CCMetric) error` | ||||
| * `Flush() error` | ||||
| * `Close()` | ||||
|  | ||||
| The data structures should be set up in `Init()` like opening a file or server connection. The `Write()` function takes a measurement, tags, fields and a timestamp and writes/sends the data. For non-blocking sinks, the `Flush()` method tells the sink to drain its internal buffers. The `Close()` function should tear down anything created in `Init()`. | ||||
| The data structures should be set up in `Init()` like opening a file or server connection. The `Write()` function writes/sends the data. For non-blocking sinks, the `Flush()` method tells the sink to drain its internal buffers. The `Close()` function should tear down anything created in `Init()`. | ||||
|  | ||||
| Finally, the sink needs to be registered in the `metric-collector.go`. There is a list of sinks called `Sinks` which is a map (sink_type_string -> pointer to sink). Add a new entry with a descriptive name and the new sink. | ||||
| Finally, the sink needs to be registered in the `sinkManager.go`. There is a list of sinks called `AvailableSinks` which is a map (`sink_type_string` -> `pointer to sink interface`). Add a new entry with a descriptive name and the new sink. | ||||
|  | ||||
| ## Sample sink | ||||
|  | ||||
| ```go | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"log" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type SampleSinkConfig struct { | ||||
| 	defaultSinkConfig  // defines JSON tags for 'name' and 'meta_as_tags' | ||||
| } | ||||
|  | ||||
| type SampleSink struct { | ||||
| 	sink              // declarate 'name' and 'meta_as_tags' | ||||
| 	config StdoutSinkConfig // entry point to the SampleSinkConfig | ||||
| } | ||||
|  | ||||
| // Initialize the sink by giving it a name and reading in the config JSON | ||||
| func (s *SampleSink) Init(config json.RawMessage) error { | ||||
| 	s.name = "SampleSink"   // Always specify a name here | ||||
|   // Read in the config JSON | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Code to submit a single CCMetric to the sink | ||||
| func (s *SampleSink) Write(point lp.CCMetric) error { | ||||
| 	log.Print(point) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // If the sink uses batched sends internally, you can tell to flush its buffers | ||||
| func (s *SampleSink) Flush() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
|  | ||||
| // Close sink: close network connection, close files, close libraries, ... | ||||
| func (s *SampleSink) Close() {} | ||||
| ``` | ||||
							
								
								
									
										50
									
								
								sinks/gangliaCommon.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								sinks/gangliaCommon.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"strings" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| func GangliaMetricName(point lp.CCMetric) string { | ||||
| 	name := point.Name() | ||||
| 	metricType, typeOK := point.GetTag("type") | ||||
| 	metricTid, tidOk := point.GetTag("type-id") | ||||
| 	gangliaType := metricType + metricTid | ||||
| 	if strings.Contains(name, metricType) && tidOk { | ||||
| 		name = strings.Replace(name, metricType, gangliaType, -1) | ||||
| 	} else if typeOK && tidOk { | ||||
| 		name = metricType + metricTid + "_" + name | ||||
| 	} else if point.HasTag("device") { | ||||
| 		device, _ := point.GetTag("device") | ||||
| 		name = name + "_" + device | ||||
| 	} | ||||
|  | ||||
| 	return name | ||||
| } | ||||
|  | ||||
| func GangliaMetricRename(point lp.CCMetric) string { | ||||
| 	name := point.Name() | ||||
| 	if name == "mem_total" || name == "swap_total" { | ||||
| 		return name | ||||
| 	} else if name == "net_bytes_in" { | ||||
| 		return "bytes_in" | ||||
| 	} else if name == "net_bytes_out" { | ||||
| 		return "bytes_out" | ||||
| 	} else if name == "net_pkts_in" { | ||||
| 		return "pkts_in" | ||||
| 	} else if name == "net_pkts_out" { | ||||
| 		return "pkts_out" | ||||
| 	} else if name == "cpu_iowait" { | ||||
| 		return "cpu_wio" | ||||
| 	} | ||||
| 	return name | ||||
| } | ||||
|  | ||||
| func GangliaSlopeType(point lp.CCMetric) uint { | ||||
| 	name := point.Name() | ||||
| 	if name == "mem_total" || name == "swap_total" { | ||||
| 		return 0 | ||||
| 	} | ||||
| 	return 3 | ||||
| } | ||||
							
								
								
									
										157
									
								
								sinks/gangliaSink.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								sinks/gangliaSink.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,157 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	//	"time" | ||||
| 	"os/exec" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const GMETRIC_EXEC = `gmetric` | ||||
| const GMETRIC_CONFIG = `/etc/ganglia/gmond.conf` | ||||
|  | ||||
| type GangliaSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	GmetricPath     string `json:"gmetric_path,omitempty"` | ||||
| 	GmetricConfig   string `json:"gmetric_config,omitempty"` | ||||
| 	AddGangliaGroup bool   `json:"add_ganglia_group,omitempty"` | ||||
| 	AddTagsAsDesc   bool   `json:"add_tags_as_desc,omitempty"` | ||||
| 	ClusterName     string `json:"cluster_name,omitempty"` | ||||
| 	AddTypeToName   bool   `json:"add_type_to_name,omitempty"` | ||||
| } | ||||
|  | ||||
| type GangliaSink struct { | ||||
| 	sink | ||||
| 	gmetric_path   string | ||||
| 	gmetric_config string | ||||
| 	config         GangliaSinkConfig | ||||
| } | ||||
|  | ||||
| func (s *GangliaSink) Init(config json.RawMessage) error { | ||||
| 	var err error = nil | ||||
| 	s.name = "GangliaSink" | ||||
| 	s.config.AddTagsAsDesc = false | ||||
| 	s.config.AddGangliaGroup = false | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError(s.name, "Error reading config for", s.name, ":", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	s.gmetric_path = "" | ||||
| 	s.gmetric_config = "" | ||||
| 	if len(s.config.GmetricPath) > 0 { | ||||
| 		p, err := exec.LookPath(s.config.GmetricPath) | ||||
| 		if err == nil { | ||||
| 			s.gmetric_path = p | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.gmetric_path) == 0 { | ||||
| 		p, err := exec.LookPath(string(GMETRIC_EXEC)) | ||||
| 		if err == nil { | ||||
| 			s.gmetric_path = p | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.gmetric_path) == 0 { | ||||
| 		err = errors.New("cannot find executable 'gmetric'") | ||||
| 	} | ||||
| 	if len(s.config.GmetricConfig) > 0 { | ||||
| 		s.gmetric_config = s.config.GmetricConfig | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (s *GangliaSink) Write(point lp.CCMetric) error { | ||||
| 	var err error = nil | ||||
| 	var tagsstr []string | ||||
| 	var argstr []string | ||||
| 	if s.config.AddGangliaGroup { | ||||
| 		if point.HasTag("group") { | ||||
| 			g, _ := point.GetTag("group") | ||||
| 			argstr = append(argstr, fmt.Sprintf("--group=%s", g)) | ||||
| 		} else if point.HasMeta("group") { | ||||
| 			g, _ := point.GetMeta("group") | ||||
| 			argstr = append(argstr, fmt.Sprintf("--group=%s", g)) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for key, value := range point.Tags() { | ||||
| 		switch key { | ||||
| 		case "unit": | ||||
| 			argstr = append(argstr, fmt.Sprintf("--units=%s", value)) | ||||
| 		default: | ||||
| 			tagsstr = append(tagsstr, fmt.Sprintf("%s=%s", key, value)) | ||||
| 		} | ||||
| 	} | ||||
| 	if s.config.MetaAsTags { | ||||
| 		for key, value := range point.Meta() { | ||||
| 			switch key { | ||||
| 			case "unit": | ||||
| 				argstr = append(argstr, fmt.Sprintf("--units=%s", value)) | ||||
| 			default: | ||||
| 				tagsstr = append(tagsstr, fmt.Sprintf("%s=%s", key, value)) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.ClusterName) > 0 { | ||||
| 		argstr = append(argstr, fmt.Sprintf("--cluster=%s", s.config.ClusterName)) | ||||
| 	} | ||||
| 	if s.config.AddTagsAsDesc && len(tagsstr) > 0 { | ||||
| 		argstr = append(argstr, fmt.Sprintf("--desc=%q", strings.Join(tagsstr, ","))) | ||||
| 	} | ||||
| 	if len(s.gmetric_config) > 0 { | ||||
| 		argstr = append(argstr, fmt.Sprintf("--conf=%s", s.gmetric_config)) | ||||
| 	} | ||||
| 	name := GangliaMetricRename(point) | ||||
| 	if s.config.AddTypeToName { | ||||
| 		argstr = append(argstr, fmt.Sprintf("--name=%s", GangliaMetricName(point))) | ||||
| 	} else { | ||||
| 		argstr = append(argstr, fmt.Sprintf("--name=%s", name)) | ||||
| 	} | ||||
| 	slope := GangliaSlopeType(point) | ||||
| 	slopeStr := "both" | ||||
| 	if slope == 0 { | ||||
| 		slopeStr = "zero" | ||||
| 	} | ||||
| 	argstr = append(argstr, fmt.Sprintf("--slope=%s", slopeStr)) | ||||
|  | ||||
| 	for k, v := range point.Fields() { | ||||
| 		if k == "value" { | ||||
| 			switch value := v.(type) { | ||||
| 			case float64: | ||||
| 				argstr = append(argstr, | ||||
| 					fmt.Sprintf("--value=%v", value), "--type=double") | ||||
| 			case float32: | ||||
| 				argstr = append(argstr, | ||||
| 					fmt.Sprintf("--value=%v", value), "--type=float") | ||||
| 			case int: | ||||
| 				argstr = append(argstr, | ||||
| 					fmt.Sprintf("--value=%d", value), "--type=int32") | ||||
| 			case int64: | ||||
| 				argstr = append(argstr, | ||||
| 					fmt.Sprintf("--value=%d", value), "--type=int32") | ||||
| 			case string: | ||||
| 				argstr = append(argstr, | ||||
| 					fmt.Sprintf("--value=%q", value), "--type=string") | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	command := exec.Command(s.gmetric_path, argstr...) | ||||
| 	command.Wait() | ||||
| 	_, err = command.Output() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (s *GangliaSink) Flush() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *GangliaSink) Close() { | ||||
| } | ||||
							
								
								
									
										21
									
								
								sinks/gangliaSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								sinks/gangliaSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| ## `ganglia` sink | ||||
|  | ||||
| The `ganglia` sink uses the `gmetric` tool of the [Ganglia Monitoring System](http://ganglia.info/) to submit the metrics | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "ganglia", | ||||
|     "meta_as_tags" : true, | ||||
|     "gmetric_path" : "/path/to/gmetric", | ||||
|     "add_ganglia_group" : true | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `ganglia` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `gmetric_path`: Path to `gmetric` executable (optional). If not given, the sink searches in `$PATH` for `gmetric`. | ||||
| - `add_ganglia_group`: Add `--group=X` based on meta information to the `gmetric` call. Some old versions of `gmetric` do not support the `--group` option.  | ||||
| @@ -2,61 +2,160 @@ package sinks | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influx "github.com/influxdata/line-protocol" | ||||
| ) | ||||
|  | ||||
| type HttpSink struct { | ||||
| 	Sink | ||||
| 	client   *http.Client | ||||
| 	url, jwt string | ||||
| 	encoder  *lp.Encoder | ||||
| 	buffer   *bytes.Buffer | ||||
| type HttpSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	URL             string `json:"url,omitempty"` | ||||
| 	JWT             string `json:"jwt,omitempty"` | ||||
| 	Timeout         string `json:"timeout,omitempty"` | ||||
| 	MaxIdleConns    int    `json:"max_idle_connections,omitempty"` | ||||
| 	IdleConnTimeout string `json:"idle_connection_timeout,omitempty"` | ||||
| 	FlushDelay      string `json:"flush_delay,omitempty"` | ||||
| } | ||||
|  | ||||
| func (s *HttpSink) Init(config SinkConfig) error { | ||||
| 	if len(config.Host) == 0 || len(config.Port) == 0 { | ||||
| 		return errors.New("`host`, `port` and `database` config options required for TCP sink") | ||||
| 	} | ||||
| type HttpSink struct { | ||||
| 	sink | ||||
| 	client          *http.Client | ||||
| 	encoder         *influx.Encoder | ||||
| 	lock            sync.Mutex // Flush() runs in another goroutine, so this lock has to protect the buffer | ||||
| 	buffer          *bytes.Buffer | ||||
| 	flushTimer      *time.Timer | ||||
| 	config          HttpSinkConfig | ||||
| 	maxIdleConns    int | ||||
| 	idleConnTimeout time.Duration | ||||
| 	timeout         time.Duration | ||||
| 	flushDelay      time.Duration | ||||
| } | ||||
|  | ||||
| 	s.client = &http.Client{} | ||||
| 	s.url = fmt.Sprintf("http://%s:%s/%s", config.Host, config.Port, config.Database) | ||||
| 	s.port = config.Port | ||||
| 	s.jwt = config.Password | ||||
| func (s *HttpSink) Init(config json.RawMessage) error { | ||||
| 	// Set default values | ||||
| 	s.name = "HttpSink" | ||||
| 	s.config.MaxIdleConns = 10 | ||||
| 	s.config.IdleConnTimeout = "5s" | ||||
| 	s.config.Timeout = "5s" | ||||
| 	s.config.FlushDelay = "1s" | ||||
|  | ||||
| 	// Read config | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.URL) == 0 { | ||||
| 		return errors.New("`url` config option is required for HTTP sink") | ||||
| 	} | ||||
| 	if s.config.MaxIdleConns > 0 { | ||||
| 		s.maxIdleConns = s.config.MaxIdleConns | ||||
| 	} | ||||
| 	if len(s.config.IdleConnTimeout) > 0 { | ||||
| 		t, err := time.ParseDuration(s.config.IdleConnTimeout) | ||||
| 		if err == nil { | ||||
| 			s.idleConnTimeout = t | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.Timeout) > 0 { | ||||
| 		t, err := time.ParseDuration(s.config.Timeout) | ||||
| 		if err == nil { | ||||
| 			s.timeout = t | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.FlushDelay) > 0 { | ||||
| 		t, err := time.ParseDuration(s.config.FlushDelay) | ||||
| 		if err == nil { | ||||
| 			s.flushDelay = t | ||||
| 		} | ||||
| 	} | ||||
| 	tr := &http.Transport{ | ||||
| 		MaxIdleConns:    s.maxIdleConns, | ||||
| 		IdleConnTimeout: s.idleConnTimeout, | ||||
| 	} | ||||
| 	s.client = &http.Client{Transport: tr, Timeout: s.timeout} | ||||
| 	s.buffer = &bytes.Buffer{} | ||||
| 	s.encoder = lp.NewEncoder(s.buffer) | ||||
| 	s.encoder = influx.NewEncoder(s.buffer) | ||||
| 	s.encoder.SetPrecision(time.Second) | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *HttpSink) Write(point lp.MutableMetric) error { | ||||
| 	_, err := s.encoder.Encode(point) | ||||
| func (s *HttpSink) Write(m lp.CCMetric) error { | ||||
| 	if s.buffer.Len() == 0 && s.flushDelay != 0 { | ||||
| 		// This is the first write since the last flush, start the flushTimer! | ||||
| 		if s.flushTimer != nil && s.flushTimer.Stop() { | ||||
| 			cclog.ComponentDebug("HttpSink", "unexpected: the flushTimer was already running?") | ||||
| 		} | ||||
|  | ||||
| 		// Run a batched flush for all lines that have arrived in the last second | ||||
| 		s.flushTimer = time.AfterFunc(s.flushDelay, func() { | ||||
| 			if err := s.Flush(); err != nil { | ||||
| 				cclog.ComponentError("HttpSink", "flush failed:", err.Error()) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
| 	p := m.ToPoint(s.config.MetaAsTags) | ||||
|  | ||||
| 	s.lock.Lock() | ||||
| 	_, err := s.encoder.Encode(p) | ||||
| 	s.lock.Unlock() // defer does not work here as Flush() takes the lock as well | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Flush synchronously if "flush_delay" is zero | ||||
| 	if s.flushDelay == 0 { | ||||
| 		return s.Flush() | ||||
| 	} | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (s *HttpSink) Flush() error { | ||||
| 	req, err := http.NewRequest(http.MethodPost, s.url, s.buffer) | ||||
| 	// buffer is read by client.Do, prevent concurrent modifications | ||||
| 	s.lock.Lock() | ||||
| 	defer s.lock.Unlock() | ||||
|  | ||||
| 	// Do not flush empty buffer | ||||
| 	if s.buffer.Len() == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	// Create new request to send buffer | ||||
| 	req, err := http.NewRequest(http.MethodPost, s.config.URL, s.buffer) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if len(s.jwt) != 0 { | ||||
| 		req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", s.jwt)) | ||||
| 	// Set authorization header | ||||
| 	if len(s.config.JWT) != 0 { | ||||
| 		req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", s.config.JWT)) | ||||
| 	} | ||||
|  | ||||
| 	// Send | ||||
| 	res, err := s.client.Do(req) | ||||
|  | ||||
| 	// Clear buffer | ||||
| 	s.buffer.Reset() | ||||
|  | ||||
| 	// Handle transport/tcp errors | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if res.StatusCode != 200 { | ||||
| 	// Handle application errors | ||||
| 	if res.StatusCode != http.StatusOK { | ||||
| 		return errors.New(res.Status) | ||||
| 	} | ||||
|  | ||||
| @@ -64,5 +163,9 @@ func (s *HttpSink) Flush() error { | ||||
| } | ||||
|  | ||||
| func (s *HttpSink) Close() { | ||||
| 	s.flushTimer.Stop() | ||||
| 	if err := s.Flush(); err != nil { | ||||
| 		cclog.ComponentError("HttpSink", "flush failed:", err.Error()) | ||||
| 	} | ||||
| 	s.client.CloseIdleConnections() | ||||
| } | ||||
|   | ||||
							
								
								
									
										29
									
								
								sinks/httpSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								sinks/httpSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| ## `http` sink | ||||
|  | ||||
| The `http` sink uses POST requests to a HTTP server to submit the metrics in the InfluxDB line-protocol format. It uses JSON web tokens for authentification. The sink creates batches of metrics before sending, to reduce the HTTP traffic. | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "http", | ||||
|     "meta_as_tags" : true, | ||||
|     "url" : "https://my-monitoring.example.com:1234/api/write", | ||||
|     "jwt" : "blabla.blabla.blabla", | ||||
|     "timeout": "5s", | ||||
|     "max_idle_connections" : 10, | ||||
|     "idle_connection_timeout" : "5s", | ||||
|     "flush_delay": "2s", | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `http` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `url`: The full URL of the endpoint | ||||
| - `jwt`: JSON web tokens for authentification (Using the *Bearer* scheme) | ||||
| - `timeout`: General timeout for the HTTP client (default '5s') | ||||
| - `max_idle_connections`: Maximally idle connections (default 10) | ||||
| - `idle_connection_timeout`: Timeout for idle connections (default '5s') | ||||
| - `flush_delay`: Batch all writes arriving in during this duration (default '1s', batching can be disabled by setting it to 0) | ||||
							
								
								
									
										120
									
								
								sinks/influxAsyncSink.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								sinks/influxAsyncSink.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,120 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influxdb2 "github.com/influxdata/influxdb-client-go/v2" | ||||
| 	influxdb2Api "github.com/influxdata/influxdb-client-go/v2/api" | ||||
| ) | ||||
|  | ||||
| type InfluxAsyncSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	Host         string `json:"host,omitempty"` | ||||
| 	Port         string `json:"port,omitempty"` | ||||
| 	Database     string `json:"database,omitempty"` | ||||
| 	User         string `json:"user,omitempty"` | ||||
| 	Password     string `json:"password,omitempty"` | ||||
| 	Organization string `json:"organization,omitempty"` | ||||
| 	SSL          bool   `json:"ssl,omitempty"` | ||||
| 	RetentionPol string `json:"retention_policy,omitempty"` | ||||
| 	// Maximum number of points sent to server in single request. Default 5000 | ||||
| 	BatchSize uint `json:"batch_size,omitempty"` | ||||
| 	// Interval, in ms, in which is buffer flushed if it has not been already written (by reaching batch size) . Default 1000ms | ||||
| 	FlushInterval uint `json:"flush_interval,omitempty"` | ||||
| } | ||||
|  | ||||
| type InfluxAsyncSink struct { | ||||
| 	sink | ||||
| 	client    influxdb2.Client | ||||
| 	writeApi  influxdb2Api.WriteAPI | ||||
| 	retPolicy string | ||||
| 	errors    <-chan error | ||||
| 	config    InfluxAsyncSinkConfig | ||||
| } | ||||
|  | ||||
| func (s *InfluxAsyncSink) connect() error { | ||||
| 	var auth string | ||||
| 	var uri string | ||||
| 	if s.config.SSL { | ||||
| 		uri = fmt.Sprintf("https://%s:%s", s.config.Host, s.config.Port) | ||||
| 	} else { | ||||
| 		uri = fmt.Sprintf("http://%s:%s", s.config.Host, s.config.Port) | ||||
| 	} | ||||
| 	if len(s.config.User) == 0 { | ||||
| 		auth = s.config.Password | ||||
| 	} else { | ||||
| 		auth = fmt.Sprintf("%s:%s", s.config.User, s.config.Password) | ||||
| 	} | ||||
| 	cclog.ComponentDebug(s.name, "Using URI", uri, "Org", s.config.Organization, "Bucket", s.config.Database) | ||||
| 	clientOptions := influxdb2.DefaultOptions() | ||||
| 	if s.config.BatchSize != 0 { | ||||
| 		clientOptions.SetBatchSize(s.config.BatchSize) | ||||
| 	} | ||||
| 	if s.config.FlushInterval != 0 { | ||||
| 		clientOptions.SetFlushInterval(s.config.FlushInterval) | ||||
| 	} | ||||
| 	clientOptions.SetTLSConfig( | ||||
| 		&tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 		}, | ||||
| 	) | ||||
| 	s.client = influxdb2.NewClientWithOptions(uri, auth, clientOptions) | ||||
| 	s.writeApi = s.client.WriteAPI(s.config.Organization, s.config.Database) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *InfluxAsyncSink) Init(config json.RawMessage) error { | ||||
| 	s.name = "InfluxSink" | ||||
|  | ||||
| 	// Set default for maximum number of points sent to server in single request. | ||||
| 	s.config.BatchSize = 100 | ||||
|  | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.Host) == 0 || | ||||
| 		len(s.config.Port) == 0 || | ||||
| 		len(s.config.Database) == 0 || | ||||
| 		len(s.config.Organization) == 0 || | ||||
| 		len(s.config.Password) == 0 { | ||||
| 		return errors.New("not all configuration variables set required by InfluxAsyncSink") | ||||
| 	} | ||||
|  | ||||
| 	// Connect to InfluxDB server | ||||
| 	err := s.connect() | ||||
|  | ||||
| 	// Start background: Read from error channel | ||||
| 	s.errors = s.writeApi.Errors() | ||||
| 	go func() { | ||||
| 		for err := range s.errors { | ||||
| 			cclog.ComponentError(s.name, err.Error()) | ||||
| 		} | ||||
| 	}() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (s *InfluxAsyncSink) Write(m lp.CCMetric) error { | ||||
| 	s.writeApi.WritePoint( | ||||
| 		m.ToPoint(s.config.MetaAsTags), | ||||
| 	) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *InfluxAsyncSink) Flush() error { | ||||
| 	s.writeApi.Flush() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *InfluxAsyncSink) Close() { | ||||
| 	cclog.ComponentDebug(s.name, "Closing InfluxDB connection") | ||||
| 	s.writeApi.Flush() | ||||
| 	s.client.Close() | ||||
| } | ||||
							
								
								
									
										34
									
								
								sinks/influxAsyncSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								sinks/influxAsyncSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| ## `influxasync` sink | ||||
|  | ||||
| The `influxasync` sink uses the official [InfluxDB golang client](https://pkg.go.dev/github.com/influxdata/influxdb-client-go/v2) to write the metrics to an InfluxDB database in a **non-blocking** fashion. It provides only support for V2 write endpoints (InfluxDB 1.8.0 or later). | ||||
|  | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "influxasync", | ||||
|     "meta_as_tags" : true, | ||||
|     "database" : "mymetrics", | ||||
|     "host": "dbhost.example.com", | ||||
|     "port": "4222", | ||||
|     "user": "exampleuser", | ||||
|     "password" : "examplepw", | ||||
|     "organization": "myorg", | ||||
|     "ssl": true, | ||||
|     "batch_size": 200, | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `influxdb` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `database`: All metrics are written to this bucket  | ||||
| - `host`: Hostname of the InfluxDB database server | ||||
| - `port`: Portnumber (as string) of the InfluxDB database server | ||||
| - `user`: Username for basic authentification | ||||
| - `password`: Password for basic authentification | ||||
| - `organization`: Organization in the InfluxDB | ||||
| - `ssl`: Use SSL connection | ||||
| - `batch_size`: batch up metrics internally, default 100 | ||||
| @@ -3,71 +3,86 @@ package sinks | ||||
| import ( | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influxdb2 "github.com/influxdata/influxdb-client-go/v2" | ||||
| 	influxdb2Api "github.com/influxdata/influxdb-client-go/v2/api" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	"log" | ||||
| ) | ||||
|  | ||||
| type InfluxSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	Host         string `json:"host,omitempty"` | ||||
| 	Port         string `json:"port,omitempty"` | ||||
| 	Database     string `json:"database,omitempty"` | ||||
| 	User         string `json:"user,omitempty"` | ||||
| 	Password     string `json:"password,omitempty"` | ||||
| 	Organization string `json:"organization,omitempty"` | ||||
| 	SSL          bool   `json:"ssl,omitempty"` | ||||
| 	RetentionPol string `json:"retention_policy,omitempty"` | ||||
| } | ||||
|  | ||||
| type InfluxSink struct { | ||||
| 	Sink | ||||
| 	client    influxdb2.Client | ||||
| 	writeApi  influxdb2Api.WriteAPIBlocking | ||||
| 	retPolicy string | ||||
| 	sink | ||||
| 	client   influxdb2.Client | ||||
| 	writeApi influxdb2Api.WriteAPIBlocking | ||||
| 	config   InfluxSinkConfig | ||||
| } | ||||
|  | ||||
| func (s *InfluxSink) connect() error { | ||||
| 	var auth string | ||||
| 	var uri string | ||||
| 	if s.ssl { | ||||
| 		uri = fmt.Sprintf("https://%s:%s", s.host, s.port) | ||||
| 	if s.config.SSL { | ||||
| 		uri = fmt.Sprintf("https://%s:%s", s.config.Host, s.config.Port) | ||||
| 	} else { | ||||
| 		uri = fmt.Sprintf("http://%s:%s", s.host, s.port) | ||||
| 		uri = fmt.Sprintf("http://%s:%s", s.config.Host, s.config.Port) | ||||
| 	} | ||||
| 	if len(s.user) == 0 { | ||||
| 		auth = s.password | ||||
| 	if len(s.config.User) == 0 { | ||||
| 		auth = s.config.Password | ||||
| 	} else { | ||||
| 		auth = fmt.Sprintf("%s:%s", s.user, s.password) | ||||
| 		auth = fmt.Sprintf("%s:%s", s.config.User, s.config.Password) | ||||
| 	} | ||||
| 	log.Print("Using URI ", uri, " Org ", s.organization, " Bucket ", s.database) | ||||
| 	s.client = influxdb2.NewClientWithOptions(uri, auth, | ||||
| 		influxdb2.DefaultOptions().SetTLSConfig(&tls.Config{InsecureSkipVerify: true})) | ||||
| 	s.writeApi = s.client.WriteAPIBlocking(s.organization, s.database) | ||||
| 	cclog.ComponentDebug(s.name, "Using URI", uri, "Org", s.config.Organization, "Bucket", s.config.Database) | ||||
| 	clientOptions := influxdb2.DefaultOptions() | ||||
| 	clientOptions.SetTLSConfig( | ||||
| 		&tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 		}, | ||||
| 	) | ||||
| 	s.client = influxdb2.NewClientWithOptions(uri, auth, clientOptions) | ||||
| 	s.writeApi = s.client.WriteAPIBlocking(s.config.Organization, s.config.Database) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *InfluxSink) Init(config SinkConfig) error { | ||||
| 	if len(config.Host) == 0 || | ||||
| 		len(config.Port) == 0 || | ||||
| 		len(config.Database) == 0 || | ||||
| 		len(config.Organization) == 0 || | ||||
| 		len(config.Password) == 0 { | ||||
| 		return errors.New("Not all configuration variables set required by InfluxSink") | ||||
| func (s *InfluxSink) Init(config json.RawMessage) error { | ||||
| 	s.name = "InfluxSink" | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	s.host = config.Host | ||||
| 	s.port = config.Port | ||||
| 	s.database = config.Database | ||||
| 	s.organization = config.Organization | ||||
| 	s.user = config.User | ||||
| 	s.password = config.Password | ||||
| 	s.ssl = config.SSL | ||||
| 	if len(s.config.Host) == 0 || | ||||
| 		len(s.config.Port) == 0 || | ||||
| 		len(s.config.Database) == 0 || | ||||
| 		len(s.config.Organization) == 0 || | ||||
| 		len(s.config.Password) == 0 { | ||||
| 		return errors.New("not all configuration variables set required by InfluxSink") | ||||
| 	} | ||||
|  | ||||
| 	// Connect to InfluxDB server | ||||
| 	return s.connect() | ||||
| } | ||||
|  | ||||
| func (s *InfluxSink) Write(point lp.MutableMetric) error { | ||||
| 	tags := map[string]string{} | ||||
| 	fields := map[string]interface{}{} | ||||
| 	for _, t := range point.TagList() { | ||||
| 		tags[t.Key] = t.Value | ||||
| 	} | ||||
| 	for _, f := range point.FieldList() { | ||||
| 		fields[f.Key] = f.Value | ||||
| 	} | ||||
| 	p := influxdb2.NewPoint(point.Name(), tags, fields, point.Time()) | ||||
| 	err := s.writeApi.WritePoint(context.Background(), p) | ||||
| func (s *InfluxSink) Write(m lp.CCMetric) error { | ||||
| 	err := | ||||
| 		s.writeApi.WritePoint( | ||||
| 			context.Background(), | ||||
| 			m.ToPoint(s.config.MetaAsTags), | ||||
| 		) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| @@ -76,6 +91,6 @@ func (s *InfluxSink) Flush() error { | ||||
| } | ||||
|  | ||||
| func (s *InfluxSink) Close() { | ||||
| 	log.Print("Closing InfluxDB connection") | ||||
| 	cclog.ComponentDebug(s.name, "Closing InfluxDB connection") | ||||
| 	s.client.Close() | ||||
| } | ||||
|   | ||||
							
								
								
									
										32
									
								
								sinks/influxSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								sinks/influxSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| ## `influxdb` sink | ||||
|  | ||||
| The `influxdb` sink uses the official [InfluxDB golang client](https://pkg.go.dev/github.com/influxdata/influxdb-client-go/v2) to write the metrics to an InfluxDB database in a **blocking** fashion. It provides only support for V2 write endpoints (InfluxDB 1.8.0 or later). | ||||
|  | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "influxdb", | ||||
|     "meta_as_tags" : true, | ||||
|     "database" : "mymetrics", | ||||
|     "host": "dbhost.example.com", | ||||
|     "port": "4222", | ||||
|     "user": "exampleuser", | ||||
|     "password" : "examplepw", | ||||
|     "organization": "myorg", | ||||
|     "ssl": true, | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `influxdb` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `database`: All metrics are written to this bucket  | ||||
| - `host`: Hostname of the InfluxDB database server | ||||
| - `port`: Portnumber (as string) of the InfluxDB database server | ||||
| - `user`: Username for basic authentification | ||||
| - `password`: Password for basic authentification | ||||
| - `organization`: Organization in the InfluxDB | ||||
| - `ssl`: Use SSL connection | ||||
							
								
								
									
										304
									
								
								sinks/libgangliaSink.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										304
									
								
								sinks/libgangliaSink.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,304 @@ | ||||
| package sinks | ||||
|  | ||||
| /* | ||||
| #cgo CFLAGS: -DGM_PROTOCOL_GUARD | ||||
| #cgo LDFLAGS: -L. -lganglia -Wl,--unresolved-symbols=ignore-in-object-files | ||||
| #include <stdlib.h> | ||||
|  | ||||
| // This is a copy&paste snippet of ganglia.h (BSD-3 license) | ||||
| // See https://github.com/ganglia/monitor-core | ||||
| // for further information | ||||
|  | ||||
| enum ganglia_slope { | ||||
|    GANGLIA_SLOPE_ZERO = 0, | ||||
|    GANGLIA_SLOPE_POSITIVE, | ||||
|    GANGLIA_SLOPE_NEGATIVE, | ||||
|    GANGLIA_SLOPE_BOTH, | ||||
|    GANGLIA_SLOPE_UNSPECIFIED, | ||||
|    GANGLIA_SLOPE_DERIVATIVE, | ||||
|    GANGLIA_SLOPE_LAST_LEGAL_VALUE=GANGLIA_SLOPE_DERIVATIVE | ||||
| }; | ||||
| typedef enum ganglia_slope ganglia_slope_t; | ||||
|  | ||||
| typedef struct Ganglia_pool* Ganglia_pool; | ||||
| typedef struct Ganglia_gmond_config* Ganglia_gmond_config; | ||||
| typedef struct Ganglia_udp_send_channels* Ganglia_udp_send_channels; | ||||
|  | ||||
| struct Ganglia_metric { | ||||
|    Ganglia_pool pool; | ||||
|    struct Ganglia_metadata_message *msg; | ||||
|    char *value; | ||||
|    void *extra; | ||||
| }; | ||||
| typedef struct Ganglia_metric * Ganglia_metric; | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| Ganglia_gmond_config Ganglia_gmond_config_create(char *path, int fallback_to_default); | ||||
| //void Ganglia_gmond_config_destroy(Ganglia_gmond_config config); | ||||
|  | ||||
| Ganglia_udp_send_channels Ganglia_udp_send_channels_create(Ganglia_pool p, Ganglia_gmond_config config); | ||||
| void Ganglia_udp_send_channels_destroy(Ganglia_udp_send_channels channels); | ||||
|  | ||||
| int Ganglia_udp_send_message(Ganglia_udp_send_channels channels, char *buf, int len ); | ||||
|  | ||||
| Ganglia_metric Ganglia_metric_create( Ganglia_pool parent_pool ); | ||||
| int Ganglia_metric_set( Ganglia_metric gmetric, char *name, char *value, char *type, char *units, unsigned int slope, unsigned int tmax, unsigned int dmax); | ||||
| int Ganglia_metric_send( Ganglia_metric gmetric, Ganglia_udp_send_channels send_channels ); | ||||
| //int Ganglia_metadata_send( Ganglia_metric gmetric, Ganglia_udp_send_channels send_channels ); | ||||
| //int Ganglia_metadata_send_real( Ganglia_metric gmetric, Ganglia_udp_send_channels send_channels, char *override_string ); | ||||
| void Ganglia_metadata_add( Ganglia_metric gmetric, char *name, char *value ); | ||||
| //int Ganglia_value_send( Ganglia_metric gmetric, Ganglia_udp_send_channels send_channels ); | ||||
| void Ganglia_metric_destroy( Ganglia_metric gmetric ); | ||||
|  | ||||
| Ganglia_pool Ganglia_pool_create( Ganglia_pool parent ); | ||||
| void Ganglia_pool_destroy( Ganglia_pool pool ); | ||||
|  | ||||
| //ganglia_slope_t cstr_to_slope(const char* str); | ||||
| //const char*     slope_to_cstr(unsigned int slope); | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } | ||||
| #endif | ||||
| */ | ||||
| import "C" | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"unsafe" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	"github.com/NVIDIA/go-nvml/pkg/dl" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	GANGLIA_LIB_NAME     = "libganglia.so" | ||||
| 	GANGLIA_LIB_DL_FLAGS = dl.RTLD_LAZY | dl.RTLD_GLOBAL | ||||
| 	GMOND_CONFIG_FILE    = `/etc/ganglia/gmond.conf` | ||||
| ) | ||||
|  | ||||
| type LibgangliaSinkSpecialMetric struct { | ||||
| 	MetricName string `json:"metric_name,omitempty"` | ||||
| 	NewName    string `json:"new_name,omitempty"` | ||||
| 	Slope      string `json:"slope,omitempty"` | ||||
| } | ||||
|  | ||||
| type LibgangliaSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	GangliaLib      string                                 `json:"libganglia_path,omitempty"` | ||||
| 	GmondConfig     string                                 `json:"gmond_config,omitempty"` | ||||
| 	AddGangliaGroup bool                                   `json:"add_ganglia_group,omitempty"` | ||||
| 	AddTypeToName   bool                                   `json:"add_type_to_name,omitempty"` | ||||
| 	AddUnits        bool                                   `json:"add_units,omitempty"` | ||||
| 	ClusterName     string                                 `json:"cluster_name,omitempty"` | ||||
| 	SpecialMetrics  map[string]LibgangliaSinkSpecialMetric `json:"rename_metrics,omitempty"` // Map to rename metric name from key to value | ||||
| 	//AddTagsAsDesc   bool              `json:"add_tags_as_desc,omitempty"` | ||||
| } | ||||
|  | ||||
| type LibgangliaSink struct { | ||||
| 	sink | ||||
| 	config         LibgangliaSinkConfig | ||||
| 	global_context C.Ganglia_pool | ||||
| 	gmond_config   C.Ganglia_gmond_config | ||||
| 	send_channels  C.Ganglia_udp_send_channels | ||||
| 	cstrCache      map[string]*C.char | ||||
| } | ||||
|  | ||||
| func (s *LibgangliaSink) Init(config json.RawMessage) error { | ||||
| 	var err error = nil | ||||
| 	s.name = "LibgangliaSink" | ||||
| 	//s.config.AddTagsAsDesc = false | ||||
| 	s.config.AddGangliaGroup = false | ||||
| 	s.config.AddTypeToName = false | ||||
| 	s.config.AddUnits = true | ||||
| 	s.config.GmondConfig = string(GMOND_CONFIG_FILE) | ||||
| 	s.config.GangliaLib = string(GANGLIA_LIB_NAME) | ||||
| 	if len(config) > 0 { | ||||
| 		err = json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(s.name, "Error reading config for", s.name, ":", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	lib := dl.New(s.config.GangliaLib, GANGLIA_LIB_DL_FLAGS) | ||||
| 	if lib == nil { | ||||
| 		return fmt.Errorf("error instantiating DynamicLibrary for %s", s.config.GangliaLib) | ||||
| 	} | ||||
|  | ||||
| 	// Set up cache for the C strings | ||||
| 	s.cstrCache = make(map[string]*C.char) | ||||
| 	// s.cstrCache["globals"] = C.CString("globals") | ||||
|  | ||||
| 	// s.cstrCache["override_hostname"] = C.CString("override_hostname") | ||||
| 	// s.cstrCache["override_ip"] = C.CString("override_ip") | ||||
|  | ||||
| 	// Add some constant strings | ||||
| 	s.cstrCache["GROUP"] = C.CString("GROUP") | ||||
| 	s.cstrCache["CLUSTER"] = C.CString("CLUSTER") | ||||
| 	s.cstrCache[""] = C.CString("") | ||||
|  | ||||
| 	// Add cluster name for lookup in Write() | ||||
| 	if len(s.config.ClusterName) > 0 { | ||||
| 		s.cstrCache[s.config.ClusterName] = C.CString(s.config.ClusterName) | ||||
| 	} | ||||
| 	// Add supported types for later lookup in Write() | ||||
| 	s.cstrCache["double"] = C.CString("double") | ||||
| 	s.cstrCache["int32"] = C.CString("int32") | ||||
| 	s.cstrCache["string"] = C.CString("string") | ||||
|  | ||||
| 	// Create Ganglia pool | ||||
| 	s.global_context = C.Ganglia_pool_create(nil) | ||||
| 	// Load Ganglia configuration | ||||
| 	s.cstrCache[s.config.GmondConfig] = C.CString(s.config.GmondConfig) | ||||
| 	s.gmond_config = C.Ganglia_gmond_config_create(s.cstrCache[s.config.GmondConfig], 0) | ||||
| 	//globals := C.cfg_getsec(gmond_config, s.cstrCache["globals"]) | ||||
| 	//override_hostname := C.cfg_getstr(globals, s.cstrCache["override_hostname"]) | ||||
| 	//override_ip := C.cfg_getstr(globals, s.cstrCache["override_ip"]) | ||||
|  | ||||
| 	s.send_channels = C.Ganglia_udp_send_channels_create(s.global_context, s.gmond_config) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *LibgangliaSink) Write(point lp.CCMetric) error { | ||||
| 	var err error = nil | ||||
| 	var c_name *C.char | ||||
| 	var c_value *C.char | ||||
| 	var c_type *C.char | ||||
| 	var c_unit *C.char | ||||
|  | ||||
| 	// helper function for looking up C strings in the cache | ||||
| 	lookup := func(key string) *C.char { | ||||
| 		if _, exist := s.cstrCache[key]; !exist { | ||||
| 			s.cstrCache[key] = C.CString(key) | ||||
| 		} | ||||
| 		return s.cstrCache[key] | ||||
| 	} | ||||
|  | ||||
| 	// Get metric name | ||||
| 	metricname := GangliaMetricRename(point) | ||||
| 	if s.config.AddTypeToName { | ||||
| 		c_name = lookup(GangliaMetricName(point)) | ||||
| 	} else { | ||||
| 		c_name = lookup(metricname) | ||||
| 	} | ||||
|  | ||||
| 	// Get the value C string and lookup the type string in the cache | ||||
| 	value, ok := point.GetField("value") | ||||
| 	if !ok { | ||||
| 		return fmt.Errorf("metric %s has no 'value' field", metricname) | ||||
| 	} | ||||
| 	switch real := value.(type) { | ||||
| 	case float64: | ||||
| 		c_value = C.CString(fmt.Sprintf("%f", real)) | ||||
| 		c_type = lookup("double") | ||||
| 	case float32: | ||||
| 		c_value = C.CString(fmt.Sprintf("%f", real)) | ||||
| 		c_type = lookup("float") | ||||
| 	case int64: | ||||
| 		c_value = C.CString(fmt.Sprintf("%d", real)) | ||||
| 		c_type = lookup("int32") | ||||
| 	case int32: | ||||
| 		c_value = C.CString(fmt.Sprintf("%d", real)) | ||||
| 		c_type = lookup("int32") | ||||
| 	case int: | ||||
| 		c_value = C.CString(fmt.Sprintf("%d", real)) | ||||
| 		c_type = lookup("int32") | ||||
| 	case string: | ||||
| 		c_value = C.CString(real) | ||||
| 		c_type = lookup("string") | ||||
| 	default: | ||||
| 		return fmt.Errorf("metric %s has invalid 'value' type for %s", point.Name(), s.name) | ||||
| 	} | ||||
|  | ||||
| 	// Add unit | ||||
| 	if s.config.AddUnits { | ||||
| 		if tagunit, tagok := point.GetTag("unit"); tagok { | ||||
| 			c_unit = lookup(tagunit) | ||||
| 		} else if metaunit, metaok := point.GetMeta("unit"); metaok { | ||||
| 			c_unit = lookup(metaunit) | ||||
| 		} else { | ||||
| 			c_unit = lookup("") | ||||
| 		} | ||||
| 	} else { | ||||
| 		c_unit = lookup("") | ||||
| 	} | ||||
|  | ||||
| 	// Determine the slope of the metric. Ganglia's own collector mostly use | ||||
| 	// 'both' but the mem and swap total uses 'zero'. | ||||
| 	slope := GangliaSlopeType(point) | ||||
| 	slope_type := C.GANGLIA_SLOPE_BOTH | ||||
| 	switch slope { | ||||
| 	case 0: | ||||
| 		slope_type = C.GANGLIA_SLOPE_ZERO | ||||
| 	} | ||||
|  | ||||
| 	// Create a new Ganglia metric | ||||
| 	gmetric := C.Ganglia_metric_create(s.global_context) | ||||
| 	// Set name, value, type and unit in the Ganglia metric | ||||
| 	// Since we don't have this information from the collectors, | ||||
| 	// we assume that the metric value can go up and down (slope), | ||||
| 	// and there is no maximum for 'dmax' and 'tmax'. | ||||
| 	// Ganglia's collectors set 'tmax' but not 'dmax' | ||||
| 	rval := C.int(0) | ||||
| 	rval = C.Ganglia_metric_set(gmetric, c_name, c_value, c_type, c_unit, C.uint(slope_type), 0, 0) | ||||
| 	switch rval { | ||||
| 	case 1: | ||||
| 		C.free(unsafe.Pointer(c_value)) | ||||
| 		return errors.New("invalid parameters") | ||||
| 	case 2: | ||||
| 		C.free(unsafe.Pointer(c_value)) | ||||
| 		return errors.New("one of your parameters has an invalid character '\"'") | ||||
| 	case 3: | ||||
| 		C.free(unsafe.Pointer(c_value)) | ||||
| 		return fmt.Errorf("the type parameter \"%s\" is not a valid type", C.GoString(c_type)) | ||||
| 	case 4: | ||||
| 		C.free(unsafe.Pointer(c_value)) | ||||
| 		return fmt.Errorf("the value parameter \"%s\" does not represent a number", C.GoString(c_value)) | ||||
| 	default: | ||||
| 	} | ||||
|  | ||||
| 	// Set the cluster name, otherwise it takes it from the configuration file | ||||
| 	if len(s.config.ClusterName) > 0 { | ||||
| 		C.Ganglia_metadata_add(gmetric, lookup("CLUSTER"), lookup(s.config.ClusterName)) | ||||
| 	} | ||||
| 	// Set the group metadata in the Ganglia metric if configured | ||||
| 	if group, ok := point.GetMeta("group"); ok && s.config.AddGangliaGroup { | ||||
| 		c_group := lookup(group) | ||||
| 		C.Ganglia_metadata_add(gmetric, lookup("GROUP"), c_group) | ||||
| 	} | ||||
|  | ||||
| 	// Now we send the metric | ||||
| 	// gmetric does provide some more options like description and other options | ||||
| 	// but they are not provided by the collectors | ||||
| 	rval = C.Ganglia_metric_send(gmetric, s.send_channels) | ||||
| 	if rval != 0 { | ||||
| 		err = fmt.Errorf("there was an error sending metric %s to %d of the send channels ", point.Name(), rval) | ||||
| 		// fall throuph to use Ganglia_metric_destroy from common cleanup | ||||
| 	} | ||||
| 	// Cleanup Ganglia metric | ||||
| 	C.Ganglia_metric_destroy(gmetric) | ||||
| 	// Free the value C string, the only one not stored in the cache | ||||
| 	C.free(unsafe.Pointer(c_value)) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (s *LibgangliaSink) Flush() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *LibgangliaSink) Close() { | ||||
| 	// Destroy Ganglia configration struct | ||||
| 	// (not done by gmetric, I thought I am more clever but no...) | ||||
| 	//C.Ganglia_gmond_config_destroy(s.gmond_config) | ||||
| 	// Destroy Ganglia pool | ||||
| 	C.Ganglia_pool_destroy(s.global_context) | ||||
|  | ||||
| 	// Cleanup C string cache | ||||
| 	for _, cstr := range s.cstrCache { | ||||
| 		C.free(unsafe.Pointer(cstr)) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										41
									
								
								sinks/libgangliaSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								sinks/libgangliaSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| ## `libganglia` sink | ||||
|  | ||||
| The `libganglia` sink interacts directly with the library of the [Ganglia Monitoring System](http://ganglia.info/) to submit the metrics. Consequently, it needs to be installed on all nodes. But this is commonly the case if you want to use Ganglia, because it requires at least a node daemon (`gmond` or `ganglia-monitor`) to work. | ||||
|  | ||||
| The `libganglia` sink has probably less overhead compared to the `ganglia` sink because it does not require any process generation but initializes the environment and UDP connections only once. | ||||
|  | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "libganglia", | ||||
|     "gmetric_config" : "/path/to/gmetric/config", | ||||
|     "cluster_name": "MyCluster", | ||||
|     "add_ganglia_group" : true, | ||||
|     "add_type_to_name": true, | ||||
|     "add_units" : true | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `libganglia` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `gmond_config`: Path to the Ganglia configuration file `gmond.conf` (default: `/etc/ganglia/gmond.conf`) | ||||
| - `cluster_name`: Set a cluster name for the metric. If not set, it is taken from `gmond_config` | ||||
| - `add_ganglia_group`: Add a Ganglia metric group based on meta information. Some old versions of `gmetric` do not support the `--group` option | ||||
| - `add_type_to_name`: Ganglia commonly uses only node-level metrics but with cc-metric-collector, there are metrics for cpus, memory domains, CPU sockets and the whole node. In order to get  eeng, this option prefixes the metric name with `<type><type-id>_` or `device_` depending on the metric tags and meta information. For metrics of the whole node `type=node`, no prefix is added | ||||
| - `add_units`: Add metric value unit if there is a `unit` entry in the metric tags or meta information | ||||
|  | ||||
| ### Ganglia Installation | ||||
|  | ||||
| My development system is Ubuntu 20.04. To install the required libraries with `apt`: | ||||
|  | ||||
| ``` | ||||
| $ sudo apt install libganglia1 | ||||
| ``` | ||||
|  | ||||
| The `libganglia.so` gets installed in `/usr/lib`. The Ganglia headers `libganglia1-dev` are **not** required. | ||||
|  | ||||
| I added a `Makefile` in the `sinks` subfolder that searches for the library in `/usr` and creates a symlink (`sinks/libganglia.so`) for running/building the cc-metric-collector. So just type `make` before running/building in the main folder or the `sinks` subfolder. | ||||
| @@ -1,34 +1,29 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	//	"time" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	"encoding/json" | ||||
|  | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type SinkConfig struct { | ||||
| 	Host         string `json:"host"` | ||||
| 	Port         string `json:"port"` | ||||
| 	Database     string `json:"database"` | ||||
| 	User         string `json:"user"` | ||||
| 	Password     string `json:"password"` | ||||
| 	Organization string `json:"organization"` | ||||
| 	Type         string `json:"type"` | ||||
| 	SSL          bool   `json:"ssl"` | ||||
| type defaultSinkConfig struct { | ||||
| 	MetaAsTags bool   `json:"meta_as_tags,omitempty"` | ||||
| 	Type       string `json:"type"` | ||||
| } | ||||
|  | ||||
| type Sink struct { | ||||
| 	host         string | ||||
| 	port         string | ||||
| 	user         string | ||||
| 	password     string | ||||
| 	database     string | ||||
| 	organization string | ||||
| 	ssl          bool | ||||
| type sink struct { | ||||
| 	meta_as_tags bool | ||||
| 	name         string | ||||
| } | ||||
|  | ||||
| type SinkFuncs interface { | ||||
| 	Init(config SinkConfig) error | ||||
| 	Write(point lp.MutableMetric) error | ||||
| type Sink interface { | ||||
| 	Init(config json.RawMessage) error | ||||
| 	Write(point lp.CCMetric) error | ||||
| 	Flush() error | ||||
| 	Close() | ||||
| 	Name() string | ||||
| } | ||||
|  | ||||
| func (s *sink) Name() string { | ||||
| 	return s.name | ||||
| } | ||||
|   | ||||
| @@ -2,90 +2,106 @@ package sinks | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| 	"log" | ||||
| 	"time" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| 	influx "github.com/influxdata/line-protocol" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| ) | ||||
|  | ||||
| type NatsSinkConfig struct { | ||||
| 	defaultSinkConfig | ||||
| 	Host     string `json:"host,omitempty"` | ||||
| 	Port     string `json:"port,omitempty"` | ||||
| 	Database string `json:"database,omitempty"` | ||||
| 	User     string `json:"user,omitempty"` | ||||
| 	Password string `json:"password,omitempty"` | ||||
| } | ||||
|  | ||||
| type NatsSink struct { | ||||
| 	Sink | ||||
| 	sink | ||||
| 	client  *nats.Conn | ||||
| 	encoder *lp.Encoder | ||||
| 	encoder *influx.Encoder | ||||
| 	buffer  *bytes.Buffer | ||||
| 	config  NatsSinkConfig | ||||
| } | ||||
|  | ||||
| func (s *NatsSink) connect() error { | ||||
| 	uinfo := nats.UserInfo(s.user, s.password) | ||||
| 	uri := fmt.Sprintf("nats://%s:%s", s.host, s.port) | ||||
| 	log.Print("Using URI ", uri) | ||||
| 	var err error | ||||
| 	var uinfo nats.Option = nil | ||||
| 	var nc *nats.Conn | ||||
| 	if len(s.config.User) > 0 && len(s.config.Password) > 0 { | ||||
| 		uinfo = nats.UserInfo(s.config.User, s.config.Password) | ||||
| 	} | ||||
| 	uri := fmt.Sprintf("nats://%s:%s", s.config.Host, s.config.Port) | ||||
| 	cclog.ComponentDebug(s.name, "Connect to", uri) | ||||
| 	s.client = nil | ||||
| 	nc, err := nats.Connect(uri, uinfo) | ||||
| 	if uinfo != nil { | ||||
| 		nc, err = nats.Connect(uri, uinfo) | ||||
| 	} else { | ||||
| 		nc, err = nats.Connect(uri) | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		log.Fatal(err) | ||||
| 		cclog.ComponentError(s.name, "Connect to", uri, "failed:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	s.client = nc | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *NatsSink) Init(config SinkConfig) error { | ||||
| 	if len(config.Host) == 0 || | ||||
| 		len(config.Port) == 0 || | ||||
| 		len(config.Database) == 0 { | ||||
| 		return errors.New("Not all configuration variables set required by NatsSink") | ||||
| func (s *NatsSink) Init(config json.RawMessage) error { | ||||
| 	s.name = "NatsSink" | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError(s.name, "Error reading config for", s.name, ":", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(s.config.Host) == 0 || | ||||
| 		len(s.config.Port) == 0 || | ||||
| 		len(s.config.Database) == 0 { | ||||
| 		return errors.New("not all configuration variables set required by NatsSink") | ||||
| 	} | ||||
| 	s.host = config.Host | ||||
| 	s.port = config.Port | ||||
| 	s.database = config.Database | ||||
| 	s.organization = config.Organization | ||||
| 	s.user = config.User | ||||
| 	s.password = config.Password | ||||
| 	// Setup Influx line protocol | ||||
| 	s.buffer = &bytes.Buffer{} | ||||
| 	s.buffer.Grow(1025) | ||||
| 	s.encoder = lp.NewEncoder(s.buffer) | ||||
| 	s.encoder = influx.NewEncoder(s.buffer) | ||||
| 	s.encoder.SetPrecision(time.Second) | ||||
| 	s.encoder.SetMaxLineBytes(1024) | ||||
| 	// Setup infos for connection | ||||
| 	return s.connect() | ||||
| } | ||||
|  | ||||
| func (s *NatsSink) Write(point lp.MutableMetric) error { | ||||
| func (s *NatsSink) Write(m lp.CCMetric) error { | ||||
| 	if s.client != nil { | ||||
| 		//	    var tags map[string]string | ||||
| 		//        var fields map[string]interface{} | ||||
| 		//        for _, t := range point.TagList() { | ||||
| 		//            tags[t.Key] = t.Value | ||||
| 		//        } | ||||
| 		//        for _, f := range point.FieldList() { | ||||
| 		//            fields[f.Key] = f.Value | ||||
| 		//        } | ||||
| 		//		m, err := protocol.New(point.Name(), tags, fields, point.Time()) | ||||
| 		//		if err != nil { | ||||
| 		//			log.Print(err) | ||||
| 		//			return err | ||||
| 		//		} | ||||
| 		_, err := s.encoder.Encode(point) | ||||
| 		_, err := s.encoder.Encode(m.ToPoint(s.config.MetaAsTags)) | ||||
| 		if err != nil { | ||||
| 			log.Print(err) | ||||
| 			cclog.ComponentError(s.name, "Write:", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 		s.client.Publish(s.database, s.buffer.Bytes()) | ||||
| 		s.buffer.Reset() | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *NatsSink) Flush() error { | ||||
| 	if s.client != nil { | ||||
| 		if err := s.client.Publish(s.config.Database, s.buffer.Bytes()); err != nil { | ||||
| 			cclog.ComponentError(s.name, "Flush:", err.Error()) | ||||
| 			return err | ||||
| 		} | ||||
| 		s.buffer.Reset() | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *NatsSink) Close() { | ||||
| 	log.Print("Closing Nats connection") | ||||
| 	if s.client != nil { | ||||
| 		cclog.ComponentDebug(s.name, "Close") | ||||
| 		s.client.Close() | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										28
									
								
								sinks/natsSink.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								sinks/natsSink.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| ## `nats` sink | ||||
|  | ||||
| The `nats` sink publishes all metrics into a NATS network. The publishing key is the database name provided in the configuration file | ||||
|  | ||||
|  | ||||
| ### Configuration structure | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "<name>": { | ||||
|     "type": "nats", | ||||
|     "meta_as_tags" : true, | ||||
|     "database" : "mymetrics", | ||||
|     "host": "dbhost.example.com", | ||||
|     "port": "4222", | ||||
|     "user": "exampleuser", | ||||
|     "password" : "examplepw" | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| - `type`: makes the sink an `nats` sink | ||||
| - `meta_as_tags`: print all meta information as tags in the output (optional) | ||||
| - `database`: All metrics are published with this subject | ||||
| - `host`: Hostname of the NATS server | ||||
| - `port`: Portnumber (as string) of the NATS server | ||||
| - `user`: Username for basic authentification | ||||
| - `password`: Password for basic authentification | ||||
							
								
								
									
										179
									
								
								sinks/sinkManager.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										179
									
								
								sinks/sinkManager.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,179 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"sync" | ||||
|  | ||||
| 	cclog "github.com/ClusterCockpit/cc-metric-collector/internal/ccLogger" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| const SINK_MAX_FORWARD = 50 | ||||
|  | ||||
| // Map of all available sinks | ||||
| var AvailableSinks = map[string]Sink{ | ||||
| 	"influxdb":    new(InfluxSink), | ||||
| 	"stdout":      new(StdoutSink), | ||||
| 	"nats":        new(NatsSink), | ||||
| 	"http":        new(HttpSink), | ||||
| 	"ganglia":     new(GangliaSink), | ||||
| 	"influxasync": new(InfluxAsyncSink), | ||||
| 	"libganglia":  new(LibgangliaSink), | ||||
| } | ||||
|  | ||||
| // Metric collector manager data structure | ||||
| type sinkManager struct { | ||||
| 	input      chan lp.CCMetric // input channel | ||||
| 	done       chan bool        // channel to finish / stop metric sink manager | ||||
| 	wg         *sync.WaitGroup  // wait group for all goroutines in cc-metric-collector | ||||
| 	sinks      map[string]Sink  // Mapping sink name to sink | ||||
| 	maxForward int              // number of metrics to write maximally in one iteration | ||||
| } | ||||
|  | ||||
| // Sink manager access functions | ||||
| type SinkManager interface { | ||||
| 	Init(wg *sync.WaitGroup, sinkConfigFile string) error | ||||
| 	AddInput(input chan lp.CCMetric) | ||||
| 	AddOutput(name string, config json.RawMessage) error | ||||
| 	Start() | ||||
| 	Close() | ||||
| } | ||||
|  | ||||
| // Init initializes the sink manager by: | ||||
| // * Reading its configuration file | ||||
| // * Adding the configured sinks and providing them with the corresponding config | ||||
| func (sm *sinkManager) Init(wg *sync.WaitGroup, sinkConfigFile string) error { | ||||
| 	sm.input = nil | ||||
| 	sm.done = make(chan bool) | ||||
| 	sm.wg = wg | ||||
| 	sm.sinks = make(map[string]Sink, 0) | ||||
| 	sm.maxForward = SINK_MAX_FORWARD | ||||
|  | ||||
| 	if len(sinkConfigFile) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	// Read sink config file | ||||
| 	configFile, err := os.Open(sinkConfigFile) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("SinkManager", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	defer configFile.Close() | ||||
|  | ||||
| 	// Parse config | ||||
| 	jsonParser := json.NewDecoder(configFile) | ||||
| 	var rawConfigs map[string]json.RawMessage | ||||
| 	err = jsonParser.Decode(&rawConfigs) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("SinkManager", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Start sinks | ||||
| 	for name, raw := range rawConfigs { | ||||
| 		err = sm.AddOutput(name, raw) | ||||
| 		if err != nil { | ||||
| 			cclog.ComponentError("SinkManager", err.Error()) | ||||
| 			continue | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Start starts the sink managers background task, which | ||||
| // distributes received metrics to the sinks | ||||
| func (sm *sinkManager) Start() { | ||||
| 	sm.wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer sm.wg.Done() | ||||
|  | ||||
| 		// Sink manager is done | ||||
| 		done := func() { | ||||
| 			for _, s := range sm.sinks { | ||||
| 				s.Close() | ||||
| 			} | ||||
|  | ||||
| 			close(sm.done) | ||||
| 			cclog.ComponentDebug("SinkManager", "DONE") | ||||
| 		} | ||||
|  | ||||
| 		toTheSinks := func(p lp.CCMetric) { | ||||
| 			// Send received metric to all outputs | ||||
| 			cclog.ComponentDebug("SinkManager", "WRITE", p) | ||||
| 			for _, s := range sm.sinks { | ||||
| 				if err := s.Write(p); err != nil { | ||||
| 					cclog.ComponentError("SinkManager", "WRITE", s.Name(), "write failed:", err.Error()) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-sm.done: | ||||
| 				done() | ||||
| 				return | ||||
|  | ||||
| 			case p := <-sm.input: | ||||
| 				toTheSinks(p) | ||||
| 				for i := 0; len(sm.input) > 0 && i < sm.maxForward; i++ { | ||||
| 					p := <-sm.input | ||||
| 					toTheSinks(p) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	// Sink manager is started | ||||
| 	cclog.ComponentDebug("SinkManager", "STARTED") | ||||
| } | ||||
|  | ||||
| // AddInput adds the input channel to the sink manager | ||||
| func (sm *sinkManager) AddInput(input chan lp.CCMetric) { | ||||
| 	sm.input = input | ||||
| } | ||||
|  | ||||
| func (sm *sinkManager) AddOutput(name string, rawConfig json.RawMessage) error { | ||||
| 	var err error | ||||
| 	var sinkConfig defaultSinkConfig | ||||
| 	if len(rawConfig) > 0 { | ||||
| 		err := json.Unmarshal(rawConfig, &sinkConfig) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	if _, found := AvailableSinks[sinkConfig.Type]; !found { | ||||
| 		cclog.ComponentError("SinkManager", "SKIP", name, "unknown sink:", sinkConfig.Type) | ||||
| 		return err | ||||
| 	} | ||||
| 	s := AvailableSinks[sinkConfig.Type] | ||||
| 	err = s.Init(rawConfig) | ||||
| 	if err != nil { | ||||
| 		cclog.ComponentError("SinkManager", "SKIP", s.Name(), "initialization failed:", err.Error()) | ||||
| 		return err | ||||
| 	} | ||||
| 	sm.sinks[name] = s | ||||
| 	cclog.ComponentDebug("SinkManager", "ADD SINK", s.Name(), "with name", fmt.Sprintf("'%s'", name)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Close finishes / stops the sink manager | ||||
| func (sm *sinkManager) Close() { | ||||
| 	cclog.ComponentDebug("SinkManager", "CLOSE") | ||||
| 	sm.done <- true | ||||
| 	// wait for close of channel sm.done | ||||
| 	<-sm.done | ||||
| } | ||||
|  | ||||
| // New creates a new initialized sink manager | ||||
| func New(wg *sync.WaitGroup, sinkConfigFile string) (SinkManager, error) { | ||||
| 	sm := new(sinkManager) | ||||
| 	err := sm.Init(wg, sinkConfigFile) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return sm, err | ||||
| } | ||||
| @@ -1,62 +1,67 @@ | ||||
| package sinks | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"os" | ||||
| 	"strings" | ||||
|  | ||||
| 	//	"time" | ||||
| 	lp "github.com/influxdata/line-protocol" | ||||
| 	lp "github.com/ClusterCockpit/cc-metric-collector/internal/ccMetric" | ||||
| ) | ||||
|  | ||||
| type StdoutSink struct { | ||||
| 	Sink | ||||
| 	sink   // meta_as_tags, name | ||||
| 	output *os.File | ||||
| 	config struct { | ||||
| 		defaultSinkConfig | ||||
| 		Output string `json:"output_file,omitempty"` | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *StdoutSink) Init(config SinkConfig) error { | ||||
| func (s *StdoutSink) Init(config json.RawMessage) error { | ||||
| 	s.name = "StdoutSink" | ||||
| 	if len(config) > 0 { | ||||
| 		err := json.Unmarshal(config, &s.config) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	s.output = os.Stdout | ||||
| 	if len(s.config.Output) > 0 { | ||||
| 		switch strings.ToLower(s.config.Output) { | ||||
| 		case "stdout": | ||||
| 			s.output = os.Stdout | ||||
| 		case "stderr": | ||||
| 			s.output = os.Stderr | ||||
| 		default: | ||||
| 			f, err := os.OpenFile(s.config.Output, os.O_CREATE|os.O_WRONLY, os.FileMode(0600)) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			s.output = f | ||||
| 		} | ||||
| 	} | ||||
| 	s.meta_as_tags = s.config.MetaAsTags | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *StdoutSink) Write(point lp.MutableMetric) error { | ||||
| 	var tagsstr []string | ||||
| 	var fieldstr []string | ||||
| 	for _, t := range point.TagList() { | ||||
| 		tagsstr = append(tagsstr, fmt.Sprintf("%s=%s", t.Key, t.Value)) | ||||
| 	} | ||||
| 	for _, f := range point.FieldList() { | ||||
| 		switch f.Value.(type) { | ||||
| 		case float64: | ||||
| 			if !math.IsNaN(f.Value.(float64)) { | ||||
| 				fieldstr = append(fieldstr, fmt.Sprintf("%s=%v", f.Key, f.Value.(float64))) | ||||
| 			} else { | ||||
| 				fieldstr = append(fieldstr, fmt.Sprintf("%s=0.0", f.Key)) | ||||
| 			} | ||||
| 		case float32: | ||||
| 			if !math.IsNaN(float64(f.Value.(float32))) { | ||||
| 				fieldstr = append(fieldstr, fmt.Sprintf("%s=%v", f.Key, f.Value.(float32))) | ||||
| 			} else { | ||||
| 				fieldstr = append(fieldstr, fmt.Sprintf("%s=0.0", f.Key)) | ||||
| 			} | ||||
| 		case int: | ||||
| 			fieldstr = append(fieldstr, fmt.Sprintf("%s=%d", f.Key, f.Value.(int))) | ||||
| 		case int64: | ||||
| 			fieldstr = append(fieldstr, fmt.Sprintf("%s=%d", f.Key, f.Value.(int64))) | ||||
| 		case string: | ||||
| 			fieldstr = append(fieldstr, fmt.Sprintf("%s=%q", f.Key, f.Value.(string))) | ||||
| 		default: | ||||
| 			fieldstr = append(fieldstr, fmt.Sprintf("%s=%v", f.Key, f.Value)) | ||||
| 		} | ||||
| 	} | ||||
| 	if len(tagsstr) > 0 { | ||||
| 		fmt.Printf("%s,%s %s %d\n", point.Name(), strings.Join(tagsstr, ","), strings.Join(fieldstr, ","), point.Time().Unix()) | ||||
| 	} else { | ||||
| 		fmt.Printf("%s %s %d\n", point.Name(), strings.Join(fieldstr, ","), point.Time().Unix()) | ||||
| 	} | ||||
| func (s *StdoutSink) Write(m lp.CCMetric) error { | ||||
| 	fmt.Fprint( | ||||
| 		s.output, | ||||
| 		m.ToLineProtocol(s.meta_as_tags), | ||||
| 	) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *StdoutSink) Flush() error { | ||||
| 	s.output.Sync() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *StdoutSink) Close() {} | ||||
| func (s *StdoutSink) Close() { | ||||
| 	if s.output != os.Stdout && s.output != os.Stderr { | ||||
| 		s.output.Close() | ||||
| 	} | ||||
| } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user