initial set of files for task driver skeleton
This commit is contained in:
commit
520e3b3ff8
1281 changed files with 510358 additions and 0 deletions
362
LICENSE
Normal file
362
LICENSE
Normal file
|
@ -0,0 +1,362 @@
|
||||||
|
Mozilla Public License, version 2.0
|
||||||
|
|
||||||
|
1. Definitions
|
||||||
|
|
||||||
|
1.1. "Contributor"
|
||||||
|
|
||||||
|
means each individual or legal entity that creates, contributes to the
|
||||||
|
creation of, or owns Covered Software.
|
||||||
|
|
||||||
|
1.2. "Contributor Version"
|
||||||
|
|
||||||
|
means the combination of the Contributions of others (if any) used by a
|
||||||
|
Contributor and that particular Contributor's Contribution.
|
||||||
|
|
||||||
|
1.3. "Contribution"
|
||||||
|
|
||||||
|
means Covered Software of a particular Contributor.
|
||||||
|
|
||||||
|
1.4. "Covered Software"
|
||||||
|
|
||||||
|
means Source Code Form to which the initial Contributor has attached the
|
||||||
|
notice in Exhibit A, the Executable Form of such Source Code Form, and
|
||||||
|
Modifications of such Source Code Form, in each case including portions
|
||||||
|
thereof.
|
||||||
|
|
||||||
|
1.5. "Incompatible With Secondary Licenses"
|
||||||
|
means
|
||||||
|
|
||||||
|
a. that the initial Contributor has attached the notice described in
|
||||||
|
Exhibit B to the Covered Software; or
|
||||||
|
|
||||||
|
b. that the Covered Software was made available under the terms of
|
||||||
|
version 1.1 or earlier of the License, but not also under the terms of
|
||||||
|
a Secondary License.
|
||||||
|
|
||||||
|
1.6. "Executable Form"
|
||||||
|
|
||||||
|
means any form of the work other than Source Code Form.
|
||||||
|
|
||||||
|
1.7. "Larger Work"
|
||||||
|
|
||||||
|
means a work that combines Covered Software with other material, in a
|
||||||
|
separate file or files, that is not Covered Software.
|
||||||
|
|
||||||
|
1.8. "License"
|
||||||
|
|
||||||
|
means this document.
|
||||||
|
|
||||||
|
1.9. "Licensable"
|
||||||
|
|
||||||
|
means having the right to grant, to the maximum extent possible, whether
|
||||||
|
at the time of the initial grant or subsequently, any and all of the
|
||||||
|
rights conveyed by this License.
|
||||||
|
|
||||||
|
1.10. "Modifications"
|
||||||
|
|
||||||
|
means any of the following:
|
||||||
|
|
||||||
|
a. any file in Source Code Form that results from an addition to,
|
||||||
|
deletion from, or modification of the contents of Covered Software; or
|
||||||
|
|
||||||
|
b. any new file in Source Code Form that contains any Covered Software.
|
||||||
|
|
||||||
|
1.11. "Patent Claims" of a Contributor
|
||||||
|
|
||||||
|
means any patent claim(s), including without limitation, method,
|
||||||
|
process, and apparatus claims, in any patent Licensable by such
|
||||||
|
Contributor that would be infringed, but for the grant of the License,
|
||||||
|
by the making, using, selling, offering for sale, having made, import,
|
||||||
|
or transfer of either its Contributions or its Contributor Version.
|
||||||
|
|
||||||
|
1.12. "Secondary License"
|
||||||
|
|
||||||
|
means either the GNU General Public License, Version 2.0, the GNU Lesser
|
||||||
|
General Public License, Version 2.1, the GNU Affero General Public
|
||||||
|
License, Version 3.0, or any later versions of those licenses.
|
||||||
|
|
||||||
|
1.13. "Source Code Form"
|
||||||
|
|
||||||
|
means the form of the work preferred for making modifications.
|
||||||
|
|
||||||
|
1.14. "You" (or "Your")
|
||||||
|
|
||||||
|
means an individual or a legal entity exercising rights under this
|
||||||
|
License. For legal entities, "You" includes any entity that controls, is
|
||||||
|
controlled by, or is under common control with You. For purposes of this
|
||||||
|
definition, "control" means (a) the power, direct or indirect, to cause
|
||||||
|
the direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (b) ownership of more than fifty percent (50%) of the
|
||||||
|
outstanding shares or beneficial ownership of such entity.
|
||||||
|
|
||||||
|
|
||||||
|
2. License Grants and Conditions
|
||||||
|
|
||||||
|
2.1. Grants
|
||||||
|
|
||||||
|
Each Contributor hereby grants You a world-wide, royalty-free,
|
||||||
|
non-exclusive license:
|
||||||
|
|
||||||
|
a. under intellectual property rights (other than patent or trademark)
|
||||||
|
Licensable by such Contributor to use, reproduce, make available,
|
||||||
|
modify, display, perform, distribute, and otherwise exploit its
|
||||||
|
Contributions, either on an unmodified basis, with Modifications, or
|
||||||
|
as part of a Larger Work; and
|
||||||
|
|
||||||
|
b. under Patent Claims of such Contributor to make, use, sell, offer for
|
||||||
|
sale, have made, import, and otherwise transfer either its
|
||||||
|
Contributions or its Contributor Version.
|
||||||
|
|
||||||
|
2.2. Effective Date
|
||||||
|
|
||||||
|
The licenses granted in Section 2.1 with respect to any Contribution
|
||||||
|
become effective for each Contribution on the date the Contributor first
|
||||||
|
distributes such Contribution.
|
||||||
|
|
||||||
|
2.3. Limitations on Grant Scope
|
||||||
|
|
||||||
|
The licenses granted in this Section 2 are the only rights granted under
|
||||||
|
this License. No additional rights or licenses will be implied from the
|
||||||
|
distribution or licensing of Covered Software under this License.
|
||||||
|
Notwithstanding Section 2.1(b) above, no patent license is granted by a
|
||||||
|
Contributor:
|
||||||
|
|
||||||
|
a. for any code that a Contributor has removed from Covered Software; or
|
||||||
|
|
||||||
|
b. for infringements caused by: (i) Your and any other third party's
|
||||||
|
modifications of Covered Software, or (ii) the combination of its
|
||||||
|
Contributions with other software (except as part of its Contributor
|
||||||
|
Version); or
|
||||||
|
|
||||||
|
c. under Patent Claims infringed by Covered Software in the absence of
|
||||||
|
its Contributions.
|
||||||
|
|
||||||
|
This License does not grant any rights in the trademarks, service marks,
|
||||||
|
or logos of any Contributor (except as may be necessary to comply with
|
||||||
|
the notice requirements in Section 3.4).
|
||||||
|
|
||||||
|
2.4. Subsequent Licenses
|
||||||
|
|
||||||
|
No Contributor makes additional grants as a result of Your choice to
|
||||||
|
distribute the Covered Software under a subsequent version of this
|
||||||
|
License (see Section 10.2) or under the terms of a Secondary License (if
|
||||||
|
permitted under the terms of Section 3.3).
|
||||||
|
|
||||||
|
2.5. Representation
|
||||||
|
|
||||||
|
Each Contributor represents that the Contributor believes its
|
||||||
|
Contributions are its original creation(s) or it has sufficient rights to
|
||||||
|
grant the rights to its Contributions conveyed by this License.
|
||||||
|
|
||||||
|
2.6. Fair Use
|
||||||
|
|
||||||
|
This License is not intended to limit any rights You have under
|
||||||
|
applicable copyright doctrines of fair use, fair dealing, or other
|
||||||
|
equivalents.
|
||||||
|
|
||||||
|
2.7. Conditions
|
||||||
|
|
||||||
|
Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted in
|
||||||
|
Section 2.1.
|
||||||
|
|
||||||
|
|
||||||
|
3. Responsibilities
|
||||||
|
|
||||||
|
3.1. Distribution of Source Form
|
||||||
|
|
||||||
|
All distribution of Covered Software in Source Code Form, including any
|
||||||
|
Modifications that You create or to which You contribute, must be under
|
||||||
|
the terms of this License. You must inform recipients that the Source
|
||||||
|
Code Form of the Covered Software is governed by the terms of this
|
||||||
|
License, and how they can obtain a copy of this License. You may not
|
||||||
|
attempt to alter or restrict the recipients' rights in the Source Code
|
||||||
|
Form.
|
||||||
|
|
||||||
|
3.2. Distribution of Executable Form
|
||||||
|
|
||||||
|
If You distribute Covered Software in Executable Form then:
|
||||||
|
|
||||||
|
a. such Covered Software must also be made available in Source Code Form,
|
||||||
|
as described in Section 3.1, and You must inform recipients of the
|
||||||
|
Executable Form how they can obtain a copy of such Source Code Form by
|
||||||
|
reasonable means in a timely manner, at a charge no more than the cost
|
||||||
|
of distribution to the recipient; and
|
||||||
|
|
||||||
|
b. You may distribute such Executable Form under the terms of this
|
||||||
|
License, or sublicense it under different terms, provided that the
|
||||||
|
license for the Executable Form does not attempt to limit or alter the
|
||||||
|
recipients' rights in the Source Code Form under this License.
|
||||||
|
|
||||||
|
3.3. Distribution of a Larger Work
|
||||||
|
|
||||||
|
You may create and distribute a Larger Work under terms of Your choice,
|
||||||
|
provided that You also comply with the requirements of this License for
|
||||||
|
the Covered Software. If the Larger Work is a combination of Covered
|
||||||
|
Software with a work governed by one or more Secondary Licenses, and the
|
||||||
|
Covered Software is not Incompatible With Secondary Licenses, this
|
||||||
|
License permits You to additionally distribute such Covered Software
|
||||||
|
under the terms of such Secondary License(s), so that the recipient of
|
||||||
|
the Larger Work may, at their option, further distribute the Covered
|
||||||
|
Software under the terms of either this License or such Secondary
|
||||||
|
License(s).
|
||||||
|
|
||||||
|
3.4. Notices
|
||||||
|
|
||||||
|
You may not remove or alter the substance of any license notices
|
||||||
|
(including copyright notices, patent notices, disclaimers of warranty, or
|
||||||
|
limitations of liability) contained within the Source Code Form of the
|
||||||
|
Covered Software, except that You may alter any license notices to the
|
||||||
|
extent required to remedy known factual inaccuracies.
|
||||||
|
|
||||||
|
3.5. Application of Additional Terms
|
||||||
|
|
||||||
|
You may choose to offer, and to charge a fee for, warranty, support,
|
||||||
|
indemnity or liability obligations to one or more recipients of Covered
|
||||||
|
Software. However, You may do so only on Your own behalf, and not on
|
||||||
|
behalf of any Contributor. You must make it absolutely clear that any
|
||||||
|
such warranty, support, indemnity, or liability obligation is offered by
|
||||||
|
You alone, and You hereby agree to indemnify every Contributor for any
|
||||||
|
liability incurred by such Contributor as a result of warranty, support,
|
||||||
|
indemnity or liability terms You offer. You may include additional
|
||||||
|
disclaimers of warranty and limitations of liability specific to any
|
||||||
|
jurisdiction.
|
||||||
|
|
||||||
|
4. Inability to Comply Due to Statute or Regulation
|
||||||
|
|
||||||
|
If it is impossible for You to comply with any of the terms of this License
|
||||||
|
with respect to some or all of the Covered Software due to statute,
|
||||||
|
judicial order, or regulation then You must: (a) comply with the terms of
|
||||||
|
this License to the maximum extent possible; and (b) describe the
|
||||||
|
limitations and the code they affect. Such description must be placed in a
|
||||||
|
text file included with all distributions of the Covered Software under
|
||||||
|
this License. Except to the extent prohibited by statute or regulation,
|
||||||
|
such description must be sufficiently detailed for a recipient of ordinary
|
||||||
|
skill to be able to understand it.
|
||||||
|
|
||||||
|
5. Termination
|
||||||
|
|
||||||
|
5.1. The rights granted under this License will terminate automatically if You
|
||||||
|
fail to comply with any of its terms. However, if You become compliant,
|
||||||
|
then the rights granted under this License from a particular Contributor
|
||||||
|
are reinstated (a) provisionally, unless and until such Contributor
|
||||||
|
explicitly and finally terminates Your grants, and (b) on an ongoing
|
||||||
|
basis, if such Contributor fails to notify You of the non-compliance by
|
||||||
|
some reasonable means prior to 60 days after You have come back into
|
||||||
|
compliance. Moreover, Your grants from a particular Contributor are
|
||||||
|
reinstated on an ongoing basis if such Contributor notifies You of the
|
||||||
|
non-compliance by some reasonable means, this is the first time You have
|
||||||
|
received notice of non-compliance with this License from such
|
||||||
|
Contributor, and You become compliant prior to 30 days after Your receipt
|
||||||
|
of the notice.
|
||||||
|
|
||||||
|
5.2. If You initiate litigation against any entity by asserting a patent
|
||||||
|
infringement claim (excluding declaratory judgment actions,
|
||||||
|
counter-claims, and cross-claims) alleging that a Contributor Version
|
||||||
|
directly or indirectly infringes any patent, then the rights granted to
|
||||||
|
You by any and all Contributors for the Covered Software under Section
|
||||||
|
2.1 of this License shall terminate.
|
||||||
|
|
||||||
|
5.3. In the event of termination under Sections 5.1 or 5.2 above, all end user
|
||||||
|
license agreements (excluding distributors and resellers) which have been
|
||||||
|
validly granted by You or Your distributors under this License prior to
|
||||||
|
termination shall survive termination.
|
||||||
|
|
||||||
|
6. Disclaimer of Warranty
|
||||||
|
|
||||||
|
Covered Software is provided under this License on an "as is" basis,
|
||||||
|
without warranty of any kind, either expressed, implied, or statutory,
|
||||||
|
including, without limitation, warranties that the Covered Software is free
|
||||||
|
of defects, merchantable, fit for a particular purpose or non-infringing.
|
||||||
|
The entire risk as to the quality and performance of the Covered Software
|
||||||
|
is with You. Should any Covered Software prove defective in any respect,
|
||||||
|
You (not any Contributor) assume the cost of any necessary servicing,
|
||||||
|
repair, or correction. This disclaimer of warranty constitutes an essential
|
||||||
|
part of this License. No use of any Covered Software is authorized under
|
||||||
|
this License except under this disclaimer.
|
||||||
|
|
||||||
|
7. Limitation of Liability
|
||||||
|
|
||||||
|
Under no circumstances and under no legal theory, whether tort (including
|
||||||
|
negligence), contract, or otherwise, shall any Contributor, or anyone who
|
||||||
|
distributes Covered Software as permitted above, be liable to You for any
|
||||||
|
direct, indirect, special, incidental, or consequential damages of any
|
||||||
|
character including, without limitation, damages for lost profits, loss of
|
||||||
|
goodwill, work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses, even if such party shall have been
|
||||||
|
informed of the possibility of such damages. This limitation of liability
|
||||||
|
shall not apply to liability for death or personal injury resulting from
|
||||||
|
such party's negligence to the extent applicable law prohibits such
|
||||||
|
limitation. Some jurisdictions do not allow the exclusion or limitation of
|
||||||
|
incidental or consequential damages, so this exclusion and limitation may
|
||||||
|
not apply to You.
|
||||||
|
|
||||||
|
8. Litigation
|
||||||
|
|
||||||
|
Any litigation relating to this License may be brought only in the courts
|
||||||
|
of a jurisdiction where the defendant maintains its principal place of
|
||||||
|
business and such litigation shall be governed by laws of that
|
||||||
|
jurisdiction, without reference to its conflict-of-law provisions. Nothing
|
||||||
|
in this Section shall prevent a party's ability to bring cross-claims or
|
||||||
|
counter-claims.
|
||||||
|
|
||||||
|
9. Miscellaneous
|
||||||
|
|
||||||
|
This License represents the complete agreement concerning the subject
|
||||||
|
matter hereof. If any provision of this License is held to be
|
||||||
|
unenforceable, such provision shall be reformed only to the extent
|
||||||
|
necessary to make it enforceable. Any law or regulation which provides that
|
||||||
|
the language of a contract shall be construed against the drafter shall not
|
||||||
|
be used to construe this License against a Contributor.
|
||||||
|
|
||||||
|
|
||||||
|
10. Versions of the License
|
||||||
|
|
||||||
|
10.1. New Versions
|
||||||
|
|
||||||
|
Mozilla Foundation is the license steward. Except as provided in Section
|
||||||
|
10.3, no one other than the license steward has the right to modify or
|
||||||
|
publish new versions of this License. Each version will be given a
|
||||||
|
distinguishing version number.
|
||||||
|
|
||||||
|
10.2. Effect of New Versions
|
||||||
|
|
||||||
|
You may distribute the Covered Software under the terms of the version
|
||||||
|
of the License under which You originally received the Covered Software,
|
||||||
|
or under the terms of any subsequent version published by the license
|
||||||
|
steward.
|
||||||
|
|
||||||
|
10.3. Modified Versions
|
||||||
|
|
||||||
|
If you create software not governed by this License, and you want to
|
||||||
|
create a new license for such software, you may create and use a
|
||||||
|
modified version of this License if you rename the license and remove
|
||||||
|
any references to the name of the license steward (except to note that
|
||||||
|
such modified license differs from this License).
|
||||||
|
|
||||||
|
10.4. Distributing Source Code Form that is Incompatible With Secondary
|
||||||
|
Licenses If You choose to distribute Source Code Form that is
|
||||||
|
Incompatible With Secondary Licenses under the terms of this version of
|
||||||
|
the License, the notice described in Exhibit B of this License must be
|
||||||
|
attached.
|
||||||
|
|
||||||
|
Exhibit A - Source Code Form License Notice
|
||||||
|
|
||||||
|
This Source Code Form is subject to the
|
||||||
|
terms of the Mozilla Public License, v.
|
||||||
|
2.0. If a copy of the MPL was not
|
||||||
|
distributed with this file, You can
|
||||||
|
obtain one at
|
||||||
|
http://mozilla.org/MPL/2.0/.
|
||||||
|
|
||||||
|
If it is not possible or desirable to put the notice in a particular file,
|
||||||
|
then You may include the notice in a location (such as a LICENSE file in a
|
||||||
|
relevant directory) where a recipient would be likely to look for such a
|
||||||
|
notice.
|
||||||
|
|
||||||
|
You may add additional accurate notices of copyright ownership.
|
||||||
|
|
||||||
|
Exhibit B - "Incompatible With Secondary Licenses" Notice
|
||||||
|
|
||||||
|
This Source Code Form is "Incompatible
|
||||||
|
With Secondary Licenses", as defined by
|
||||||
|
the Mozilla Public License, v. 2.0.
|
55
Makefile
Normal file
55
Makefile
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
NAME := nomad-driver-skeleton
|
||||||
|
PKG := github.com/hashicorp/$(NAME)
|
||||||
|
|
||||||
|
CGO_ENABLED := 0
|
||||||
|
|
||||||
|
# Set any default go build tags.
|
||||||
|
BUILDTAGS :=
|
||||||
|
# Set an output prefix, which is the local directory if not specified
|
||||||
|
PREFIX?=$(shell pwd)
|
||||||
|
|
||||||
|
GOOSARCHES=linux/amd64
|
||||||
|
|
||||||
|
GO=go
|
||||||
|
|
||||||
|
.PHONY: build
|
||||||
|
build: $(NAME) ## Builds a dynamic executable or package.
|
||||||
|
|
||||||
|
$(NAME): $(wildcard *.go) $(wildcard */*.go)
|
||||||
|
@echo "+ $@"
|
||||||
|
$(V)GO111MODULE=on GOOS=linux $(GO) build -tags "$(BUILDTAGS)" ${GO_LDFLAGS} -o $(NAME) ./cmd/driver/main.go
|
||||||
|
|
||||||
|
.PHONY: static
|
||||||
|
static: ## Builds a static executable.
|
||||||
|
@echo "+ $@"
|
||||||
|
CGO_ENABLED=$(CGO_ENABLED) $(GO) build \
|
||||||
|
-tags "$(BUILDTAGS) static_build" \
|
||||||
|
${GO_LDFLAGS_STATIC} -o $(NAME) .
|
||||||
|
|
||||||
|
all: clean build fmt lint test vet
|
||||||
|
|
||||||
|
.PHONY: fmt
|
||||||
|
fmt: ## Verifies all files have been `gofmt`ed.
|
||||||
|
@echo "+ $@"
|
||||||
|
@gofmt -s -l . | grep -v '.pb.go:' | grep -v vendor | tee /dev/stderr
|
||||||
|
|
||||||
|
.PHONY: test
|
||||||
|
test: ## Runs the go tests.
|
||||||
|
@echo "+ $@"
|
||||||
|
@$(GO) test -v -tags "$(BUILDTAGS) cgo" $(shell $(GO) list ./... | grep -v vendor)
|
||||||
|
|
||||||
|
.PHONY: vet
|
||||||
|
vet: ## Verifies `go vet` passes.
|
||||||
|
@echo "+ $@"
|
||||||
|
@$(GO) vet -printfuncs Error,ErrorDepth,Errorf,Errorln,Exit,ExitDepth,Exitf,Exitln,Fatal,FatalDepth,Fatalf,Fatalln,Info,InfoDepth,Infof,Infoln,Warning,WarningDepth,Warningf,Warningln -all ./...
|
||||||
|
|
||||||
|
.PHONY: clean
|
||||||
|
clean: ## Cleanup any build binaries or packages.
|
||||||
|
@echo "+ $@"
|
||||||
|
$(RM) $(NAME)
|
||||||
|
$(RM) -r $(BUILDDIR)
|
||||||
|
|
||||||
|
dep:
|
||||||
|
$(V)GO111MODULE=on go mod download
|
||||||
|
$(V)GO111MODULE=on go mod vendor
|
||||||
|
$(V)GO111MODULE=on go mod tidy
|
50
README.md
Normal file
50
README.md
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
# nomad-driver-skeleton
|
||||||
|
This project is a [Hashicorp Nomad](https://www.nomadproject.io/) runtime driver plugin skeleton implementation.
|
||||||
|
You can clone and modify this to get started writing your
|
||||||
|
own driver plugin.
|
||||||
|
|
||||||
|
Note that this is only a skeleton to help you get started. The Makefile only builds
|
||||||
|
for linux/amd64
|
||||||
|
|
||||||
|
Requirements
|
||||||
|
------------
|
||||||
|
|
||||||
|
- [Nomad](https://www.nomadproject.io/downloads.html) v0.9.5+
|
||||||
|
- [Go](https://golang.org/doc/install) v1.11+ (to build the plugin)
|
||||||
|
|
||||||
|
Get Started
|
||||||
|
-----------
|
||||||
|
|
||||||
|
Clone repository to: `$GOPATH/src/github.com/hashicorp/nomad-driver-skeleton
|
||||||
|
|
||||||
|
Copy the files into your new driver's package and change package names as needed.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
$ cd $GOPATH/src/github.com/hashicorp/nomad-driver-skeleton
|
||||||
|
$ make dep
|
||||||
|
$ make build
|
||||||
|
```
|
||||||
|
|
||||||
|
Requirements
|
||||||
|
------------
|
||||||
|
|
||||||
|
[Go](http://www.golang.org) should be installed on your machine.
|
||||||
|
You will also need to correctly setup a [GOPATH](http://golang.org/doc/code.html#GOPATH)
|
||||||
|
and add `$GOPATH/bin` to your `$PATH`.
|
||||||
|
|
||||||
|
To compile the driver plugin, run `make build`. This will build the plugin and put the task driver binary under the
|
||||||
|
NOMAD plugin dir, which by default is located under `<nomad-data-dir>/plugins/`.
|
||||||
|
Check Nomad `-data-dir` and `-plugin-dir` flags for more information.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
$ make build
|
||||||
|
```
|
||||||
|
|
||||||
|
Makefile and project module structure were inspired by [Sylab's singularity]() driver.
|
||||||
|
|
||||||
|
Code Organization
|
||||||
|
-----------------
|
||||||
|
The skeleton driver comes with an in memory task state store (see state.go).
|
||||||
|
This in-memory map is for storing runtime details (like a process id). The details
|
||||||
|
of what to store is left to the implementation of the runtime driver.
|
||||||
|
Fields stored in the base implementation in this skeleton can be found in the `taskHandle` struct.
|
21
cmd/driver/main.go
Normal file
21
cmd/driver/main.go
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// +build linux
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
log "github.com/hashicorp/go-hclog"
|
||||||
|
|
||||||
|
"github.com/hashicorp/nomad/plugins"
|
||||||
|
"github.com/hashicorp/nomad-driver-skeleton/pkg/plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// Serve the plugin
|
||||||
|
plugins.Serve(factory)
|
||||||
|
}
|
||||||
|
|
||||||
|
// factory returns a new instance of a nomad driver plugin
|
||||||
|
func factory(log log.Logger) interface{} {
|
||||||
|
return skeleton.NewSkeletonDriver(log)
|
||||||
|
}
|
||||||
|
|
49
go.mod
Normal file
49
go.mod
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
module github.com/hashicorp/nomad-driver-skeleton
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/LK4D4/joincontext v0.0.0-20171026170139-1724345da6d5 // indirect
|
||||||
|
github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6 // indirect
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7 // indirect
|
||||||
|
github.com/go-ole/go-ole v1.2.2 // indirect
|
||||||
|
github.com/golang/snappy v0.0.1 // indirect
|
||||||
|
github.com/gorhill/cronexpr v0.0.0-20140423231348-a557574d6c02 // indirect
|
||||||
|
github.com/hashicorp/consul v1.0.7 // indirect
|
||||||
|
github.com/hashicorp/go-hclog v0.8.0
|
||||||
|
github.com/hashicorp/go-plugin v0.0.0-20190212232519-b838ffee39ce // indirect
|
||||||
|
github.com/hashicorp/go-retryablehttp v0.5.3 // indirect
|
||||||
|
github.com/hashicorp/go-rootcerts v0.0.0-20160503143440-6bb64b370b90 // indirect
|
||||||
|
github.com/hashicorp/go-uuid v1.0.1 // indirect
|
||||||
|
github.com/hashicorp/go-version v1.0.0 // indirect
|
||||||
|
github.com/hashicorp/hcl v0.0.0-20161101180025-6e968a3fcdcb // indirect
|
||||||
|
github.com/hashicorp/memberlist v0.1.3 // indirect
|
||||||
|
github.com/hashicorp/nomad v0.9.0-rc2
|
||||||
|
github.com/hashicorp/raft v0.0.0-20190104133720-9c733b2b7f53 // indirect
|
||||||
|
github.com/hashicorp/serf v0.0.0-20180119224300-b6017ae61f44 // indirect
|
||||||
|
github.com/hashicorp/vault v1.1.0 // indirect
|
||||||
|
github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d // indirect
|
||||||
|
github.com/hpcloud/tail v0.0.0-20180514194441-a1dbeea552b7 // indirect
|
||||||
|
github.com/kr/pretty v0.1.0 // indirect
|
||||||
|
github.com/mitchellh/copystructure v0.0.0-20170525013902-d23ffcb85de3 // indirect
|
||||||
|
github.com/mitchellh/go-homedir v1.1.0 // indirect
|
||||||
|
github.com/mitchellh/go-testing-interface v1.0.0 // indirect
|
||||||
|
github.com/mitchellh/hashstructure v0.0.0-20160118175604-1ef5c71b025a // indirect
|
||||||
|
github.com/mitchellh/mapstructure v1.1.2 // indirect
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.0 // indirect
|
||||||
|
github.com/pascaldekloe/goe v0.1.0 // indirect
|
||||||
|
github.com/pierrec/lz4 v2.0.5+incompatible // indirect
|
||||||
|
github.com/pkg/errors v0.8.1 // indirect
|
||||||
|
github.com/ryanuber/go-glob v0.0.0-20170128012129-256dc444b735 // indirect
|
||||||
|
github.com/shirou/gopsutil v0.0.0-20190131151121-071446942108 // indirect
|
||||||
|
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4 // indirect
|
||||||
|
github.com/ugorji/go v0.0.0-20170620060102-0053ebfd9d0e // indirect
|
||||||
|
github.com/vmihailenco/msgpack v4.0.2+incompatible // indirect
|
||||||
|
github.com/zclconf/go-cty v0.0.0-20180718220526-02bd58e97b57 // indirect
|
||||||
|
golang.org/x/sys v0.0.0-20190220154126-629670e5acc5 // indirect
|
||||||
|
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 // indirect
|
||||||
|
google.golang.org/appengine v1.4.0 // indirect
|
||||||
|
google.golang.org/genproto v0.0.0-20190219182410-082222b4a5c5 // indirect
|
||||||
|
google.golang.org/grpc v1.18.0 // indirect
|
||||||
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect
|
||||||
|
gopkg.in/fsnotify/fsnotify.v1 v1.4.7 // indirect
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect
|
||||||
|
)
|
165
go.sum
Normal file
165
go.sum
Normal file
|
@ -0,0 +1,165 @@
|
||||||
|
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
|
github.com/LK4D4/joincontext v0.0.0-20171026170139-1724345da6d5 h1:U7q69tqXiCf6m097GRlNQB0/6SI1qWIOHYHhCEvDxF4=
|
||||||
|
github.com/LK4D4/joincontext v0.0.0-20171026170139-1724345da6d5/go.mod h1:nxQPcNPR/34g+HcK2hEsF99O+GJgIkW/OmPl8wtzhmk=
|
||||||
|
github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6 h1:fLjPD/aNc3UIOA6tDi6QXUemppXK3P9BI7mr2hd6gx8=
|
||||||
|
github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6/go.mod h1:3eOhrUMpNV+6aFIbp5/iudMxNCF27Vw2OZgy4xEx0Fg=
|
||||||
|
github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da h1:8GUt8eRujhVEGZFFEjBj46YV4rDjvGrNxb0KMWYkL2I=
|
||||||
|
github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY=
|
||||||
|
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
||||||
|
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/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I=
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
||||||
|
github.com/go-ole/go-ole v1.2.2 h1:QNWhweRd9D5Py2rRVboZ2L4SEoW/dyraWJCc8bgS8kE=
|
||||||
|
github.com/go-ole/go-ole v1.2.2/go.mod h1:pnvuG7BrDMZ8ifMurTQmxwhQM/odqm9sSqNe5BUI7v4=
|
||||||
|
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58=
|
||||||
|
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
||||||
|
github.com/golang/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:tluoj9z5200jBnyusfRPU2LqT6J+DAorxEvtC7LHB+E=
|
||||||
|
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||||
|
github.com/golang/protobuf v1.2.0 h1:P3YflyNX/ehuJFLhxviNdFxQPkGK5cDcApsge1SqnvM=
|
||||||
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
|
github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
|
||||||
|
github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
||||||
|
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c h1:964Od4U6p2jUkFxvCydnIczKteheJEzHRToSGK3Bnlw=
|
||||||
|
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
||||||
|
github.com/gorhill/cronexpr v0.0.0-20140423231348-a557574d6c02 h1:Spo+4PFAGDqULAsZ7J69MOxq4/fwgZ0zvmDTBqpq7yU=
|
||||||
|
github.com/gorhill/cronexpr v0.0.0-20140423231348-a557574d6c02/go.mod h1:g2644b03hfBX9Ov0ZBDgXXens4rxSxmqFBbhvKv2yVA=
|
||||||
|
github.com/hashicorp/consul v1.0.7 h1:GuHjalgSkzFdlBVxfKzlLJJ6bT4VCvAXV678UWpuJbw=
|
||||||
|
github.com/hashicorp/consul v1.0.7/go.mod h1:mFrjN1mfidgJfYP1xrJCF+AfRhr6Eaqhb2+sfyn/OOI=
|
||||||
|
github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA=
|
||||||
|
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
|
||||||
|
github.com/hashicorp/go-cleanhttp v0.5.0 h1:wvCrVc9TjDls6+YGAF2hAifE1E5U1+b4tH6KdvN3Gig=
|
||||||
|
github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
|
||||||
|
github.com/hashicorp/go-hclog v0.0.0-20180709165350-ff2cf002a8dd/go.mod h1:9bjs9uLqI8l75knNv3lV1kA55veR+WUPSiKIWcQHudI=
|
||||||
|
github.com/hashicorp/go-hclog v0.8.0 h1:z3ollgGRg8RjfJH6UVBaG54R70GFd++QOkvnJH3VSBY=
|
||||||
|
github.com/hashicorp/go-hclog v0.8.0/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ=
|
||||||
|
github.com/hashicorp/go-immutable-radix v1.0.0 h1:AKDB1HM5PWEA7i4nhcpwOrO2byshxBjXVn/J/3+z5/0=
|
||||||
|
github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60=
|
||||||
|
github.com/hashicorp/go-msgpack v0.5.3 h1:zKjpN5BK/P5lMYrLmBHdBULWbJ0XpYR+7NGzqkZzoD4=
|
||||||
|
github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM=
|
||||||
|
github.com/hashicorp/go-multierror v1.0.0 h1:iVjPR7a6H0tWELX5NxNe7bYopibicUzc7uPribsnS6o=
|
||||||
|
github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
|
||||||
|
github.com/hashicorp/go-plugin v0.0.0-20190212232519-b838ffee39ce h1:I3KJUf8jyMubLFeHit2ibr9YeVxJX2CXMXVM6xlB+Qk=
|
||||||
|
github.com/hashicorp/go-plugin v0.0.0-20190212232519-b838ffee39ce/go.mod h1:++UyYGoz3o5w9ZzAdZxtQKrWWP+iqPBn3cQptSMzBuY=
|
||||||
|
github.com/hashicorp/go-retryablehttp v0.5.3 h1:QlWt0KvWT0lq8MFppF9tsJGF+ynG7ztc2KIPhzRGk7s=
|
||||||
|
github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs=
|
||||||
|
github.com/hashicorp/go-rootcerts v0.0.0-20160503143440-6bb64b370b90 h1:VBj0QYQ0u2MCJzBfeYXGexnAl17GsH1yidnoxCqqD9E=
|
||||||
|
github.com/hashicorp/go-rootcerts v0.0.0-20160503143440-6bb64b370b90/go.mod h1:o4zcYY1e0GEZI6eSEr+43QDYmuGglw1qSO6qdHUHCgg=
|
||||||
|
github.com/hashicorp/go-sockaddr v1.0.0 h1:GeH6tui99pF4NJgfnhp+L6+FfobzVW3Ah46sLo0ICXs=
|
||||||
|
github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU=
|
||||||
|
github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
|
||||||
|
github.com/hashicorp/go-uuid v1.0.1 h1:fv1ep09latC32wFoVwnqcnKJGnMSdBanPczbHAYm1BE=
|
||||||
|
github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
|
||||||
|
github.com/hashicorp/go-version v1.0.0 h1:21MVWPKDphxa7ineQQTrCU5brh7OuVVAzGOCnnCPtE8=
|
||||||
|
github.com/hashicorp/go-version v1.0.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
|
||||||
|
github.com/hashicorp/golang-lru v0.5.0 h1:CL2msUPvZTLb5O648aiLNJw3hnBxN2+1Jq8rCOH9wdo=
|
||||||
|
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
|
github.com/hashicorp/hcl v0.0.0-20161101180025-6e968a3fcdcb h1:S76lj//ZzHoMTiOOi3P9NcBQhoUxi4ZnYTcGqQnkCcE=
|
||||||
|
github.com/hashicorp/hcl v0.0.0-20161101180025-6e968a3fcdcb/go.mod h1:oZtUIOe8dh44I2q6ScRibXws4Ajl+d+nod3AaR9vL5w=
|
||||||
|
github.com/hashicorp/memberlist v0.1.3 h1:EmmoJme1matNzb+hMpDuR/0sbJSUisxyqBGG676r31M=
|
||||||
|
github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I=
|
||||||
|
github.com/hashicorp/nomad v0.9.0-rc2 h1:sN6Dy8ve/fwrWFSNzYSikYaXqqoUm1o4pqtJIOTaRQo=
|
||||||
|
github.com/hashicorp/nomad v0.9.0-rc2/go.mod h1:WRaKjdO1G2iqi86TvTjIYtKTyxg4pl7NLr9InxtWaI0=
|
||||||
|
github.com/hashicorp/raft v0.0.0-20190104133720-9c733b2b7f53 h1:Kcs5PApYrQyn+RLWUvk3eQhKNGDj8M/r1mt0X8p9HAA=
|
||||||
|
github.com/hashicorp/raft v0.0.0-20190104133720-9c733b2b7f53/go.mod h1:DVSAWItjLjTOkVbSpWQ0j0kUADIvDaCtBxIcbNAQLkI=
|
||||||
|
github.com/hashicorp/serf v0.0.0-20180119224300-b6017ae61f44 h1:uY64+3NuMxFyc9+nchedxCBabBhU67QF+ymrIedtexE=
|
||||||
|
github.com/hashicorp/serf v0.0.0-20180119224300-b6017ae61f44/go.mod h1:h/Ru6tmZazX7WO/GDmwdpS975F019L4t5ng5IgwbNrE=
|
||||||
|
github.com/hashicorp/vault v1.1.0 h1:v79NUgO5xCZnXVzUkIqFOXtP8YhpnHAi1fk3eo9cuOE=
|
||||||
|
github.com/hashicorp/vault v1.1.0/go.mod h1:KfSyffbKxoVyspOdlaGVjIuwLobi07qD1bAbosPMpP0=
|
||||||
|
github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM=
|
||||||
|
github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d h1:kJCB4vdITiW1eC1vq2e6IsrXKrZit1bv/TDYFGMp4BQ=
|
||||||
|
github.com/hashicorp/yamux v0.0.0-20181012175058-2f1d1f20f75d/go.mod h1:+NfK9FKeTrX5uv1uIXGdwYDTeHna2qgaIlx54MXqjAM=
|
||||||
|
github.com/hpcloud/tail v0.0.0-20180514194441-a1dbeea552b7 h1:lus8hJKTrh146vNoUWNHv2F1jdtsud5ajNL0/YndJUw=
|
||||||
|
github.com/hpcloud/tail v0.0.0-20180514194441-a1dbeea552b7/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
|
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
|
||||||
|
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/miekg/dns v1.0.14 h1:9jZdLNd/P4+SfEJ0TNyxYpsK8N4GtfylBLqtbYN1sbA=
|
||||||
|
github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg=
|
||||||
|
github.com/mitchellh/copystructure v0.0.0-20170525013902-d23ffcb85de3 h1:dECZqiJYhKdj9QlLpiQaRDXHDXRTdiyZI3owdDGhlYY=
|
||||||
|
github.com/mitchellh/copystructure v0.0.0-20170525013902-d23ffcb85de3/go.mod h1:eOsF2yLPlBBJPvD+nhl5QMTBSOBbOph6N7j/IDUw7PY=
|
||||||
|
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
|
||||||
|
github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
|
||||||
|
github.com/mitchellh/go-testing-interface v0.0.0-20171004221916-a61a99592b77/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
|
||||||
|
github.com/mitchellh/go-testing-interface v1.0.0 h1:fzU/JVNcaqHQEcVFAKeR41fkiLdIPrefOvVG1VZ96U0=
|
||||||
|
github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
|
||||||
|
github.com/mitchellh/hashstructure v0.0.0-20160118175604-1ef5c71b025a h1:2p2+qxgbcNvMPt/Q/Nfd0mEHJokUf7rHAYDVQH1dKpM=
|
||||||
|
github.com/mitchellh/hashstructure v0.0.0-20160118175604-1ef5c71b025a/go.mod h1:QjSHrPWS+BGUVBYkbTZWEnOh3G1DutKwClXU/ABz6AQ=
|
||||||
|
github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE=
|
||||||
|
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.0 h1:9D+8oIskB4VJBN5SFlmc27fSlIBZaov1Wpk/IfikLNY=
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.0/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
|
||||||
|
github.com/oklog/run v1.0.0 h1:Ru7dDtJNOyC66gQ5dQmaCa0qIsAUFY3sFpK1Xk8igrw=
|
||||||
|
github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA=
|
||||||
|
github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc=
|
||||||
|
github.com/pascaldekloe/goe v0.1.0 h1:cBOtyMzM9HTpWjXfbbunk26uA6nG3a8n06Wieeh0MwY=
|
||||||
|
github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc=
|
||||||
|
github.com/pierrec/lz4 v2.0.5+incompatible h1:2xWsjqPFWcplujydGg4WmhC/6fZqK42wMM8aXeqhl0I=
|
||||||
|
github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY=
|
||||||
|
github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
|
||||||
|
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
|
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/ryanuber/go-glob v0.0.0-20170128012129-256dc444b735 h1:7YvPJVmEeFHR1Tj9sZEYsmarJEQfMVYpd/Vyy/A8dqE=
|
||||||
|
github.com/ryanuber/go-glob v0.0.0-20170128012129-256dc444b735/go.mod h1:807d1WSdnB0XRJzKNil9Om6lcp/3a0v4qIHxIXzX/Yc=
|
||||||
|
github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529 h1:nn5Wsu0esKSJiIVhscUtVbo7ada43DJhG55ua/hjS5I=
|
||||||
|
github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc=
|
||||||
|
github.com/shirou/gopsutil v0.0.0-20190131151121-071446942108 h1:XXgDK65TPH+Qbo2sdYHldM5avclwThBXVYZHxroFkTQ=
|
||||||
|
github.com/shirou/gopsutil v0.0.0-20190131151121-071446942108/go.mod h1:5b4v6he4MtMOwMlS0TUMTu2PcXUg8+E1lC7eC3UO/RA=
|
||||||
|
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4 h1:udFKJ0aHUL60LboW/A+DfgoHVedieIzIXE8uylPue0U=
|
||||||
|
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
|
||||||
|
github.com/stretchr/objx v0.1.0/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 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q=
|
||||||
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
|
github.com/ugorji/go v0.0.0-20170620060102-0053ebfd9d0e h1:Eurc/1QbldPTy6eU1WSHOH1vuFc7BAUdKDWSsZd4ieo=
|
||||||
|
github.com/ugorji/go v0.0.0-20170620060102-0053ebfd9d0e/go.mod h1:hnLbHMwcvSihnDhEfx2/BzKp2xb0Y+ErdfYcrs9tkJQ=
|
||||||
|
github.com/vmihailenco/msgpack v4.0.2+incompatible h1:6ujmmycMfB62Mwv2N4atpnf8CKLSzhgodqMenpELKIQ=
|
||||||
|
github.com/vmihailenco/msgpack v4.0.2+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk=
|
||||||
|
github.com/zclconf/go-cty v0.0.0-20180718220526-02bd58e97b57 h1:FWjfWQtNK83IJMe58FfMCSNkySe0PO9TvrQyLOKcPxw=
|
||||||
|
github.com/zclconf/go-cty v0.0.0-20180718220526-02bd58e97b57/go.mod h1:LnDKxj8gN4aatfXUqmUNooaDjvmDcLPbAN3hYBIVoJE=
|
||||||
|
golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3 h1:KYQXGkl6vs02hK7pK4eIbw0NpNPedieTSTEiJ//bwGs=
|
||||||
|
golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
|
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
|
golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
|
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
|
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-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
golang.org/x/net v0.0.0-20181106065722-10aee1819953 h1:LuZIitY8waaxUfNIdtajyE/YzA/zyf0YxXG27VpLrkg=
|
||||||
|
golang.org/x/net v0.0.0-20181106065722-10aee1819953/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 h1:YUO/7uOKsKeq9UokNS62b8FYywz3ker1l1vDZRCRefw=
|
||||||
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/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-20190129075346-302c3dd5f1cc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190220154126-629670e5acc5 h1:3Nsfe5Xa1wTt01QxlAFIY5j9ycDtS+d7mhvI8ZY5bn0=
|
||||||
|
golang.org/x/sys v0.0.0-20190220154126-629670e5acc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg=
|
||||||
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
|
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ=
|
||||||
|
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
|
golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
||||||
|
google.golang.org/appengine v1.4.0 h1:/wp5JvzpHIxhs/dumFmF7BXTf3Z+dd4uXta4kVyO508=
|
||||||
|
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-20190219182410-082222b4a5c5 h1:SdCO7As+ChE1iV3IjBleIIWlj8VjZWuYEUF5pjELOOQ=
|
||||||
|
google.golang.org/genproto v0.0.0-20190219182410-082222b4a5c5/go.mod h1:L3J43x8/uS+qIUoksaLKe6OS3nUKxOKuIFz1sl2/jx4=
|
||||||
|
google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw=
|
||||||
|
google.golang.org/grpc v1.16.0/go.mod h1:0JHn/cJsOMiMfNA9+DeHDlAU7KAAB5GDlYFpa9MZMio=
|
||||||
|
google.golang.org/grpc v1.18.0 h1:IZl7mfBGfbhYx2p2rKRtYgDFw6SBz+kclmxYrCksPPA=
|
||||||
|
google.golang.org/grpc v1.18.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs=
|
||||||
|
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/fsnotify/fsnotify.v1 v1.4.7 h1:XNNYLJHt73EyYiCZi6+xjupS9CpvmiDgjPTAjrBlQbo=
|
||||||
|
gopkg.in/fsnotify/fsnotify.v1 v1.4.7/go.mod h1:Fyux9zXlo4rWoMSIzpn9fDAYjalPqJ/K1qJ27s+7ltE=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
||||||
|
honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
330
pkg/plugin/driver.go
Normal file
330
pkg/plugin/driver.go
Normal file
|
@ -0,0 +1,330 @@
|
||||||
|
package skeleton
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
hclog "github.com/hashicorp/go-hclog"
|
||||||
|
"github.com/hashicorp/nomad/drivers/shared/eventer"
|
||||||
|
"github.com/hashicorp/nomad/plugins/base"
|
||||||
|
"github.com/hashicorp/nomad/plugins/drivers"
|
||||||
|
"github.com/hashicorp/nomad/plugins/shared/hclspec"
|
||||||
|
structs "github.com/hashicorp/nomad/plugins/shared/structs"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// pluginName is the name of the plugin
|
||||||
|
pluginName = "test-plugin"
|
||||||
|
|
||||||
|
// fingerprintPeriod is the interval at which the driver will send fingerprint responses
|
||||||
|
fingerprintPeriod = 30 * time.Second
|
||||||
|
|
||||||
|
// taskHandleVersion is the version of task handle which this driver sets
|
||||||
|
// and understands how to decode driver state
|
||||||
|
taskHandleVersion = 1
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// pluginInfo is the response returned for the PluginInfo RPC
|
||||||
|
pluginInfo = &base.PluginInfoResponse{
|
||||||
|
Type: base.PluginTypeDriver,
|
||||||
|
PluginApiVersions: []string{"0.1.0"},
|
||||||
|
PluginVersion: "0.0.1",
|
||||||
|
Name: pluginName,
|
||||||
|
}
|
||||||
|
|
||||||
|
// configSpec is the hcl specification returned by the ConfigSchema RPC
|
||||||
|
configSpec = hclspec.NewObject(map[string]*hclspec.Spec{
|
||||||
|
"enabled": hclspec.NewDefault(
|
||||||
|
hclspec.NewAttr("enabled", "bool", false),
|
||||||
|
hclspec.NewLiteral("true"),
|
||||||
|
),
|
||||||
|
})
|
||||||
|
|
||||||
|
// taskConfigSpec is the hcl specification for the driver config section of
|
||||||
|
// a taskConfig within a job. It is returned in the TaskConfigSchema RPC
|
||||||
|
taskConfigSpec = hclspec.NewObject(map[string]*hclspec.Spec{
|
||||||
|
"strconfig": hclspec.NewAttr("strconfig", "string", true),
|
||||||
|
"boolcfg": hclspec.NewAttr("boolcfg", "bool", false),
|
||||||
|
})
|
||||||
|
|
||||||
|
// capabilities is returned by the Capabilities RPC and indicates what
|
||||||
|
// optional features this driver supports. This should be set according
|
||||||
|
// to the target run time
|
||||||
|
capabilities = &drivers.Capabilities{
|
||||||
|
SendSignals: true,
|
||||||
|
Exec: false,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// Driver is a skeleton driver implementation. The fields in this struct
|
||||||
|
// are to help plugin authors get started writing their runtime plugins.
|
||||||
|
type Driver struct {
|
||||||
|
// eventer is used to handle multiplexing of TaskEvents calls such that an
|
||||||
|
// event can be broadcast to all callers
|
||||||
|
eventer *eventer.Eventer
|
||||||
|
|
||||||
|
// config is the driver configuration set by the SetConfig RPC
|
||||||
|
config *Config
|
||||||
|
|
||||||
|
// nomadConfig is the client config from nomad
|
||||||
|
nomadConfig *base.ClientDriverConfig
|
||||||
|
|
||||||
|
// tasks is the in memory datastore mapping taskIDs to rawExecDriverHandles
|
||||||
|
tasks *taskStore
|
||||||
|
|
||||||
|
// ctx is the context for the driver. It is passed to other subsystems to
|
||||||
|
// coordinate shutdown
|
||||||
|
ctx context.Context
|
||||||
|
|
||||||
|
// signalShutdown is called when the driver is shutting down and cancels the
|
||||||
|
// ctx passed to any subsystems
|
||||||
|
signalShutdown context.CancelFunc
|
||||||
|
|
||||||
|
// logger will log to the Nomad agent
|
||||||
|
logger hclog.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// Config is the driver configuration set by the SetConfig RPC call
|
||||||
|
type Config struct {
|
||||||
|
// Enabled is set to true to enable this driver
|
||||||
|
Enabled bool `codec:"enabled"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// TaskConfig is the driver configuration of a task within a job
|
||||||
|
type TaskConfig struct {
|
||||||
|
StrConfig string `codec:"strconfig"`
|
||||||
|
BoolConfig bool `codec:"boolconfig"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// TaskState is the state which is encoded in the handle returned in
|
||||||
|
// StartTask. This information is needed to rebuild the task state and handler
|
||||||
|
// during recovery.
|
||||||
|
type TaskState struct {
|
||||||
|
TaskConfig *drivers.TaskConfig
|
||||||
|
ContainerName string
|
||||||
|
StartedAt time.Time
|
||||||
|
PID int
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSkeletonDriver returns a new DriverPlugin implementation
|
||||||
|
func NewSkeletonDriver(logger hclog.Logger) drivers.DriverPlugin {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
logger = logger.Named(pluginName)
|
||||||
|
|
||||||
|
return &Driver{
|
||||||
|
eventer: eventer.NewEventer(ctx, logger),
|
||||||
|
config: &Config{},
|
||||||
|
tasks: newTaskStore(),
|
||||||
|
ctx: ctx,
|
||||||
|
signalShutdown: cancel,
|
||||||
|
logger: logger,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) PluginInfo() (*base.PluginInfoResponse, error) {
|
||||||
|
return pluginInfo, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) ConfigSchema() (*hclspec.Spec, error) {
|
||||||
|
return configSpec, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) SetConfig(cfg *base.Config) error {
|
||||||
|
var config Config
|
||||||
|
if len(cfg.PluginConfig) != 0 {
|
||||||
|
if err := base.MsgPackDecode(cfg.PluginConfig, &config); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
d.config = &config
|
||||||
|
if cfg.AgentConfig != nil {
|
||||||
|
d.nomadConfig = cfg.AgentConfig.Driver
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) Shutdown(ctx context.Context) error {
|
||||||
|
d.signalShutdown()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) TaskConfigSchema() (*hclspec.Spec, error) {
|
||||||
|
return taskConfigSpec, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) Capabilities() (*drivers.Capabilities, error) {
|
||||||
|
return capabilities, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) Fingerprint(ctx context.Context) (<-chan *drivers.Fingerprint, error) {
|
||||||
|
ch := make(chan *drivers.Fingerprint)
|
||||||
|
go d.handleFingerprint(ctx, ch)
|
||||||
|
return ch, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) handleFingerprint(ctx context.Context, ch chan<- *drivers.Fingerprint) {
|
||||||
|
defer close(ch)
|
||||||
|
ticker := time.NewTimer(0)
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
return
|
||||||
|
case <-d.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-ticker.C:
|
||||||
|
ticker.Reset(fingerprintPeriod)
|
||||||
|
ch <- d.buildFingerprint()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) buildFingerprint() *drivers.Fingerprint {
|
||||||
|
var health drivers.HealthState
|
||||||
|
var desc string
|
||||||
|
attrs := map[string]*structs.Attribute{}
|
||||||
|
|
||||||
|
// Implement finger printing logic here to populate health and driver attributes
|
||||||
|
|
||||||
|
return &drivers.Fingerprint{
|
||||||
|
Attributes: attrs,
|
||||||
|
Health: health,
|
||||||
|
HealthDescription: desc,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) RecoverTask(handle *drivers.TaskHandle) error {
|
||||||
|
if handle == nil {
|
||||||
|
return fmt.Errorf("error: handle cannot be nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, ok := d.tasks.Get(handle.Config.ID); ok {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var taskState TaskState
|
||||||
|
if err := handle.GetDriverState(&taskState); err != nil {
|
||||||
|
return fmt.Errorf("failed to decode task state from handle: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var driverConfig TaskConfig
|
||||||
|
if err := taskState.TaskConfig.DecodeDriverConfig(&driverConfig); err != nil {
|
||||||
|
return fmt.Errorf("failed to decode driver config: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implement driver specific way to reattach to a running task or restart
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// StartTask setup the task exec and calls the container excecutor
|
||||||
|
func (d *Driver) StartTask(cfg *drivers.TaskConfig) (*drivers.TaskHandle, *drivers.DriverNetwork, error) {
|
||||||
|
if _, ok := d.tasks.Get(cfg.ID); ok {
|
||||||
|
return nil, nil, fmt.Errorf("task with ID %q already started", cfg.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
var driverConfig TaskConfig
|
||||||
|
if err := cfg.DecodeDriverConfig(&driverConfig); err != nil {
|
||||||
|
return nil, nil, fmt.Errorf("failed to decode driver config: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
handle := drivers.NewTaskHandle(taskHandleVersion)
|
||||||
|
handle.Config = cfg
|
||||||
|
|
||||||
|
// Implement driver specific mechanism to start the task here
|
||||||
|
|
||||||
|
return nil, nil, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) WaitTask(ctx context.Context, taskID string) (<-chan *drivers.ExitResult, error) {
|
||||||
|
handle, ok := d.tasks.Get(taskID)
|
||||||
|
if !ok {
|
||||||
|
return nil, drivers.ErrTaskNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
ch := make(chan *drivers.ExitResult)
|
||||||
|
go d.handleWait(ctx, handle, ch)
|
||||||
|
|
||||||
|
return ch, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) handleWait(ctx context.Context, handle *taskHandle, ch chan *drivers.ExitResult) {
|
||||||
|
defer close(ch)
|
||||||
|
|
||||||
|
ticker := time.NewTicker(1 * time.Second)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
return
|
||||||
|
case <-d.ctx.Done():
|
||||||
|
return
|
||||||
|
case <-ticker.C:
|
||||||
|
s := handle.TaskStatus()
|
||||||
|
if s.State == drivers.TaskStateExited {
|
||||||
|
ch <- handle.exitResult
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) StopTask(taskID string, timeout time.Duration, signal string) error {
|
||||||
|
handle, ok := d.tasks.Get(taskID)
|
||||||
|
if !ok {
|
||||||
|
return drivers.ErrTaskNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := handle.shutdown(timeout); err != nil {
|
||||||
|
return fmt.Errorf("executor Shutdown failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) DestroyTask(taskID string, force bool) error {
|
||||||
|
handle, ok := d.tasks.Get(taskID)
|
||||||
|
if !ok {
|
||||||
|
return drivers.ErrTaskNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
if handle.IsRunning() && !force {
|
||||||
|
return fmt.Errorf("cannot destroy running task")
|
||||||
|
}
|
||||||
|
|
||||||
|
d.tasks.Delete(taskID)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) InspectTask(taskID string) (*drivers.TaskStatus, error) {
|
||||||
|
handle, ok := d.tasks.Get(taskID)
|
||||||
|
if !ok {
|
||||||
|
return nil, drivers.ErrTaskNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
return handle.TaskStatus(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) TaskStats(ctx context.Context, taskID string, interval time.Duration) (<-chan *drivers.TaskResourceUsage, error) {
|
||||||
|
handle, ok := d.tasks.Get(taskID)
|
||||||
|
if !ok {
|
||||||
|
return nil, drivers.ErrTaskNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
return handle.stats(ctx, interval)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) TaskEvents(ctx context.Context) (<-chan *drivers.TaskEvent, error) {
|
||||||
|
return d.eventer.TaskEvents(ctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) SignalTask(taskID string, signal string) error {
|
||||||
|
return fmt.Errorf("This driver does not support signals")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Driver) ExecTask(taskID string, cmd []string, timeout time.Duration) (*drivers.ExecTaskResult, error) {
|
||||||
|
return nil, fmt.Errorf("This driver does not support exec") //TODO
|
||||||
|
}
|
79
pkg/plugin/handle.go
Normal file
79
pkg/plugin/handle.go
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
package skeleton
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strconv"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/hashicorp/go-hclog"
|
||||||
|
"github.com/hashicorp/nomad/plugins/drivers"
|
||||||
|
)
|
||||||
|
|
||||||
|
// taskHandle should store all relevant runtime information
|
||||||
|
// such as process ID if this is a local task or other meta
|
||||||
|
// data if this driver deals with external APIs
|
||||||
|
type taskHandle struct {
|
||||||
|
pid int
|
||||||
|
logger hclog.Logger
|
||||||
|
|
||||||
|
// stateLock syncs access to all fields below
|
||||||
|
stateLock sync.RWMutex
|
||||||
|
|
||||||
|
taskConfig *drivers.TaskConfig
|
||||||
|
procState drivers.TaskState
|
||||||
|
startedAt time.Time
|
||||||
|
completedAt time.Time
|
||||||
|
exitResult *drivers.ExitResult
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *taskHandle) TaskStatus() *drivers.TaskStatus {
|
||||||
|
h.stateLock.RLock()
|
||||||
|
defer h.stateLock.RUnlock()
|
||||||
|
|
||||||
|
return &drivers.TaskStatus{
|
||||||
|
ID: h.taskConfig.ID,
|
||||||
|
Name: h.taskConfig.Name,
|
||||||
|
State: h.procState,
|
||||||
|
StartedAt: h.startedAt,
|
||||||
|
CompletedAt: h.completedAt,
|
||||||
|
ExitResult: h.exitResult,
|
||||||
|
DriverAttributes: map[string]string{
|
||||||
|
"pid": strconv.Itoa(h.pid),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *taskHandle) IsRunning() bool {
|
||||||
|
h.stateLock.RLock()
|
||||||
|
defer h.stateLock.RUnlock()
|
||||||
|
return h.procState == drivers.TaskStateRunning
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *taskHandle) run() {
|
||||||
|
h.stateLock.Lock()
|
||||||
|
if h.exitResult == nil {
|
||||||
|
h.exitResult = &drivers.ExitResult{}
|
||||||
|
}
|
||||||
|
h.stateLock.Unlock()
|
||||||
|
|
||||||
|
h.stateLock.Lock()
|
||||||
|
defer h.stateLock.Unlock()
|
||||||
|
|
||||||
|
// Implement running the task here, this is driver specific
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *taskHandle) stats(ctx context.Context, interval time.Duration) (<-chan *drivers.TaskResourceUsage, error) {
|
||||||
|
ch := make(chan *drivers.TaskResourceUsage)
|
||||||
|
|
||||||
|
return ch, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// shutdown shuts down the container, with `timeout` grace period
|
||||||
|
// before killing the container with SIGKILL.
|
||||||
|
func (h *taskHandle) shutdown(timeout time.Duration) error {
|
||||||
|
// Implement shutdown of the task. Shutdown should be tried
|
||||||
|
// gracefully first. If its past the timeout, then this method
|
||||||
|
// should do a hard shutdown and return
|
||||||
|
return nil
|
||||||
|
}
|
36
pkg/plugin/state.go
Normal file
36
pkg/plugin/state.go
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
package skeleton
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
// taskStore provides a mechanism to store and retrieve
|
||||||
|
// task handles given a string identifier. The ID should
|
||||||
|
// be unique per task
|
||||||
|
type taskStore struct {
|
||||||
|
store map[string]*taskHandle
|
||||||
|
lock sync.RWMutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func newTaskStore() *taskStore {
|
||||||
|
return &taskStore{store: map[string]*taskHandle{}}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ts *taskStore) Set(id string, handle *taskHandle) {
|
||||||
|
ts.lock.Lock()
|
||||||
|
defer ts.lock.Unlock()
|
||||||
|
ts.store[id] = handle
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ts *taskStore) Get(id string) (*taskHandle, bool) {
|
||||||
|
ts.lock.RLock()
|
||||||
|
defer ts.lock.RUnlock()
|
||||||
|
t, ok := ts.store[id]
|
||||||
|
return t, ok
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ts *taskStore) Delete(id string) {
|
||||||
|
ts.lock.Lock()
|
||||||
|
defer ts.lock.Unlock()
|
||||||
|
delete(ts.store, id)
|
||||||
|
}
|
13
vendor/github.com/LK4D4/joincontext/.travis.yml
generated
vendored
Normal file
13
vendor/github.com/LK4D4/joincontext/.travis.yml
generated
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
language: go
|
||||||
|
sudo: false
|
||||||
|
go:
|
||||||
|
- 1.9
|
||||||
|
- master
|
||||||
|
install:
|
||||||
|
- go get -t ./...
|
||||||
|
- go get github.com/golang/lint/golint
|
||||||
|
script:
|
||||||
|
- go vet ./...
|
||||||
|
- golint -set_exit_status ./...
|
||||||
|
- test -z "$(gofmt -s -l . | tee /dev/stderr)"
|
||||||
|
- go test -race -v ./...
|
21
vendor/github.com/LK4D4/joincontext/LICENSE
generated
vendored
Normal file
21
vendor/github.com/LK4D4/joincontext/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2016 Alexander Morozov
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
39
vendor/github.com/LK4D4/joincontext/README.md
generated
vendored
Normal file
39
vendor/github.com/LK4D4/joincontext/README.md
generated
vendored
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
# joincontext
|
||||||
|
|
||||||
|
[![Build Status](https://travis-ci.org/LK4D4/joincontext.svg?branch=master)](https://travis-ci.org/LK4D4/joincontext)
|
||||||
|
[![GoDoc](https://godoc.org/github.com/LK4D4/joincontext?status.svg)](https://godoc.org/github.com/LK4D4/joincontext)
|
||||||
|
|
||||||
|
Package joincontext provides a way to combine two contexts.
|
||||||
|
For example it might be useful for grpc server to cancel all handlers in
|
||||||
|
addition to provided handler context.
|
||||||
|
|
||||||
|
For additional info see [godoc page](https://godoc.org/github.com/LK4D4/joincontext)
|
||||||
|
|
||||||
|
## Example
|
||||||
|
```go
|
||||||
|
ctx1, cancel1 := context.WithCancel(context.Background())
|
||||||
|
defer cancel1()
|
||||||
|
ctx2 := context.Background()
|
||||||
|
|
||||||
|
ctx, cancel := joincontext.Join(ctx1, ctx2)
|
||||||
|
defer cancel()
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
default:
|
||||||
|
fmt.Println("context alive")
|
||||||
|
}
|
||||||
|
|
||||||
|
cancel1()
|
||||||
|
|
||||||
|
// give some time to propagate
|
||||||
|
time.Sleep(100 * time.Millisecond)
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
fmt.Println(ctx.Err())
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output: context alive
|
||||||
|
// context canceled
|
||||||
|
```
|
102
vendor/github.com/LK4D4/joincontext/context.go
generated
vendored
Normal file
102
vendor/github.com/LK4D4/joincontext/context.go
generated
vendored
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
// Package joincontext provides a way to combine two contexts.
|
||||||
|
// For example it might be useful for grpc server to cancel all handlers in
|
||||||
|
// addition to provided handler context.
|
||||||
|
package joincontext
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/net/context"
|
||||||
|
)
|
||||||
|
|
||||||
|
type joinContext struct {
|
||||||
|
mu sync.Mutex
|
||||||
|
ctx1 context.Context
|
||||||
|
ctx2 context.Context
|
||||||
|
done chan struct{}
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
// Join returns new context which is child for two passed contexts.
|
||||||
|
// It starts new goroutine which tracks both contexts.
|
||||||
|
//
|
||||||
|
// Done() channel is closed when one of parents contexts is done.
|
||||||
|
//
|
||||||
|
// Deadline() returns earliest deadline between parent contexts.
|
||||||
|
//
|
||||||
|
// Err() returns error from first done parent context.
|
||||||
|
//
|
||||||
|
// Value(key) looks for key in parent contexts. First found is returned.
|
||||||
|
func Join(ctx1, ctx2 context.Context) (context.Context, context.CancelFunc) {
|
||||||
|
c := &joinContext{ctx1: ctx1, ctx2: ctx2, done: make(chan struct{})}
|
||||||
|
go c.run()
|
||||||
|
return c, c.cancel
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) Deadline() (deadline time.Time, ok bool) {
|
||||||
|
d1, ok1 := c.ctx1.Deadline()
|
||||||
|
if !ok1 {
|
||||||
|
return c.ctx2.Deadline()
|
||||||
|
}
|
||||||
|
d2, ok2 := c.ctx2.Deadline()
|
||||||
|
if !ok2 {
|
||||||
|
return d1, true
|
||||||
|
}
|
||||||
|
|
||||||
|
if d2.Before(d1) {
|
||||||
|
return d2, true
|
||||||
|
}
|
||||||
|
return d1, true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) Done() <-chan struct{} {
|
||||||
|
return c.done
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) Err() error {
|
||||||
|
c.mu.Lock()
|
||||||
|
defer c.mu.Unlock()
|
||||||
|
return c.err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) Value(key interface{}) interface{} {
|
||||||
|
v := c.ctx1.Value(key)
|
||||||
|
if v == nil {
|
||||||
|
v = c.ctx2.Value(key)
|
||||||
|
}
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) run() {
|
||||||
|
var doneCtx context.Context
|
||||||
|
select {
|
||||||
|
case <-c.ctx1.Done():
|
||||||
|
doneCtx = c.ctx1
|
||||||
|
case <-c.ctx2.Done():
|
||||||
|
doneCtx = c.ctx2
|
||||||
|
case <-c.done:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.mu.Lock()
|
||||||
|
if c.err != nil {
|
||||||
|
c.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.err = doneCtx.Err()
|
||||||
|
c.mu.Unlock()
|
||||||
|
close(c.done)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *joinContext) cancel() {
|
||||||
|
c.mu.Lock()
|
||||||
|
if c.err != nil {
|
||||||
|
c.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.err = context.Canceled
|
||||||
|
|
||||||
|
c.mu.Unlock()
|
||||||
|
close(c.done)
|
||||||
|
}
|
20
vendor/github.com/StackExchange/wmi/LICENSE
generated
vendored
Normal file
20
vendor/github.com/StackExchange/wmi/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2013 Stack Exchange
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
this software and associated documentation files (the "Software"), to deal in
|
||||||
|
the Software without restriction, including without limitation the rights to
|
||||||
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
6
vendor/github.com/StackExchange/wmi/README.md
generated
vendored
Normal file
6
vendor/github.com/StackExchange/wmi/README.md
generated
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
wmi
|
||||||
|
===
|
||||||
|
|
||||||
|
Package wmi provides a WQL interface to Windows WMI.
|
||||||
|
|
||||||
|
Note: It interfaces with WMI on the local machine, therefore it only runs on Windows.
|
260
vendor/github.com/StackExchange/wmi/swbemservices.go
generated
vendored
Normal file
260
vendor/github.com/StackExchange/wmi/swbemservices.go
generated
vendored
Normal file
|
@ -0,0 +1,260 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package wmi
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/go-ole/go-ole"
|
||||||
|
"github.com/go-ole/go-ole/oleutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SWbemServices is used to access wmi. See https://msdn.microsoft.com/en-us/library/aa393719(v=vs.85).aspx
|
||||||
|
type SWbemServices struct {
|
||||||
|
//TODO: track namespace. Not sure if we can re connect to a different namespace using the same instance
|
||||||
|
cWMIClient *Client //This could also be an embedded struct, but then we would need to branch on Client vs SWbemServices in the Query method
|
||||||
|
sWbemLocatorIUnknown *ole.IUnknown
|
||||||
|
sWbemLocatorIDispatch *ole.IDispatch
|
||||||
|
queries chan *queryRequest
|
||||||
|
closeError chan error
|
||||||
|
lQueryorClose sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
type queryRequest struct {
|
||||||
|
query string
|
||||||
|
dst interface{}
|
||||||
|
args []interface{}
|
||||||
|
finished chan error
|
||||||
|
}
|
||||||
|
|
||||||
|
// InitializeSWbemServices will return a new SWbemServices object that can be used to query WMI
|
||||||
|
func InitializeSWbemServices(c *Client, connectServerArgs ...interface{}) (*SWbemServices, error) {
|
||||||
|
//fmt.Println("InitializeSWbemServices: Starting")
|
||||||
|
//TODO: implement connectServerArgs as optional argument for init with connectServer call
|
||||||
|
s := new(SWbemServices)
|
||||||
|
s.cWMIClient = c
|
||||||
|
s.queries = make(chan *queryRequest)
|
||||||
|
initError := make(chan error)
|
||||||
|
go s.process(initError)
|
||||||
|
|
||||||
|
err, ok := <-initError
|
||||||
|
if ok {
|
||||||
|
return nil, err //Send error to caller
|
||||||
|
}
|
||||||
|
//fmt.Println("InitializeSWbemServices: Finished")
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close will clear and release all of the SWbemServices resources
|
||||||
|
func (s *SWbemServices) Close() error {
|
||||||
|
s.lQueryorClose.Lock()
|
||||||
|
if s == nil || s.sWbemLocatorIDispatch == nil {
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
return fmt.Errorf("SWbemServices is not Initialized")
|
||||||
|
}
|
||||||
|
if s.queries == nil {
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
return fmt.Errorf("SWbemServices has been closed")
|
||||||
|
}
|
||||||
|
//fmt.Println("Close: sending close request")
|
||||||
|
var result error
|
||||||
|
ce := make(chan error)
|
||||||
|
s.closeError = ce //Race condition if multiple callers to close. May need to lock here
|
||||||
|
close(s.queries) //Tell background to shut things down
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
err, ok := <-ce
|
||||||
|
if ok {
|
||||||
|
result = err
|
||||||
|
}
|
||||||
|
//fmt.Println("Close: finished")
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SWbemServices) process(initError chan error) {
|
||||||
|
//fmt.Println("process: starting background thread initialization")
|
||||||
|
//All OLE/WMI calls must happen on the same initialized thead, so lock this goroutine
|
||||||
|
runtime.LockOSThread()
|
||||||
|
defer runtime.LockOSThread()
|
||||||
|
|
||||||
|
err := ole.CoInitializeEx(0, ole.COINIT_MULTITHREADED)
|
||||||
|
if err != nil {
|
||||||
|
oleCode := err.(*ole.OleError).Code()
|
||||||
|
if oleCode != ole.S_OK && oleCode != S_FALSE {
|
||||||
|
initError <- fmt.Errorf("ole.CoInitializeEx error: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
defer ole.CoUninitialize()
|
||||||
|
|
||||||
|
unknown, err := oleutil.CreateObject("WbemScripting.SWbemLocator")
|
||||||
|
if err != nil {
|
||||||
|
initError <- fmt.Errorf("CreateObject SWbemLocator error: %v", err)
|
||||||
|
return
|
||||||
|
} else if unknown == nil {
|
||||||
|
initError <- ErrNilCreateObject
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer unknown.Release()
|
||||||
|
s.sWbemLocatorIUnknown = unknown
|
||||||
|
|
||||||
|
dispatch, err := s.sWbemLocatorIUnknown.QueryInterface(ole.IID_IDispatch)
|
||||||
|
if err != nil {
|
||||||
|
initError <- fmt.Errorf("SWbemLocator QueryInterface error: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer dispatch.Release()
|
||||||
|
s.sWbemLocatorIDispatch = dispatch
|
||||||
|
|
||||||
|
// we can't do the ConnectServer call outside the loop unless we find a way to track and re-init the connectServerArgs
|
||||||
|
//fmt.Println("process: initialized. closing initError")
|
||||||
|
close(initError)
|
||||||
|
//fmt.Println("process: waiting for queries")
|
||||||
|
for q := range s.queries {
|
||||||
|
//fmt.Printf("process: new query: len(query)=%d\n", len(q.query))
|
||||||
|
errQuery := s.queryBackground(q)
|
||||||
|
//fmt.Println("process: s.queryBackground finished")
|
||||||
|
if errQuery != nil {
|
||||||
|
q.finished <- errQuery
|
||||||
|
}
|
||||||
|
close(q.finished)
|
||||||
|
}
|
||||||
|
//fmt.Println("process: queries channel closed")
|
||||||
|
s.queries = nil //set channel to nil so we know it is closed
|
||||||
|
//TODO: I think the Release/Clear calls can panic if things are in a bad state.
|
||||||
|
//TODO: May need to recover from panics and send error to method caller instead.
|
||||||
|
close(s.closeError)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query runs the WQL query using a SWbemServices instance and appends the values to dst.
|
||||||
|
//
|
||||||
|
// dst must have type *[]S or *[]*S, for some struct type S. Fields selected in
|
||||||
|
// the query must have the same name in dst. Supported types are all signed and
|
||||||
|
// unsigned integers, time.Time, string, bool, or a pointer to one of those.
|
||||||
|
// Array types are not supported.
|
||||||
|
//
|
||||||
|
// By default, the local machine and default namespace are used. These can be
|
||||||
|
// changed using connectServerArgs. See
|
||||||
|
// http://msdn.microsoft.com/en-us/library/aa393720.aspx for details.
|
||||||
|
func (s *SWbemServices) Query(query string, dst interface{}, connectServerArgs ...interface{}) error {
|
||||||
|
s.lQueryorClose.Lock()
|
||||||
|
if s == nil || s.sWbemLocatorIDispatch == nil {
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
return fmt.Errorf("SWbemServices is not Initialized")
|
||||||
|
}
|
||||||
|
if s.queries == nil {
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
return fmt.Errorf("SWbemServices has been closed")
|
||||||
|
}
|
||||||
|
|
||||||
|
//fmt.Println("Query: Sending query request")
|
||||||
|
qr := queryRequest{
|
||||||
|
query: query,
|
||||||
|
dst: dst,
|
||||||
|
args: connectServerArgs,
|
||||||
|
finished: make(chan error),
|
||||||
|
}
|
||||||
|
s.queries <- &qr
|
||||||
|
s.lQueryorClose.Unlock()
|
||||||
|
err, ok := <-qr.finished
|
||||||
|
if ok {
|
||||||
|
//fmt.Println("Query: Finished with error")
|
||||||
|
return err //Send error to caller
|
||||||
|
}
|
||||||
|
//fmt.Println("Query: Finished")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *SWbemServices) queryBackground(q *queryRequest) error {
|
||||||
|
if s == nil || s.sWbemLocatorIDispatch == nil {
|
||||||
|
return fmt.Errorf("SWbemServices is not Initialized")
|
||||||
|
}
|
||||||
|
wmi := s.sWbemLocatorIDispatch //Should just rename in the code, but this will help as we break things apart
|
||||||
|
//fmt.Println("queryBackground: Starting")
|
||||||
|
|
||||||
|
dv := reflect.ValueOf(q.dst)
|
||||||
|
if dv.Kind() != reflect.Ptr || dv.IsNil() {
|
||||||
|
return ErrInvalidEntityType
|
||||||
|
}
|
||||||
|
dv = dv.Elem()
|
||||||
|
mat, elemType := checkMultiArg(dv)
|
||||||
|
if mat == multiArgTypeInvalid {
|
||||||
|
return ErrInvalidEntityType
|
||||||
|
}
|
||||||
|
|
||||||
|
// service is a SWbemServices
|
||||||
|
serviceRaw, err := oleutil.CallMethod(wmi, "ConnectServer", q.args...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
service := serviceRaw.ToIDispatch()
|
||||||
|
defer serviceRaw.Clear()
|
||||||
|
|
||||||
|
// result is a SWBemObjectSet
|
||||||
|
resultRaw, err := oleutil.CallMethod(service, "ExecQuery", q.query)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
result := resultRaw.ToIDispatch()
|
||||||
|
defer resultRaw.Clear()
|
||||||
|
|
||||||
|
count, err := oleInt64(result, "Count")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
enumProperty, err := result.GetProperty("_NewEnum")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer enumProperty.Clear()
|
||||||
|
|
||||||
|
enum, err := enumProperty.ToIUnknown().IEnumVARIANT(ole.IID_IEnumVariant)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if enum == nil {
|
||||||
|
return fmt.Errorf("can't get IEnumVARIANT, enum is nil")
|
||||||
|
}
|
||||||
|
defer enum.Release()
|
||||||
|
|
||||||
|
// Initialize a slice with Count capacity
|
||||||
|
dv.Set(reflect.MakeSlice(dv.Type(), 0, int(count)))
|
||||||
|
|
||||||
|
var errFieldMismatch error
|
||||||
|
for itemRaw, length, err := enum.Next(1); length > 0; itemRaw, length, err = enum.Next(1) {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := func() error {
|
||||||
|
// item is a SWbemObject, but really a Win32_Process
|
||||||
|
item := itemRaw.ToIDispatch()
|
||||||
|
defer item.Release()
|
||||||
|
|
||||||
|
ev := reflect.New(elemType)
|
||||||
|
if err = s.cWMIClient.loadEntity(ev.Interface(), item); err != nil {
|
||||||
|
if _, ok := err.(*ErrFieldMismatch); ok {
|
||||||
|
// We continue loading entities even in the face of field mismatch errors.
|
||||||
|
// If we encounter any other error, that other error is returned. Otherwise,
|
||||||
|
// an ErrFieldMismatch is returned.
|
||||||
|
errFieldMismatch = err
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if mat != multiArgTypeStructPtr {
|
||||||
|
ev = ev.Elem()
|
||||||
|
}
|
||||||
|
dv.Set(reflect.Append(dv, ev))
|
||||||
|
return nil
|
||||||
|
}()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//fmt.Println("queryBackground: Finished")
|
||||||
|
return errFieldMismatch
|
||||||
|
}
|
486
vendor/github.com/StackExchange/wmi/wmi.go
generated
vendored
Normal file
486
vendor/github.com/StackExchange/wmi/wmi.go
generated
vendored
Normal file
|
@ -0,0 +1,486 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
/*
|
||||||
|
Package wmi provides a WQL interface for WMI on Windows.
|
||||||
|
|
||||||
|
Example code to print names of running processes:
|
||||||
|
|
||||||
|
type Win32_Process struct {
|
||||||
|
Name string
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
var dst []Win32_Process
|
||||||
|
q := wmi.CreateQuery(&dst, "")
|
||||||
|
err := wmi.Query(q, &dst)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
for i, v := range dst {
|
||||||
|
println(i, v.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*/
|
||||||
|
package wmi
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-ole/go-ole"
|
||||||
|
"github.com/go-ole/go-ole/oleutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
var l = log.New(os.Stdout, "", log.LstdFlags)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrInvalidEntityType = errors.New("wmi: invalid entity type")
|
||||||
|
// ErrNilCreateObject is the error returned if CreateObject returns nil even
|
||||||
|
// if the error was nil.
|
||||||
|
ErrNilCreateObject = errors.New("wmi: create object returned nil")
|
||||||
|
lock sync.Mutex
|
||||||
|
)
|
||||||
|
|
||||||
|
// S_FALSE is returned by CoInitializeEx if it was already called on this thread.
|
||||||
|
const S_FALSE = 0x00000001
|
||||||
|
|
||||||
|
// QueryNamespace invokes Query with the given namespace on the local machine.
|
||||||
|
func QueryNamespace(query string, dst interface{}, namespace string) error {
|
||||||
|
return Query(query, dst, nil, namespace)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Query runs the WQL query and appends the values to dst.
|
||||||
|
//
|
||||||
|
// dst must have type *[]S or *[]*S, for some struct type S. Fields selected in
|
||||||
|
// the query must have the same name in dst. Supported types are all signed and
|
||||||
|
// unsigned integers, time.Time, string, bool, or a pointer to one of those.
|
||||||
|
// Array types are not supported.
|
||||||
|
//
|
||||||
|
// By default, the local machine and default namespace are used. These can be
|
||||||
|
// changed using connectServerArgs. See
|
||||||
|
// http://msdn.microsoft.com/en-us/library/aa393720.aspx for details.
|
||||||
|
//
|
||||||
|
// Query is a wrapper around DefaultClient.Query.
|
||||||
|
func Query(query string, dst interface{}, connectServerArgs ...interface{}) error {
|
||||||
|
if DefaultClient.SWbemServicesClient == nil {
|
||||||
|
return DefaultClient.Query(query, dst, connectServerArgs...)
|
||||||
|
}
|
||||||
|
return DefaultClient.SWbemServicesClient.Query(query, dst, connectServerArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// A Client is an WMI query client.
|
||||||
|
//
|
||||||
|
// Its zero value (DefaultClient) is a usable client.
|
||||||
|
type Client struct {
|
||||||
|
// NonePtrZero specifies if nil values for fields which aren't pointers
|
||||||
|
// should be returned as the field types zero value.
|
||||||
|
//
|
||||||
|
// Setting this to true allows stucts without pointer fields to be used
|
||||||
|
// without the risk failure should a nil value returned from WMI.
|
||||||
|
NonePtrZero bool
|
||||||
|
|
||||||
|
// PtrNil specifies if nil values for pointer fields should be returned
|
||||||
|
// as nil.
|
||||||
|
//
|
||||||
|
// Setting this to true will set pointer fields to nil where WMI
|
||||||
|
// returned nil, otherwise the types zero value will be returned.
|
||||||
|
PtrNil bool
|
||||||
|
|
||||||
|
// AllowMissingFields specifies that struct fields not present in the
|
||||||
|
// query result should not result in an error.
|
||||||
|
//
|
||||||
|
// Setting this to true allows custom queries to be used with full
|
||||||
|
// struct definitions instead of having to define multiple structs.
|
||||||
|
AllowMissingFields bool
|
||||||
|
|
||||||
|
// SWbemServiceClient is an optional SWbemServices object that can be
|
||||||
|
// initialized and then reused across multiple queries. If it is null
|
||||||
|
// then the method will initialize a new temporary client each time.
|
||||||
|
SWbemServicesClient *SWbemServices
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultClient is the default Client and is used by Query, QueryNamespace
|
||||||
|
var DefaultClient = &Client{}
|
||||||
|
|
||||||
|
// Query runs the WQL query and appends the values to dst.
|
||||||
|
//
|
||||||
|
// dst must have type *[]S or *[]*S, for some struct type S. Fields selected in
|
||||||
|
// the query must have the same name in dst. Supported types are all signed and
|
||||||
|
// unsigned integers, time.Time, string, bool, or a pointer to one of those.
|
||||||
|
// Array types are not supported.
|
||||||
|
//
|
||||||
|
// By default, the local machine and default namespace are used. These can be
|
||||||
|
// changed using connectServerArgs. See
|
||||||
|
// http://msdn.microsoft.com/en-us/library/aa393720.aspx for details.
|
||||||
|
func (c *Client) Query(query string, dst interface{}, connectServerArgs ...interface{}) error {
|
||||||
|
dv := reflect.ValueOf(dst)
|
||||||
|
if dv.Kind() != reflect.Ptr || dv.IsNil() {
|
||||||
|
return ErrInvalidEntityType
|
||||||
|
}
|
||||||
|
dv = dv.Elem()
|
||||||
|
mat, elemType := checkMultiArg(dv)
|
||||||
|
if mat == multiArgTypeInvalid {
|
||||||
|
return ErrInvalidEntityType
|
||||||
|
}
|
||||||
|
|
||||||
|
lock.Lock()
|
||||||
|
defer lock.Unlock()
|
||||||
|
runtime.LockOSThread()
|
||||||
|
defer runtime.UnlockOSThread()
|
||||||
|
|
||||||
|
err := ole.CoInitializeEx(0, ole.COINIT_MULTITHREADED)
|
||||||
|
if err != nil {
|
||||||
|
oleCode := err.(*ole.OleError).Code()
|
||||||
|
if oleCode != ole.S_OK && oleCode != S_FALSE {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
defer ole.CoUninitialize()
|
||||||
|
|
||||||
|
unknown, err := oleutil.CreateObject("WbemScripting.SWbemLocator")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
} else if unknown == nil {
|
||||||
|
return ErrNilCreateObject
|
||||||
|
}
|
||||||
|
defer unknown.Release()
|
||||||
|
|
||||||
|
wmi, err := unknown.QueryInterface(ole.IID_IDispatch)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer wmi.Release()
|
||||||
|
|
||||||
|
// service is a SWbemServices
|
||||||
|
serviceRaw, err := oleutil.CallMethod(wmi, "ConnectServer", connectServerArgs...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
service := serviceRaw.ToIDispatch()
|
||||||
|
defer serviceRaw.Clear()
|
||||||
|
|
||||||
|
// result is a SWBemObjectSet
|
||||||
|
resultRaw, err := oleutil.CallMethod(service, "ExecQuery", query)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
result := resultRaw.ToIDispatch()
|
||||||
|
defer resultRaw.Clear()
|
||||||
|
|
||||||
|
count, err := oleInt64(result, "Count")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
enumProperty, err := result.GetProperty("_NewEnum")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer enumProperty.Clear()
|
||||||
|
|
||||||
|
enum, err := enumProperty.ToIUnknown().IEnumVARIANT(ole.IID_IEnumVariant)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if enum == nil {
|
||||||
|
return fmt.Errorf("can't get IEnumVARIANT, enum is nil")
|
||||||
|
}
|
||||||
|
defer enum.Release()
|
||||||
|
|
||||||
|
// Initialize a slice with Count capacity
|
||||||
|
dv.Set(reflect.MakeSlice(dv.Type(), 0, int(count)))
|
||||||
|
|
||||||
|
var errFieldMismatch error
|
||||||
|
for itemRaw, length, err := enum.Next(1); length > 0; itemRaw, length, err = enum.Next(1) {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := func() error {
|
||||||
|
// item is a SWbemObject, but really a Win32_Process
|
||||||
|
item := itemRaw.ToIDispatch()
|
||||||
|
defer item.Release()
|
||||||
|
|
||||||
|
ev := reflect.New(elemType)
|
||||||
|
if err = c.loadEntity(ev.Interface(), item); err != nil {
|
||||||
|
if _, ok := err.(*ErrFieldMismatch); ok {
|
||||||
|
// We continue loading entities even in the face of field mismatch errors.
|
||||||
|
// If we encounter any other error, that other error is returned. Otherwise,
|
||||||
|
// an ErrFieldMismatch is returned.
|
||||||
|
errFieldMismatch = err
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if mat != multiArgTypeStructPtr {
|
||||||
|
ev = ev.Elem()
|
||||||
|
}
|
||||||
|
dv.Set(reflect.Append(dv, ev))
|
||||||
|
return nil
|
||||||
|
}()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errFieldMismatch
|
||||||
|
}
|
||||||
|
|
||||||
|
// ErrFieldMismatch is returned when a field is to be loaded into a different
|
||||||
|
// type than the one it was stored from, or when a field is missing or
|
||||||
|
// unexported in the destination struct.
|
||||||
|
// StructType is the type of the struct pointed to by the destination argument.
|
||||||
|
type ErrFieldMismatch struct {
|
||||||
|
StructType reflect.Type
|
||||||
|
FieldName string
|
||||||
|
Reason string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ErrFieldMismatch) Error() string {
|
||||||
|
return fmt.Sprintf("wmi: cannot load field %q into a %q: %s",
|
||||||
|
e.FieldName, e.StructType, e.Reason)
|
||||||
|
}
|
||||||
|
|
||||||
|
var timeType = reflect.TypeOf(time.Time{})
|
||||||
|
|
||||||
|
// loadEntity loads a SWbemObject into a struct pointer.
|
||||||
|
func (c *Client) loadEntity(dst interface{}, src *ole.IDispatch) (errFieldMismatch error) {
|
||||||
|
v := reflect.ValueOf(dst).Elem()
|
||||||
|
for i := 0; i < v.NumField(); i++ {
|
||||||
|
f := v.Field(i)
|
||||||
|
of := f
|
||||||
|
isPtr := f.Kind() == reflect.Ptr
|
||||||
|
if isPtr {
|
||||||
|
ptr := reflect.New(f.Type().Elem())
|
||||||
|
f.Set(ptr)
|
||||||
|
f = f.Elem()
|
||||||
|
}
|
||||||
|
n := v.Type().Field(i).Name
|
||||||
|
if !f.CanSet() {
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "CanSet() is false",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
prop, err := oleutil.GetProperty(src, n)
|
||||||
|
if err != nil {
|
||||||
|
if !c.AllowMissingFields {
|
||||||
|
errFieldMismatch = &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "no such struct field",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
defer prop.Clear()
|
||||||
|
|
||||||
|
switch val := prop.Value().(type) {
|
||||||
|
case int8, int16, int32, int64, int:
|
||||||
|
v := reflect.ValueOf(val).Int()
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
f.SetInt(v)
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
f.SetUint(uint64(v))
|
||||||
|
default:
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "not an integer class",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case uint8, uint16, uint32, uint64:
|
||||||
|
v := reflect.ValueOf(val).Uint()
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
f.SetInt(int64(v))
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
f.SetUint(v)
|
||||||
|
default:
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "not an integer class",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case string:
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.String:
|
||||||
|
f.SetString(val)
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
iv, err := strconv.ParseInt(val, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
f.SetInt(iv)
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
uv, err := strconv.ParseUint(val, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
f.SetUint(uv)
|
||||||
|
case reflect.Struct:
|
||||||
|
switch f.Type() {
|
||||||
|
case timeType:
|
||||||
|
if len(val) == 25 {
|
||||||
|
mins, err := strconv.Atoi(val[22:])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
val = val[:22] + fmt.Sprintf("%02d%02d", mins/60, mins%60)
|
||||||
|
}
|
||||||
|
t, err := time.Parse("20060102150405.000000-0700", val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
f.Set(reflect.ValueOf(t))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case bool:
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
f.SetBool(val)
|
||||||
|
default:
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "not a bool",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case float32:
|
||||||
|
switch f.Kind() {
|
||||||
|
case reflect.Float32:
|
||||||
|
f.SetFloat(float64(val))
|
||||||
|
default:
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: "not a Float32",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
if f.Kind() == reflect.Slice {
|
||||||
|
switch f.Type().Elem().Kind() {
|
||||||
|
case reflect.String:
|
||||||
|
safeArray := prop.ToArray()
|
||||||
|
if safeArray != nil {
|
||||||
|
arr := safeArray.ToValueArray()
|
||||||
|
fArr := reflect.MakeSlice(f.Type(), len(arr), len(arr))
|
||||||
|
for i, v := range arr {
|
||||||
|
s := fArr.Index(i)
|
||||||
|
s.SetString(v.(string))
|
||||||
|
}
|
||||||
|
f.Set(fArr)
|
||||||
|
}
|
||||||
|
case reflect.Uint8:
|
||||||
|
safeArray := prop.ToArray()
|
||||||
|
if safeArray != nil {
|
||||||
|
arr := safeArray.ToValueArray()
|
||||||
|
fArr := reflect.MakeSlice(f.Type(), len(arr), len(arr))
|
||||||
|
for i, v := range arr {
|
||||||
|
s := fArr.Index(i)
|
||||||
|
s.SetUint(reflect.ValueOf(v).Uint())
|
||||||
|
}
|
||||||
|
f.Set(fArr)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: fmt.Sprintf("unsupported slice type (%T)", val),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
typeof := reflect.TypeOf(val)
|
||||||
|
if typeof == nil && (isPtr || c.NonePtrZero) {
|
||||||
|
if (isPtr && c.PtrNil) || (!isPtr && c.NonePtrZero) {
|
||||||
|
of.Set(reflect.Zero(of.Type()))
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
return &ErrFieldMismatch{
|
||||||
|
StructType: of.Type(),
|
||||||
|
FieldName: n,
|
||||||
|
Reason: fmt.Sprintf("unsupported type (%T)", val),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errFieldMismatch
|
||||||
|
}
|
||||||
|
|
||||||
|
type multiArgType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
multiArgTypeInvalid multiArgType = iota
|
||||||
|
multiArgTypeStruct
|
||||||
|
multiArgTypeStructPtr
|
||||||
|
)
|
||||||
|
|
||||||
|
// checkMultiArg checks that v has type []S, []*S for some struct type S.
|
||||||
|
//
|
||||||
|
// It returns what category the slice's elements are, and the reflect.Type
|
||||||
|
// that represents S.
|
||||||
|
func checkMultiArg(v reflect.Value) (m multiArgType, elemType reflect.Type) {
|
||||||
|
if v.Kind() != reflect.Slice {
|
||||||
|
return multiArgTypeInvalid, nil
|
||||||
|
}
|
||||||
|
elemType = v.Type().Elem()
|
||||||
|
switch elemType.Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
return multiArgTypeStruct, elemType
|
||||||
|
case reflect.Ptr:
|
||||||
|
elemType = elemType.Elem()
|
||||||
|
if elemType.Kind() == reflect.Struct {
|
||||||
|
return multiArgTypeStructPtr, elemType
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return multiArgTypeInvalid, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func oleInt64(item *ole.IDispatch, prop string) (int64, error) {
|
||||||
|
v, err := oleutil.GetProperty(item, prop)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
defer v.Clear()
|
||||||
|
|
||||||
|
i := int64(v.Val)
|
||||||
|
return i, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateQuery returns a WQL query string that queries all columns of src. where
|
||||||
|
// is an optional string that is appended to the query, to be used with WHERE
|
||||||
|
// clauses. In such a case, the "WHERE" string should appear at the beginning.
|
||||||
|
func CreateQuery(src interface{}, where string) string {
|
||||||
|
var b bytes.Buffer
|
||||||
|
b.WriteString("SELECT ")
|
||||||
|
s := reflect.Indirect(reflect.ValueOf(src))
|
||||||
|
t := s.Type()
|
||||||
|
if s.Kind() == reflect.Slice {
|
||||||
|
t = t.Elem()
|
||||||
|
}
|
||||||
|
if t.Kind() != reflect.Struct {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
var fields []string
|
||||||
|
for i := 0; i < t.NumField(); i++ {
|
||||||
|
fields = append(fields, t.Field(i).Name)
|
||||||
|
}
|
||||||
|
b.WriteString(strings.Join(fields, ", "))
|
||||||
|
b.WriteString(" FROM ")
|
||||||
|
b.WriteString(t.Name())
|
||||||
|
b.WriteString(" " + where)
|
||||||
|
return b.String()
|
||||||
|
}
|
24
vendor/github.com/armon/go-metrics/.gitignore
generated
vendored
Normal file
24
vendor/github.com/armon/go-metrics/.gitignore
generated
vendored
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
# Compiled Object files, Static and Dynamic libs (Shared Objects)
|
||||||
|
*.o
|
||||||
|
*.a
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Folders
|
||||||
|
_obj
|
||||||
|
_test
|
||||||
|
|
||||||
|
# Architecture specific extensions/prefixes
|
||||||
|
*.[568vq]
|
||||||
|
[568vq].out
|
||||||
|
|
||||||
|
*.cgo1.go
|
||||||
|
*.cgo2.c
|
||||||
|
_cgo_defun.c
|
||||||
|
_cgo_gotypes.go
|
||||||
|
_cgo_export.*
|
||||||
|
|
||||||
|
_testmain.go
|
||||||
|
|
||||||
|
*.exe
|
||||||
|
|
||||||
|
/metrics.out
|
20
vendor/github.com/armon/go-metrics/LICENSE
generated
vendored
Normal file
20
vendor/github.com/armon/go-metrics/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2013 Armon Dadgar
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
this software and associated documentation files (the "Software"), to deal in
|
||||||
|
the Software without restriction, including without limitation the rights to
|
||||||
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
91
vendor/github.com/armon/go-metrics/README.md
generated
vendored
Normal file
91
vendor/github.com/armon/go-metrics/README.md
generated
vendored
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
go-metrics
|
||||||
|
==========
|
||||||
|
|
||||||
|
This library provides a `metrics` package which can be used to instrument code,
|
||||||
|
expose application metrics, and profile runtime performance in a flexible manner.
|
||||||
|
|
||||||
|
Current API: [![GoDoc](https://godoc.org/github.com/armon/go-metrics?status.svg)](https://godoc.org/github.com/armon/go-metrics)
|
||||||
|
|
||||||
|
Sinks
|
||||||
|
-----
|
||||||
|
|
||||||
|
The `metrics` package makes use of a `MetricSink` interface to support delivery
|
||||||
|
to any type of backend. Currently the following sinks are provided:
|
||||||
|
|
||||||
|
* StatsiteSink : Sinks to a [statsite](https://github.com/armon/statsite/) instance (TCP)
|
||||||
|
* StatsdSink: Sinks to a [StatsD](https://github.com/etsy/statsd/) / statsite instance (UDP)
|
||||||
|
* PrometheusSink: Sinks to a [Prometheus](http://prometheus.io/) metrics endpoint (exposed via HTTP for scrapes)
|
||||||
|
* InmemSink : Provides in-memory aggregation, can be used to export stats
|
||||||
|
* FanoutSink : Sinks to multiple sinks. Enables writing to multiple statsite instances for example.
|
||||||
|
* BlackholeSink : Sinks to nowhere
|
||||||
|
|
||||||
|
In addition to the sinks, the `InmemSignal` can be used to catch a signal,
|
||||||
|
and dump a formatted output of recent metrics. For example, when a process gets
|
||||||
|
a SIGUSR1, it can dump to stderr recent performance metrics for debugging.
|
||||||
|
|
||||||
|
Labels
|
||||||
|
------
|
||||||
|
|
||||||
|
Most metrics do have an equivalent ending with `WithLabels`, such methods
|
||||||
|
allow to push metrics with labels and use some features of underlying Sinks
|
||||||
|
(ex: translated into Prometheus labels).
|
||||||
|
|
||||||
|
Since some of these labels may increase greatly cardinality of metrics, the
|
||||||
|
library allow to filter labels using a blacklist/whitelist filtering system
|
||||||
|
which is global to all metrics.
|
||||||
|
|
||||||
|
* If `Config.AllowedLabels` is not nil, then only labels specified in this value will be sent to underlying Sink, otherwise, all labels are sent by default.
|
||||||
|
* If `Config.BlockedLabels` is not nil, any label specified in this value will not be sent to underlying Sinks.
|
||||||
|
|
||||||
|
By default, both `Config.AllowedLabels` and `Config.BlockedLabels` are nil, meaning that
|
||||||
|
no tags are filetered at all, but it allow to a user to globally block some tags with high
|
||||||
|
cardinality at application level.
|
||||||
|
|
||||||
|
Examples
|
||||||
|
--------
|
||||||
|
|
||||||
|
Here is an example of using the package:
|
||||||
|
|
||||||
|
```go
|
||||||
|
func SlowMethod() {
|
||||||
|
// Profiling the runtime of a method
|
||||||
|
defer metrics.MeasureSince([]string{"SlowMethod"}, time.Now())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configure a statsite sink as the global metrics sink
|
||||||
|
sink, _ := metrics.NewStatsiteSink("statsite:8125")
|
||||||
|
metrics.NewGlobal(metrics.DefaultConfig("service-name"), sink)
|
||||||
|
|
||||||
|
// Emit a Key/Value pair
|
||||||
|
metrics.EmitKey([]string{"questions", "meaning of life"}, 42)
|
||||||
|
```
|
||||||
|
|
||||||
|
Here is an example of setting up a signal handler:
|
||||||
|
|
||||||
|
```go
|
||||||
|
// Setup the inmem sink and signal handler
|
||||||
|
inm := metrics.NewInmemSink(10*time.Second, time.Minute)
|
||||||
|
sig := metrics.DefaultInmemSignal(inm)
|
||||||
|
metrics.NewGlobal(metrics.DefaultConfig("service-name"), inm)
|
||||||
|
|
||||||
|
// Run some code
|
||||||
|
inm.SetGauge([]string{"foo"}, 42)
|
||||||
|
inm.EmitKey([]string{"bar"}, 30)
|
||||||
|
|
||||||
|
inm.IncrCounter([]string{"baz"}, 42)
|
||||||
|
inm.IncrCounter([]string{"baz"}, 1)
|
||||||
|
inm.IncrCounter([]string{"baz"}, 80)
|
||||||
|
|
||||||
|
inm.AddSample([]string{"method", "wow"}, 42)
|
||||||
|
inm.AddSample([]string{"method", "wow"}, 100)
|
||||||
|
inm.AddSample([]string{"method", "wow"}, 22)
|
||||||
|
|
||||||
|
....
|
||||||
|
```
|
||||||
|
|
||||||
|
When a signal comes in, output like the following will be dumped to stderr:
|
||||||
|
|
||||||
|
[2014-01-28 14:57:33.04 -0800 PST][G] 'foo': 42.000
|
||||||
|
[2014-01-28 14:57:33.04 -0800 PST][P] 'bar': 30.000
|
||||||
|
[2014-01-28 14:57:33.04 -0800 PST][C] 'baz': Count: 3 Min: 1.000 Mean: 41.000 Max: 80.000 Stddev: 39.509
|
||||||
|
[2014-01-28 14:57:33.04 -0800 PST][S] 'method.wow': Count: 3 Min: 22.000 Mean: 54.667 Max: 100.000 Stddev: 40.513
|
12
vendor/github.com/armon/go-metrics/const_unix.go
generated
vendored
Normal file
12
vendor/github.com/armon/go-metrics/const_unix.go
generated
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// DefaultSignal is used with DefaultInmemSignal
|
||||||
|
DefaultSignal = syscall.SIGUSR1
|
||||||
|
)
|
13
vendor/github.com/armon/go-metrics/const_windows.go
generated
vendored
Normal file
13
vendor/github.com/armon/go-metrics/const_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// DefaultSignal is used with DefaultInmemSignal
|
||||||
|
// Windows has no SIGUSR1, use SIGBREAK
|
||||||
|
DefaultSignal = syscall.Signal(21)
|
||||||
|
)
|
348
vendor/github.com/armon/go-metrics/inmem.go
generated
vendored
Normal file
348
vendor/github.com/armon/go-metrics/inmem.go
generated
vendored
Normal file
|
@ -0,0 +1,348 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// InmemSink provides a MetricSink that does in-memory aggregation
|
||||||
|
// without sending metrics over a network. It can be embedded within
|
||||||
|
// an application to provide profiling information.
|
||||||
|
type InmemSink struct {
|
||||||
|
// How long is each aggregation interval
|
||||||
|
interval time.Duration
|
||||||
|
|
||||||
|
// Retain controls how many metrics interval we keep
|
||||||
|
retain time.Duration
|
||||||
|
|
||||||
|
// maxIntervals is the maximum length of intervals.
|
||||||
|
// It is retain / interval.
|
||||||
|
maxIntervals int
|
||||||
|
|
||||||
|
// intervals is a slice of the retained intervals
|
||||||
|
intervals []*IntervalMetrics
|
||||||
|
intervalLock sync.RWMutex
|
||||||
|
|
||||||
|
rateDenom float64
|
||||||
|
}
|
||||||
|
|
||||||
|
// IntervalMetrics stores the aggregated metrics
|
||||||
|
// for a specific interval
|
||||||
|
type IntervalMetrics struct {
|
||||||
|
sync.RWMutex
|
||||||
|
|
||||||
|
// The start time of the interval
|
||||||
|
Interval time.Time
|
||||||
|
|
||||||
|
// Gauges maps the key to the last set value
|
||||||
|
Gauges map[string]GaugeValue
|
||||||
|
|
||||||
|
// Points maps the string to the list of emitted values
|
||||||
|
// from EmitKey
|
||||||
|
Points map[string][]float32
|
||||||
|
|
||||||
|
// Counters maps the string key to a sum of the counter
|
||||||
|
// values
|
||||||
|
Counters map[string]SampledValue
|
||||||
|
|
||||||
|
// Samples maps the key to an AggregateSample,
|
||||||
|
// which has the rolled up view of a sample
|
||||||
|
Samples map[string]SampledValue
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewIntervalMetrics creates a new IntervalMetrics for a given interval
|
||||||
|
func NewIntervalMetrics(intv time.Time) *IntervalMetrics {
|
||||||
|
return &IntervalMetrics{
|
||||||
|
Interval: intv,
|
||||||
|
Gauges: make(map[string]GaugeValue),
|
||||||
|
Points: make(map[string][]float32),
|
||||||
|
Counters: make(map[string]SampledValue),
|
||||||
|
Samples: make(map[string]SampledValue),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AggregateSample is used to hold aggregate metrics
|
||||||
|
// about a sample
|
||||||
|
type AggregateSample struct {
|
||||||
|
Count int // The count of emitted pairs
|
||||||
|
Rate float64 // The values rate per time unit (usually 1 second)
|
||||||
|
Sum float64 // The sum of values
|
||||||
|
SumSq float64 `json:"-"` // The sum of squared values
|
||||||
|
Min float64 // Minimum value
|
||||||
|
Max float64 // Maximum value
|
||||||
|
LastUpdated time.Time `json:"-"` // When value was last updated
|
||||||
|
}
|
||||||
|
|
||||||
|
// Computes a Stddev of the values
|
||||||
|
func (a *AggregateSample) Stddev() float64 {
|
||||||
|
num := (float64(a.Count) * a.SumSq) - math.Pow(a.Sum, 2)
|
||||||
|
div := float64(a.Count * (a.Count - 1))
|
||||||
|
if div == 0 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return math.Sqrt(num / div)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Computes a mean of the values
|
||||||
|
func (a *AggregateSample) Mean() float64 {
|
||||||
|
if a.Count == 0 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return a.Sum / float64(a.Count)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ingest is used to update a sample
|
||||||
|
func (a *AggregateSample) Ingest(v float64, rateDenom float64) {
|
||||||
|
a.Count++
|
||||||
|
a.Sum += v
|
||||||
|
a.SumSq += (v * v)
|
||||||
|
if v < a.Min || a.Count == 1 {
|
||||||
|
a.Min = v
|
||||||
|
}
|
||||||
|
if v > a.Max || a.Count == 1 {
|
||||||
|
a.Max = v
|
||||||
|
}
|
||||||
|
a.Rate = float64(a.Sum) / rateDenom
|
||||||
|
a.LastUpdated = time.Now()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *AggregateSample) String() string {
|
||||||
|
if a.Count == 0 {
|
||||||
|
return "Count: 0"
|
||||||
|
} else if a.Stddev() == 0 {
|
||||||
|
return fmt.Sprintf("Count: %d Sum: %0.3f LastUpdated: %s", a.Count, a.Sum, a.LastUpdated)
|
||||||
|
} else {
|
||||||
|
return fmt.Sprintf("Count: %d Min: %0.3f Mean: %0.3f Max: %0.3f Stddev: %0.3f Sum: %0.3f LastUpdated: %s",
|
||||||
|
a.Count, a.Min, a.Mean(), a.Max, a.Stddev(), a.Sum, a.LastUpdated)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewInmemSinkFromURL creates an InmemSink from a URL. It is used
|
||||||
|
// (and tested) from NewMetricSinkFromURL.
|
||||||
|
func NewInmemSinkFromURL(u *url.URL) (MetricSink, error) {
|
||||||
|
params := u.Query()
|
||||||
|
|
||||||
|
interval, err := time.ParseDuration(params.Get("interval"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Bad 'interval' param: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
retain, err := time.ParseDuration(params.Get("retain"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Bad 'retain' param: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewInmemSink(interval, retain), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewInmemSink is used to construct a new in-memory sink.
|
||||||
|
// Uses an aggregation interval and maximum retention period.
|
||||||
|
func NewInmemSink(interval, retain time.Duration) *InmemSink {
|
||||||
|
rateTimeUnit := time.Second
|
||||||
|
i := &InmemSink{
|
||||||
|
interval: interval,
|
||||||
|
retain: retain,
|
||||||
|
maxIntervals: int(retain / interval),
|
||||||
|
rateDenom: float64(interval.Nanoseconds()) / float64(rateTimeUnit.Nanoseconds()),
|
||||||
|
}
|
||||||
|
i.intervals = make([]*IntervalMetrics, 0, i.maxIntervals)
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) SetGauge(key []string, val float32) {
|
||||||
|
i.SetGaugeWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
k, name := i.flattenKeyLabels(key, labels)
|
||||||
|
intv := i.getInterval()
|
||||||
|
|
||||||
|
intv.Lock()
|
||||||
|
defer intv.Unlock()
|
||||||
|
intv.Gauges[k] = GaugeValue{Name: name, Value: val, Labels: labels}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) EmitKey(key []string, val float32) {
|
||||||
|
k := i.flattenKey(key)
|
||||||
|
intv := i.getInterval()
|
||||||
|
|
||||||
|
intv.Lock()
|
||||||
|
defer intv.Unlock()
|
||||||
|
vals := intv.Points[k]
|
||||||
|
intv.Points[k] = append(vals, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) IncrCounter(key []string, val float32) {
|
||||||
|
i.IncrCounterWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
k, name := i.flattenKeyLabels(key, labels)
|
||||||
|
intv := i.getInterval()
|
||||||
|
|
||||||
|
intv.Lock()
|
||||||
|
defer intv.Unlock()
|
||||||
|
|
||||||
|
agg, ok := intv.Counters[k]
|
||||||
|
if !ok {
|
||||||
|
agg = SampledValue{
|
||||||
|
Name: name,
|
||||||
|
AggregateSample: &AggregateSample{},
|
||||||
|
Labels: labels,
|
||||||
|
}
|
||||||
|
intv.Counters[k] = agg
|
||||||
|
}
|
||||||
|
agg.Ingest(float64(val), i.rateDenom)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) AddSample(key []string, val float32) {
|
||||||
|
i.AddSampleWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
k, name := i.flattenKeyLabels(key, labels)
|
||||||
|
intv := i.getInterval()
|
||||||
|
|
||||||
|
intv.Lock()
|
||||||
|
defer intv.Unlock()
|
||||||
|
|
||||||
|
agg, ok := intv.Samples[k]
|
||||||
|
if !ok {
|
||||||
|
agg = SampledValue{
|
||||||
|
Name: name,
|
||||||
|
AggregateSample: &AggregateSample{},
|
||||||
|
Labels: labels,
|
||||||
|
}
|
||||||
|
intv.Samples[k] = agg
|
||||||
|
}
|
||||||
|
agg.Ingest(float64(val), i.rateDenom)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Data is used to retrieve all the aggregated metrics
|
||||||
|
// Intervals may be in use, and a read lock should be acquired
|
||||||
|
func (i *InmemSink) Data() []*IntervalMetrics {
|
||||||
|
// Get the current interval, forces creation
|
||||||
|
i.getInterval()
|
||||||
|
|
||||||
|
i.intervalLock.RLock()
|
||||||
|
defer i.intervalLock.RUnlock()
|
||||||
|
|
||||||
|
n := len(i.intervals)
|
||||||
|
intervals := make([]*IntervalMetrics, n)
|
||||||
|
|
||||||
|
copy(intervals[:n-1], i.intervals[:n-1])
|
||||||
|
current := i.intervals[n-1]
|
||||||
|
|
||||||
|
// make its own copy for current interval
|
||||||
|
intervals[n-1] = &IntervalMetrics{}
|
||||||
|
copyCurrent := intervals[n-1]
|
||||||
|
current.RLock()
|
||||||
|
*copyCurrent = *current
|
||||||
|
|
||||||
|
copyCurrent.Gauges = make(map[string]GaugeValue, len(current.Gauges))
|
||||||
|
for k, v := range current.Gauges {
|
||||||
|
copyCurrent.Gauges[k] = v
|
||||||
|
}
|
||||||
|
// saved values will be not change, just copy its link
|
||||||
|
copyCurrent.Points = make(map[string][]float32, len(current.Points))
|
||||||
|
for k, v := range current.Points {
|
||||||
|
copyCurrent.Points[k] = v
|
||||||
|
}
|
||||||
|
copyCurrent.Counters = make(map[string]SampledValue, len(current.Counters))
|
||||||
|
for k, v := range current.Counters {
|
||||||
|
copyCurrent.Counters[k] = v
|
||||||
|
}
|
||||||
|
copyCurrent.Samples = make(map[string]SampledValue, len(current.Samples))
|
||||||
|
for k, v := range current.Samples {
|
||||||
|
copyCurrent.Samples[k] = v
|
||||||
|
}
|
||||||
|
current.RUnlock()
|
||||||
|
|
||||||
|
return intervals
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) getExistingInterval(intv time.Time) *IntervalMetrics {
|
||||||
|
i.intervalLock.RLock()
|
||||||
|
defer i.intervalLock.RUnlock()
|
||||||
|
|
||||||
|
n := len(i.intervals)
|
||||||
|
if n > 0 && i.intervals[n-1].Interval == intv {
|
||||||
|
return i.intervals[n-1]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *InmemSink) createInterval(intv time.Time) *IntervalMetrics {
|
||||||
|
i.intervalLock.Lock()
|
||||||
|
defer i.intervalLock.Unlock()
|
||||||
|
|
||||||
|
// Check for an existing interval
|
||||||
|
n := len(i.intervals)
|
||||||
|
if n > 0 && i.intervals[n-1].Interval == intv {
|
||||||
|
return i.intervals[n-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the current interval
|
||||||
|
current := NewIntervalMetrics(intv)
|
||||||
|
i.intervals = append(i.intervals, current)
|
||||||
|
n++
|
||||||
|
|
||||||
|
// Truncate the intervals if they are too long
|
||||||
|
if n >= i.maxIntervals {
|
||||||
|
copy(i.intervals[0:], i.intervals[n-i.maxIntervals:])
|
||||||
|
i.intervals = i.intervals[:i.maxIntervals]
|
||||||
|
}
|
||||||
|
return current
|
||||||
|
}
|
||||||
|
|
||||||
|
// getInterval returns the current interval to write to
|
||||||
|
func (i *InmemSink) getInterval() *IntervalMetrics {
|
||||||
|
intv := time.Now().Truncate(i.interval)
|
||||||
|
if m := i.getExistingInterval(intv); m != nil {
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
return i.createInterval(intv)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key for formatting, removes spaces
|
||||||
|
func (i *InmemSink) flattenKey(parts []string) string {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
replacer := strings.NewReplacer(" ", "_")
|
||||||
|
|
||||||
|
if len(parts) > 0 {
|
||||||
|
replacer.WriteString(buf, parts[0])
|
||||||
|
}
|
||||||
|
for _, part := range parts[1:] {
|
||||||
|
replacer.WriteString(buf, ".")
|
||||||
|
replacer.WriteString(buf, part)
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key for formatting along with its labels, removes spaces
|
||||||
|
func (i *InmemSink) flattenKeyLabels(parts []string, labels []Label) (string, string) {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
replacer := strings.NewReplacer(" ", "_")
|
||||||
|
|
||||||
|
if len(parts) > 0 {
|
||||||
|
replacer.WriteString(buf, parts[0])
|
||||||
|
}
|
||||||
|
for _, part := range parts[1:] {
|
||||||
|
replacer.WriteString(buf, ".")
|
||||||
|
replacer.WriteString(buf, part)
|
||||||
|
}
|
||||||
|
|
||||||
|
key := buf.String()
|
||||||
|
|
||||||
|
for _, label := range labels {
|
||||||
|
replacer.WriteString(buf, fmt.Sprintf(";%s=%s", label.Name, label.Value))
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.String(), key
|
||||||
|
}
|
118
vendor/github.com/armon/go-metrics/inmem_endpoint.go
generated
vendored
Normal file
118
vendor/github.com/armon/go-metrics/inmem_endpoint.go
generated
vendored
Normal file
|
@ -0,0 +1,118 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MetricsSummary holds a roll-up of metrics info for a given interval
|
||||||
|
type MetricsSummary struct {
|
||||||
|
Timestamp string
|
||||||
|
Gauges []GaugeValue
|
||||||
|
Points []PointValue
|
||||||
|
Counters []SampledValue
|
||||||
|
Samples []SampledValue
|
||||||
|
}
|
||||||
|
|
||||||
|
type GaugeValue struct {
|
||||||
|
Name string
|
||||||
|
Hash string `json:"-"`
|
||||||
|
Value float32
|
||||||
|
|
||||||
|
Labels []Label `json:"-"`
|
||||||
|
DisplayLabels map[string]string `json:"Labels"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PointValue struct {
|
||||||
|
Name string
|
||||||
|
Points []float32
|
||||||
|
}
|
||||||
|
|
||||||
|
type SampledValue struct {
|
||||||
|
Name string
|
||||||
|
Hash string `json:"-"`
|
||||||
|
*AggregateSample
|
||||||
|
Mean float64
|
||||||
|
Stddev float64
|
||||||
|
|
||||||
|
Labels []Label `json:"-"`
|
||||||
|
DisplayLabels map[string]string `json:"Labels"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// DisplayMetrics returns a summary of the metrics from the most recent finished interval.
|
||||||
|
func (i *InmemSink) DisplayMetrics(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
||||||
|
data := i.Data()
|
||||||
|
|
||||||
|
var interval *IntervalMetrics
|
||||||
|
n := len(data)
|
||||||
|
switch {
|
||||||
|
case n == 0:
|
||||||
|
return nil, fmt.Errorf("no metric intervals have been initialized yet")
|
||||||
|
case n == 1:
|
||||||
|
// Show the current interval if it's all we have
|
||||||
|
interval = i.intervals[0]
|
||||||
|
default:
|
||||||
|
// Show the most recent finished interval if we have one
|
||||||
|
interval = i.intervals[n-2]
|
||||||
|
}
|
||||||
|
|
||||||
|
summary := MetricsSummary{
|
||||||
|
Timestamp: interval.Interval.Round(time.Second).UTC().String(),
|
||||||
|
Gauges: make([]GaugeValue, 0, len(interval.Gauges)),
|
||||||
|
Points: make([]PointValue, 0, len(interval.Points)),
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format and sort the output of each metric type, so it gets displayed in a
|
||||||
|
// deterministic order.
|
||||||
|
for name, points := range interval.Points {
|
||||||
|
summary.Points = append(summary.Points, PointValue{name, points})
|
||||||
|
}
|
||||||
|
sort.Slice(summary.Points, func(i, j int) bool {
|
||||||
|
return summary.Points[i].Name < summary.Points[j].Name
|
||||||
|
})
|
||||||
|
|
||||||
|
for hash, value := range interval.Gauges {
|
||||||
|
value.Hash = hash
|
||||||
|
value.DisplayLabels = make(map[string]string)
|
||||||
|
for _, label := range value.Labels {
|
||||||
|
value.DisplayLabels[label.Name] = label.Value
|
||||||
|
}
|
||||||
|
value.Labels = nil
|
||||||
|
|
||||||
|
summary.Gauges = append(summary.Gauges, value)
|
||||||
|
}
|
||||||
|
sort.Slice(summary.Gauges, func(i, j int) bool {
|
||||||
|
return summary.Gauges[i].Hash < summary.Gauges[j].Hash
|
||||||
|
})
|
||||||
|
|
||||||
|
summary.Counters = formatSamples(interval.Counters)
|
||||||
|
summary.Samples = formatSamples(interval.Samples)
|
||||||
|
|
||||||
|
return summary, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatSamples(source map[string]SampledValue) []SampledValue {
|
||||||
|
output := make([]SampledValue, 0, len(source))
|
||||||
|
for hash, sample := range source {
|
||||||
|
displayLabels := make(map[string]string)
|
||||||
|
for _, label := range sample.Labels {
|
||||||
|
displayLabels[label.Name] = label.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
output = append(output, SampledValue{
|
||||||
|
Name: sample.Name,
|
||||||
|
Hash: hash,
|
||||||
|
AggregateSample: sample.AggregateSample,
|
||||||
|
Mean: sample.AggregateSample.Mean(),
|
||||||
|
Stddev: sample.AggregateSample.Stddev(),
|
||||||
|
DisplayLabels: displayLabels,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
sort.Slice(output, func(i, j int) bool {
|
||||||
|
return output[i].Hash < output[j].Hash
|
||||||
|
})
|
||||||
|
|
||||||
|
return output
|
||||||
|
}
|
117
vendor/github.com/armon/go-metrics/inmem_signal.go
generated
vendored
Normal file
117
vendor/github.com/armon/go-metrics/inmem_signal.go
generated
vendored
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
// InmemSignal is used to listen for a given signal, and when received,
|
||||||
|
// to dump the current metrics from the InmemSink to an io.Writer
|
||||||
|
type InmemSignal struct {
|
||||||
|
signal syscall.Signal
|
||||||
|
inm *InmemSink
|
||||||
|
w io.Writer
|
||||||
|
sigCh chan os.Signal
|
||||||
|
|
||||||
|
stop bool
|
||||||
|
stopCh chan struct{}
|
||||||
|
stopLock sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewInmemSignal creates a new InmemSignal which listens for a given signal,
|
||||||
|
// and dumps the current metrics out to a writer
|
||||||
|
func NewInmemSignal(inmem *InmemSink, sig syscall.Signal, w io.Writer) *InmemSignal {
|
||||||
|
i := &InmemSignal{
|
||||||
|
signal: sig,
|
||||||
|
inm: inmem,
|
||||||
|
w: w,
|
||||||
|
sigCh: make(chan os.Signal, 1),
|
||||||
|
stopCh: make(chan struct{}),
|
||||||
|
}
|
||||||
|
signal.Notify(i.sigCh, sig)
|
||||||
|
go i.run()
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultInmemSignal returns a new InmemSignal that responds to SIGUSR1
|
||||||
|
// and writes output to stderr. Windows uses SIGBREAK
|
||||||
|
func DefaultInmemSignal(inmem *InmemSink) *InmemSignal {
|
||||||
|
return NewInmemSignal(inmem, DefaultSignal, os.Stderr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stop is used to stop the InmemSignal from listening
|
||||||
|
func (i *InmemSignal) Stop() {
|
||||||
|
i.stopLock.Lock()
|
||||||
|
defer i.stopLock.Unlock()
|
||||||
|
|
||||||
|
if i.stop {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
i.stop = true
|
||||||
|
close(i.stopCh)
|
||||||
|
signal.Stop(i.sigCh)
|
||||||
|
}
|
||||||
|
|
||||||
|
// run is a long running routine that handles signals
|
||||||
|
func (i *InmemSignal) run() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-i.sigCh:
|
||||||
|
i.dumpStats()
|
||||||
|
case <-i.stopCh:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// dumpStats is used to dump the data to output writer
|
||||||
|
func (i *InmemSignal) dumpStats() {
|
||||||
|
buf := bytes.NewBuffer(nil)
|
||||||
|
|
||||||
|
data := i.inm.Data()
|
||||||
|
// Skip the last period which is still being aggregated
|
||||||
|
for j := 0; j < len(data)-1; j++ {
|
||||||
|
intv := data[j]
|
||||||
|
intv.RLock()
|
||||||
|
for _, val := range intv.Gauges {
|
||||||
|
name := i.flattenLabels(val.Name, val.Labels)
|
||||||
|
fmt.Fprintf(buf, "[%v][G] '%s': %0.3f\n", intv.Interval, name, val.Value)
|
||||||
|
}
|
||||||
|
for name, vals := range intv.Points {
|
||||||
|
for _, val := range vals {
|
||||||
|
fmt.Fprintf(buf, "[%v][P] '%s': %0.3f\n", intv.Interval, name, val)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, agg := range intv.Counters {
|
||||||
|
name := i.flattenLabels(agg.Name, agg.Labels)
|
||||||
|
fmt.Fprintf(buf, "[%v][C] '%s': %s\n", intv.Interval, name, agg.AggregateSample)
|
||||||
|
}
|
||||||
|
for _, agg := range intv.Samples {
|
||||||
|
name := i.flattenLabels(agg.Name, agg.Labels)
|
||||||
|
fmt.Fprintf(buf, "[%v][S] '%s': %s\n", intv.Interval, name, agg.AggregateSample)
|
||||||
|
}
|
||||||
|
intv.RUnlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write out the bytes
|
||||||
|
i.w.Write(buf.Bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key for formatting along with its labels, removes spaces
|
||||||
|
func (i *InmemSignal) flattenLabels(name string, labels []Label) string {
|
||||||
|
buf := bytes.NewBufferString(name)
|
||||||
|
replacer := strings.NewReplacer(" ", "_", ":", "_")
|
||||||
|
|
||||||
|
for _, label := range labels {
|
||||||
|
replacer.WriteString(buf, ".")
|
||||||
|
replacer.WriteString(buf, label.Value)
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.String()
|
||||||
|
}
|
278
vendor/github.com/armon/go-metrics/metrics.go
generated
vendored
Normal file
278
vendor/github.com/armon/go-metrics/metrics.go
generated
vendored
Normal file
|
@ -0,0 +1,278 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"runtime"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/hashicorp/go-immutable-radix"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Label struct {
|
||||||
|
Name string
|
||||||
|
Value string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) SetGauge(key []string, val float32) {
|
||||||
|
m.SetGaugeWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
if m.HostName != "" {
|
||||||
|
if m.EnableHostnameLabel {
|
||||||
|
labels = append(labels, Label{"host", m.HostName})
|
||||||
|
} else if m.EnableHostname {
|
||||||
|
key = insert(0, m.HostName, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if m.EnableTypePrefix {
|
||||||
|
key = insert(0, "gauge", key)
|
||||||
|
}
|
||||||
|
if m.ServiceName != "" {
|
||||||
|
if m.EnableServiceLabel {
|
||||||
|
labels = append(labels, Label{"service", m.ServiceName})
|
||||||
|
} else {
|
||||||
|
key = insert(0, m.ServiceName, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
allowed, labelsFiltered := m.allowMetric(key, labels)
|
||||||
|
if !allowed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.sink.SetGaugeWithLabels(key, val, labelsFiltered)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) EmitKey(key []string, val float32) {
|
||||||
|
if m.EnableTypePrefix {
|
||||||
|
key = insert(0, "kv", key)
|
||||||
|
}
|
||||||
|
if m.ServiceName != "" {
|
||||||
|
key = insert(0, m.ServiceName, key)
|
||||||
|
}
|
||||||
|
allowed, _ := m.allowMetric(key, nil)
|
||||||
|
if !allowed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.sink.EmitKey(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) IncrCounter(key []string, val float32) {
|
||||||
|
m.IncrCounterWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
if m.HostName != "" && m.EnableHostnameLabel {
|
||||||
|
labels = append(labels, Label{"host", m.HostName})
|
||||||
|
}
|
||||||
|
if m.EnableTypePrefix {
|
||||||
|
key = insert(0, "counter", key)
|
||||||
|
}
|
||||||
|
if m.ServiceName != "" {
|
||||||
|
if m.EnableServiceLabel {
|
||||||
|
labels = append(labels, Label{"service", m.ServiceName})
|
||||||
|
} else {
|
||||||
|
key = insert(0, m.ServiceName, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
allowed, labelsFiltered := m.allowMetric(key, labels)
|
||||||
|
if !allowed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.sink.IncrCounterWithLabels(key, val, labelsFiltered)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) AddSample(key []string, val float32) {
|
||||||
|
m.AddSampleWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
if m.HostName != "" && m.EnableHostnameLabel {
|
||||||
|
labels = append(labels, Label{"host", m.HostName})
|
||||||
|
}
|
||||||
|
if m.EnableTypePrefix {
|
||||||
|
key = insert(0, "sample", key)
|
||||||
|
}
|
||||||
|
if m.ServiceName != "" {
|
||||||
|
if m.EnableServiceLabel {
|
||||||
|
labels = append(labels, Label{"service", m.ServiceName})
|
||||||
|
} else {
|
||||||
|
key = insert(0, m.ServiceName, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
allowed, labelsFiltered := m.allowMetric(key, labels)
|
||||||
|
if !allowed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
m.sink.AddSampleWithLabels(key, val, labelsFiltered)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) MeasureSince(key []string, start time.Time) {
|
||||||
|
m.MeasureSinceWithLabels(key, start, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Metrics) MeasureSinceWithLabels(key []string, start time.Time, labels []Label) {
|
||||||
|
if m.HostName != "" && m.EnableHostnameLabel {
|
||||||
|
labels = append(labels, Label{"host", m.HostName})
|
||||||
|
}
|
||||||
|
if m.EnableTypePrefix {
|
||||||
|
key = insert(0, "timer", key)
|
||||||
|
}
|
||||||
|
if m.ServiceName != "" {
|
||||||
|
if m.EnableServiceLabel {
|
||||||
|
labels = append(labels, Label{"service", m.ServiceName})
|
||||||
|
} else {
|
||||||
|
key = insert(0, m.ServiceName, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
allowed, labelsFiltered := m.allowMetric(key, labels)
|
||||||
|
if !allowed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
now := time.Now()
|
||||||
|
elapsed := now.Sub(start)
|
||||||
|
msec := float32(elapsed.Nanoseconds()) / float32(m.TimerGranularity)
|
||||||
|
m.sink.AddSampleWithLabels(key, msec, labelsFiltered)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UpdateFilter overwrites the existing filter with the given rules.
|
||||||
|
func (m *Metrics) UpdateFilter(allow, block []string) {
|
||||||
|
m.UpdateFilterAndLabels(allow, block, m.AllowedLabels, m.BlockedLabels)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UpdateFilterAndLabels overwrites the existing filter with the given rules.
|
||||||
|
func (m *Metrics) UpdateFilterAndLabels(allow, block, allowedLabels, blockedLabels []string) {
|
||||||
|
m.filterLock.Lock()
|
||||||
|
defer m.filterLock.Unlock()
|
||||||
|
|
||||||
|
m.AllowedPrefixes = allow
|
||||||
|
m.BlockedPrefixes = block
|
||||||
|
|
||||||
|
if allowedLabels == nil {
|
||||||
|
// Having a white list means we take only elements from it
|
||||||
|
m.allowedLabels = nil
|
||||||
|
} else {
|
||||||
|
m.allowedLabels = make(map[string]bool)
|
||||||
|
for _, v := range allowedLabels {
|
||||||
|
m.allowedLabels[v] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m.blockedLabels = make(map[string]bool)
|
||||||
|
for _, v := range blockedLabels {
|
||||||
|
m.blockedLabels[v] = true
|
||||||
|
}
|
||||||
|
m.AllowedLabels = allowedLabels
|
||||||
|
m.BlockedLabels = blockedLabels
|
||||||
|
|
||||||
|
m.filter = iradix.New()
|
||||||
|
for _, prefix := range m.AllowedPrefixes {
|
||||||
|
m.filter, _, _ = m.filter.Insert([]byte(prefix), true)
|
||||||
|
}
|
||||||
|
for _, prefix := range m.BlockedPrefixes {
|
||||||
|
m.filter, _, _ = m.filter.Insert([]byte(prefix), false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// labelIsAllowed return true if a should be included in metric
|
||||||
|
// the caller should lock m.filterLock while calling this method
|
||||||
|
func (m *Metrics) labelIsAllowed(label *Label) bool {
|
||||||
|
labelName := (*label).Name
|
||||||
|
if m.blockedLabels != nil {
|
||||||
|
_, ok := m.blockedLabels[labelName]
|
||||||
|
if ok {
|
||||||
|
// If present, let's remove this label
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if m.allowedLabels != nil {
|
||||||
|
_, ok := m.allowedLabels[labelName]
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
// Allow by default
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// filterLabels return only allowed labels
|
||||||
|
// the caller should lock m.filterLock while calling this method
|
||||||
|
func (m *Metrics) filterLabels(labels []Label) []Label {
|
||||||
|
if labels == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
toReturn := labels[:0]
|
||||||
|
for _, label := range labels {
|
||||||
|
if m.labelIsAllowed(&label) {
|
||||||
|
toReturn = append(toReturn, label)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return toReturn
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns whether the metric should be allowed based on configured prefix filters
|
||||||
|
// Also return the applicable labels
|
||||||
|
func (m *Metrics) allowMetric(key []string, labels []Label) (bool, []Label) {
|
||||||
|
m.filterLock.RLock()
|
||||||
|
defer m.filterLock.RUnlock()
|
||||||
|
|
||||||
|
if m.filter == nil || m.filter.Len() == 0 {
|
||||||
|
return m.Config.FilterDefault, m.filterLabels(labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, allowed, ok := m.filter.Root().LongestPrefix([]byte(strings.Join(key, ".")))
|
||||||
|
if !ok {
|
||||||
|
return m.Config.FilterDefault, m.filterLabels(labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
return allowed.(bool), m.filterLabels(labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Periodically collects runtime stats to publish
|
||||||
|
func (m *Metrics) collectStats() {
|
||||||
|
for {
|
||||||
|
time.Sleep(m.ProfileInterval)
|
||||||
|
m.emitRuntimeStats()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Emits various runtime statsitics
|
||||||
|
func (m *Metrics) emitRuntimeStats() {
|
||||||
|
// Export number of Goroutines
|
||||||
|
numRoutines := runtime.NumGoroutine()
|
||||||
|
m.SetGauge([]string{"runtime", "num_goroutines"}, float32(numRoutines))
|
||||||
|
|
||||||
|
// Export memory stats
|
||||||
|
var stats runtime.MemStats
|
||||||
|
runtime.ReadMemStats(&stats)
|
||||||
|
m.SetGauge([]string{"runtime", "alloc_bytes"}, float32(stats.Alloc))
|
||||||
|
m.SetGauge([]string{"runtime", "sys_bytes"}, float32(stats.Sys))
|
||||||
|
m.SetGauge([]string{"runtime", "malloc_count"}, float32(stats.Mallocs))
|
||||||
|
m.SetGauge([]string{"runtime", "free_count"}, float32(stats.Frees))
|
||||||
|
m.SetGauge([]string{"runtime", "heap_objects"}, float32(stats.HeapObjects))
|
||||||
|
m.SetGauge([]string{"runtime", "total_gc_pause_ns"}, float32(stats.PauseTotalNs))
|
||||||
|
m.SetGauge([]string{"runtime", "total_gc_runs"}, float32(stats.NumGC))
|
||||||
|
|
||||||
|
// Export info about the last few GC runs
|
||||||
|
num := stats.NumGC
|
||||||
|
|
||||||
|
// Handle wrap around
|
||||||
|
if num < m.lastNumGC {
|
||||||
|
m.lastNumGC = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure we don't scan more than 256
|
||||||
|
if num-m.lastNumGC >= 256 {
|
||||||
|
m.lastNumGC = num - 255
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := m.lastNumGC; i < num; i++ {
|
||||||
|
pause := stats.PauseNs[i%256]
|
||||||
|
m.AddSample([]string{"runtime", "gc_pause_ns"}, float32(pause))
|
||||||
|
}
|
||||||
|
m.lastNumGC = num
|
||||||
|
}
|
||||||
|
|
||||||
|
// Inserts a string value at an index into the slice
|
||||||
|
func insert(i int, v string, s []string) []string {
|
||||||
|
s = append(s, "")
|
||||||
|
copy(s[i+1:], s[i:])
|
||||||
|
s[i] = v
|
||||||
|
return s
|
||||||
|
}
|
115
vendor/github.com/armon/go-metrics/sink.go
generated
vendored
Normal file
115
vendor/github.com/armon/go-metrics/sink.go
generated
vendored
Normal file
|
@ -0,0 +1,115 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
)
|
||||||
|
|
||||||
|
// The MetricSink interface is used to transmit metrics information
|
||||||
|
// to an external system
|
||||||
|
type MetricSink interface {
|
||||||
|
// A Gauge should retain the last value it is set to
|
||||||
|
SetGauge(key []string, val float32)
|
||||||
|
SetGaugeWithLabels(key []string, val float32, labels []Label)
|
||||||
|
|
||||||
|
// Should emit a Key/Value pair for each call
|
||||||
|
EmitKey(key []string, val float32)
|
||||||
|
|
||||||
|
// Counters should accumulate values
|
||||||
|
IncrCounter(key []string, val float32)
|
||||||
|
IncrCounterWithLabels(key []string, val float32, labels []Label)
|
||||||
|
|
||||||
|
// Samples are for timing information, where quantiles are used
|
||||||
|
AddSample(key []string, val float32)
|
||||||
|
AddSampleWithLabels(key []string, val float32, labels []Label)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BlackholeSink is used to just blackhole messages
|
||||||
|
type BlackholeSink struct{}
|
||||||
|
|
||||||
|
func (*BlackholeSink) SetGauge(key []string, val float32) {}
|
||||||
|
func (*BlackholeSink) SetGaugeWithLabels(key []string, val float32, labels []Label) {}
|
||||||
|
func (*BlackholeSink) EmitKey(key []string, val float32) {}
|
||||||
|
func (*BlackholeSink) IncrCounter(key []string, val float32) {}
|
||||||
|
func (*BlackholeSink) IncrCounterWithLabels(key []string, val float32, labels []Label) {}
|
||||||
|
func (*BlackholeSink) AddSample(key []string, val float32) {}
|
||||||
|
func (*BlackholeSink) AddSampleWithLabels(key []string, val float32, labels []Label) {}
|
||||||
|
|
||||||
|
// FanoutSink is used to sink to fanout values to multiple sinks
|
||||||
|
type FanoutSink []MetricSink
|
||||||
|
|
||||||
|
func (fh FanoutSink) SetGauge(key []string, val float32) {
|
||||||
|
fh.SetGaugeWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
for _, s := range fh {
|
||||||
|
s.SetGaugeWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) EmitKey(key []string, val float32) {
|
||||||
|
for _, s := range fh {
|
||||||
|
s.EmitKey(key, val)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) IncrCounter(key []string, val float32) {
|
||||||
|
fh.IncrCounterWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
for _, s := range fh {
|
||||||
|
s.IncrCounterWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) AddSample(key []string, val float32) {
|
||||||
|
fh.AddSampleWithLabels(key, val, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fh FanoutSink) AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
for _, s := range fh {
|
||||||
|
s.AddSampleWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// sinkURLFactoryFunc is an generic interface around the *SinkFromURL() function provided
|
||||||
|
// by each sink type
|
||||||
|
type sinkURLFactoryFunc func(*url.URL) (MetricSink, error)
|
||||||
|
|
||||||
|
// sinkRegistry supports the generic NewMetricSink function by mapping URL
|
||||||
|
// schemes to metric sink factory functions
|
||||||
|
var sinkRegistry = map[string]sinkURLFactoryFunc{
|
||||||
|
"statsd": NewStatsdSinkFromURL,
|
||||||
|
"statsite": NewStatsiteSinkFromURL,
|
||||||
|
"inmem": NewInmemSinkFromURL,
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMetricSinkFromURL allows a generic URL input to configure any of the
|
||||||
|
// supported sinks. The scheme of the URL identifies the type of the sink, the
|
||||||
|
// and query parameters are used to set options.
|
||||||
|
//
|
||||||
|
// "statsd://" - Initializes a StatsdSink. The host and port are passed through
|
||||||
|
// as the "addr" of the sink
|
||||||
|
//
|
||||||
|
// "statsite://" - Initializes a StatsiteSink. The host and port become the
|
||||||
|
// "addr" of the sink
|
||||||
|
//
|
||||||
|
// "inmem://" - Initializes an InmemSink. The host and port are ignored. The
|
||||||
|
// "interval" and "duration" query parameters must be specified with valid
|
||||||
|
// durations, see NewInmemSink for details.
|
||||||
|
func NewMetricSinkFromURL(urlStr string) (MetricSink, error) {
|
||||||
|
u, err := url.Parse(urlStr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
sinkURLFactoryFunc := sinkRegistry[u.Scheme]
|
||||||
|
if sinkURLFactoryFunc == nil {
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"cannot create metric sink, unrecognized sink name: %q", u.Scheme)
|
||||||
|
}
|
||||||
|
|
||||||
|
return sinkURLFactoryFunc(u)
|
||||||
|
}
|
141
vendor/github.com/armon/go-metrics/start.go
generated
vendored
Normal file
141
vendor/github.com/armon/go-metrics/start.go
generated
vendored
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"sync"
|
||||||
|
"sync/atomic"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/hashicorp/go-immutable-radix"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Config is used to configure metrics settings
|
||||||
|
type Config struct {
|
||||||
|
ServiceName string // Prefixed with keys to separate services
|
||||||
|
HostName string // Hostname to use. If not provided and EnableHostname, it will be os.Hostname
|
||||||
|
EnableHostname bool // Enable prefixing gauge values with hostname
|
||||||
|
EnableHostnameLabel bool // Enable adding hostname to labels
|
||||||
|
EnableServiceLabel bool // Enable adding service to labels
|
||||||
|
EnableRuntimeMetrics bool // Enables profiling of runtime metrics (GC, Goroutines, Memory)
|
||||||
|
EnableTypePrefix bool // Prefixes key with a type ("counter", "gauge", "timer")
|
||||||
|
TimerGranularity time.Duration // Granularity of timers.
|
||||||
|
ProfileInterval time.Duration // Interval to profile runtime metrics
|
||||||
|
|
||||||
|
AllowedPrefixes []string // A list of metric prefixes to allow, with '.' as the separator
|
||||||
|
BlockedPrefixes []string // A list of metric prefixes to block, with '.' as the separator
|
||||||
|
AllowedLabels []string // A list of metric labels to allow, with '.' as the separator
|
||||||
|
BlockedLabels []string // A list of metric labels to block, with '.' as the separator
|
||||||
|
FilterDefault bool // Whether to allow metrics by default
|
||||||
|
}
|
||||||
|
|
||||||
|
// Metrics represents an instance of a metrics sink that can
|
||||||
|
// be used to emit
|
||||||
|
type Metrics struct {
|
||||||
|
Config
|
||||||
|
lastNumGC uint32
|
||||||
|
sink MetricSink
|
||||||
|
filter *iradix.Tree
|
||||||
|
allowedLabels map[string]bool
|
||||||
|
blockedLabels map[string]bool
|
||||||
|
filterLock sync.RWMutex // Lock filters and allowedLabels/blockedLabels access
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shared global metrics instance
|
||||||
|
var globalMetrics atomic.Value // *Metrics
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
// Initialize to a blackhole sink to avoid errors
|
||||||
|
globalMetrics.Store(&Metrics{sink: &BlackholeSink{}})
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultConfig provides a sane default configuration
|
||||||
|
func DefaultConfig(serviceName string) *Config {
|
||||||
|
c := &Config{
|
||||||
|
ServiceName: serviceName, // Use client provided service
|
||||||
|
HostName: "",
|
||||||
|
EnableHostname: true, // Enable hostname prefix
|
||||||
|
EnableRuntimeMetrics: true, // Enable runtime profiling
|
||||||
|
EnableTypePrefix: false, // Disable type prefix
|
||||||
|
TimerGranularity: time.Millisecond, // Timers are in milliseconds
|
||||||
|
ProfileInterval: time.Second, // Poll runtime every second
|
||||||
|
FilterDefault: true, // Don't filter metrics by default
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to get the hostname
|
||||||
|
name, _ := os.Hostname()
|
||||||
|
c.HostName = name
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// New is used to create a new instance of Metrics
|
||||||
|
func New(conf *Config, sink MetricSink) (*Metrics, error) {
|
||||||
|
met := &Metrics{}
|
||||||
|
met.Config = *conf
|
||||||
|
met.sink = sink
|
||||||
|
met.UpdateFilterAndLabels(conf.AllowedPrefixes, conf.BlockedPrefixes, conf.AllowedLabels, conf.BlockedLabels)
|
||||||
|
|
||||||
|
// Start the runtime collector
|
||||||
|
if conf.EnableRuntimeMetrics {
|
||||||
|
go met.collectStats()
|
||||||
|
}
|
||||||
|
return met, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGlobal is the same as New, but it assigns the metrics object to be
|
||||||
|
// used globally as well as returning it.
|
||||||
|
func NewGlobal(conf *Config, sink MetricSink) (*Metrics, error) {
|
||||||
|
metrics, err := New(conf, sink)
|
||||||
|
if err == nil {
|
||||||
|
globalMetrics.Store(metrics)
|
||||||
|
}
|
||||||
|
return metrics, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Proxy all the methods to the globalMetrics instance
|
||||||
|
func SetGauge(key []string, val float32) {
|
||||||
|
globalMetrics.Load().(*Metrics).SetGauge(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
globalMetrics.Load().(*Metrics).SetGaugeWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitKey(key []string, val float32) {
|
||||||
|
globalMetrics.Load().(*Metrics).EmitKey(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func IncrCounter(key []string, val float32) {
|
||||||
|
globalMetrics.Load().(*Metrics).IncrCounter(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
globalMetrics.Load().(*Metrics).IncrCounterWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddSample(key []string, val float32) {
|
||||||
|
globalMetrics.Load().(*Metrics).AddSample(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
globalMetrics.Load().(*Metrics).AddSampleWithLabels(key, val, labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
func MeasureSince(key []string, start time.Time) {
|
||||||
|
globalMetrics.Load().(*Metrics).MeasureSince(key, start)
|
||||||
|
}
|
||||||
|
|
||||||
|
func MeasureSinceWithLabels(key []string, start time.Time, labels []Label) {
|
||||||
|
globalMetrics.Load().(*Metrics).MeasureSinceWithLabels(key, start, labels)
|
||||||
|
}
|
||||||
|
|
||||||
|
func UpdateFilter(allow, block []string) {
|
||||||
|
globalMetrics.Load().(*Metrics).UpdateFilter(allow, block)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UpdateFilterAndLabels set allow/block prefixes of metrics while allowedLabels
|
||||||
|
// and blockedLabels - when not nil - allow filtering of labels in order to
|
||||||
|
// block/allow globally labels (especially useful when having large number of
|
||||||
|
// values for a given label). See README.md for more information about usage.
|
||||||
|
func UpdateFilterAndLabels(allow, block, allowedLabels, blockedLabels []string) {
|
||||||
|
globalMetrics.Load().(*Metrics).UpdateFilterAndLabels(allow, block, allowedLabels, blockedLabels)
|
||||||
|
}
|
184
vendor/github.com/armon/go-metrics/statsd.go
generated
vendored
Normal file
184
vendor/github.com/armon/go-metrics/statsd.go
generated
vendored
Normal file
|
@ -0,0 +1,184 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// statsdMaxLen is the maximum size of a packet
|
||||||
|
// to send to statsd
|
||||||
|
statsdMaxLen = 1400
|
||||||
|
)
|
||||||
|
|
||||||
|
// StatsdSink provides a MetricSink that can be used
|
||||||
|
// with a statsite or statsd metrics server. It uses
|
||||||
|
// only UDP packets, while StatsiteSink uses TCP.
|
||||||
|
type StatsdSink struct {
|
||||||
|
addr string
|
||||||
|
metricQueue chan string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStatsdSinkFromURL creates an StatsdSink from a URL. It is used
|
||||||
|
// (and tested) from NewMetricSinkFromURL.
|
||||||
|
func NewStatsdSinkFromURL(u *url.URL) (MetricSink, error) {
|
||||||
|
return NewStatsdSink(u.Host)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStatsdSink is used to create a new StatsdSink
|
||||||
|
func NewStatsdSink(addr string) (*StatsdSink, error) {
|
||||||
|
s := &StatsdSink{
|
||||||
|
addr: addr,
|
||||||
|
metricQueue: make(chan string, 4096),
|
||||||
|
}
|
||||||
|
go s.flushMetrics()
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close is used to stop flushing to statsd
|
||||||
|
func (s *StatsdSink) Shutdown() {
|
||||||
|
close(s.metricQueue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) SetGauge(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) EmitKey(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|kv\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) IncrCounter(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) AddSample(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsdSink) AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key for formatting, removes spaces
|
||||||
|
func (s *StatsdSink) flattenKey(parts []string) string {
|
||||||
|
joined := strings.Join(parts, ".")
|
||||||
|
return strings.Map(func(r rune) rune {
|
||||||
|
switch r {
|
||||||
|
case ':':
|
||||||
|
fallthrough
|
||||||
|
case ' ':
|
||||||
|
return '_'
|
||||||
|
default:
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
}, joined)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key along with labels for formatting, removes spaces
|
||||||
|
func (s *StatsdSink) flattenKeyLabels(parts []string, labels []Label) string {
|
||||||
|
for _, label := range labels {
|
||||||
|
parts = append(parts, label.Value)
|
||||||
|
}
|
||||||
|
return s.flattenKey(parts)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Does a non-blocking push to the metrics queue
|
||||||
|
func (s *StatsdSink) pushMetric(m string) {
|
||||||
|
select {
|
||||||
|
case s.metricQueue <- m:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flushes metrics
|
||||||
|
func (s *StatsdSink) flushMetrics() {
|
||||||
|
var sock net.Conn
|
||||||
|
var err error
|
||||||
|
var wait <-chan time.Time
|
||||||
|
ticker := time.NewTicker(flushInterval)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
CONNECT:
|
||||||
|
// Create a buffer
|
||||||
|
buf := bytes.NewBuffer(nil)
|
||||||
|
|
||||||
|
// Attempt to connect
|
||||||
|
sock, err = net.Dial("udp", s.addr)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERR] Error connecting to statsd! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case metric, ok := <-s.metricQueue:
|
||||||
|
// Get a metric from the queue
|
||||||
|
if !ok {
|
||||||
|
goto QUIT
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if this would overflow the packet size
|
||||||
|
if len(metric)+buf.Len() > statsdMaxLen {
|
||||||
|
_, err := sock.Write(buf.Bytes())
|
||||||
|
buf.Reset()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERR] Error writing to statsd! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Append to the buffer
|
||||||
|
buf.WriteString(metric)
|
||||||
|
|
||||||
|
case <-ticker.C:
|
||||||
|
if buf.Len() == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := sock.Write(buf.Bytes())
|
||||||
|
buf.Reset()
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERR] Error flushing to statsd! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WAIT:
|
||||||
|
// Wait for a while
|
||||||
|
wait = time.After(time.Duration(5) * time.Second)
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
// Dequeue the messages to avoid backlog
|
||||||
|
case _, ok := <-s.metricQueue:
|
||||||
|
if !ok {
|
||||||
|
goto QUIT
|
||||||
|
}
|
||||||
|
case <-wait:
|
||||||
|
goto CONNECT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
QUIT:
|
||||||
|
s.metricQueue = nil
|
||||||
|
}
|
172
vendor/github.com/armon/go-metrics/statsite.go
generated
vendored
Normal file
172
vendor/github.com/armon/go-metrics/statsite.go
generated
vendored
Normal file
|
@ -0,0 +1,172 @@
|
||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// We force flush the statsite metrics after this period of
|
||||||
|
// inactivity. Prevents stats from getting stuck in a buffer
|
||||||
|
// forever.
|
||||||
|
flushInterval = 100 * time.Millisecond
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewStatsiteSinkFromURL creates an StatsiteSink from a URL. It is used
|
||||||
|
// (and tested) from NewMetricSinkFromURL.
|
||||||
|
func NewStatsiteSinkFromURL(u *url.URL) (MetricSink, error) {
|
||||||
|
return NewStatsiteSink(u.Host)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StatsiteSink provides a MetricSink that can be used with a
|
||||||
|
// statsite metrics server
|
||||||
|
type StatsiteSink struct {
|
||||||
|
addr string
|
||||||
|
metricQueue chan string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewStatsiteSink is used to create a new StatsiteSink
|
||||||
|
func NewStatsiteSink(addr string) (*StatsiteSink, error) {
|
||||||
|
s := &StatsiteSink{
|
||||||
|
addr: addr,
|
||||||
|
metricQueue: make(chan string, 4096),
|
||||||
|
}
|
||||||
|
go s.flushMetrics()
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close is used to stop flushing to statsite
|
||||||
|
func (s *StatsiteSink) Shutdown() {
|
||||||
|
close(s.metricQueue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) SetGauge(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) SetGaugeWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) EmitKey(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|kv\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) IncrCounter(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) IncrCounterWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) AddSample(key []string, val float32) {
|
||||||
|
flatKey := s.flattenKey(key)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatsiteSink) AddSampleWithLabels(key []string, val float32, labels []Label) {
|
||||||
|
flatKey := s.flattenKeyLabels(key, labels)
|
||||||
|
s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key for formatting, removes spaces
|
||||||
|
func (s *StatsiteSink) flattenKey(parts []string) string {
|
||||||
|
joined := strings.Join(parts, ".")
|
||||||
|
return strings.Map(func(r rune) rune {
|
||||||
|
switch r {
|
||||||
|
case ':':
|
||||||
|
fallthrough
|
||||||
|
case ' ':
|
||||||
|
return '_'
|
||||||
|
default:
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
}, joined)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flattens the key along with labels for formatting, removes spaces
|
||||||
|
func (s *StatsiteSink) flattenKeyLabels(parts []string, labels []Label) string {
|
||||||
|
for _, label := range labels {
|
||||||
|
parts = append(parts, label.Value)
|
||||||
|
}
|
||||||
|
return s.flattenKey(parts)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Does a non-blocking push to the metrics queue
|
||||||
|
func (s *StatsiteSink) pushMetric(m string) {
|
||||||
|
select {
|
||||||
|
case s.metricQueue <- m:
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Flushes metrics
|
||||||
|
func (s *StatsiteSink) flushMetrics() {
|
||||||
|
var sock net.Conn
|
||||||
|
var err error
|
||||||
|
var wait <-chan time.Time
|
||||||
|
var buffered *bufio.Writer
|
||||||
|
ticker := time.NewTicker(flushInterval)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
CONNECT:
|
||||||
|
// Attempt to connect
|
||||||
|
sock, err = net.Dial("tcp", s.addr)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERR] Error connecting to statsite! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a buffered writer
|
||||||
|
buffered = bufio.NewWriter(sock)
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case metric, ok := <-s.metricQueue:
|
||||||
|
// Get a metric from the queue
|
||||||
|
if !ok {
|
||||||
|
goto QUIT
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to send to statsite
|
||||||
|
_, err := buffered.Write([]byte(metric))
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("[ERR] Error writing to statsite! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
case <-ticker.C:
|
||||||
|
if err := buffered.Flush(); err != nil {
|
||||||
|
log.Printf("[ERR] Error flushing to statsite! Err: %s", err)
|
||||||
|
goto WAIT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WAIT:
|
||||||
|
// Wait for a while
|
||||||
|
wait = time.After(time.Duration(5) * time.Second)
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
// Dequeue the messages to avoid backlog
|
||||||
|
case _, ok := <-s.metricQueue:
|
||||||
|
if !ok {
|
||||||
|
goto QUIT
|
||||||
|
}
|
||||||
|
case <-wait:
|
||||||
|
goto CONNECT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
QUIT:
|
||||||
|
s.metricQueue = nil
|
||||||
|
}
|
9
vendor/github.com/go-ole/go-ole/.travis.yml
generated
vendored
Normal file
9
vendor/github.com/go-ole/go-ole/.travis.yml
generated
vendored
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
language: go
|
||||||
|
sudo: false
|
||||||
|
|
||||||
|
go:
|
||||||
|
- 1.1
|
||||||
|
- 1.2
|
||||||
|
- 1.3
|
||||||
|
- 1.4
|
||||||
|
- tip
|
49
vendor/github.com/go-ole/go-ole/ChangeLog.md
generated
vendored
Normal file
49
vendor/github.com/go-ole/go-ole/ChangeLog.md
generated
vendored
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
# Version 1.x.x
|
||||||
|
|
||||||
|
* **Add more test cases and reference new test COM server project.** (Placeholder for future additions)
|
||||||
|
|
||||||
|
# Version 1.2.0-alphaX
|
||||||
|
|
||||||
|
**Minimum supported version is now Go 1.4. Go 1.1 support is deprecated, but should still build.**
|
||||||
|
|
||||||
|
* Added CI configuration for Travis-CI and AppVeyor.
|
||||||
|
* Added test InterfaceID and ClassID for the COM Test Server project.
|
||||||
|
* Added more inline documentation (#83).
|
||||||
|
* Added IEnumVARIANT implementation (#88).
|
||||||
|
* Added IEnumVARIANT test cases (#99, #100, #101).
|
||||||
|
* Added support for retrieving `time.Time` from VARIANT (#92).
|
||||||
|
* Added test case for IUnknown (#64).
|
||||||
|
* Added test case for IDispatch (#64).
|
||||||
|
* Added test cases for scalar variants (#64, #76).
|
||||||
|
|
||||||
|
# Version 1.1.1
|
||||||
|
|
||||||
|
* Fixes for Linux build.
|
||||||
|
* Fixes for Windows build.
|
||||||
|
|
||||||
|
# Version 1.1.0
|
||||||
|
|
||||||
|
The change to provide building on all platforms is a new feature. The increase in minor version reflects that and allows those who wish to stay on 1.0.x to continue to do so. Support for 1.0.x will be limited to bug fixes.
|
||||||
|
|
||||||
|
* Move GUID out of variables.go into its own file to make new documentation available.
|
||||||
|
* Move OleError out of ole.go into its own file to make new documentation available.
|
||||||
|
* Add documentation to utility functions.
|
||||||
|
* Add documentation to variant receiver functions.
|
||||||
|
* Add documentation to ole structures.
|
||||||
|
* Make variant available to other systems outside of Windows.
|
||||||
|
* Make OLE structures available to other systems outside of Windows.
|
||||||
|
|
||||||
|
## New Features
|
||||||
|
|
||||||
|
* Library should now be built on all platforms supported by Go. Library will NOOP on any platform that is not Windows.
|
||||||
|
* More functions are now documented and available on godoc.org.
|
||||||
|
|
||||||
|
# Version 1.0.1
|
||||||
|
|
||||||
|
1. Fix package references from repository location change.
|
||||||
|
|
||||||
|
# Version 1.0.0
|
||||||
|
|
||||||
|
This version is stable enough for use. The COM API is still incomplete, but provides enough functionality for accessing COM servers using IDispatch interface.
|
||||||
|
|
||||||
|
There is no changelog for this version. Check commits for history.
|
21
vendor/github.com/go-ole/go-ole/LICENSE
generated
vendored
Normal file
21
vendor/github.com/go-ole/go-ole/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright © 2013-2017 Yasuhiro Matsumoto, <mattn.jp@gmail.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
this software and associated documentation files (the “Software”), to deal in
|
||||||
|
the Software without restriction, including without limitation the rights to
|
||||||
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||||||
|
of the Software, and to permit persons to whom the Software is furnished to do
|
||||||
|
so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
46
vendor/github.com/go-ole/go-ole/README.md
generated
vendored
Normal file
46
vendor/github.com/go-ole/go-ole/README.md
generated
vendored
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
# Go OLE
|
||||||
|
|
||||||
|
[![Build status](https://ci.appveyor.com/api/projects/status/qr0u2sf7q43us9fj?svg=true)](https://ci.appveyor.com/project/jacobsantos/go-ole-jgs28)
|
||||||
|
[![Build Status](https://travis-ci.org/go-ole/go-ole.svg?branch=master)](https://travis-ci.org/go-ole/go-ole)
|
||||||
|
[![GoDoc](https://godoc.org/github.com/go-ole/go-ole?status.svg)](https://godoc.org/github.com/go-ole/go-ole)
|
||||||
|
|
||||||
|
Go bindings for Windows COM using shared libraries instead of cgo.
|
||||||
|
|
||||||
|
By Yasuhiro Matsumoto.
|
||||||
|
|
||||||
|
## Install
|
||||||
|
|
||||||
|
To experiment with go-ole, you can just compile and run the example program:
|
||||||
|
|
||||||
|
```
|
||||||
|
go get github.com/go-ole/go-ole
|
||||||
|
cd /path/to/go-ole/
|
||||||
|
go test
|
||||||
|
|
||||||
|
cd /path/to/go-ole/example/excel
|
||||||
|
go run excel.go
|
||||||
|
```
|
||||||
|
|
||||||
|
## Continuous Integration
|
||||||
|
|
||||||
|
Continuous integration configuration has been added for both Travis-CI and AppVeyor. You will have to add these to your own account for your fork in order for it to run.
|
||||||
|
|
||||||
|
**Travis-CI**
|
||||||
|
|
||||||
|
Travis-CI was added to check builds on Linux to ensure that `go get` works when cross building. Currently, Travis-CI is not used to test cross-building, but this may be changed in the future. It is also not currently possible to test the library on Linux, since COM API is specific to Windows and it is not currently possible to run a COM server on Linux or even connect to a remote COM server.
|
||||||
|
|
||||||
|
**AppVeyor**
|
||||||
|
|
||||||
|
AppVeyor is used to build on Windows using the (in-development) test COM server. It is currently only used to test the build and ensure that the code works on Windows. It will be used to register a COM server and then run the test cases based on the test COM server.
|
||||||
|
|
||||||
|
The tests currently do run and do pass and this should be maintained with commits.
|
||||||
|
|
||||||
|
## Versioning
|
||||||
|
|
||||||
|
Go OLE uses [semantic versioning](http://semver.org) for version numbers, which is similar to the version contract of the Go language. Which means that the major version will always maintain backwards compatibility with minor versions. Minor versions will only add new additions and changes. Fixes will always be in patch.
|
||||||
|
|
||||||
|
This contract should allow you to upgrade to new minor and patch versions without breakage or modifications to your existing code. Leave a ticket, if there is breakage, so that it could be fixed.
|
||||||
|
|
||||||
|
## LICENSE
|
||||||
|
|
||||||
|
Under the MIT License: http://mattn.mit-license.org/2013
|
54
vendor/github.com/go-ole/go-ole/appveyor.yml
generated
vendored
Normal file
54
vendor/github.com/go-ole/go-ole/appveyor.yml
generated
vendored
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
# Notes:
|
||||||
|
# - Minimal appveyor.yml file is an empty file. All sections are optional.
|
||||||
|
# - Indent each level of configuration with 2 spaces. Do not use tabs!
|
||||||
|
# - All section names are case-sensitive.
|
||||||
|
# - Section names should be unique on each level.
|
||||||
|
|
||||||
|
version: "1.3.0.{build}-alpha-{branch}"
|
||||||
|
|
||||||
|
os: Windows Server 2012 R2
|
||||||
|
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- master
|
||||||
|
- v1.2
|
||||||
|
- v1.1
|
||||||
|
- v1.0
|
||||||
|
|
||||||
|
skip_tags: true
|
||||||
|
|
||||||
|
clone_folder: c:\gopath\src\github.com\go-ole\go-ole
|
||||||
|
|
||||||
|
environment:
|
||||||
|
GOPATH: c:\gopath
|
||||||
|
matrix:
|
||||||
|
- GOARCH: amd64
|
||||||
|
GOVERSION: 1.5
|
||||||
|
GOROOT: c:\go
|
||||||
|
DOWNLOADPLATFORM: "x64"
|
||||||
|
|
||||||
|
install:
|
||||||
|
- choco install mingw
|
||||||
|
- SET PATH=c:\tools\mingw64\bin;%PATH%
|
||||||
|
# - Download COM Server
|
||||||
|
- ps: Start-FileDownload "https://github.com/go-ole/test-com-server/releases/download/v1.0.2/test-com-server-${env:DOWNLOADPLATFORM}.zip"
|
||||||
|
- 7z e test-com-server-%DOWNLOADPLATFORM%.zip -oc:\gopath\src\github.com\go-ole\go-ole > NUL
|
||||||
|
- c:\gopath\src\github.com\go-ole\go-ole\build\register-assembly.bat
|
||||||
|
# - set
|
||||||
|
- go version
|
||||||
|
- go env
|
||||||
|
- go get -u golang.org/x/tools/cmd/cover
|
||||||
|
- go get -u golang.org/x/tools/cmd/godoc
|
||||||
|
- go get -u golang.org/x/tools/cmd/stringer
|
||||||
|
|
||||||
|
build_script:
|
||||||
|
- cd c:\gopath\src\github.com\go-ole\go-ole
|
||||||
|
- go get -v -t ./...
|
||||||
|
- go build
|
||||||
|
- go test -v -cover ./...
|
||||||
|
|
||||||
|
# disable automatic tests
|
||||||
|
test: off
|
||||||
|
|
||||||
|
# disable deployment
|
||||||
|
deploy: off
|
344
vendor/github.com/go-ole/go-ole/com.go
generated
vendored
Normal file
344
vendor/github.com/go-ole/go-ole/com.go
generated
vendored
Normal file
|
@ -0,0 +1,344 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unicode/utf16"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
procCoInitialize, _ = modole32.FindProc("CoInitialize")
|
||||||
|
procCoInitializeEx, _ = modole32.FindProc("CoInitializeEx")
|
||||||
|
procCoUninitialize, _ = modole32.FindProc("CoUninitialize")
|
||||||
|
procCoCreateInstance, _ = modole32.FindProc("CoCreateInstance")
|
||||||
|
procCoTaskMemFree, _ = modole32.FindProc("CoTaskMemFree")
|
||||||
|
procCLSIDFromProgID, _ = modole32.FindProc("CLSIDFromProgID")
|
||||||
|
procCLSIDFromString, _ = modole32.FindProc("CLSIDFromString")
|
||||||
|
procStringFromCLSID, _ = modole32.FindProc("StringFromCLSID")
|
||||||
|
procStringFromIID, _ = modole32.FindProc("StringFromIID")
|
||||||
|
procIIDFromString, _ = modole32.FindProc("IIDFromString")
|
||||||
|
procCoGetObject, _ = modole32.FindProc("CoGetObject")
|
||||||
|
procGetUserDefaultLCID, _ = modkernel32.FindProc("GetUserDefaultLCID")
|
||||||
|
procCopyMemory, _ = modkernel32.FindProc("RtlMoveMemory")
|
||||||
|
procVariantInit, _ = modoleaut32.FindProc("VariantInit")
|
||||||
|
procVariantClear, _ = modoleaut32.FindProc("VariantClear")
|
||||||
|
procVariantTimeToSystemTime, _ = modoleaut32.FindProc("VariantTimeToSystemTime")
|
||||||
|
procSysAllocString, _ = modoleaut32.FindProc("SysAllocString")
|
||||||
|
procSysAllocStringLen, _ = modoleaut32.FindProc("SysAllocStringLen")
|
||||||
|
procSysFreeString, _ = modoleaut32.FindProc("SysFreeString")
|
||||||
|
procSysStringLen, _ = modoleaut32.FindProc("SysStringLen")
|
||||||
|
procCreateDispTypeInfo, _ = modoleaut32.FindProc("CreateDispTypeInfo")
|
||||||
|
procCreateStdDispatch, _ = modoleaut32.FindProc("CreateStdDispatch")
|
||||||
|
procGetActiveObject, _ = modoleaut32.FindProc("GetActiveObject")
|
||||||
|
|
||||||
|
procGetMessageW, _ = moduser32.FindProc("GetMessageW")
|
||||||
|
procDispatchMessageW, _ = moduser32.FindProc("DispatchMessageW")
|
||||||
|
)
|
||||||
|
|
||||||
|
// coInitialize initializes COM library on current thread.
|
||||||
|
//
|
||||||
|
// MSDN documentation suggests that this function should not be called. Call
|
||||||
|
// CoInitializeEx() instead. The reason has to do with threading and this
|
||||||
|
// function is only for single-threaded apartments.
|
||||||
|
//
|
||||||
|
// That said, most users of the library have gotten away with just this
|
||||||
|
// function. If you are experiencing threading issues, then use
|
||||||
|
// CoInitializeEx().
|
||||||
|
func coInitialize() (err error) {
|
||||||
|
// http://msdn.microsoft.com/en-us/library/windows/desktop/ms678543(v=vs.85).aspx
|
||||||
|
// Suggests that no value should be passed to CoInitialized.
|
||||||
|
// Could just be Call() since the parameter is optional. <-- Needs testing to be sure.
|
||||||
|
hr, _, _ := procCoInitialize.Call(uintptr(0))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// coInitializeEx initializes COM library with concurrency model.
|
||||||
|
func coInitializeEx(coinit uint32) (err error) {
|
||||||
|
// http://msdn.microsoft.com/en-us/library/windows/desktop/ms695279(v=vs.85).aspx
|
||||||
|
// Suggests that the first parameter is not only optional but should always be NULL.
|
||||||
|
hr, _, _ := procCoInitializeEx.Call(uintptr(0), uintptr(coinit))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoInitialize initializes COM library on current thread.
|
||||||
|
//
|
||||||
|
// MSDN documentation suggests that this function should not be called. Call
|
||||||
|
// CoInitializeEx() instead. The reason has to do with threading and this
|
||||||
|
// function is only for single-threaded apartments.
|
||||||
|
//
|
||||||
|
// That said, most users of the library have gotten away with just this
|
||||||
|
// function. If you are experiencing threading issues, then use
|
||||||
|
// CoInitializeEx().
|
||||||
|
func CoInitialize(p uintptr) (err error) {
|
||||||
|
// p is ignored and won't be used.
|
||||||
|
// Avoid any variable not used errors.
|
||||||
|
p = uintptr(0)
|
||||||
|
return coInitialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoInitializeEx initializes COM library with concurrency model.
|
||||||
|
func CoInitializeEx(p uintptr, coinit uint32) (err error) {
|
||||||
|
// Avoid any variable not used errors.
|
||||||
|
p = uintptr(0)
|
||||||
|
return coInitializeEx(coinit)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoUninitialize uninitializes COM Library.
|
||||||
|
func CoUninitialize() {
|
||||||
|
procCoUninitialize.Call()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoTaskMemFree frees memory pointer.
|
||||||
|
func CoTaskMemFree(memptr uintptr) {
|
||||||
|
procCoTaskMemFree.Call(memptr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CLSIDFromProgID retrieves Class Identifier with the given Program Identifier.
|
||||||
|
//
|
||||||
|
// The Programmatic Identifier must be registered, because it will be looked up
|
||||||
|
// in the Windows Registry. The registry entry has the following keys: CLSID,
|
||||||
|
// Insertable, Protocol and Shell
|
||||||
|
// (https://msdn.microsoft.com/en-us/library/dd542719(v=vs.85).aspx).
|
||||||
|
//
|
||||||
|
// programID identifies the class id with less precision and is not guaranteed
|
||||||
|
// to be unique. These are usually found in the registry under
|
||||||
|
// HKEY_LOCAL_MACHINE\SOFTWARE\Classes, usually with the format of
|
||||||
|
// "Program.Component.Version" with version being optional.
|
||||||
|
//
|
||||||
|
// CLSIDFromProgID in Windows API.
|
||||||
|
func CLSIDFromProgID(progId string) (clsid *GUID, err error) {
|
||||||
|
var guid GUID
|
||||||
|
lpszProgID := uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(progId)))
|
||||||
|
hr, _, _ := procCLSIDFromProgID.Call(lpszProgID, uintptr(unsafe.Pointer(&guid)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
clsid = &guid
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CLSIDFromString retrieves Class ID from string representation.
|
||||||
|
//
|
||||||
|
// This is technically the string version of the GUID and will convert the
|
||||||
|
// string to object.
|
||||||
|
//
|
||||||
|
// CLSIDFromString in Windows API.
|
||||||
|
func CLSIDFromString(str string) (clsid *GUID, err error) {
|
||||||
|
var guid GUID
|
||||||
|
lpsz := uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(str)))
|
||||||
|
hr, _, _ := procCLSIDFromString.Call(lpsz, uintptr(unsafe.Pointer(&guid)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
clsid = &guid
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// StringFromCLSID returns GUID formated string from GUID object.
|
||||||
|
func StringFromCLSID(clsid *GUID) (str string, err error) {
|
||||||
|
var p *uint16
|
||||||
|
hr, _, _ := procStringFromCLSID.Call(uintptr(unsafe.Pointer(clsid)), uintptr(unsafe.Pointer(&p)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
str = LpOleStrToString(p)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// IIDFromString returns GUID from program ID.
|
||||||
|
func IIDFromString(progId string) (clsid *GUID, err error) {
|
||||||
|
var guid GUID
|
||||||
|
lpsz := uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(progId)))
|
||||||
|
hr, _, _ := procIIDFromString.Call(lpsz, uintptr(unsafe.Pointer(&guid)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
clsid = &guid
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// StringFromIID returns GUID formatted string from GUID object.
|
||||||
|
func StringFromIID(iid *GUID) (str string, err error) {
|
||||||
|
var p *uint16
|
||||||
|
hr, _, _ := procStringFromIID.Call(uintptr(unsafe.Pointer(iid)), uintptr(unsafe.Pointer(&p)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
str = LpOleStrToString(p)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateInstance of single uninitialized object with GUID.
|
||||||
|
func CreateInstance(clsid *GUID, iid *GUID) (unk *IUnknown, err error) {
|
||||||
|
if iid == nil {
|
||||||
|
iid = IID_IUnknown
|
||||||
|
}
|
||||||
|
hr, _, _ := procCoCreateInstance.Call(
|
||||||
|
uintptr(unsafe.Pointer(clsid)),
|
||||||
|
0,
|
||||||
|
CLSCTX_SERVER,
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(&unk)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetActiveObject retrieves pointer to active object.
|
||||||
|
func GetActiveObject(clsid *GUID, iid *GUID) (unk *IUnknown, err error) {
|
||||||
|
if iid == nil {
|
||||||
|
iid = IID_IUnknown
|
||||||
|
}
|
||||||
|
hr, _, _ := procGetActiveObject.Call(
|
||||||
|
uintptr(unsafe.Pointer(clsid)),
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(&unk)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
type BindOpts struct {
|
||||||
|
CbStruct uint32
|
||||||
|
GrfFlags uint32
|
||||||
|
GrfMode uint32
|
||||||
|
TickCountDeadline uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetObject retrieves pointer to active object.
|
||||||
|
func GetObject(programID string, bindOpts *BindOpts, iid *GUID) (unk *IUnknown, err error) {
|
||||||
|
if bindOpts != nil {
|
||||||
|
bindOpts.CbStruct = uint32(unsafe.Sizeof(BindOpts{}))
|
||||||
|
}
|
||||||
|
if iid == nil {
|
||||||
|
iid = IID_IUnknown
|
||||||
|
}
|
||||||
|
hr, _, _ := procCoGetObject.Call(
|
||||||
|
uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(programID))),
|
||||||
|
uintptr(unsafe.Pointer(bindOpts)),
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(&unk)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// VariantInit initializes variant.
|
||||||
|
func VariantInit(v *VARIANT) (err error) {
|
||||||
|
hr, _, _ := procVariantInit.Call(uintptr(unsafe.Pointer(v)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// VariantClear clears value in Variant settings to VT_EMPTY.
|
||||||
|
func VariantClear(v *VARIANT) (err error) {
|
||||||
|
hr, _, _ := procVariantClear.Call(uintptr(unsafe.Pointer(v)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysAllocString allocates memory for string and copies string into memory.
|
||||||
|
func SysAllocString(v string) (ss *int16) {
|
||||||
|
pss, _, _ := procSysAllocString.Call(uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(v))))
|
||||||
|
ss = (*int16)(unsafe.Pointer(pss))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysAllocStringLen copies up to length of given string returning pointer.
|
||||||
|
func SysAllocStringLen(v string) (ss *int16) {
|
||||||
|
utf16 := utf16.Encode([]rune(v + "\x00"))
|
||||||
|
ptr := &utf16[0]
|
||||||
|
|
||||||
|
pss, _, _ := procSysAllocStringLen.Call(uintptr(unsafe.Pointer(ptr)), uintptr(len(utf16)-1))
|
||||||
|
ss = (*int16)(unsafe.Pointer(pss))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysFreeString frees string system memory. This must be called with SysAllocString.
|
||||||
|
func SysFreeString(v *int16) (err error) {
|
||||||
|
hr, _, _ := procSysFreeString.Call(uintptr(unsafe.Pointer(v)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysStringLen is the length of the system allocated string.
|
||||||
|
func SysStringLen(v *int16) uint32 {
|
||||||
|
l, _, _ := procSysStringLen.Call(uintptr(unsafe.Pointer(v)))
|
||||||
|
return uint32(l)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateStdDispatch provides default IDispatch implementation for IUnknown.
|
||||||
|
//
|
||||||
|
// This handles default IDispatch implementation for objects. It haves a few
|
||||||
|
// limitations with only supporting one language. It will also only return
|
||||||
|
// default exception codes.
|
||||||
|
func CreateStdDispatch(unk *IUnknown, v uintptr, ptinfo *IUnknown) (disp *IDispatch, err error) {
|
||||||
|
hr, _, _ := procCreateStdDispatch.Call(
|
||||||
|
uintptr(unsafe.Pointer(unk)),
|
||||||
|
v,
|
||||||
|
uintptr(unsafe.Pointer(ptinfo)),
|
||||||
|
uintptr(unsafe.Pointer(&disp)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateDispTypeInfo provides default ITypeInfo implementation for IDispatch.
|
||||||
|
//
|
||||||
|
// This will not handle the full implementation of the interface.
|
||||||
|
func CreateDispTypeInfo(idata *INTERFACEDATA) (pptinfo *IUnknown, err error) {
|
||||||
|
hr, _, _ := procCreateDispTypeInfo.Call(
|
||||||
|
uintptr(unsafe.Pointer(idata)),
|
||||||
|
uintptr(GetUserDefaultLCID()),
|
||||||
|
uintptr(unsafe.Pointer(&pptinfo)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// copyMemory moves location of a block of memory.
|
||||||
|
func copyMemory(dest unsafe.Pointer, src unsafe.Pointer, length uint32) {
|
||||||
|
procCopyMemory.Call(uintptr(dest), uintptr(src), uintptr(length))
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetUserDefaultLCID retrieves current user default locale.
|
||||||
|
func GetUserDefaultLCID() (lcid uint32) {
|
||||||
|
ret, _, _ := procGetUserDefaultLCID.Call()
|
||||||
|
lcid = uint32(ret)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMessage in message queue from runtime.
|
||||||
|
//
|
||||||
|
// This function appears to block. PeekMessage does not block.
|
||||||
|
func GetMessage(msg *Msg, hwnd uint32, MsgFilterMin uint32, MsgFilterMax uint32) (ret int32, err error) {
|
||||||
|
r0, _, err := procGetMessageW.Call(uintptr(unsafe.Pointer(msg)), uintptr(hwnd), uintptr(MsgFilterMin), uintptr(MsgFilterMax))
|
||||||
|
ret = int32(r0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DispatchMessage to window procedure.
|
||||||
|
func DispatchMessage(msg *Msg) (ret int32) {
|
||||||
|
r0, _, _ := procDispatchMessageW.Call(uintptr(unsafe.Pointer(msg)))
|
||||||
|
ret = int32(r0)
|
||||||
|
return
|
||||||
|
}
|
174
vendor/github.com/go-ole/go-ole/com_func.go
generated
vendored
Normal file
174
vendor/github.com/go-ole/go-ole/com_func.go
generated
vendored
Normal file
|
@ -0,0 +1,174 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// coInitialize initializes COM library on current thread.
|
||||||
|
//
|
||||||
|
// MSDN documentation suggests that this function should not be called. Call
|
||||||
|
// CoInitializeEx() instead. The reason has to do with threading and this
|
||||||
|
// function is only for single-threaded apartments.
|
||||||
|
//
|
||||||
|
// That said, most users of the library have gotten away with just this
|
||||||
|
// function. If you are experiencing threading issues, then use
|
||||||
|
// CoInitializeEx().
|
||||||
|
func coInitialize() error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// coInitializeEx initializes COM library with concurrency model.
|
||||||
|
func coInitializeEx(coinit uint32) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoInitialize initializes COM library on current thread.
|
||||||
|
//
|
||||||
|
// MSDN documentation suggests that this function should not be called. Call
|
||||||
|
// CoInitializeEx() instead. The reason has to do with threading and this
|
||||||
|
// function is only for single-threaded apartments.
|
||||||
|
//
|
||||||
|
// That said, most users of the library have gotten away with just this
|
||||||
|
// function. If you are experiencing threading issues, then use
|
||||||
|
// CoInitializeEx().
|
||||||
|
func CoInitialize(p uintptr) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoInitializeEx initializes COM library with concurrency model.
|
||||||
|
func CoInitializeEx(p uintptr, coinit uint32) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CoUninitialize uninitializes COM Library.
|
||||||
|
func CoUninitialize() {}
|
||||||
|
|
||||||
|
// CoTaskMemFree frees memory pointer.
|
||||||
|
func CoTaskMemFree(memptr uintptr) {}
|
||||||
|
|
||||||
|
// CLSIDFromProgID retrieves Class Identifier with the given Program Identifier.
|
||||||
|
//
|
||||||
|
// The Programmatic Identifier must be registered, because it will be looked up
|
||||||
|
// in the Windows Registry. The registry entry has the following keys: CLSID,
|
||||||
|
// Insertable, Protocol and Shell
|
||||||
|
// (https://msdn.microsoft.com/en-us/library/dd542719(v=vs.85).aspx).
|
||||||
|
//
|
||||||
|
// programID identifies the class id with less precision and is not guaranteed
|
||||||
|
// to be unique. These are usually found in the registry under
|
||||||
|
// HKEY_LOCAL_MACHINE\SOFTWARE\Classes, usually with the format of
|
||||||
|
// "Program.Component.Version" with version being optional.
|
||||||
|
//
|
||||||
|
// CLSIDFromProgID in Windows API.
|
||||||
|
func CLSIDFromProgID(progId string) (*GUID, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CLSIDFromString retrieves Class ID from string representation.
|
||||||
|
//
|
||||||
|
// This is technically the string version of the GUID and will convert the
|
||||||
|
// string to object.
|
||||||
|
//
|
||||||
|
// CLSIDFromString in Windows API.
|
||||||
|
func CLSIDFromString(str string) (*GUID, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StringFromCLSID returns GUID formated string from GUID object.
|
||||||
|
func StringFromCLSID(clsid *GUID) (string, error) {
|
||||||
|
return "", NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// IIDFromString returns GUID from program ID.
|
||||||
|
func IIDFromString(progId string) (*GUID, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StringFromIID returns GUID formatted string from GUID object.
|
||||||
|
func StringFromIID(iid *GUID) (string, error) {
|
||||||
|
return "", NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateInstance of single uninitialized object with GUID.
|
||||||
|
func CreateInstance(clsid *GUID, iid *GUID) (*IUnknown, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetActiveObject retrieves pointer to active object.
|
||||||
|
func GetActiveObject(clsid *GUID, iid *GUID) (*IUnknown, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// VariantInit initializes variant.
|
||||||
|
func VariantInit(v *VARIANT) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// VariantClear clears value in Variant settings to VT_EMPTY.
|
||||||
|
func VariantClear(v *VARIANT) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysAllocString allocates memory for string and copies string into memory.
|
||||||
|
func SysAllocString(v string) *int16 {
|
||||||
|
u := int16(0)
|
||||||
|
return &u
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysAllocStringLen copies up to length of given string returning pointer.
|
||||||
|
func SysAllocStringLen(v string) *int16 {
|
||||||
|
u := int16(0)
|
||||||
|
return &u
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysFreeString frees string system memory. This must be called with SysAllocString.
|
||||||
|
func SysFreeString(v *int16) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SysStringLen is the length of the system allocated string.
|
||||||
|
func SysStringLen(v *int16) uint32 {
|
||||||
|
return uint32(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateStdDispatch provides default IDispatch implementation for IUnknown.
|
||||||
|
//
|
||||||
|
// This handles default IDispatch implementation for objects. It haves a few
|
||||||
|
// limitations with only supporting one language. It will also only return
|
||||||
|
// default exception codes.
|
||||||
|
func CreateStdDispatch(unk *IUnknown, v uintptr, ptinfo *IUnknown) (*IDispatch, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateDispTypeInfo provides default ITypeInfo implementation for IDispatch.
|
||||||
|
//
|
||||||
|
// This will not handle the full implementation of the interface.
|
||||||
|
func CreateDispTypeInfo(idata *INTERFACEDATA) (*IUnknown, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// copyMemory moves location of a block of memory.
|
||||||
|
func copyMemory(dest unsafe.Pointer, src unsafe.Pointer, length uint32) {}
|
||||||
|
|
||||||
|
// GetUserDefaultLCID retrieves current user default locale.
|
||||||
|
func GetUserDefaultLCID() uint32 {
|
||||||
|
return uint32(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMessage in message queue from runtime.
|
||||||
|
//
|
||||||
|
// This function appears to block. PeekMessage does not block.
|
||||||
|
func GetMessage(msg *Msg, hwnd uint32, MsgFilterMin uint32, MsgFilterMax uint32) (int32, error) {
|
||||||
|
return int32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DispatchMessage to window procedure.
|
||||||
|
func DispatchMessage(msg *Msg) int32 {
|
||||||
|
return int32(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetVariantDate(value uint64) (time.Time, error) {
|
||||||
|
return time.Now(), NewError(E_NOTIMPL)
|
||||||
|
}
|
192
vendor/github.com/go-ole/go-ole/connect.go
generated
vendored
Normal file
192
vendor/github.com/go-ole/go-ole/connect.go
generated
vendored
Normal file
|
@ -0,0 +1,192 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
// Connection contains IUnknown for fluent interface interaction.
|
||||||
|
//
|
||||||
|
// Deprecated. Use oleutil package instead.
|
||||||
|
type Connection struct {
|
||||||
|
Object *IUnknown // Access COM
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize COM.
|
||||||
|
func (*Connection) Initialize() (err error) {
|
||||||
|
return coInitialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Uninitialize COM.
|
||||||
|
func (*Connection) Uninitialize() {
|
||||||
|
CoUninitialize()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create IUnknown object based first on ProgId and then from String.
|
||||||
|
func (c *Connection) Create(progId string) (err error) {
|
||||||
|
var clsid *GUID
|
||||||
|
clsid, err = CLSIDFromProgID(progId)
|
||||||
|
if err != nil {
|
||||||
|
clsid, err = CLSIDFromString(progId)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unknown, err := CreateInstance(clsid, IID_IUnknown)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.Object = unknown
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Release IUnknown object.
|
||||||
|
func (c *Connection) Release() {
|
||||||
|
c.Object.Release()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load COM object from list of programIDs or strings.
|
||||||
|
func (c *Connection) Load(names ...string) (errors []error) {
|
||||||
|
var tempErrors []error = make([]error, len(names))
|
||||||
|
var numErrors int = 0
|
||||||
|
for _, name := range names {
|
||||||
|
err := c.Create(name)
|
||||||
|
if err != nil {
|
||||||
|
tempErrors = append(tempErrors, err)
|
||||||
|
numErrors += 1
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
copy(errors, tempErrors[0:numErrors])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dispatch returns Dispatch object.
|
||||||
|
func (c *Connection) Dispatch() (object *Dispatch, err error) {
|
||||||
|
dispatch, err := c.Object.QueryInterface(IID_IDispatch)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
object = &Dispatch{dispatch}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dispatch stores IDispatch object.
|
||||||
|
type Dispatch struct {
|
||||||
|
Object *IDispatch // Dispatch object.
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call method on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) Call(method string, params ...interface{}) (result *VARIANT, err error) {
|
||||||
|
id, err := d.GetId(method)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err = d.Invoke(id, DISPATCH_METHOD, params)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustCall method on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) MustCall(method string, params ...interface{}) (result *VARIANT) {
|
||||||
|
id, err := d.GetId(method)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err = d.Invoke(id, DISPATCH_METHOD, params)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get property on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) Get(name string, params ...interface{}) (result *VARIANT, err error) {
|
||||||
|
id, err := d.GetId(name)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
result, err = d.Invoke(id, DISPATCH_PROPERTYGET, params)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustGet property on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) MustGet(name string, params ...interface{}) (result *VARIANT) {
|
||||||
|
id, err := d.GetId(name)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err = d.Invoke(id, DISPATCH_PROPERTYGET, params)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set property on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) Set(name string, params ...interface{}) (result *VARIANT, err error) {
|
||||||
|
id, err := d.GetId(name)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
result, err = d.Invoke(id, DISPATCH_PROPERTYPUT, params)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustSet property on IDispatch with parameters.
|
||||||
|
func (d *Dispatch) MustSet(name string, params ...interface{}) (result *VARIANT) {
|
||||||
|
id, err := d.GetId(name)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err = d.Invoke(id, DISPATCH_PROPERTYPUT, params)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetId retrieves ID of name on IDispatch.
|
||||||
|
func (d *Dispatch) GetId(name string) (id int32, err error) {
|
||||||
|
var dispid []int32
|
||||||
|
dispid, err = d.Object.GetIDsOfName([]string{name})
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
id = dispid[0]
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetIds retrieves all IDs of names on IDispatch.
|
||||||
|
func (d *Dispatch) GetIds(names ...string) (dispid []int32, err error) {
|
||||||
|
dispid, err = d.Object.GetIDsOfName(names)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Invoke IDispatch on DisplayID of dispatch type with parameters.
|
||||||
|
//
|
||||||
|
// There have been problems where if send cascading params..., it would error
|
||||||
|
// out because the parameters would be empty.
|
||||||
|
func (d *Dispatch) Invoke(id int32, dispatch int16, params []interface{}) (result *VARIANT, err error) {
|
||||||
|
if len(params) < 1 {
|
||||||
|
result, err = d.Object.Invoke(id, dispatch)
|
||||||
|
} else {
|
||||||
|
result, err = d.Object.Invoke(id, dispatch, params...)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Release IDispatch object.
|
||||||
|
func (d *Dispatch) Release() {
|
||||||
|
d.Object.Release()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Connect initializes COM and attempts to load IUnknown based on given names.
|
||||||
|
func Connect(names ...string) (connection *Connection) {
|
||||||
|
connection.Initialize()
|
||||||
|
connection.Load(names...)
|
||||||
|
return
|
||||||
|
}
|
153
vendor/github.com/go-ole/go-ole/constants.go
generated
vendored
Normal file
153
vendor/github.com/go-ole/go-ole/constants.go
generated
vendored
Normal file
|
@ -0,0 +1,153 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
const (
|
||||||
|
CLSCTX_INPROC_SERVER = 1
|
||||||
|
CLSCTX_INPROC_HANDLER = 2
|
||||||
|
CLSCTX_LOCAL_SERVER = 4
|
||||||
|
CLSCTX_INPROC_SERVER16 = 8
|
||||||
|
CLSCTX_REMOTE_SERVER = 16
|
||||||
|
CLSCTX_ALL = CLSCTX_INPROC_SERVER | CLSCTX_INPROC_HANDLER | CLSCTX_LOCAL_SERVER
|
||||||
|
CLSCTX_INPROC = CLSCTX_INPROC_SERVER | CLSCTX_INPROC_HANDLER
|
||||||
|
CLSCTX_SERVER = CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER | CLSCTX_REMOTE_SERVER
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
COINIT_APARTMENTTHREADED = 0x2
|
||||||
|
COINIT_MULTITHREADED = 0x0
|
||||||
|
COINIT_DISABLE_OLE1DDE = 0x4
|
||||||
|
COINIT_SPEED_OVER_MEMORY = 0x8
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DISPATCH_METHOD = 1
|
||||||
|
DISPATCH_PROPERTYGET = 2
|
||||||
|
DISPATCH_PROPERTYPUT = 4
|
||||||
|
DISPATCH_PROPERTYPUTREF = 8
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
S_OK = 0x00000000
|
||||||
|
E_UNEXPECTED = 0x8000FFFF
|
||||||
|
E_NOTIMPL = 0x80004001
|
||||||
|
E_OUTOFMEMORY = 0x8007000E
|
||||||
|
E_INVALIDARG = 0x80070057
|
||||||
|
E_NOINTERFACE = 0x80004002
|
||||||
|
E_POINTER = 0x80004003
|
||||||
|
E_HANDLE = 0x80070006
|
||||||
|
E_ABORT = 0x80004004
|
||||||
|
E_FAIL = 0x80004005
|
||||||
|
E_ACCESSDENIED = 0x80070005
|
||||||
|
E_PENDING = 0x8000000A
|
||||||
|
|
||||||
|
CO_E_CLASSSTRING = 0x800401F3
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
CC_FASTCALL = iota
|
||||||
|
CC_CDECL
|
||||||
|
CC_MSCPASCAL
|
||||||
|
CC_PASCAL = CC_MSCPASCAL
|
||||||
|
CC_MACPASCAL
|
||||||
|
CC_STDCALL
|
||||||
|
CC_FPFASTCALL
|
||||||
|
CC_SYSCALL
|
||||||
|
CC_MPWCDECL
|
||||||
|
CC_MPWPASCAL
|
||||||
|
CC_MAX = CC_MPWPASCAL
|
||||||
|
)
|
||||||
|
|
||||||
|
type VT uint16
|
||||||
|
|
||||||
|
const (
|
||||||
|
VT_EMPTY VT = 0x0
|
||||||
|
VT_NULL VT = 0x1
|
||||||
|
VT_I2 VT = 0x2
|
||||||
|
VT_I4 VT = 0x3
|
||||||
|
VT_R4 VT = 0x4
|
||||||
|
VT_R8 VT = 0x5
|
||||||
|
VT_CY VT = 0x6
|
||||||
|
VT_DATE VT = 0x7
|
||||||
|
VT_BSTR VT = 0x8
|
||||||
|
VT_DISPATCH VT = 0x9
|
||||||
|
VT_ERROR VT = 0xa
|
||||||
|
VT_BOOL VT = 0xb
|
||||||
|
VT_VARIANT VT = 0xc
|
||||||
|
VT_UNKNOWN VT = 0xd
|
||||||
|
VT_DECIMAL VT = 0xe
|
||||||
|
VT_I1 VT = 0x10
|
||||||
|
VT_UI1 VT = 0x11
|
||||||
|
VT_UI2 VT = 0x12
|
||||||
|
VT_UI4 VT = 0x13
|
||||||
|
VT_I8 VT = 0x14
|
||||||
|
VT_UI8 VT = 0x15
|
||||||
|
VT_INT VT = 0x16
|
||||||
|
VT_UINT VT = 0x17
|
||||||
|
VT_VOID VT = 0x18
|
||||||
|
VT_HRESULT VT = 0x19
|
||||||
|
VT_PTR VT = 0x1a
|
||||||
|
VT_SAFEARRAY VT = 0x1b
|
||||||
|
VT_CARRAY VT = 0x1c
|
||||||
|
VT_USERDEFINED VT = 0x1d
|
||||||
|
VT_LPSTR VT = 0x1e
|
||||||
|
VT_LPWSTR VT = 0x1f
|
||||||
|
VT_RECORD VT = 0x24
|
||||||
|
VT_INT_PTR VT = 0x25
|
||||||
|
VT_UINT_PTR VT = 0x26
|
||||||
|
VT_FILETIME VT = 0x40
|
||||||
|
VT_BLOB VT = 0x41
|
||||||
|
VT_STREAM VT = 0x42
|
||||||
|
VT_STORAGE VT = 0x43
|
||||||
|
VT_STREAMED_OBJECT VT = 0x44
|
||||||
|
VT_STORED_OBJECT VT = 0x45
|
||||||
|
VT_BLOB_OBJECT VT = 0x46
|
||||||
|
VT_CF VT = 0x47
|
||||||
|
VT_CLSID VT = 0x48
|
||||||
|
VT_BSTR_BLOB VT = 0xfff
|
||||||
|
VT_VECTOR VT = 0x1000
|
||||||
|
VT_ARRAY VT = 0x2000
|
||||||
|
VT_BYREF VT = 0x4000
|
||||||
|
VT_RESERVED VT = 0x8000
|
||||||
|
VT_ILLEGAL VT = 0xffff
|
||||||
|
VT_ILLEGALMASKED VT = 0xfff
|
||||||
|
VT_TYPEMASK VT = 0xfff
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DISPID_UNKNOWN = -1
|
||||||
|
DISPID_VALUE = 0
|
||||||
|
DISPID_PROPERTYPUT = -3
|
||||||
|
DISPID_NEWENUM = -4
|
||||||
|
DISPID_EVALUATE = -5
|
||||||
|
DISPID_CONSTRUCTOR = -6
|
||||||
|
DISPID_DESTRUCTOR = -7
|
||||||
|
DISPID_COLLECT = -8
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
TKIND_ENUM = 1
|
||||||
|
TKIND_RECORD = 2
|
||||||
|
TKIND_MODULE = 3
|
||||||
|
TKIND_INTERFACE = 4
|
||||||
|
TKIND_DISPATCH = 5
|
||||||
|
TKIND_COCLASS = 6
|
||||||
|
TKIND_ALIAS = 7
|
||||||
|
TKIND_UNION = 8
|
||||||
|
TKIND_MAX = 9
|
||||||
|
)
|
||||||
|
|
||||||
|
// Safe Array Feature Flags
|
||||||
|
|
||||||
|
const (
|
||||||
|
FADF_AUTO = 0x0001
|
||||||
|
FADF_STATIC = 0x0002
|
||||||
|
FADF_EMBEDDED = 0x0004
|
||||||
|
FADF_FIXEDSIZE = 0x0010
|
||||||
|
FADF_RECORD = 0x0020
|
||||||
|
FADF_HAVEIID = 0x0040
|
||||||
|
FADF_HAVEVARTYPE = 0x0080
|
||||||
|
FADF_BSTR = 0x0100
|
||||||
|
FADF_UNKNOWN = 0x0200
|
||||||
|
FADF_DISPATCH = 0x0400
|
||||||
|
FADF_VARIANT = 0x0800
|
||||||
|
FADF_RESERVED = 0xF008
|
||||||
|
)
|
51
vendor/github.com/go-ole/go-ole/error.go
generated
vendored
Normal file
51
vendor/github.com/go-ole/go-ole/error.go
generated
vendored
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
// OleError stores COM errors.
|
||||||
|
type OleError struct {
|
||||||
|
hr uintptr
|
||||||
|
description string
|
||||||
|
subError error
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewError creates new error with HResult.
|
||||||
|
func NewError(hr uintptr) *OleError {
|
||||||
|
return &OleError{hr: hr}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewErrorWithDescription creates new COM error with HResult and description.
|
||||||
|
func NewErrorWithDescription(hr uintptr, description string) *OleError {
|
||||||
|
return &OleError{hr: hr, description: description}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewErrorWithSubError creates new COM error with parent error.
|
||||||
|
func NewErrorWithSubError(hr uintptr, description string, err error) *OleError {
|
||||||
|
return &OleError{hr: hr, description: description, subError: err}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Code is the HResult.
|
||||||
|
func (v *OleError) Code() uintptr {
|
||||||
|
return uintptr(v.hr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// String description, either manually set or format message with error code.
|
||||||
|
func (v *OleError) String() string {
|
||||||
|
if v.description != "" {
|
||||||
|
return errstr(int(v.hr)) + " (" + v.description + ")"
|
||||||
|
}
|
||||||
|
return errstr(int(v.hr))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error implements error interface.
|
||||||
|
func (v *OleError) Error() string {
|
||||||
|
return v.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Description retrieves error summary, if there is one.
|
||||||
|
func (v *OleError) Description() string {
|
||||||
|
return v.description
|
||||||
|
}
|
||||||
|
|
||||||
|
// SubError returns parent error, if there is one.
|
||||||
|
func (v *OleError) SubError() error {
|
||||||
|
return v.subError
|
||||||
|
}
|
8
vendor/github.com/go-ole/go-ole/error_func.go
generated
vendored
Normal file
8
vendor/github.com/go-ole/go-ole/error_func.go
generated
vendored
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
// errstr converts error code to string.
|
||||||
|
func errstr(errno int) string {
|
||||||
|
return ""
|
||||||
|
}
|
24
vendor/github.com/go-ole/go-ole/error_windows.go
generated
vendored
Normal file
24
vendor/github.com/go-ole/go-ole/error_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"syscall"
|
||||||
|
"unicode/utf16"
|
||||||
|
)
|
||||||
|
|
||||||
|
// errstr converts error code to string.
|
||||||
|
func errstr(errno int) string {
|
||||||
|
// ask windows for the remaining errors
|
||||||
|
var flags uint32 = syscall.FORMAT_MESSAGE_FROM_SYSTEM | syscall.FORMAT_MESSAGE_ARGUMENT_ARRAY | syscall.FORMAT_MESSAGE_IGNORE_INSERTS
|
||||||
|
b := make([]uint16, 300)
|
||||||
|
n, err := syscall.FormatMessage(flags, 0, uint32(errno), 0, b, nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Sprintf("error %d (FormatMessage failed with: %v)", errno, err)
|
||||||
|
}
|
||||||
|
// trim terminating \r and \n
|
||||||
|
for ; n > 0 && (b[n-1] == '\n' || b[n-1] == '\r'); n-- {
|
||||||
|
}
|
||||||
|
return string(utf16.Decode(b[:n]))
|
||||||
|
}
|
5
vendor/github.com/go-ole/go-ole/go.mod
generated
vendored
Normal file
5
vendor/github.com/go-ole/go-ole/go.mod
generated
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
module github.com/go-ole/go-ole
|
||||||
|
|
||||||
|
go 1.12
|
||||||
|
|
||||||
|
require github.com/go-ole/go-ole v1.2.2
|
2
vendor/github.com/go-ole/go-ole/go.sum
generated
vendored
Normal file
2
vendor/github.com/go-ole/go-ole/go.sum
generated
vendored
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
github.com/go-ole/go-ole v1.2.2 h1:HXmymm3IQ8iAfpqlbbUGLHd+SZrnmI4y1pv+WL/3R7c=
|
||||||
|
github.com/go-ole/go-ole v1.2.2/go.mod h1:7FAglXiTm7HKlQRDeOQ6ZNUHidzCWXuZWq/1dTyBNF8=
|
284
vendor/github.com/go-ole/go-ole/guid.go
generated
vendored
Normal file
284
vendor/github.com/go-ole/go-ole/guid.go
generated
vendored
Normal file
|
@ -0,0 +1,284 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
var (
|
||||||
|
// IID_NULL is null Interface ID, used when no other Interface ID is known.
|
||||||
|
IID_NULL = NewGUID("{00000000-0000-0000-0000-000000000000}")
|
||||||
|
|
||||||
|
// IID_IUnknown is for IUnknown interfaces.
|
||||||
|
IID_IUnknown = NewGUID("{00000000-0000-0000-C000-000000000046}")
|
||||||
|
|
||||||
|
// IID_IDispatch is for IDispatch interfaces.
|
||||||
|
IID_IDispatch = NewGUID("{00020400-0000-0000-C000-000000000046}")
|
||||||
|
|
||||||
|
// IID_IEnumVariant is for IEnumVariant interfaces
|
||||||
|
IID_IEnumVariant = NewGUID("{00020404-0000-0000-C000-000000000046}")
|
||||||
|
|
||||||
|
// IID_IConnectionPointContainer is for IConnectionPointContainer interfaces.
|
||||||
|
IID_IConnectionPointContainer = NewGUID("{B196B284-BAB4-101A-B69C-00AA00341D07}")
|
||||||
|
|
||||||
|
// IID_IConnectionPoint is for IConnectionPoint interfaces.
|
||||||
|
IID_IConnectionPoint = NewGUID("{B196B286-BAB4-101A-B69C-00AA00341D07}")
|
||||||
|
|
||||||
|
// IID_IInspectable is for IInspectable interfaces.
|
||||||
|
IID_IInspectable = NewGUID("{AF86E2E0-B12D-4C6A-9C5A-D7AA65101E90}")
|
||||||
|
|
||||||
|
// IID_IProvideClassInfo is for IProvideClassInfo interfaces.
|
||||||
|
IID_IProvideClassInfo = NewGUID("{B196B283-BAB4-101A-B69C-00AA00341D07}")
|
||||||
|
)
|
||||||
|
|
||||||
|
// These are for testing and not part of any library.
|
||||||
|
var (
|
||||||
|
// IID_ICOMTestString is for ICOMTestString interfaces.
|
||||||
|
//
|
||||||
|
// {E0133EB4-C36F-469A-9D3D-C66B84BE19ED}
|
||||||
|
IID_ICOMTestString = NewGUID("{E0133EB4-C36F-469A-9D3D-C66B84BE19ED}")
|
||||||
|
|
||||||
|
// IID_ICOMTestInt8 is for ICOMTestInt8 interfaces.
|
||||||
|
//
|
||||||
|
// {BEB06610-EB84-4155-AF58-E2BFF53680B4}
|
||||||
|
IID_ICOMTestInt8 = NewGUID("{BEB06610-EB84-4155-AF58-E2BFF53680B4}")
|
||||||
|
|
||||||
|
// IID_ICOMTestInt16 is for ICOMTestInt16 interfaces.
|
||||||
|
//
|
||||||
|
// {DAA3F9FA-761E-4976-A860-8364CE55F6FC}
|
||||||
|
IID_ICOMTestInt16 = NewGUID("{DAA3F9FA-761E-4976-A860-8364CE55F6FC}")
|
||||||
|
|
||||||
|
// IID_ICOMTestInt32 is for ICOMTestInt32 interfaces.
|
||||||
|
//
|
||||||
|
// {E3DEDEE7-38A2-4540-91D1-2EEF1D8891B0}
|
||||||
|
IID_ICOMTestInt32 = NewGUID("{E3DEDEE7-38A2-4540-91D1-2EEF1D8891B0}")
|
||||||
|
|
||||||
|
// IID_ICOMTestInt64 is for ICOMTestInt64 interfaces.
|
||||||
|
//
|
||||||
|
// {8D437CBC-B3ED-485C-BC32-C336432A1623}
|
||||||
|
IID_ICOMTestInt64 = NewGUID("{8D437CBC-B3ED-485C-BC32-C336432A1623}")
|
||||||
|
|
||||||
|
// IID_ICOMTestFloat is for ICOMTestFloat interfaces.
|
||||||
|
//
|
||||||
|
// {BF1ED004-EA02-456A-AA55-2AC8AC6B054C}
|
||||||
|
IID_ICOMTestFloat = NewGUID("{BF1ED004-EA02-456A-AA55-2AC8AC6B054C}")
|
||||||
|
|
||||||
|
// IID_ICOMTestDouble is for ICOMTestDouble interfaces.
|
||||||
|
//
|
||||||
|
// {BF908A81-8687-4E93-999F-D86FAB284BA0}
|
||||||
|
IID_ICOMTestDouble = NewGUID("{BF908A81-8687-4E93-999F-D86FAB284BA0}")
|
||||||
|
|
||||||
|
// IID_ICOMTestBoolean is for ICOMTestBoolean interfaces.
|
||||||
|
//
|
||||||
|
// {D530E7A6-4EE8-40D1-8931-3D63B8605010}
|
||||||
|
IID_ICOMTestBoolean = NewGUID("{D530E7A6-4EE8-40D1-8931-3D63B8605010}")
|
||||||
|
|
||||||
|
// IID_ICOMEchoTestObject is for ICOMEchoTestObject interfaces.
|
||||||
|
//
|
||||||
|
// {6485B1EF-D780-4834-A4FE-1EBB51746CA3}
|
||||||
|
IID_ICOMEchoTestObject = NewGUID("{6485B1EF-D780-4834-A4FE-1EBB51746CA3}")
|
||||||
|
|
||||||
|
// IID_ICOMTestTypes is for ICOMTestTypes interfaces.
|
||||||
|
//
|
||||||
|
// {CCA8D7AE-91C0-4277-A8B3-FF4EDF28D3C0}
|
||||||
|
IID_ICOMTestTypes = NewGUID("{CCA8D7AE-91C0-4277-A8B3-FF4EDF28D3C0}")
|
||||||
|
|
||||||
|
// CLSID_COMEchoTestObject is for COMEchoTestObject class.
|
||||||
|
//
|
||||||
|
// {3C24506A-AE9E-4D50-9157-EF317281F1B0}
|
||||||
|
CLSID_COMEchoTestObject = NewGUID("{3C24506A-AE9E-4D50-9157-EF317281F1B0}")
|
||||||
|
|
||||||
|
// CLSID_COMTestScalarClass is for COMTestScalarClass class.
|
||||||
|
//
|
||||||
|
// {865B85C5-0334-4AC6-9EF6-AACEC8FC5E86}
|
||||||
|
CLSID_COMTestScalarClass = NewGUID("{865B85C5-0334-4AC6-9EF6-AACEC8FC5E86}")
|
||||||
|
)
|
||||||
|
|
||||||
|
const hextable = "0123456789ABCDEF"
|
||||||
|
const emptyGUID = "{00000000-0000-0000-0000-000000000000}"
|
||||||
|
|
||||||
|
// GUID is Windows API specific GUID type.
|
||||||
|
//
|
||||||
|
// This exists to match Windows GUID type for direct passing for COM.
|
||||||
|
// Format is in xxxxxxxx-xxxx-xxxx-xxxxxxxxxxxxxxxx.
|
||||||
|
type GUID struct {
|
||||||
|
Data1 uint32
|
||||||
|
Data2 uint16
|
||||||
|
Data3 uint16
|
||||||
|
Data4 [8]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGUID converts the given string into a globally unique identifier that is
|
||||||
|
// compliant with the Windows API.
|
||||||
|
//
|
||||||
|
// The supplied string may be in any of these formats:
|
||||||
|
//
|
||||||
|
// XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
|
||||||
|
// XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX
|
||||||
|
// {XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX}
|
||||||
|
//
|
||||||
|
// The conversion of the supplied string is not case-sensitive.
|
||||||
|
func NewGUID(guid string) *GUID {
|
||||||
|
d := []byte(guid)
|
||||||
|
var d1, d2, d3, d4a, d4b []byte
|
||||||
|
|
||||||
|
switch len(d) {
|
||||||
|
case 38:
|
||||||
|
if d[0] != '{' || d[37] != '}' {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
d = d[1:37]
|
||||||
|
fallthrough
|
||||||
|
case 36:
|
||||||
|
if d[8] != '-' || d[13] != '-' || d[18] != '-' || d[23] != '-' {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
d1 = d[0:8]
|
||||||
|
d2 = d[9:13]
|
||||||
|
d3 = d[14:18]
|
||||||
|
d4a = d[19:23]
|
||||||
|
d4b = d[24:36]
|
||||||
|
case 32:
|
||||||
|
d1 = d[0:8]
|
||||||
|
d2 = d[8:12]
|
||||||
|
d3 = d[12:16]
|
||||||
|
d4a = d[16:20]
|
||||||
|
d4b = d[20:32]
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var g GUID
|
||||||
|
var ok1, ok2, ok3, ok4 bool
|
||||||
|
g.Data1, ok1 = decodeHexUint32(d1)
|
||||||
|
g.Data2, ok2 = decodeHexUint16(d2)
|
||||||
|
g.Data3, ok3 = decodeHexUint16(d3)
|
||||||
|
g.Data4, ok4 = decodeHexByte64(d4a, d4b)
|
||||||
|
if ok1 && ok2 && ok3 && ok4 {
|
||||||
|
return &g
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func decodeHexUint32(src []byte) (value uint32, ok bool) {
|
||||||
|
var b1, b2, b3, b4 byte
|
||||||
|
var ok1, ok2, ok3, ok4 bool
|
||||||
|
b1, ok1 = decodeHexByte(src[0], src[1])
|
||||||
|
b2, ok2 = decodeHexByte(src[2], src[3])
|
||||||
|
b3, ok3 = decodeHexByte(src[4], src[5])
|
||||||
|
b4, ok4 = decodeHexByte(src[6], src[7])
|
||||||
|
value = (uint32(b1) << 24) | (uint32(b2) << 16) | (uint32(b3) << 8) | uint32(b4)
|
||||||
|
ok = ok1 && ok2 && ok3 && ok4
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func decodeHexUint16(src []byte) (value uint16, ok bool) {
|
||||||
|
var b1, b2 byte
|
||||||
|
var ok1, ok2 bool
|
||||||
|
b1, ok1 = decodeHexByte(src[0], src[1])
|
||||||
|
b2, ok2 = decodeHexByte(src[2], src[3])
|
||||||
|
value = (uint16(b1) << 8) | uint16(b2)
|
||||||
|
ok = ok1 && ok2
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func decodeHexByte64(s1 []byte, s2 []byte) (value [8]byte, ok bool) {
|
||||||
|
var ok1, ok2, ok3, ok4, ok5, ok6, ok7, ok8 bool
|
||||||
|
value[0], ok1 = decodeHexByte(s1[0], s1[1])
|
||||||
|
value[1], ok2 = decodeHexByte(s1[2], s1[3])
|
||||||
|
value[2], ok3 = decodeHexByte(s2[0], s2[1])
|
||||||
|
value[3], ok4 = decodeHexByte(s2[2], s2[3])
|
||||||
|
value[4], ok5 = decodeHexByte(s2[4], s2[5])
|
||||||
|
value[5], ok6 = decodeHexByte(s2[6], s2[7])
|
||||||
|
value[6], ok7 = decodeHexByte(s2[8], s2[9])
|
||||||
|
value[7], ok8 = decodeHexByte(s2[10], s2[11])
|
||||||
|
ok = ok1 && ok2 && ok3 && ok4 && ok5 && ok6 && ok7 && ok8
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func decodeHexByte(c1, c2 byte) (value byte, ok bool) {
|
||||||
|
var n1, n2 byte
|
||||||
|
var ok1, ok2 bool
|
||||||
|
n1, ok1 = decodeHexChar(c1)
|
||||||
|
n2, ok2 = decodeHexChar(c2)
|
||||||
|
value = (n1 << 4) | n2
|
||||||
|
ok = ok1 && ok2
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func decodeHexChar(c byte) (byte, bool) {
|
||||||
|
switch {
|
||||||
|
case '0' <= c && c <= '9':
|
||||||
|
return c - '0', true
|
||||||
|
case 'a' <= c && c <= 'f':
|
||||||
|
return c - 'a' + 10, true
|
||||||
|
case 'A' <= c && c <= 'F':
|
||||||
|
return c - 'A' + 10, true
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// String converts the GUID to string form. It will adhere to this pattern:
|
||||||
|
//
|
||||||
|
// {XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX}
|
||||||
|
//
|
||||||
|
// If the GUID is nil, the string representation of an empty GUID is returned:
|
||||||
|
//
|
||||||
|
// {00000000-0000-0000-0000-000000000000}
|
||||||
|
func (guid *GUID) String() string {
|
||||||
|
if guid == nil {
|
||||||
|
return emptyGUID
|
||||||
|
}
|
||||||
|
|
||||||
|
var c [38]byte
|
||||||
|
c[0] = '{'
|
||||||
|
putUint32Hex(c[1:9], guid.Data1)
|
||||||
|
c[9] = '-'
|
||||||
|
putUint16Hex(c[10:14], guid.Data2)
|
||||||
|
c[14] = '-'
|
||||||
|
putUint16Hex(c[15:19], guid.Data3)
|
||||||
|
c[19] = '-'
|
||||||
|
putByteHex(c[20:24], guid.Data4[0:2])
|
||||||
|
c[24] = '-'
|
||||||
|
putByteHex(c[25:37], guid.Data4[2:8])
|
||||||
|
c[37] = '}'
|
||||||
|
return string(c[:])
|
||||||
|
}
|
||||||
|
|
||||||
|
func putUint32Hex(b []byte, v uint32) {
|
||||||
|
b[0] = hextable[byte(v>>24)>>4]
|
||||||
|
b[1] = hextable[byte(v>>24)&0x0f]
|
||||||
|
b[2] = hextable[byte(v>>16)>>4]
|
||||||
|
b[3] = hextable[byte(v>>16)&0x0f]
|
||||||
|
b[4] = hextable[byte(v>>8)>>4]
|
||||||
|
b[5] = hextable[byte(v>>8)&0x0f]
|
||||||
|
b[6] = hextable[byte(v)>>4]
|
||||||
|
b[7] = hextable[byte(v)&0x0f]
|
||||||
|
}
|
||||||
|
|
||||||
|
func putUint16Hex(b []byte, v uint16) {
|
||||||
|
b[0] = hextable[byte(v>>8)>>4]
|
||||||
|
b[1] = hextable[byte(v>>8)&0x0f]
|
||||||
|
b[2] = hextable[byte(v)>>4]
|
||||||
|
b[3] = hextable[byte(v)&0x0f]
|
||||||
|
}
|
||||||
|
|
||||||
|
func putByteHex(dst, src []byte) {
|
||||||
|
for i := 0; i < len(src); i++ {
|
||||||
|
dst[i*2] = hextable[src[i]>>4]
|
||||||
|
dst[i*2+1] = hextable[src[i]&0x0f]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsEqualGUID compares two GUID.
|
||||||
|
//
|
||||||
|
// Not constant time comparison.
|
||||||
|
func IsEqualGUID(guid1 *GUID, guid2 *GUID) bool {
|
||||||
|
return guid1.Data1 == guid2.Data1 &&
|
||||||
|
guid1.Data2 == guid2.Data2 &&
|
||||||
|
guid1.Data3 == guid2.Data3 &&
|
||||||
|
guid1.Data4[0] == guid2.Data4[0] &&
|
||||||
|
guid1.Data4[1] == guid2.Data4[1] &&
|
||||||
|
guid1.Data4[2] == guid2.Data4[2] &&
|
||||||
|
guid1.Data4[3] == guid2.Data4[3] &&
|
||||||
|
guid1.Data4[4] == guid2.Data4[4] &&
|
||||||
|
guid1.Data4[5] == guid2.Data4[5] &&
|
||||||
|
guid1.Data4[6] == guid2.Data4[6] &&
|
||||||
|
guid1.Data4[7] == guid2.Data4[7]
|
||||||
|
}
|
20
vendor/github.com/go-ole/go-ole/iconnectionpoint.go
generated
vendored
Normal file
20
vendor/github.com/go-ole/go-ole/iconnectionpoint.go
generated
vendored
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IConnectionPoint struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IConnectionPointVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
GetConnectionInterface uintptr
|
||||||
|
GetConnectionPointContainer uintptr
|
||||||
|
Advise uintptr
|
||||||
|
Unadvise uintptr
|
||||||
|
EnumConnections uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) VTable() *IConnectionPointVtbl {
|
||||||
|
return (*IConnectionPointVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
21
vendor/github.com/go-ole/go-ole/iconnectionpoint_func.go
generated
vendored
Normal file
21
vendor/github.com/go-ole/go-ole/iconnectionpoint_func.go
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) GetConnectionInterface(piid **GUID) int32 {
|
||||||
|
return int32(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) Advise(unknown *IUnknown) (uint32, error) {
|
||||||
|
return uint32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) Unadvise(cookie uint32) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) EnumConnections(p *unsafe.Pointer) (err error) {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
43
vendor/github.com/go-ole/go-ole/iconnectionpoint_windows.go
generated
vendored
Normal file
43
vendor/github.com/go-ole/go-ole/iconnectionpoint_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) GetConnectionInterface(piid **GUID) int32 {
|
||||||
|
// XXX: This doesn't look like it does what it's supposed to
|
||||||
|
return release((*IUnknown)(unsafe.Pointer(v)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) Advise(unknown *IUnknown) (cookie uint32, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().Advise,
|
||||||
|
3,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(unknown)),
|
||||||
|
uintptr(unsafe.Pointer(&cookie)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) Unadvise(cookie uint32) (err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().Unadvise,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(cookie),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPoint) EnumConnections(p *unsafe.Pointer) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
17
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer.go
generated
vendored
Normal file
17
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer.go
generated
vendored
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IConnectionPointContainer struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IConnectionPointContainerVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
EnumConnectionPoints uintptr
|
||||||
|
FindConnectionPoint uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPointContainer) VTable() *IConnectionPointContainerVtbl {
|
||||||
|
return (*IConnectionPointContainerVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
11
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer_func.go
generated
vendored
Normal file
11
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer_func.go
generated
vendored
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func (v *IConnectionPointContainer) EnumConnectionPoints(points interface{}) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPointContainer) FindConnectionPoint(iid *GUID, point **IConnectionPoint) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
25
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer_windows.go
generated
vendored
Normal file
25
vendor/github.com/go-ole/go-ole/iconnectionpointcontainer_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (v *IConnectionPointContainer) EnumConnectionPoints(points interface{}) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IConnectionPointContainer) FindConnectionPoint(iid *GUID, point **IConnectionPoint) (err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().FindConnectionPoint,
|
||||||
|
3,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(point)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
94
vendor/github.com/go-ole/go-ole/idispatch.go
generated
vendored
Normal file
94
vendor/github.com/go-ole/go-ole/idispatch.go
generated
vendored
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IDispatch struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IDispatchVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
GetTypeInfoCount uintptr
|
||||||
|
GetTypeInfo uintptr
|
||||||
|
GetIDsOfNames uintptr
|
||||||
|
Invoke uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IDispatch) VTable() *IDispatchVtbl {
|
||||||
|
return (*IDispatchVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IDispatch) GetIDsOfName(names []string) (dispid []int32, err error) {
|
||||||
|
dispid, err = getIDsOfName(v, names)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IDispatch) Invoke(dispid int32, dispatch int16, params ...interface{}) (result *VARIANT, err error) {
|
||||||
|
result, err = invoke(v, dispid, dispatch, params...)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IDispatch) GetTypeInfoCount() (c uint32, err error) {
|
||||||
|
c, err = getTypeInfoCount(v)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IDispatch) GetTypeInfo() (tinfo *ITypeInfo, err error) {
|
||||||
|
tinfo, err = getTypeInfo(v)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetSingleIDOfName is a helper that returns single display ID for IDispatch name.
|
||||||
|
//
|
||||||
|
// This replaces the common pattern of attempting to get a single name from the list of available
|
||||||
|
// IDs. It gives the first ID, if it is available.
|
||||||
|
func (v *IDispatch) GetSingleIDOfName(name string) (displayID int32, err error) {
|
||||||
|
var displayIDs []int32
|
||||||
|
displayIDs, err = v.GetIDsOfName([]string{name})
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
displayID = displayIDs[0]
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// InvokeWithOptionalArgs accepts arguments as an array, works like Invoke.
|
||||||
|
//
|
||||||
|
// Accepts name and will attempt to retrieve Display ID to pass to Invoke.
|
||||||
|
//
|
||||||
|
// Passing params as an array is a workaround that could be fixed in later versions of Go that
|
||||||
|
// prevent passing empty params. During testing it was discovered that this is an acceptable way of
|
||||||
|
// getting around not being able to pass params normally.
|
||||||
|
func (v *IDispatch) InvokeWithOptionalArgs(name string, dispatch int16, params []interface{}) (result *VARIANT, err error) {
|
||||||
|
displayID, err := v.GetSingleIDOfName(name)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(params) < 1 {
|
||||||
|
result, err = v.Invoke(displayID, dispatch)
|
||||||
|
} else {
|
||||||
|
result, err = v.Invoke(displayID, dispatch, params...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CallMethod invokes named function with arguments on object.
|
||||||
|
func (v *IDispatch) CallMethod(name string, params ...interface{}) (*VARIANT, error) {
|
||||||
|
return v.InvokeWithOptionalArgs(name, DISPATCH_METHOD, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetProperty retrieves the property with the name with the ability to pass arguments.
|
||||||
|
//
|
||||||
|
// Most of the time you will not need to pass arguments as most objects do not allow for this
|
||||||
|
// feature. Or at least, should not allow for this feature. Some servers don't follow best practices
|
||||||
|
// and this is provided for those edge cases.
|
||||||
|
func (v *IDispatch) GetProperty(name string, params ...interface{}) (*VARIANT, error) {
|
||||||
|
return v.InvokeWithOptionalArgs(name, DISPATCH_PROPERTYGET, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PutProperty attempts to mutate a property in the object.
|
||||||
|
func (v *IDispatch) PutProperty(name string, params ...interface{}) (*VARIANT, error) {
|
||||||
|
return v.InvokeWithOptionalArgs(name, DISPATCH_PROPERTYPUT, params)
|
||||||
|
}
|
19
vendor/github.com/go-ole/go-ole/idispatch_func.go
generated
vendored
Normal file
19
vendor/github.com/go-ole/go-ole/idispatch_func.go
generated
vendored
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func getIDsOfName(disp *IDispatch, names []string) ([]int32, error) {
|
||||||
|
return []int32{}, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTypeInfoCount(disp *IDispatch) (uint32, error) {
|
||||||
|
return uint32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTypeInfo(disp *IDispatch) (*ITypeInfo, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func invoke(disp *IDispatch, dispid int32, dispatch int16, params ...interface{}) (*VARIANT, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
200
vendor/github.com/go-ole/go-ole/idispatch_windows.go
generated
vendored
Normal file
200
vendor/github.com/go-ole/go-ole/idispatch_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,200 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/big"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func getIDsOfName(disp *IDispatch, names []string) (dispid []int32, err error) {
|
||||||
|
wnames := make([]*uint16, len(names))
|
||||||
|
for i := 0; i < len(names); i++ {
|
||||||
|
wnames[i] = syscall.StringToUTF16Ptr(names[i])
|
||||||
|
}
|
||||||
|
dispid = make([]int32, len(names))
|
||||||
|
namelen := uint32(len(names))
|
||||||
|
hr, _, _ := syscall.Syscall6(
|
||||||
|
disp.VTable().GetIDsOfNames,
|
||||||
|
6,
|
||||||
|
uintptr(unsafe.Pointer(disp)),
|
||||||
|
uintptr(unsafe.Pointer(IID_NULL)),
|
||||||
|
uintptr(unsafe.Pointer(&wnames[0])),
|
||||||
|
uintptr(namelen),
|
||||||
|
uintptr(GetUserDefaultLCID()),
|
||||||
|
uintptr(unsafe.Pointer(&dispid[0])))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTypeInfoCount(disp *IDispatch) (c uint32, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
disp.VTable().GetTypeInfoCount,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(disp)),
|
||||||
|
uintptr(unsafe.Pointer(&c)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTypeInfo(disp *IDispatch) (tinfo *ITypeInfo, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
disp.VTable().GetTypeInfo,
|
||||||
|
3,
|
||||||
|
uintptr(unsafe.Pointer(disp)),
|
||||||
|
uintptr(GetUserDefaultLCID()),
|
||||||
|
uintptr(unsafe.Pointer(&tinfo)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func invoke(disp *IDispatch, dispid int32, dispatch int16, params ...interface{}) (result *VARIANT, err error) {
|
||||||
|
var dispparams DISPPARAMS
|
||||||
|
|
||||||
|
if dispatch&DISPATCH_PROPERTYPUT != 0 {
|
||||||
|
dispnames := [1]int32{DISPID_PROPERTYPUT}
|
||||||
|
dispparams.rgdispidNamedArgs = uintptr(unsafe.Pointer(&dispnames[0]))
|
||||||
|
dispparams.cNamedArgs = 1
|
||||||
|
} else if dispatch&DISPATCH_PROPERTYPUTREF != 0 {
|
||||||
|
dispnames := [1]int32{DISPID_PROPERTYPUT}
|
||||||
|
dispparams.rgdispidNamedArgs = uintptr(unsafe.Pointer(&dispnames[0]))
|
||||||
|
dispparams.cNamedArgs = 1
|
||||||
|
}
|
||||||
|
var vargs []VARIANT
|
||||||
|
if len(params) > 0 {
|
||||||
|
vargs = make([]VARIANT, len(params))
|
||||||
|
for i, v := range params {
|
||||||
|
//n := len(params)-i-1
|
||||||
|
n := len(params) - i - 1
|
||||||
|
VariantInit(&vargs[n])
|
||||||
|
switch vv := v.(type) {
|
||||||
|
case bool:
|
||||||
|
if vv {
|
||||||
|
vargs[n] = NewVariant(VT_BOOL, 0xffff)
|
||||||
|
} else {
|
||||||
|
vargs[n] = NewVariant(VT_BOOL, 0)
|
||||||
|
}
|
||||||
|
case *bool:
|
||||||
|
vargs[n] = NewVariant(VT_BOOL|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*bool)))))
|
||||||
|
case uint8:
|
||||||
|
vargs[n] = NewVariant(VT_I1, int64(v.(uint8)))
|
||||||
|
case *uint8:
|
||||||
|
vargs[n] = NewVariant(VT_I1|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint8)))))
|
||||||
|
case int8:
|
||||||
|
vargs[n] = NewVariant(VT_I1, int64(v.(int8)))
|
||||||
|
case *int8:
|
||||||
|
vargs[n] = NewVariant(VT_I1|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint8)))))
|
||||||
|
case int16:
|
||||||
|
vargs[n] = NewVariant(VT_I2, int64(v.(int16)))
|
||||||
|
case *int16:
|
||||||
|
vargs[n] = NewVariant(VT_I2|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*int16)))))
|
||||||
|
case uint16:
|
||||||
|
vargs[n] = NewVariant(VT_UI2, int64(v.(uint16)))
|
||||||
|
case *uint16:
|
||||||
|
vargs[n] = NewVariant(VT_UI2|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint16)))))
|
||||||
|
case int32:
|
||||||
|
vargs[n] = NewVariant(VT_I4, int64(v.(int32)))
|
||||||
|
case *int32:
|
||||||
|
vargs[n] = NewVariant(VT_I4|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*int32)))))
|
||||||
|
case uint32:
|
||||||
|
vargs[n] = NewVariant(VT_UI4, int64(v.(uint32)))
|
||||||
|
case *uint32:
|
||||||
|
vargs[n] = NewVariant(VT_UI4|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint32)))))
|
||||||
|
case int64:
|
||||||
|
vargs[n] = NewVariant(VT_I8, int64(v.(int64)))
|
||||||
|
case *int64:
|
||||||
|
vargs[n] = NewVariant(VT_I8|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*int64)))))
|
||||||
|
case uint64:
|
||||||
|
vargs[n] = NewVariant(VT_UI8, int64(uintptr(v.(uint64))))
|
||||||
|
case *uint64:
|
||||||
|
vargs[n] = NewVariant(VT_UI8|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint64)))))
|
||||||
|
case int:
|
||||||
|
vargs[n] = NewVariant(VT_I4, int64(v.(int)))
|
||||||
|
case *int:
|
||||||
|
vargs[n] = NewVariant(VT_I4|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*int)))))
|
||||||
|
case uint:
|
||||||
|
vargs[n] = NewVariant(VT_UI4, int64(v.(uint)))
|
||||||
|
case *uint:
|
||||||
|
vargs[n] = NewVariant(VT_UI4|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*uint)))))
|
||||||
|
case float32:
|
||||||
|
vargs[n] = NewVariant(VT_R4, *(*int64)(unsafe.Pointer(&vv)))
|
||||||
|
case *float32:
|
||||||
|
vargs[n] = NewVariant(VT_R4|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*float32)))))
|
||||||
|
case float64:
|
||||||
|
vargs[n] = NewVariant(VT_R8, *(*int64)(unsafe.Pointer(&vv)))
|
||||||
|
case *float64:
|
||||||
|
vargs[n] = NewVariant(VT_R8|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*float64)))))
|
||||||
|
case *big.Int:
|
||||||
|
vargs[n] = NewVariant(VT_DECIMAL, v.(*big.Int).Int64())
|
||||||
|
case string:
|
||||||
|
vargs[n] = NewVariant(VT_BSTR, int64(uintptr(unsafe.Pointer(SysAllocStringLen(v.(string))))))
|
||||||
|
case *string:
|
||||||
|
vargs[n] = NewVariant(VT_BSTR|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*string)))))
|
||||||
|
case time.Time:
|
||||||
|
s := vv.Format("2006-01-02 15:04:05")
|
||||||
|
vargs[n] = NewVariant(VT_BSTR, int64(uintptr(unsafe.Pointer(SysAllocStringLen(s)))))
|
||||||
|
case *time.Time:
|
||||||
|
s := vv.Format("2006-01-02 15:04:05")
|
||||||
|
vargs[n] = NewVariant(VT_BSTR|VT_BYREF, int64(uintptr(unsafe.Pointer(&s))))
|
||||||
|
case *IDispatch:
|
||||||
|
vargs[n] = NewVariant(VT_DISPATCH, int64(uintptr(unsafe.Pointer(v.(*IDispatch)))))
|
||||||
|
case **IDispatch:
|
||||||
|
vargs[n] = NewVariant(VT_DISPATCH|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(**IDispatch)))))
|
||||||
|
case nil:
|
||||||
|
vargs[n] = NewVariant(VT_NULL, 0)
|
||||||
|
case *VARIANT:
|
||||||
|
vargs[n] = NewVariant(VT_VARIANT|VT_BYREF, int64(uintptr(unsafe.Pointer(v.(*VARIANT)))))
|
||||||
|
case []byte:
|
||||||
|
safeByteArray := safeArrayFromByteSlice(v.([]byte))
|
||||||
|
vargs[n] = NewVariant(VT_ARRAY|VT_UI1, int64(uintptr(unsafe.Pointer(safeByteArray))))
|
||||||
|
defer VariantClear(&vargs[n])
|
||||||
|
case []string:
|
||||||
|
safeByteArray := safeArrayFromStringSlice(v.([]string))
|
||||||
|
vargs[n] = NewVariant(VT_ARRAY|VT_BSTR, int64(uintptr(unsafe.Pointer(safeByteArray))))
|
||||||
|
defer VariantClear(&vargs[n])
|
||||||
|
default:
|
||||||
|
panic("unknown type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dispparams.rgvarg = uintptr(unsafe.Pointer(&vargs[0]))
|
||||||
|
dispparams.cArgs = uint32(len(params))
|
||||||
|
}
|
||||||
|
|
||||||
|
result = new(VARIANT)
|
||||||
|
var excepInfo EXCEPINFO
|
||||||
|
VariantInit(result)
|
||||||
|
hr, _, _ := syscall.Syscall9(
|
||||||
|
disp.VTable().Invoke,
|
||||||
|
9,
|
||||||
|
uintptr(unsafe.Pointer(disp)),
|
||||||
|
uintptr(dispid),
|
||||||
|
uintptr(unsafe.Pointer(IID_NULL)),
|
||||||
|
uintptr(GetUserDefaultLCID()),
|
||||||
|
uintptr(dispatch),
|
||||||
|
uintptr(unsafe.Pointer(&dispparams)),
|
||||||
|
uintptr(unsafe.Pointer(result)),
|
||||||
|
uintptr(unsafe.Pointer(&excepInfo)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewErrorWithSubError(hr, BstrToString(excepInfo.bstrDescription), excepInfo)
|
||||||
|
}
|
||||||
|
for i, varg := range vargs {
|
||||||
|
n := len(params) - i - 1
|
||||||
|
if varg.VT == VT_BSTR && varg.Val != 0 {
|
||||||
|
SysFreeString(((*int16)(unsafe.Pointer(uintptr(varg.Val)))))
|
||||||
|
}
|
||||||
|
if varg.VT == (VT_BSTR|VT_BYREF) && varg.Val != 0 {
|
||||||
|
*(params[n].(*string)) = LpOleStrToString(*(**uint16)(unsafe.Pointer(uintptr(varg.Val))))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
19
vendor/github.com/go-ole/go-ole/ienumvariant.go
generated
vendored
Normal file
19
vendor/github.com/go-ole/go-ole/ienumvariant.go
generated
vendored
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IEnumVARIANT struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IEnumVARIANTVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
Next uintptr
|
||||||
|
Skip uintptr
|
||||||
|
Reset uintptr
|
||||||
|
Clone uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IEnumVARIANT) VTable() *IEnumVARIANTVtbl {
|
||||||
|
return (*IEnumVARIANTVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
19
vendor/github.com/go-ole/go-ole/ienumvariant_func.go
generated
vendored
Normal file
19
vendor/github.com/go-ole/go-ole/ienumvariant_func.go
generated
vendored
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Clone() (*IEnumVARIANT, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Reset() error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Skip(celt uint) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Next(celt uint) (VARIANT, uint, error) {
|
||||||
|
return NewVariant(VT_NULL, int64(0)), 0, NewError(E_NOTIMPL)
|
||||||
|
}
|
63
vendor/github.com/go-ole/go-ole/ienumvariant_windows.go
generated
vendored
Normal file
63
vendor/github.com/go-ole/go-ole/ienumvariant_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Clone() (cloned *IEnumVARIANT, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
enum.VTable().Clone,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(enum)),
|
||||||
|
uintptr(unsafe.Pointer(&cloned)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Reset() (err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
enum.VTable().Reset,
|
||||||
|
1,
|
||||||
|
uintptr(unsafe.Pointer(enum)),
|
||||||
|
0,
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Skip(celt uint) (err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
enum.VTable().Skip,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(enum)),
|
||||||
|
uintptr(celt),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (enum *IEnumVARIANT) Next(celt uint) (array VARIANT, length uint, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall6(
|
||||||
|
enum.VTable().Next,
|
||||||
|
4,
|
||||||
|
uintptr(unsafe.Pointer(enum)),
|
||||||
|
uintptr(celt),
|
||||||
|
uintptr(unsafe.Pointer(&array)),
|
||||||
|
uintptr(unsafe.Pointer(&length)),
|
||||||
|
0,
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
18
vendor/github.com/go-ole/go-ole/iinspectable.go
generated
vendored
Normal file
18
vendor/github.com/go-ole/go-ole/iinspectable.go
generated
vendored
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IInspectable struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IInspectableVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
GetIIds uintptr
|
||||||
|
GetRuntimeClassName uintptr
|
||||||
|
GetTrustLevel uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IInspectable) VTable() *IInspectableVtbl {
|
||||||
|
return (*IInspectableVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
15
vendor/github.com/go-ole/go-ole/iinspectable_func.go
generated
vendored
Normal file
15
vendor/github.com/go-ole/go-ole/iinspectable_func.go
generated
vendored
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func (v *IInspectable) GetIids() ([]*GUID, error) {
|
||||||
|
return []*GUID{}, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IInspectable) GetRuntimeClassName() (string, error) {
|
||||||
|
return "", NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IInspectable) GetTrustLevel() (uint32, error) {
|
||||||
|
return uint32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
72
vendor/github.com/go-ole/go-ole/iinspectable_windows.go
generated
vendored
Normal file
72
vendor/github.com/go-ole/go-ole/iinspectable_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"reflect"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (v *IInspectable) GetIids() (iids []*GUID, err error) {
|
||||||
|
var count uint32
|
||||||
|
var array uintptr
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().GetIIds,
|
||||||
|
3,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(&count)),
|
||||||
|
uintptr(unsafe.Pointer(&array)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer CoTaskMemFree(array)
|
||||||
|
|
||||||
|
iids = make([]*GUID, count)
|
||||||
|
byteCount := count * uint32(unsafe.Sizeof(GUID{}))
|
||||||
|
slicehdr := reflect.SliceHeader{Data: array, Len: int(byteCount), Cap: int(byteCount)}
|
||||||
|
byteSlice := *(*[]byte)(unsafe.Pointer(&slicehdr))
|
||||||
|
reader := bytes.NewReader(byteSlice)
|
||||||
|
for i := range iids {
|
||||||
|
guid := GUID{}
|
||||||
|
err = binary.Read(reader, binary.LittleEndian, &guid)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
iids[i] = &guid
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IInspectable) GetRuntimeClassName() (s string, err error) {
|
||||||
|
var hstring HString
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().GetRuntimeClassName,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(&hstring)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
s = hstring.String()
|
||||||
|
DeleteHString(hstring)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IInspectable) GetTrustLevel() (level uint32, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
v.VTable().GetTrustLevel,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(&level)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
21
vendor/github.com/go-ole/go-ole/iprovideclassinfo.go
generated
vendored
Normal file
21
vendor/github.com/go-ole/go-ole/iprovideclassinfo.go
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IProvideClassInfo struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type IProvideClassInfoVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
GetClassInfo uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IProvideClassInfo) VTable() *IProvideClassInfoVtbl {
|
||||||
|
return (*IProvideClassInfoVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IProvideClassInfo) GetClassInfo() (cinfo *ITypeInfo, err error) {
|
||||||
|
cinfo, err = getClassInfo(v)
|
||||||
|
return
|
||||||
|
}
|
7
vendor/github.com/go-ole/go-ole/iprovideclassinfo_func.go
generated
vendored
Normal file
7
vendor/github.com/go-ole/go-ole/iprovideclassinfo_func.go
generated
vendored
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func getClassInfo(disp *IProvideClassInfo) (tinfo *ITypeInfo, err error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
21
vendor/github.com/go-ole/go-ole/iprovideclassinfo_windows.go
generated
vendored
Normal file
21
vendor/github.com/go-ole/go-ole/iprovideclassinfo_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func getClassInfo(disp *IProvideClassInfo) (tinfo *ITypeInfo, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
disp.VTable().GetClassInfo,
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(disp)),
|
||||||
|
uintptr(unsafe.Pointer(&tinfo)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
34
vendor/github.com/go-ole/go-ole/itypeinfo.go
generated
vendored
Normal file
34
vendor/github.com/go-ole/go-ole/itypeinfo.go
generated
vendored
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type ITypeInfo struct {
|
||||||
|
IUnknown
|
||||||
|
}
|
||||||
|
|
||||||
|
type ITypeInfoVtbl struct {
|
||||||
|
IUnknownVtbl
|
||||||
|
GetTypeAttr uintptr
|
||||||
|
GetTypeComp uintptr
|
||||||
|
GetFuncDesc uintptr
|
||||||
|
GetVarDesc uintptr
|
||||||
|
GetNames uintptr
|
||||||
|
GetRefTypeOfImplType uintptr
|
||||||
|
GetImplTypeFlags uintptr
|
||||||
|
GetIDsOfNames uintptr
|
||||||
|
Invoke uintptr
|
||||||
|
GetDocumentation uintptr
|
||||||
|
GetDllEntry uintptr
|
||||||
|
GetRefTypeInfo uintptr
|
||||||
|
AddressOfMember uintptr
|
||||||
|
CreateInstance uintptr
|
||||||
|
GetMops uintptr
|
||||||
|
GetContainingTypeLib uintptr
|
||||||
|
ReleaseTypeAttr uintptr
|
||||||
|
ReleaseFuncDesc uintptr
|
||||||
|
ReleaseVarDesc uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *ITypeInfo) VTable() *ITypeInfoVtbl {
|
||||||
|
return (*ITypeInfoVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
7
vendor/github.com/go-ole/go-ole/itypeinfo_func.go
generated
vendored
Normal file
7
vendor/github.com/go-ole/go-ole/itypeinfo_func.go
generated
vendored
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func (v *ITypeInfo) GetTypeAttr() (*TYPEATTR, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
21
vendor/github.com/go-ole/go-ole/itypeinfo_windows.go
generated
vendored
Normal file
21
vendor/github.com/go-ole/go-ole/itypeinfo_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (v *ITypeInfo) GetTypeAttr() (tattr *TYPEATTR, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
uintptr(v.VTable().GetTypeAttr),
|
||||||
|
2,
|
||||||
|
uintptr(unsafe.Pointer(v)),
|
||||||
|
uintptr(unsafe.Pointer(&tattr)),
|
||||||
|
0)
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
57
vendor/github.com/go-ole/go-ole/iunknown.go
generated
vendored
Normal file
57
vendor/github.com/go-ole/go-ole/iunknown.go
generated
vendored
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
type IUnknown struct {
|
||||||
|
RawVTable *interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type IUnknownVtbl struct {
|
||||||
|
QueryInterface uintptr
|
||||||
|
AddRef uintptr
|
||||||
|
Release uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
type UnknownLike interface {
|
||||||
|
QueryInterface(iid *GUID) (disp *IDispatch, err error)
|
||||||
|
AddRef() int32
|
||||||
|
Release() int32
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) VTable() *IUnknownVtbl {
|
||||||
|
return (*IUnknownVtbl)(unsafe.Pointer(v.RawVTable))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) PutQueryInterface(interfaceID *GUID, obj interface{}) error {
|
||||||
|
return reflectQueryInterface(v, v.VTable().QueryInterface, interfaceID, obj)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) IDispatch(interfaceID *GUID) (dispatch *IDispatch, err error) {
|
||||||
|
err = v.PutQueryInterface(interfaceID, &dispatch)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) IEnumVARIANT(interfaceID *GUID) (enum *IEnumVARIANT, err error) {
|
||||||
|
err = v.PutQueryInterface(interfaceID, &enum)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) QueryInterface(iid *GUID) (*IDispatch, error) {
|
||||||
|
return queryInterface(v, iid)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) MustQueryInterface(iid *GUID) (disp *IDispatch) {
|
||||||
|
unk, err := queryInterface(v, iid)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return unk
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) AddRef() int32 {
|
||||||
|
return addRef(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *IUnknown) Release() int32 {
|
||||||
|
return release(v)
|
||||||
|
}
|
19
vendor/github.com/go-ole/go-ole/iunknown_func.go
generated
vendored
Normal file
19
vendor/github.com/go-ole/go-ole/iunknown_func.go
generated
vendored
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
func reflectQueryInterface(self interface{}, method uintptr, interfaceID *GUID, obj interface{}) (err error) {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func queryInterface(unk *IUnknown, iid *GUID) (disp *IDispatch, err error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func addRef(unk *IUnknown) int32 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func release(unk *IUnknown) int32 {
|
||||||
|
return 0
|
||||||
|
}
|
58
vendor/github.com/go-ole/go-ole/iunknown_windows.go
generated
vendored
Normal file
58
vendor/github.com/go-ole/go-ole/iunknown_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func reflectQueryInterface(self interface{}, method uintptr, interfaceID *GUID, obj interface{}) (err error) {
|
||||||
|
selfValue := reflect.ValueOf(self).Elem()
|
||||||
|
objValue := reflect.ValueOf(obj).Elem()
|
||||||
|
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
method,
|
||||||
|
3,
|
||||||
|
selfValue.UnsafeAddr(),
|
||||||
|
uintptr(unsafe.Pointer(interfaceID)),
|
||||||
|
objValue.Addr().Pointer())
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func queryInterface(unk *IUnknown, iid *GUID) (disp *IDispatch, err error) {
|
||||||
|
hr, _, _ := syscall.Syscall(
|
||||||
|
unk.VTable().QueryInterface,
|
||||||
|
3,
|
||||||
|
uintptr(unsafe.Pointer(unk)),
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(&disp)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func addRef(unk *IUnknown) int32 {
|
||||||
|
ret, _, _ := syscall.Syscall(
|
||||||
|
unk.VTable().AddRef,
|
||||||
|
1,
|
||||||
|
uintptr(unsafe.Pointer(unk)),
|
||||||
|
0,
|
||||||
|
0)
|
||||||
|
return int32(ret)
|
||||||
|
}
|
||||||
|
|
||||||
|
func release(unk *IUnknown) int32 {
|
||||||
|
ret, _, _ := syscall.Syscall(
|
||||||
|
unk.VTable().Release,
|
||||||
|
1,
|
||||||
|
uintptr(unsafe.Pointer(unk)),
|
||||||
|
0,
|
||||||
|
0)
|
||||||
|
return int32(ret)
|
||||||
|
}
|
157
vendor/github.com/go-ole/go-ole/ole.go
generated
vendored
Normal file
157
vendor/github.com/go-ole/go-ole/ole.go
generated
vendored
Normal file
|
@ -0,0 +1,157 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// DISPPARAMS are the arguments that passed to methods or property.
|
||||||
|
type DISPPARAMS struct {
|
||||||
|
rgvarg uintptr
|
||||||
|
rgdispidNamedArgs uintptr
|
||||||
|
cArgs uint32
|
||||||
|
cNamedArgs uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// EXCEPINFO defines exception info.
|
||||||
|
type EXCEPINFO struct {
|
||||||
|
wCode uint16
|
||||||
|
wReserved uint16
|
||||||
|
bstrSource *uint16
|
||||||
|
bstrDescription *uint16
|
||||||
|
bstrHelpFile *uint16
|
||||||
|
dwHelpContext uint32
|
||||||
|
pvReserved uintptr
|
||||||
|
pfnDeferredFillIn uintptr
|
||||||
|
scode uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// WCode return wCode in EXCEPINFO.
|
||||||
|
func (e EXCEPINFO) WCode() uint16 {
|
||||||
|
return e.wCode
|
||||||
|
}
|
||||||
|
|
||||||
|
// SCODE return scode in EXCEPINFO.
|
||||||
|
func (e EXCEPINFO) SCODE() uint32 {
|
||||||
|
return e.scode
|
||||||
|
}
|
||||||
|
|
||||||
|
// String convert EXCEPINFO to string.
|
||||||
|
func (e EXCEPINFO) String() string {
|
||||||
|
var src, desc, hlp string
|
||||||
|
if e.bstrSource == nil {
|
||||||
|
src = "<nil>"
|
||||||
|
} else {
|
||||||
|
src = BstrToString(e.bstrSource)
|
||||||
|
}
|
||||||
|
|
||||||
|
if e.bstrDescription == nil {
|
||||||
|
desc = "<nil>"
|
||||||
|
} else {
|
||||||
|
desc = BstrToString(e.bstrDescription)
|
||||||
|
}
|
||||||
|
|
||||||
|
if e.bstrHelpFile == nil {
|
||||||
|
hlp = "<nil>"
|
||||||
|
} else {
|
||||||
|
hlp = BstrToString(e.bstrHelpFile)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Sprintf(
|
||||||
|
"wCode: %#x, bstrSource: %v, bstrDescription: %v, bstrHelpFile: %v, dwHelpContext: %#x, scode: %#x",
|
||||||
|
e.wCode, src, desc, hlp, e.dwHelpContext, e.scode,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error implements error interface and returns error string.
|
||||||
|
func (e EXCEPINFO) Error() string {
|
||||||
|
if e.bstrDescription != nil {
|
||||||
|
return strings.TrimSpace(BstrToString(e.bstrDescription))
|
||||||
|
}
|
||||||
|
|
||||||
|
src := "Unknown"
|
||||||
|
if e.bstrSource != nil {
|
||||||
|
src = BstrToString(e.bstrSource)
|
||||||
|
}
|
||||||
|
|
||||||
|
code := e.scode
|
||||||
|
if e.wCode != 0 {
|
||||||
|
code = uint32(e.wCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Sprintf("%v: %#x", src, code)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PARAMDATA defines parameter data type.
|
||||||
|
type PARAMDATA struct {
|
||||||
|
Name *int16
|
||||||
|
Vt uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
// METHODDATA defines method info.
|
||||||
|
type METHODDATA struct {
|
||||||
|
Name *uint16
|
||||||
|
Data *PARAMDATA
|
||||||
|
Dispid int32
|
||||||
|
Meth uint32
|
||||||
|
CC int32
|
||||||
|
CArgs uint32
|
||||||
|
Flags uint16
|
||||||
|
VtReturn uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// INTERFACEDATA defines interface info.
|
||||||
|
type INTERFACEDATA struct {
|
||||||
|
MethodData *METHODDATA
|
||||||
|
CMembers uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// Point is 2D vector type.
|
||||||
|
type Point struct {
|
||||||
|
X int32
|
||||||
|
Y int32
|
||||||
|
}
|
||||||
|
|
||||||
|
// Msg is message between processes.
|
||||||
|
type Msg struct {
|
||||||
|
Hwnd uint32
|
||||||
|
Message uint32
|
||||||
|
Wparam int32
|
||||||
|
Lparam int32
|
||||||
|
Time uint32
|
||||||
|
Pt Point
|
||||||
|
}
|
||||||
|
|
||||||
|
// TYPEDESC defines data type.
|
||||||
|
type TYPEDESC struct {
|
||||||
|
Hreftype uint32
|
||||||
|
VT uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
// IDLDESC defines IDL info.
|
||||||
|
type IDLDESC struct {
|
||||||
|
DwReserved uint32
|
||||||
|
WIDLFlags uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
// TYPEATTR defines type info.
|
||||||
|
type TYPEATTR struct {
|
||||||
|
Guid GUID
|
||||||
|
Lcid uint32
|
||||||
|
dwReserved uint32
|
||||||
|
MemidConstructor int32
|
||||||
|
MemidDestructor int32
|
||||||
|
LpstrSchema *uint16
|
||||||
|
CbSizeInstance uint32
|
||||||
|
Typekind int32
|
||||||
|
CFuncs uint16
|
||||||
|
CVars uint16
|
||||||
|
CImplTypes uint16
|
||||||
|
CbSizeVft uint16
|
||||||
|
CbAlignment uint16
|
||||||
|
WTypeFlags uint16
|
||||||
|
WMajorVerNum uint16
|
||||||
|
WMinorVerNum uint16
|
||||||
|
TdescAlias TYPEDESC
|
||||||
|
IdldescType IDLDESC
|
||||||
|
}
|
100
vendor/github.com/go-ole/go-ole/oleutil/connection.go
generated
vendored
Normal file
100
vendor/github.com/go-ole/go-ole/oleutil/connection.go
generated
vendored
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package oleutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
ole "github.com/go-ole/go-ole"
|
||||||
|
)
|
||||||
|
|
||||||
|
type stdDispatch struct {
|
||||||
|
lpVtbl *stdDispatchVtbl
|
||||||
|
ref int32
|
||||||
|
iid *ole.GUID
|
||||||
|
iface interface{}
|
||||||
|
funcMap map[string]int32
|
||||||
|
}
|
||||||
|
|
||||||
|
type stdDispatchVtbl struct {
|
||||||
|
pQueryInterface uintptr
|
||||||
|
pAddRef uintptr
|
||||||
|
pRelease uintptr
|
||||||
|
pGetTypeInfoCount uintptr
|
||||||
|
pGetTypeInfo uintptr
|
||||||
|
pGetIDsOfNames uintptr
|
||||||
|
pInvoke uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispQueryInterface(this *ole.IUnknown, iid *ole.GUID, punk **ole.IUnknown) uint32 {
|
||||||
|
pthis := (*stdDispatch)(unsafe.Pointer(this))
|
||||||
|
*punk = nil
|
||||||
|
if ole.IsEqualGUID(iid, ole.IID_IUnknown) ||
|
||||||
|
ole.IsEqualGUID(iid, ole.IID_IDispatch) {
|
||||||
|
dispAddRef(this)
|
||||||
|
*punk = this
|
||||||
|
return ole.S_OK
|
||||||
|
}
|
||||||
|
if ole.IsEqualGUID(iid, pthis.iid) {
|
||||||
|
dispAddRef(this)
|
||||||
|
*punk = this
|
||||||
|
return ole.S_OK
|
||||||
|
}
|
||||||
|
return ole.E_NOINTERFACE
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispAddRef(this *ole.IUnknown) int32 {
|
||||||
|
pthis := (*stdDispatch)(unsafe.Pointer(this))
|
||||||
|
pthis.ref++
|
||||||
|
return pthis.ref
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispRelease(this *ole.IUnknown) int32 {
|
||||||
|
pthis := (*stdDispatch)(unsafe.Pointer(this))
|
||||||
|
pthis.ref--
|
||||||
|
return pthis.ref
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispGetIDsOfNames(this *ole.IUnknown, iid *ole.GUID, wnames []*uint16, namelen int, lcid int, pdisp []int32) uintptr {
|
||||||
|
pthis := (*stdDispatch)(unsafe.Pointer(this))
|
||||||
|
names := make([]string, len(wnames))
|
||||||
|
for i := 0; i < len(names); i++ {
|
||||||
|
names[i] = ole.LpOleStrToString(wnames[i])
|
||||||
|
}
|
||||||
|
for n := 0; n < namelen; n++ {
|
||||||
|
if id, ok := pthis.funcMap[names[n]]; ok {
|
||||||
|
pdisp[n] = id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ole.S_OK
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispGetTypeInfoCount(pcount *int) uintptr {
|
||||||
|
if pcount != nil {
|
||||||
|
*pcount = 0
|
||||||
|
}
|
||||||
|
return ole.S_OK
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispGetTypeInfo(ptypeif *uintptr) uintptr {
|
||||||
|
return ole.E_NOTIMPL
|
||||||
|
}
|
||||||
|
|
||||||
|
func dispInvoke(this *ole.IDispatch, dispid int32, riid *ole.GUID, lcid int, flags int16, dispparams *ole.DISPPARAMS, result *ole.VARIANT, pexcepinfo *ole.EXCEPINFO, nerr *uint) uintptr {
|
||||||
|
pthis := (*stdDispatch)(unsafe.Pointer(this))
|
||||||
|
found := ""
|
||||||
|
for name, id := range pthis.funcMap {
|
||||||
|
if id == dispid {
|
||||||
|
found = name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if found != "" {
|
||||||
|
rv := reflect.ValueOf(pthis.iface).Elem()
|
||||||
|
rm := rv.MethodByName(found)
|
||||||
|
rr := rm.Call([]reflect.Value{})
|
||||||
|
println(len(rr))
|
||||||
|
return ole.S_OK
|
||||||
|
}
|
||||||
|
return ole.E_NOTIMPL
|
||||||
|
}
|
10
vendor/github.com/go-ole/go-ole/oleutil/connection_func.go
generated
vendored
Normal file
10
vendor/github.com/go-ole/go-ole/oleutil/connection_func.go
generated
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package oleutil
|
||||||
|
|
||||||
|
import ole "github.com/go-ole/go-ole"
|
||||||
|
|
||||||
|
// ConnectObject creates a connection point between two services for communication.
|
||||||
|
func ConnectObject(disp *ole.IDispatch, iid *ole.GUID, idisp interface{}) (uint32, error) {
|
||||||
|
return 0, ole.NewError(ole.E_NOTIMPL)
|
||||||
|
}
|
58
vendor/github.com/go-ole/go-ole/oleutil/connection_windows.go
generated
vendored
Normal file
58
vendor/github.com/go-ole/go-ole/oleutil/connection_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package oleutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
ole "github.com/go-ole/go-ole"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ConnectObject creates a connection point between two services for communication.
|
||||||
|
func ConnectObject(disp *ole.IDispatch, iid *ole.GUID, idisp interface{}) (cookie uint32, err error) {
|
||||||
|
unknown, err := disp.QueryInterface(ole.IID_IConnectionPointContainer)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
container := (*ole.IConnectionPointContainer)(unsafe.Pointer(unknown))
|
||||||
|
var point *ole.IConnectionPoint
|
||||||
|
err = container.FindConnectionPoint(iid, &point)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if edisp, ok := idisp.(*ole.IUnknown); ok {
|
||||||
|
cookie, err = point.Advise(edisp)
|
||||||
|
container.Release()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rv := reflect.ValueOf(disp).Elem()
|
||||||
|
if rv.Type().Kind() == reflect.Struct {
|
||||||
|
dest := &stdDispatch{}
|
||||||
|
dest.lpVtbl = &stdDispatchVtbl{}
|
||||||
|
dest.lpVtbl.pQueryInterface = syscall.NewCallback(dispQueryInterface)
|
||||||
|
dest.lpVtbl.pAddRef = syscall.NewCallback(dispAddRef)
|
||||||
|
dest.lpVtbl.pRelease = syscall.NewCallback(dispRelease)
|
||||||
|
dest.lpVtbl.pGetTypeInfoCount = syscall.NewCallback(dispGetTypeInfoCount)
|
||||||
|
dest.lpVtbl.pGetTypeInfo = syscall.NewCallback(dispGetTypeInfo)
|
||||||
|
dest.lpVtbl.pGetIDsOfNames = syscall.NewCallback(dispGetIDsOfNames)
|
||||||
|
dest.lpVtbl.pInvoke = syscall.NewCallback(dispInvoke)
|
||||||
|
dest.iface = disp
|
||||||
|
dest.iid = iid
|
||||||
|
cookie, err = point.Advise((*ole.IUnknown)(unsafe.Pointer(dest)))
|
||||||
|
container.Release()
|
||||||
|
if err != nil {
|
||||||
|
point.Release()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
container.Release()
|
||||||
|
|
||||||
|
return 0, ole.NewError(ole.E_INVALIDARG)
|
||||||
|
}
|
6
vendor/github.com/go-ole/go-ole/oleutil/go-get.go
generated
vendored
Normal file
6
vendor/github.com/go-ole/go-ole/oleutil/go-get.go
generated
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
// This file is here so go get succeeds as without it errors with:
|
||||||
|
// no buildable Go source files in ...
|
||||||
|
//
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package oleutil
|
127
vendor/github.com/go-ole/go-ole/oleutil/oleutil.go
generated
vendored
Normal file
127
vendor/github.com/go-ole/go-ole/oleutil/oleutil.go
generated
vendored
Normal file
|
@ -0,0 +1,127 @@
|
||||||
|
package oleutil
|
||||||
|
|
||||||
|
import ole "github.com/go-ole/go-ole"
|
||||||
|
|
||||||
|
// ClassIDFrom retrieves class ID whether given is program ID or application string.
|
||||||
|
func ClassIDFrom(programID string) (classID *ole.GUID, err error) {
|
||||||
|
return ole.ClassIDFrom(programID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateObject creates object from programID based on interface type.
|
||||||
|
//
|
||||||
|
// Only supports IUnknown.
|
||||||
|
//
|
||||||
|
// Program ID can be either program ID or application string.
|
||||||
|
func CreateObject(programID string) (unknown *ole.IUnknown, err error) {
|
||||||
|
classID, err := ole.ClassIDFrom(programID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
unknown, err = ole.CreateInstance(classID, ole.IID_IUnknown)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetActiveObject retrieves active object for program ID and interface ID based
|
||||||
|
// on interface type.
|
||||||
|
//
|
||||||
|
// Only supports IUnknown.
|
||||||
|
//
|
||||||
|
// Program ID can be either program ID or application string.
|
||||||
|
func GetActiveObject(programID string) (unknown *ole.IUnknown, err error) {
|
||||||
|
classID, err := ole.ClassIDFrom(programID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
unknown, err = ole.GetActiveObject(classID, ole.IID_IUnknown)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// CallMethod calls method on IDispatch with parameters.
|
||||||
|
func CallMethod(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT, err error) {
|
||||||
|
return disp.InvokeWithOptionalArgs(name, ole.DISPATCH_METHOD, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustCallMethod calls method on IDispatch with parameters or panics.
|
||||||
|
func MustCallMethod(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT) {
|
||||||
|
r, err := CallMethod(disp, name, params...)
|
||||||
|
if err != nil {
|
||||||
|
panic(err.Error())
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetProperty retrieves property from IDispatch.
|
||||||
|
func GetProperty(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT, err error) {
|
||||||
|
return disp.InvokeWithOptionalArgs(name, ole.DISPATCH_PROPERTYGET, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustGetProperty retrieves property from IDispatch or panics.
|
||||||
|
func MustGetProperty(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT) {
|
||||||
|
r, err := GetProperty(disp, name, params...)
|
||||||
|
if err != nil {
|
||||||
|
panic(err.Error())
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
// PutProperty mutates property.
|
||||||
|
func PutProperty(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT, err error) {
|
||||||
|
return disp.InvokeWithOptionalArgs(name, ole.DISPATCH_PROPERTYPUT, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustPutProperty mutates property or panics.
|
||||||
|
func MustPutProperty(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT) {
|
||||||
|
r, err := PutProperty(disp, name, params...)
|
||||||
|
if err != nil {
|
||||||
|
panic(err.Error())
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
// PutPropertyRef mutates property reference.
|
||||||
|
func PutPropertyRef(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT, err error) {
|
||||||
|
return disp.InvokeWithOptionalArgs(name, ole.DISPATCH_PROPERTYPUTREF, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MustPutPropertyRef mutates property reference or panics.
|
||||||
|
func MustPutPropertyRef(disp *ole.IDispatch, name string, params ...interface{}) (result *ole.VARIANT) {
|
||||||
|
r, err := PutPropertyRef(disp, name, params...)
|
||||||
|
if err != nil {
|
||||||
|
panic(err.Error())
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func ForEach(disp *ole.IDispatch, f func(v *ole.VARIANT) error) error {
|
||||||
|
newEnum, err := disp.GetProperty("_NewEnum")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer newEnum.Clear()
|
||||||
|
|
||||||
|
enum, err := newEnum.ToIUnknown().IEnumVARIANT(ole.IID_IEnumVariant)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer enum.Release()
|
||||||
|
|
||||||
|
for item, length, err := enum.Next(1); length > 0; item, length, err = enum.Next(1) {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if ferr := f(&item); ferr != nil {
|
||||||
|
return ferr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
27
vendor/github.com/go-ole/go-ole/safearray.go
generated
vendored
Normal file
27
vendor/github.com/go-ole/go-ole/safearray.go
generated
vendored
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
// Package is meant to retrieve and process safe array data returned from COM.
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
// SafeArrayBound defines the SafeArray boundaries.
|
||||||
|
type SafeArrayBound struct {
|
||||||
|
Elements uint32
|
||||||
|
LowerBound int32
|
||||||
|
}
|
||||||
|
|
||||||
|
// SafeArray is how COM handles arrays.
|
||||||
|
type SafeArray struct {
|
||||||
|
Dimensions uint16
|
||||||
|
FeaturesFlag uint16
|
||||||
|
ElementsSize uint32
|
||||||
|
LocksAmount uint32
|
||||||
|
Data uint32
|
||||||
|
Bounds [16]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFEARRAY is obsolete, exists for backwards compatibility.
|
||||||
|
// Use SafeArray
|
||||||
|
type SAFEARRAY SafeArray
|
||||||
|
|
||||||
|
// SAFEARRAYBOUND is obsolete, exists for backwards compatibility.
|
||||||
|
// Use SafeArrayBound
|
||||||
|
type SAFEARRAYBOUND SafeArrayBound
|
211
vendor/github.com/go-ole/go-ole/safearray_func.go
generated
vendored
Normal file
211
vendor/github.com/go-ole/go-ole/safearray_func.go
generated
vendored
Normal file
|
@ -0,0 +1,211 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// safeArrayAccessData returns raw array pointer.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAccessData in Windows API.
|
||||||
|
func safeArrayAccessData(safearray *SafeArray) (uintptr, error) {
|
||||||
|
return uintptr(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayUnaccessData releases raw array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayUnaccessData in Windows API.
|
||||||
|
func safeArrayUnaccessData(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocData allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocData in Windows API.
|
||||||
|
func safeArrayAllocData(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocDescriptor allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocDescriptor in Windows API.
|
||||||
|
func safeArrayAllocDescriptor(dimensions uint32) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocDescriptorEx allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocDescriptorEx in Windows API.
|
||||||
|
func safeArrayAllocDescriptorEx(variantType VT, dimensions uint32) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCopy returns copy of SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCopy in Windows API.
|
||||||
|
func safeArrayCopy(original *SafeArray) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCopyData duplicates SafeArray into another SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCopyData in Windows API.
|
||||||
|
func safeArrayCopyData(original *SafeArray, duplicate *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreate creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreate in Windows API.
|
||||||
|
func safeArrayCreate(variantType VT, dimensions uint32, bounds *SafeArrayBound) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateEx creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateEx in Windows API.
|
||||||
|
func safeArrayCreateEx(variantType VT, dimensions uint32, bounds *SafeArrayBound, extra uintptr) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateVector creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateVector in Windows API.
|
||||||
|
func safeArrayCreateVector(variantType VT, lowerBound int32, length uint32) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateVectorEx creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateVectorEx in Windows API.
|
||||||
|
func safeArrayCreateVectorEx(variantType VT, lowerBound int32, length uint32, extra uintptr) (*SafeArray, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroy destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroy in Windows API.
|
||||||
|
func safeArrayDestroy(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroyData destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroyData in Windows API.
|
||||||
|
func safeArrayDestroyData(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroyDescriptor destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroyDescriptor in Windows API.
|
||||||
|
func safeArrayDestroyDescriptor(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetDim is the amount of dimensions in the SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetDim in Windows API.
|
||||||
|
func safeArrayGetDim(safearray *SafeArray) (*uint32, error) {
|
||||||
|
u := uint32(0)
|
||||||
|
return &u, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElementSize is the element size in bytes.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetElemsize in Windows API.
|
||||||
|
func safeArrayGetElementSize(safearray *SafeArray) (*uint32, error) {
|
||||||
|
u := uint32(0)
|
||||||
|
return &u, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElement retrieves element at given index.
|
||||||
|
func safeArrayGetElement(safearray *SafeArray, index int32, pv unsafe.Pointer) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElement retrieves element at given index and converts to string.
|
||||||
|
func safeArrayGetElementString(safearray *SafeArray, index int32) (string, error) {
|
||||||
|
return "", NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetIID is the InterfaceID of the elements in the SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetIID in Windows API.
|
||||||
|
func safeArrayGetIID(safearray *SafeArray) (*GUID, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetLBound returns lower bounds of SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetLBound in Windows API.
|
||||||
|
func safeArrayGetLBound(safearray *SafeArray, dimension uint32) (int32, error) {
|
||||||
|
return int32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetUBound returns upper bounds of SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetUBound in Windows API.
|
||||||
|
func safeArrayGetUBound(safearray *SafeArray, dimension uint32) (int32, error) {
|
||||||
|
return int32(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetVartype returns data type of SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetVartype in Windows API.
|
||||||
|
func safeArrayGetVartype(safearray *SafeArray) (uint16, error) {
|
||||||
|
return uint16(0), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayLock locks SafeArray for reading to modify SafeArray.
|
||||||
|
//
|
||||||
|
// This must be called during some calls to ensure that another process does not
|
||||||
|
// read or write to the SafeArray during editing.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayLock in Windows API.
|
||||||
|
func safeArrayLock(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayUnlock unlocks SafeArray for reading.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayUnlock in Windows API.
|
||||||
|
func safeArrayUnlock(safearray *SafeArray) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayPutElement stores the data element at the specified location in the
|
||||||
|
// array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayPutElement in Windows API.
|
||||||
|
func safeArrayPutElement(safearray *SafeArray, index int64, element uintptr) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetRecordInfo accesses IRecordInfo info for custom types.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetRecordInfo in Windows API.
|
||||||
|
//
|
||||||
|
// XXX: Must implement IRecordInfo interface for this to return.
|
||||||
|
func safeArrayGetRecordInfo(safearray *SafeArray) (interface{}, error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArraySetRecordInfo mutates IRecordInfo info for custom types.
|
||||||
|
//
|
||||||
|
// AKA: SafeArraySetRecordInfo in Windows API.
|
||||||
|
//
|
||||||
|
// XXX: Must implement IRecordInfo interface for this to return.
|
||||||
|
func safeArraySetRecordInfo(safearray *SafeArray, recordInfo interface{}) error {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
337
vendor/github.com/go-ole/go-ole/safearray_windows.go
generated
vendored
Normal file
337
vendor/github.com/go-ole/go-ole/safearray_windows.go
generated
vendored
Normal file
|
@ -0,0 +1,337 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
procSafeArrayAccessData, _ = modoleaut32.FindProc("SafeArrayAccessData")
|
||||||
|
procSafeArrayAllocData, _ = modoleaut32.FindProc("SafeArrayAllocData")
|
||||||
|
procSafeArrayAllocDescriptor, _ = modoleaut32.FindProc("SafeArrayAllocDescriptor")
|
||||||
|
procSafeArrayAllocDescriptorEx, _ = modoleaut32.FindProc("SafeArrayAllocDescriptorEx")
|
||||||
|
procSafeArrayCopy, _ = modoleaut32.FindProc("SafeArrayCopy")
|
||||||
|
procSafeArrayCopyData, _ = modoleaut32.FindProc("SafeArrayCopyData")
|
||||||
|
procSafeArrayCreate, _ = modoleaut32.FindProc("SafeArrayCreate")
|
||||||
|
procSafeArrayCreateEx, _ = modoleaut32.FindProc("SafeArrayCreateEx")
|
||||||
|
procSafeArrayCreateVector, _ = modoleaut32.FindProc("SafeArrayCreateVector")
|
||||||
|
procSafeArrayCreateVectorEx, _ = modoleaut32.FindProc("SafeArrayCreateVectorEx")
|
||||||
|
procSafeArrayDestroy, _ = modoleaut32.FindProc("SafeArrayDestroy")
|
||||||
|
procSafeArrayDestroyData, _ = modoleaut32.FindProc("SafeArrayDestroyData")
|
||||||
|
procSafeArrayDestroyDescriptor, _ = modoleaut32.FindProc("SafeArrayDestroyDescriptor")
|
||||||
|
procSafeArrayGetDim, _ = modoleaut32.FindProc("SafeArrayGetDim")
|
||||||
|
procSafeArrayGetElement, _ = modoleaut32.FindProc("SafeArrayGetElement")
|
||||||
|
procSafeArrayGetElemsize, _ = modoleaut32.FindProc("SafeArrayGetElemsize")
|
||||||
|
procSafeArrayGetIID, _ = modoleaut32.FindProc("SafeArrayGetIID")
|
||||||
|
procSafeArrayGetLBound, _ = modoleaut32.FindProc("SafeArrayGetLBound")
|
||||||
|
procSafeArrayGetUBound, _ = modoleaut32.FindProc("SafeArrayGetUBound")
|
||||||
|
procSafeArrayGetVartype, _ = modoleaut32.FindProc("SafeArrayGetVartype")
|
||||||
|
procSafeArrayLock, _ = modoleaut32.FindProc("SafeArrayLock")
|
||||||
|
procSafeArrayPtrOfIndex, _ = modoleaut32.FindProc("SafeArrayPtrOfIndex")
|
||||||
|
procSafeArrayUnaccessData, _ = modoleaut32.FindProc("SafeArrayUnaccessData")
|
||||||
|
procSafeArrayUnlock, _ = modoleaut32.FindProc("SafeArrayUnlock")
|
||||||
|
procSafeArrayPutElement, _ = modoleaut32.FindProc("SafeArrayPutElement")
|
||||||
|
//procSafeArrayRedim, _ = modoleaut32.FindProc("SafeArrayRedim") // TODO
|
||||||
|
//procSafeArraySetIID, _ = modoleaut32.FindProc("SafeArraySetIID") // TODO
|
||||||
|
procSafeArrayGetRecordInfo, _ = modoleaut32.FindProc("SafeArrayGetRecordInfo")
|
||||||
|
procSafeArraySetRecordInfo, _ = modoleaut32.FindProc("SafeArraySetRecordInfo")
|
||||||
|
)
|
||||||
|
|
||||||
|
// safeArrayAccessData returns raw array pointer.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAccessData in Windows API.
|
||||||
|
// Todo: Test
|
||||||
|
func safeArrayAccessData(safearray *SafeArray) (element uintptr, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayAccessData.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&element))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayUnaccessData releases raw array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayUnaccessData in Windows API.
|
||||||
|
func safeArrayUnaccessData(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayUnaccessData.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocData allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocData in Windows API.
|
||||||
|
func safeArrayAllocData(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayAllocData.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocDescriptor allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocDescriptor in Windows API.
|
||||||
|
func safeArrayAllocDescriptor(dimensions uint32) (safearray *SafeArray, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayAllocDescriptor.Call(uintptr(dimensions), uintptr(unsafe.Pointer(&safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayAllocDescriptorEx allocates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayAllocDescriptorEx in Windows API.
|
||||||
|
func safeArrayAllocDescriptorEx(variantType VT, dimensions uint32) (safearray *SafeArray, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayAllocDescriptorEx.Call(
|
||||||
|
uintptr(variantType),
|
||||||
|
uintptr(dimensions),
|
||||||
|
uintptr(unsafe.Pointer(&safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCopy returns copy of SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCopy in Windows API.
|
||||||
|
func safeArrayCopy(original *SafeArray) (safearray *SafeArray, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayCopy.Call(
|
||||||
|
uintptr(unsafe.Pointer(original)),
|
||||||
|
uintptr(unsafe.Pointer(&safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCopyData duplicates SafeArray into another SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCopyData in Windows API.
|
||||||
|
func safeArrayCopyData(original *SafeArray, duplicate *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayCopyData.Call(
|
||||||
|
uintptr(unsafe.Pointer(original)),
|
||||||
|
uintptr(unsafe.Pointer(duplicate))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreate creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreate in Windows API.
|
||||||
|
func safeArrayCreate(variantType VT, dimensions uint32, bounds *SafeArrayBound) (safearray *SafeArray, err error) {
|
||||||
|
sa, _, err := procSafeArrayCreate.Call(
|
||||||
|
uintptr(variantType),
|
||||||
|
uintptr(dimensions),
|
||||||
|
uintptr(unsafe.Pointer(bounds)))
|
||||||
|
safearray = (*SafeArray)(unsafe.Pointer(&sa))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateEx creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateEx in Windows API.
|
||||||
|
func safeArrayCreateEx(variantType VT, dimensions uint32, bounds *SafeArrayBound, extra uintptr) (safearray *SafeArray, err error) {
|
||||||
|
sa, _, err := procSafeArrayCreateEx.Call(
|
||||||
|
uintptr(variantType),
|
||||||
|
uintptr(dimensions),
|
||||||
|
uintptr(unsafe.Pointer(bounds)),
|
||||||
|
extra)
|
||||||
|
safearray = (*SafeArray)(unsafe.Pointer(sa))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateVector creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateVector in Windows API.
|
||||||
|
func safeArrayCreateVector(variantType VT, lowerBound int32, length uint32) (safearray *SafeArray, err error) {
|
||||||
|
sa, _, err := procSafeArrayCreateVector.Call(
|
||||||
|
uintptr(variantType),
|
||||||
|
uintptr(lowerBound),
|
||||||
|
uintptr(length))
|
||||||
|
safearray = (*SafeArray)(unsafe.Pointer(sa))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayCreateVectorEx creates SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayCreateVectorEx in Windows API.
|
||||||
|
func safeArrayCreateVectorEx(variantType VT, lowerBound int32, length uint32, extra uintptr) (safearray *SafeArray, err error) {
|
||||||
|
sa, _, err := procSafeArrayCreateVectorEx.Call(
|
||||||
|
uintptr(variantType),
|
||||||
|
uintptr(lowerBound),
|
||||||
|
uintptr(length),
|
||||||
|
extra)
|
||||||
|
safearray = (*SafeArray)(unsafe.Pointer(sa))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroy destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroy in Windows API.
|
||||||
|
func safeArrayDestroy(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayDestroy.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroyData destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroyData in Windows API.
|
||||||
|
func safeArrayDestroyData(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayDestroyData.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayDestroyDescriptor destroys SafeArray object.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayDestroyDescriptor in Windows API.
|
||||||
|
func safeArrayDestroyDescriptor(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayDestroyDescriptor.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetDim is the amount of dimensions in the SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetDim in Windows API.
|
||||||
|
func safeArrayGetDim(safearray *SafeArray) (dimensions *uint32, err error) {
|
||||||
|
l, _, err := procSafeArrayGetDim.Call(uintptr(unsafe.Pointer(safearray)))
|
||||||
|
dimensions = (*uint32)(unsafe.Pointer(l))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElementSize is the element size in bytes.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetElemsize in Windows API.
|
||||||
|
func safeArrayGetElementSize(safearray *SafeArray) (length *uint32, err error) {
|
||||||
|
l, _, err := procSafeArrayGetElemsize.Call(uintptr(unsafe.Pointer(safearray)))
|
||||||
|
length = (*uint32)(unsafe.Pointer(l))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElement retrieves element at given index.
|
||||||
|
func safeArrayGetElement(safearray *SafeArray, index int32, pv unsafe.Pointer) error {
|
||||||
|
return convertHresultToError(
|
||||||
|
procSafeArrayGetElement.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&index)),
|
||||||
|
uintptr(pv)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetElementString retrieves element at given index and converts to string.
|
||||||
|
func safeArrayGetElementString(safearray *SafeArray, index int32) (str string, err error) {
|
||||||
|
var element *int16
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetElement.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&index)),
|
||||||
|
uintptr(unsafe.Pointer(&element))))
|
||||||
|
str = BstrToString(*(**uint16)(unsafe.Pointer(&element)))
|
||||||
|
SysFreeString(element)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetIID is the InterfaceID of the elements in the SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetIID in Windows API.
|
||||||
|
func safeArrayGetIID(safearray *SafeArray) (guid *GUID, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetIID.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&guid))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetLBound returns lower bounds of SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetLBound in Windows API.
|
||||||
|
func safeArrayGetLBound(safearray *SafeArray, dimension uint32) (lowerBound int32, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetLBound.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(dimension),
|
||||||
|
uintptr(unsafe.Pointer(&lowerBound))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetUBound returns upper bounds of SafeArray.
|
||||||
|
//
|
||||||
|
// SafeArrays may have multiple dimensions. Meaning, it could be
|
||||||
|
// multidimensional array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetUBound in Windows API.
|
||||||
|
func safeArrayGetUBound(safearray *SafeArray, dimension uint32) (upperBound int32, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetUBound.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(dimension),
|
||||||
|
uintptr(unsafe.Pointer(&upperBound))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetVartype returns data type of SafeArray.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetVartype in Windows API.
|
||||||
|
func safeArrayGetVartype(safearray *SafeArray) (varType uint16, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetVartype.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&varType))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayLock locks SafeArray for reading to modify SafeArray.
|
||||||
|
//
|
||||||
|
// This must be called during some calls to ensure that another process does not
|
||||||
|
// read or write to the SafeArray during editing.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayLock in Windows API.
|
||||||
|
func safeArrayLock(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayLock.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayUnlock unlocks SafeArray for reading.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayUnlock in Windows API.
|
||||||
|
func safeArrayUnlock(safearray *SafeArray) (err error) {
|
||||||
|
err = convertHresultToError(procSafeArrayUnlock.Call(uintptr(unsafe.Pointer(safearray))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayPutElement stores the data element at the specified location in the
|
||||||
|
// array.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayPutElement in Windows API.
|
||||||
|
func safeArrayPutElement(safearray *SafeArray, index int64, element uintptr) (err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayPutElement.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&index)),
|
||||||
|
uintptr(unsafe.Pointer(element))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArrayGetRecordInfo accesses IRecordInfo info for custom types.
|
||||||
|
//
|
||||||
|
// AKA: SafeArrayGetRecordInfo in Windows API.
|
||||||
|
//
|
||||||
|
// XXX: Must implement IRecordInfo interface for this to return.
|
||||||
|
func safeArrayGetRecordInfo(safearray *SafeArray) (recordInfo interface{}, err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArrayGetRecordInfo.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&recordInfo))))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// safeArraySetRecordInfo mutates IRecordInfo info for custom types.
|
||||||
|
//
|
||||||
|
// AKA: SafeArraySetRecordInfo in Windows API.
|
||||||
|
//
|
||||||
|
// XXX: Must implement IRecordInfo interface for this to return.
|
||||||
|
func safeArraySetRecordInfo(safearray *SafeArray, recordInfo interface{}) (err error) {
|
||||||
|
err = convertHresultToError(
|
||||||
|
procSafeArraySetRecordInfo.Call(
|
||||||
|
uintptr(unsafe.Pointer(safearray)),
|
||||||
|
uintptr(unsafe.Pointer(&recordInfo))))
|
||||||
|
return
|
||||||
|
}
|
140
vendor/github.com/go-ole/go-ole/safearrayconversion.go
generated
vendored
Normal file
140
vendor/github.com/go-ole/go-ole/safearrayconversion.go
generated
vendored
Normal file
|
@ -0,0 +1,140 @@
|
||||||
|
// Helper for converting SafeArray to array of objects.
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
type SafeArrayConversion struct {
|
||||||
|
Array *SafeArray
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) ToStringArray() (strings []string) {
|
||||||
|
totalElements, _ := sac.TotalElements(0)
|
||||||
|
strings = make([]string, totalElements)
|
||||||
|
|
||||||
|
for i := int32(0); i < totalElements; i++ {
|
||||||
|
strings[int32(i)], _ = safeArrayGetElementString(sac.Array, i)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) ToByteArray() (bytes []byte) {
|
||||||
|
totalElements, _ := sac.TotalElements(0)
|
||||||
|
bytes = make([]byte, totalElements)
|
||||||
|
|
||||||
|
for i := int32(0); i < totalElements; i++ {
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&bytes[int32(i)]))
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) ToValueArray() (values []interface{}) {
|
||||||
|
totalElements, _ := sac.TotalElements(0)
|
||||||
|
values = make([]interface{}, totalElements)
|
||||||
|
vt, _ := safeArrayGetVartype(sac.Array)
|
||||||
|
|
||||||
|
for i := int32(0); i < totalElements; i++ {
|
||||||
|
switch VT(vt) {
|
||||||
|
case VT_BOOL:
|
||||||
|
var v bool
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_I1:
|
||||||
|
var v int8
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_I2:
|
||||||
|
var v int16
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_I4:
|
||||||
|
var v int32
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_I8:
|
||||||
|
var v int64
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_UI1:
|
||||||
|
var v uint8
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_UI2:
|
||||||
|
var v uint16
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_UI4:
|
||||||
|
var v uint32
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_UI8:
|
||||||
|
var v uint64
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_R4:
|
||||||
|
var v float32
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_R8:
|
||||||
|
var v float64
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_BSTR:
|
||||||
|
var v string
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v
|
||||||
|
case VT_VARIANT:
|
||||||
|
var v VARIANT
|
||||||
|
safeArrayGetElement(sac.Array, i, unsafe.Pointer(&v))
|
||||||
|
values[i] = v.Value()
|
||||||
|
default:
|
||||||
|
// TODO
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) GetType() (varType uint16, err error) {
|
||||||
|
return safeArrayGetVartype(sac.Array)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) GetDimensions() (dimensions *uint32, err error) {
|
||||||
|
return safeArrayGetDim(sac.Array)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) GetSize() (length *uint32, err error) {
|
||||||
|
return safeArrayGetElementSize(sac.Array)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sac *SafeArrayConversion) TotalElements(index uint32) (totalElements int32, err error) {
|
||||||
|
if index < 1 {
|
||||||
|
index = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get array bounds
|
||||||
|
var LowerBounds int32
|
||||||
|
var UpperBounds int32
|
||||||
|
|
||||||
|
LowerBounds, err = safeArrayGetLBound(sac.Array, index)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
UpperBounds, err = safeArrayGetUBound(sac.Array, index)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
totalElements = UpperBounds - LowerBounds + 1
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Release Safe Array memory
|
||||||
|
func (sac *SafeArrayConversion) Release() {
|
||||||
|
safeArrayDestroy(sac.Array)
|
||||||
|
}
|
33
vendor/github.com/go-ole/go-ole/safearrayslices.go
generated
vendored
Normal file
33
vendor/github.com/go-ole/go-ole/safearrayslices.go
generated
vendored
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func safeArrayFromByteSlice(slice []byte) *SafeArray {
|
||||||
|
array, _ := safeArrayCreateVector(VT_UI1, 0, uint32(len(slice)))
|
||||||
|
|
||||||
|
if array == nil {
|
||||||
|
panic("Could not convert []byte to SAFEARRAY")
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range slice {
|
||||||
|
safeArrayPutElement(array, int64(i), uintptr(unsafe.Pointer(&v)))
|
||||||
|
}
|
||||||
|
return array
|
||||||
|
}
|
||||||
|
|
||||||
|
func safeArrayFromStringSlice(slice []string) *SafeArray {
|
||||||
|
array, _ := safeArrayCreateVector(VT_BSTR, 0, uint32(len(slice)))
|
||||||
|
|
||||||
|
if array == nil {
|
||||||
|
panic("Could not convert []string to SAFEARRAY")
|
||||||
|
}
|
||||||
|
// SysAllocStringLen(s)
|
||||||
|
for i, v := range slice {
|
||||||
|
safeArrayPutElement(array, int64(i), uintptr(unsafe.Pointer(SysAllocStringLen(v))))
|
||||||
|
}
|
||||||
|
return array
|
||||||
|
}
|
101
vendor/github.com/go-ole/go-ole/utility.go
generated
vendored
Normal file
101
vendor/github.com/go-ole/go-ole/utility.go
generated
vendored
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"unicode/utf16"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ClassIDFrom retrieves class ID whether given is program ID or application string.
|
||||||
|
//
|
||||||
|
// Helper that provides check against both Class ID from Program ID and Class ID from string. It is
|
||||||
|
// faster, if you know which you are using, to use the individual functions, but this will check
|
||||||
|
// against available functions for you.
|
||||||
|
func ClassIDFrom(programID string) (classID *GUID, err error) {
|
||||||
|
classID, err = CLSIDFromProgID(programID)
|
||||||
|
if err != nil {
|
||||||
|
classID, err = CLSIDFromString(programID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// BytePtrToString converts byte pointer to a Go string.
|
||||||
|
func BytePtrToString(p *byte) string {
|
||||||
|
a := (*[10000]uint8)(unsafe.Pointer(p))
|
||||||
|
i := 0
|
||||||
|
for a[i] != 0 {
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
return string(a[:i])
|
||||||
|
}
|
||||||
|
|
||||||
|
// UTF16PtrToString is alias for LpOleStrToString.
|
||||||
|
//
|
||||||
|
// Kept for compatibility reasons.
|
||||||
|
func UTF16PtrToString(p *uint16) string {
|
||||||
|
return LpOleStrToString(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
// LpOleStrToString converts COM Unicode to Go string.
|
||||||
|
func LpOleStrToString(p *uint16) string {
|
||||||
|
if p == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
length := lpOleStrLen(p)
|
||||||
|
a := make([]uint16, length)
|
||||||
|
|
||||||
|
ptr := unsafe.Pointer(p)
|
||||||
|
|
||||||
|
for i := 0; i < int(length); i++ {
|
||||||
|
a[i] = *(*uint16)(ptr)
|
||||||
|
ptr = unsafe.Pointer(uintptr(ptr) + 2)
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(utf16.Decode(a))
|
||||||
|
}
|
||||||
|
|
||||||
|
// BstrToString converts COM binary string to Go string.
|
||||||
|
func BstrToString(p *uint16) string {
|
||||||
|
if p == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
length := SysStringLen((*int16)(unsafe.Pointer(p)))
|
||||||
|
a := make([]uint16, length)
|
||||||
|
|
||||||
|
ptr := unsafe.Pointer(p)
|
||||||
|
|
||||||
|
for i := 0; i < int(length); i++ {
|
||||||
|
a[i] = *(*uint16)(ptr)
|
||||||
|
ptr = unsafe.Pointer(uintptr(ptr) + 2)
|
||||||
|
}
|
||||||
|
return string(utf16.Decode(a))
|
||||||
|
}
|
||||||
|
|
||||||
|
// lpOleStrLen returns the length of Unicode string.
|
||||||
|
func lpOleStrLen(p *uint16) (length int64) {
|
||||||
|
if p == nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr := unsafe.Pointer(p)
|
||||||
|
|
||||||
|
for i := 0; ; i++ {
|
||||||
|
if 0 == *(*uint16)(ptr) {
|
||||||
|
length = int64(i)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
ptr = unsafe.Pointer(uintptr(ptr) + 2)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// convertHresultToError converts syscall to error, if call is unsuccessful.
|
||||||
|
func convertHresultToError(hr uintptr, r2 uintptr, ignore error) (err error) {
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
16
vendor/github.com/go-ole/go-ole/variables.go
generated
vendored
Normal file
16
vendor/github.com/go-ole/go-ole/variables.go
generated
vendored
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
modcombase = syscall.NewLazyDLL("combase.dll")
|
||||||
|
modkernel32, _ = syscall.LoadDLL("kernel32.dll")
|
||||||
|
modole32, _ = syscall.LoadDLL("ole32.dll")
|
||||||
|
modoleaut32, _ = syscall.LoadDLL("oleaut32.dll")
|
||||||
|
modmsvcrt, _ = syscall.LoadDLL("msvcrt.dll")
|
||||||
|
moduser32, _ = syscall.LoadDLL("user32.dll")
|
||||||
|
)
|
105
vendor/github.com/go-ole/go-ole/variant.go
generated
vendored
Normal file
105
vendor/github.com/go-ole/go-ole/variant.go
generated
vendored
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
// NewVariant returns new variant based on type and value.
|
||||||
|
func NewVariant(vt VT, val int64) VARIANT {
|
||||||
|
return VARIANT{VT: vt, Val: val}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToIUnknown converts Variant to Unknown object.
|
||||||
|
func (v *VARIANT) ToIUnknown() *IUnknown {
|
||||||
|
if v.VT != VT_UNKNOWN {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return (*IUnknown)(unsafe.Pointer(uintptr(v.Val)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToIDispatch converts variant to dispatch object.
|
||||||
|
func (v *VARIANT) ToIDispatch() *IDispatch {
|
||||||
|
if v.VT != VT_DISPATCH {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return (*IDispatch)(unsafe.Pointer(uintptr(v.Val)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToArray converts variant to SafeArray helper.
|
||||||
|
func (v *VARIANT) ToArray() *SafeArrayConversion {
|
||||||
|
if v.VT != VT_SAFEARRAY {
|
||||||
|
if v.VT&VT_ARRAY == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var safeArray *SafeArray = (*SafeArray)(unsafe.Pointer(uintptr(v.Val)))
|
||||||
|
return &SafeArrayConversion{safeArray}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ToString converts variant to Go string.
|
||||||
|
func (v *VARIANT) ToString() string {
|
||||||
|
if v.VT != VT_BSTR {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return BstrToString(*(**uint16)(unsafe.Pointer(&v.Val)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear the memory of variant object.
|
||||||
|
func (v *VARIANT) Clear() error {
|
||||||
|
return VariantClear(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Value returns variant value based on its type.
|
||||||
|
//
|
||||||
|
// Currently supported types: 2- and 4-byte integers, strings, bools.
|
||||||
|
// Note that 64-bit integers, datetimes, and other types are stored as strings
|
||||||
|
// and will be returned as strings.
|
||||||
|
//
|
||||||
|
// Needs to be further converted, because this returns an interface{}.
|
||||||
|
func (v *VARIANT) Value() interface{} {
|
||||||
|
switch v.VT {
|
||||||
|
case VT_I1:
|
||||||
|
return int8(v.Val)
|
||||||
|
case VT_UI1:
|
||||||
|
return uint8(v.Val)
|
||||||
|
case VT_I2:
|
||||||
|
return int16(v.Val)
|
||||||
|
case VT_UI2:
|
||||||
|
return uint16(v.Val)
|
||||||
|
case VT_I4:
|
||||||
|
return int32(v.Val)
|
||||||
|
case VT_UI4:
|
||||||
|
return uint32(v.Val)
|
||||||
|
case VT_I8:
|
||||||
|
return int64(v.Val)
|
||||||
|
case VT_UI8:
|
||||||
|
return uint64(v.Val)
|
||||||
|
case VT_INT:
|
||||||
|
return int(v.Val)
|
||||||
|
case VT_UINT:
|
||||||
|
return uint(v.Val)
|
||||||
|
case VT_INT_PTR:
|
||||||
|
return uintptr(v.Val) // TODO
|
||||||
|
case VT_UINT_PTR:
|
||||||
|
return uintptr(v.Val)
|
||||||
|
case VT_R4:
|
||||||
|
return *(*float32)(unsafe.Pointer(&v.Val))
|
||||||
|
case VT_R8:
|
||||||
|
return *(*float64)(unsafe.Pointer(&v.Val))
|
||||||
|
case VT_BSTR:
|
||||||
|
return v.ToString()
|
||||||
|
case VT_DATE:
|
||||||
|
// VT_DATE type will either return float64 or time.Time.
|
||||||
|
d := uint64(v.Val)
|
||||||
|
date, err := GetVariantDate(d)
|
||||||
|
if err != nil {
|
||||||
|
return float64(v.Val)
|
||||||
|
}
|
||||||
|
return date
|
||||||
|
case VT_UNKNOWN:
|
||||||
|
return v.ToIUnknown()
|
||||||
|
case VT_DISPATCH:
|
||||||
|
return v.ToIDispatch()
|
||||||
|
case VT_BOOL:
|
||||||
|
return v.Val != 0
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
11
vendor/github.com/go-ole/go-ole/variant_386.go
generated
vendored
Normal file
11
vendor/github.com/go-ole/go-ole/variant_386.go
generated
vendored
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
// +build 386
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
type VARIANT struct {
|
||||||
|
VT VT // 2
|
||||||
|
wReserved1 uint16 // 4
|
||||||
|
wReserved2 uint16 // 6
|
||||||
|
wReserved3 uint16 // 8
|
||||||
|
Val int64 // 16
|
||||||
|
}
|
12
vendor/github.com/go-ole/go-ole/variant_amd64.go
generated
vendored
Normal file
12
vendor/github.com/go-ole/go-ole/variant_amd64.go
generated
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
// +build amd64
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
type VARIANT struct {
|
||||||
|
VT VT // 2
|
||||||
|
wReserved1 uint16 // 4
|
||||||
|
wReserved2 uint16 // 6
|
||||||
|
wReserved3 uint16 // 8
|
||||||
|
Val int64 // 16
|
||||||
|
_ [8]byte // 24
|
||||||
|
}
|
22
vendor/github.com/go-ole/go-ole/variant_date_386.go
generated
vendored
Normal file
22
vendor/github.com/go-ole/go-ole/variant_date_386.go
generated
vendored
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
// +build windows,386
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetVariantDate converts COM Variant Time value to Go time.Time.
|
||||||
|
func GetVariantDate(value uint64) (time.Time, error) {
|
||||||
|
var st syscall.Systemtime
|
||||||
|
v1 := uint32(value)
|
||||||
|
v2 := uint32(value >> 32)
|
||||||
|
r, _, _ := procVariantTimeToSystemTime.Call(uintptr(v1), uintptr(v2), uintptr(unsafe.Pointer(&st)))
|
||||||
|
if r != 0 {
|
||||||
|
return time.Date(int(st.Year), time.Month(st.Month), int(st.Day), int(st.Hour), int(st.Minute), int(st.Second), int(st.Milliseconds/1000), time.UTC), nil
|
||||||
|
}
|
||||||
|
return time.Now(), errors.New("Could not convert to time, passing current time.")
|
||||||
|
}
|
20
vendor/github.com/go-ole/go-ole/variant_date_amd64.go
generated
vendored
Normal file
20
vendor/github.com/go-ole/go-ole/variant_date_amd64.go
generated
vendored
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
// +build windows,amd64
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetVariantDate converts COM Variant Time value to Go time.Time.
|
||||||
|
func GetVariantDate(value uint64) (time.Time, error) {
|
||||||
|
var st syscall.Systemtime
|
||||||
|
r, _, _ := procVariantTimeToSystemTime.Call(uintptr(value), uintptr(unsafe.Pointer(&st)))
|
||||||
|
if r != 0 {
|
||||||
|
return time.Date(int(st.Year), time.Month(st.Month), int(st.Day), int(st.Hour), int(st.Minute), int(st.Second), int(st.Milliseconds/1000), time.UTC), nil
|
||||||
|
}
|
||||||
|
return time.Now(), errors.New("Could not convert to time, passing current time.")
|
||||||
|
}
|
12
vendor/github.com/go-ole/go-ole/variant_ppc64le.go
generated
vendored
Normal file
12
vendor/github.com/go-ole/go-ole/variant_ppc64le.go
generated
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
// +build ppc64le
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
type VARIANT struct {
|
||||||
|
VT VT // 2
|
||||||
|
wReserved1 uint16 // 4
|
||||||
|
wReserved2 uint16 // 6
|
||||||
|
wReserved3 uint16 // 8
|
||||||
|
Val int64 // 16
|
||||||
|
_ [8]byte // 24
|
||||||
|
}
|
12
vendor/github.com/go-ole/go-ole/variant_s390x.go
generated
vendored
Normal file
12
vendor/github.com/go-ole/go-ole/variant_s390x.go
generated
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
// +build s390x
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
type VARIANT struct {
|
||||||
|
VT VT // 2
|
||||||
|
wReserved1 uint16 // 4
|
||||||
|
wReserved2 uint16 // 6
|
||||||
|
wReserved3 uint16 // 8
|
||||||
|
Val int64 // 16
|
||||||
|
_ [8]byte // 24
|
||||||
|
}
|
58
vendor/github.com/go-ole/go-ole/vt_string.go
generated
vendored
Normal file
58
vendor/github.com/go-ole/go-ole/vt_string.go
generated
vendored
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
// generated by stringer -output vt_string.go -type VT; DO NOT EDIT
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
const (
|
||||||
|
_VT_name_0 = "VT_EMPTYVT_NULLVT_I2VT_I4VT_R4VT_R8VT_CYVT_DATEVT_BSTRVT_DISPATCHVT_ERRORVT_BOOLVT_VARIANTVT_UNKNOWNVT_DECIMAL"
|
||||||
|
_VT_name_1 = "VT_I1VT_UI1VT_UI2VT_UI4VT_I8VT_UI8VT_INTVT_UINTVT_VOIDVT_HRESULTVT_PTRVT_SAFEARRAYVT_CARRAYVT_USERDEFINEDVT_LPSTRVT_LPWSTR"
|
||||||
|
_VT_name_2 = "VT_RECORDVT_INT_PTRVT_UINT_PTR"
|
||||||
|
_VT_name_3 = "VT_FILETIMEVT_BLOBVT_STREAMVT_STORAGEVT_STREAMED_OBJECTVT_STORED_OBJECTVT_BLOB_OBJECTVT_CFVT_CLSID"
|
||||||
|
_VT_name_4 = "VT_BSTR_BLOBVT_VECTOR"
|
||||||
|
_VT_name_5 = "VT_ARRAY"
|
||||||
|
_VT_name_6 = "VT_BYREF"
|
||||||
|
_VT_name_7 = "VT_RESERVED"
|
||||||
|
_VT_name_8 = "VT_ILLEGAL"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
_VT_index_0 = [...]uint8{0, 8, 15, 20, 25, 30, 35, 40, 47, 54, 65, 73, 80, 90, 100, 110}
|
||||||
|
_VT_index_1 = [...]uint8{0, 5, 11, 17, 23, 28, 34, 40, 47, 54, 64, 70, 82, 91, 105, 113, 122}
|
||||||
|
_VT_index_2 = [...]uint8{0, 9, 19, 30}
|
||||||
|
_VT_index_3 = [...]uint8{0, 11, 18, 27, 37, 55, 71, 85, 90, 98}
|
||||||
|
_VT_index_4 = [...]uint8{0, 12, 21}
|
||||||
|
_VT_index_5 = [...]uint8{0, 8}
|
||||||
|
_VT_index_6 = [...]uint8{0, 8}
|
||||||
|
_VT_index_7 = [...]uint8{0, 11}
|
||||||
|
_VT_index_8 = [...]uint8{0, 10}
|
||||||
|
)
|
||||||
|
|
||||||
|
func (i VT) String() string {
|
||||||
|
switch {
|
||||||
|
case 0 <= i && i <= 14:
|
||||||
|
return _VT_name_0[_VT_index_0[i]:_VT_index_0[i+1]]
|
||||||
|
case 16 <= i && i <= 31:
|
||||||
|
i -= 16
|
||||||
|
return _VT_name_1[_VT_index_1[i]:_VT_index_1[i+1]]
|
||||||
|
case 36 <= i && i <= 38:
|
||||||
|
i -= 36
|
||||||
|
return _VT_name_2[_VT_index_2[i]:_VT_index_2[i+1]]
|
||||||
|
case 64 <= i && i <= 72:
|
||||||
|
i -= 64
|
||||||
|
return _VT_name_3[_VT_index_3[i]:_VT_index_3[i+1]]
|
||||||
|
case 4095 <= i && i <= 4096:
|
||||||
|
i -= 4095
|
||||||
|
return _VT_name_4[_VT_index_4[i]:_VT_index_4[i+1]]
|
||||||
|
case i == 8192:
|
||||||
|
return _VT_name_5
|
||||||
|
case i == 16384:
|
||||||
|
return _VT_name_6
|
||||||
|
case i == 32768:
|
||||||
|
return _VT_name_7
|
||||||
|
case i == 65535:
|
||||||
|
return _VT_name_8
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("VT(%d)", i)
|
||||||
|
}
|
||||||
|
}
|
99
vendor/github.com/go-ole/go-ole/winrt.go
generated
vendored
Normal file
99
vendor/github.com/go-ole/go-ole/winrt.go
generated
vendored
Normal file
|
@ -0,0 +1,99 @@
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"syscall"
|
||||||
|
"unicode/utf8"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
procRoInitialize = modcombase.NewProc("RoInitialize")
|
||||||
|
procRoActivateInstance = modcombase.NewProc("RoActivateInstance")
|
||||||
|
procRoGetActivationFactory = modcombase.NewProc("RoGetActivationFactory")
|
||||||
|
procWindowsCreateString = modcombase.NewProc("WindowsCreateString")
|
||||||
|
procWindowsDeleteString = modcombase.NewProc("WindowsDeleteString")
|
||||||
|
procWindowsGetStringRawBuffer = modcombase.NewProc("WindowsGetStringRawBuffer")
|
||||||
|
)
|
||||||
|
|
||||||
|
func RoInitialize(thread_type uint32) (err error) {
|
||||||
|
hr, _, _ := procRoInitialize.Call(uintptr(thread_type))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func RoActivateInstance(clsid string) (ins *IInspectable, err error) {
|
||||||
|
hClsid, err := NewHString(clsid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer DeleteHString(hClsid)
|
||||||
|
|
||||||
|
hr, _, _ := procRoActivateInstance.Call(
|
||||||
|
uintptr(unsafe.Pointer(hClsid)),
|
||||||
|
uintptr(unsafe.Pointer(&ins)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func RoGetActivationFactory(clsid string, iid *GUID) (ins *IInspectable, err error) {
|
||||||
|
hClsid, err := NewHString(clsid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer DeleteHString(hClsid)
|
||||||
|
|
||||||
|
hr, _, _ := procRoGetActivationFactory.Call(
|
||||||
|
uintptr(unsafe.Pointer(hClsid)),
|
||||||
|
uintptr(unsafe.Pointer(iid)),
|
||||||
|
uintptr(unsafe.Pointer(&ins)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// HString is handle string for pointers.
|
||||||
|
type HString uintptr
|
||||||
|
|
||||||
|
// NewHString returns a new HString for Go string.
|
||||||
|
func NewHString(s string) (hstring HString, err error) {
|
||||||
|
u16 := syscall.StringToUTF16Ptr(s)
|
||||||
|
len := uint32(utf8.RuneCountInString(s))
|
||||||
|
hr, _, _ := procWindowsCreateString.Call(
|
||||||
|
uintptr(unsafe.Pointer(u16)),
|
||||||
|
uintptr(len),
|
||||||
|
uintptr(unsafe.Pointer(&hstring)))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteHString deletes HString.
|
||||||
|
func DeleteHString(hstring HString) (err error) {
|
||||||
|
hr, _, _ := procWindowsDeleteString.Call(uintptr(hstring))
|
||||||
|
if hr != 0 {
|
||||||
|
err = NewError(hr)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns Go string value of HString.
|
||||||
|
func (h HString) String() string {
|
||||||
|
var u16buf uintptr
|
||||||
|
var u16len uint32
|
||||||
|
u16buf, _, _ = procWindowsGetStringRawBuffer.Call(
|
||||||
|
uintptr(h),
|
||||||
|
uintptr(unsafe.Pointer(&u16len)))
|
||||||
|
|
||||||
|
u16hdr := reflect.SliceHeader{Data: u16buf, Len: int(u16len), Cap: int(u16len)}
|
||||||
|
u16 := *(*[]uint16)(unsafe.Pointer(&u16hdr))
|
||||||
|
return syscall.UTF16ToString(u16)
|
||||||
|
}
|
36
vendor/github.com/go-ole/go-ole/winrt_doc.go
generated
vendored
Normal file
36
vendor/github.com/go-ole/go-ole/winrt_doc.go
generated
vendored
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package ole
|
||||||
|
|
||||||
|
// RoInitialize
|
||||||
|
func RoInitialize(thread_type uint32) (err error) {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RoActivateInstance
|
||||||
|
func RoActivateInstance(clsid string) (ins *IInspectable, err error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RoGetActivationFactory
|
||||||
|
func RoGetActivationFactory(clsid string, iid *GUID) (ins *IInspectable, err error) {
|
||||||
|
return nil, NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HString is handle string for pointers.
|
||||||
|
type HString uintptr
|
||||||
|
|
||||||
|
// NewHString returns a new HString for Go string.
|
||||||
|
func NewHString(s string) (hstring HString, err error) {
|
||||||
|
return HString(uintptr(0)), NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteHString deletes HString.
|
||||||
|
func DeleteHString(hstring HString) (err error) {
|
||||||
|
return NewError(E_NOTIMPL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns Go string value of HString.
|
||||||
|
func (h HString) String() string {
|
||||||
|
return ""
|
||||||
|
}
|
3
vendor/github.com/golang/protobuf/AUTHORS
generated
vendored
Normal file
3
vendor/github.com/golang/protobuf/AUTHORS
generated
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# This source code refers to The Go Authors for copyright purposes.
|
||||||
|
# The master list of authors is in the main Go distribution,
|
||||||
|
# visible at http://tip.golang.org/AUTHORS.
|
3
vendor/github.com/golang/protobuf/CONTRIBUTORS
generated
vendored
Normal file
3
vendor/github.com/golang/protobuf/CONTRIBUTORS
generated
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# This source code was written by the Go contributors.
|
||||||
|
# The master list of contributors is in the main Go distribution,
|
||||||
|
# visible at http://tip.golang.org/CONTRIBUTORS.
|
28
vendor/github.com/golang/protobuf/LICENSE
generated
vendored
Normal file
28
vendor/github.com/golang/protobuf/LICENSE
generated
vendored
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
Copyright 2010 The Go Authors. All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions are
|
||||||
|
met:
|
||||||
|
|
||||||
|
* Redistributions of source code must retain the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer.
|
||||||
|
* Redistributions in binary form must reproduce the above
|
||||||
|
copyright notice, this list of conditions and the following disclaimer
|
||||||
|
in the documentation and/or other materials provided with the
|
||||||
|
distribution.
|
||||||
|
* Neither the name of Google Inc. nor the names of its
|
||||||
|
contributors may be used to endorse or promote products derived from
|
||||||
|
this software without specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
253
vendor/github.com/golang/protobuf/proto/clone.go
generated
vendored
Normal file
253
vendor/github.com/golang/protobuf/proto/clone.go
generated
vendored
Normal file
|
@ -0,0 +1,253 @@
|
||||||
|
// Go support for Protocol Buffers - Google's data interchange format
|
||||||
|
//
|
||||||
|
// Copyright 2011 The Go Authors. All rights reserved.
|
||||||
|
// https://github.com/golang/protobuf
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are
|
||||||
|
// met:
|
||||||
|
//
|
||||||
|
// * Redistributions of source code must retain the above copyright
|
||||||
|
// notice, this list of conditions and the following disclaimer.
|
||||||
|
// * Redistributions in binary form must reproduce the above
|
||||||
|
// copyright notice, this list of conditions and the following disclaimer
|
||||||
|
// in the documentation and/or other materials provided with the
|
||||||
|
// distribution.
|
||||||
|
// * Neither the name of Google Inc. nor the names of its
|
||||||
|
// contributors may be used to endorse or promote products derived from
|
||||||
|
// this software without specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
// Protocol buffer deep copy and merge.
|
||||||
|
// TODO: RawMessage.
|
||||||
|
|
||||||
|
package proto
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Clone returns a deep copy of a protocol buffer.
|
||||||
|
func Clone(src Message) Message {
|
||||||
|
in := reflect.ValueOf(src)
|
||||||
|
if in.IsNil() {
|
||||||
|
return src
|
||||||
|
}
|
||||||
|
out := reflect.New(in.Type().Elem())
|
||||||
|
dst := out.Interface().(Message)
|
||||||
|
Merge(dst, src)
|
||||||
|
return dst
|
||||||
|
}
|
||||||
|
|
||||||
|
// Merger is the interface representing objects that can merge messages of the same type.
|
||||||
|
type Merger interface {
|
||||||
|
// Merge merges src into this message.
|
||||||
|
// Required and optional fields that are set in src will be set to that value in dst.
|
||||||
|
// Elements of repeated fields will be appended.
|
||||||
|
//
|
||||||
|
// Merge may panic if called with a different argument type than the receiver.
|
||||||
|
Merge(src Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
// generatedMerger is the custom merge method that generated protos will have.
|
||||||
|
// We must add this method since a generate Merge method will conflict with
|
||||||
|
// many existing protos that have a Merge data field already defined.
|
||||||
|
type generatedMerger interface {
|
||||||
|
XXX_Merge(src Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Merge merges src into dst.
|
||||||
|
// Required and optional fields that are set in src will be set to that value in dst.
|
||||||
|
// Elements of repeated fields will be appended.
|
||||||
|
// Merge panics if src and dst are not the same type, or if dst is nil.
|
||||||
|
func Merge(dst, src Message) {
|
||||||
|
if m, ok := dst.(Merger); ok {
|
||||||
|
m.Merge(src)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
in := reflect.ValueOf(src)
|
||||||
|
out := reflect.ValueOf(dst)
|
||||||
|
if out.IsNil() {
|
||||||
|
panic("proto: nil destination")
|
||||||
|
}
|
||||||
|
if in.Type() != out.Type() {
|
||||||
|
panic(fmt.Sprintf("proto.Merge(%T, %T) type mismatch", dst, src))
|
||||||
|
}
|
||||||
|
if in.IsNil() {
|
||||||
|
return // Merge from nil src is a noop
|
||||||
|
}
|
||||||
|
if m, ok := dst.(generatedMerger); ok {
|
||||||
|
m.XXX_Merge(src)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
mergeStruct(out.Elem(), in.Elem())
|
||||||
|
}
|
||||||
|
|
||||||
|
func mergeStruct(out, in reflect.Value) {
|
||||||
|
sprop := GetProperties(in.Type())
|
||||||
|
for i := 0; i < in.NumField(); i++ {
|
||||||
|
f := in.Type().Field(i)
|
||||||
|
if strings.HasPrefix(f.Name, "XXX_") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
mergeAny(out.Field(i), in.Field(i), false, sprop.Prop[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
if emIn, err := extendable(in.Addr().Interface()); err == nil {
|
||||||
|
emOut, _ := extendable(out.Addr().Interface())
|
||||||
|
mIn, muIn := emIn.extensionsRead()
|
||||||
|
if mIn != nil {
|
||||||
|
mOut := emOut.extensionsWrite()
|
||||||
|
muIn.Lock()
|
||||||
|
mergeExtension(mOut, mIn)
|
||||||
|
muIn.Unlock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uf := in.FieldByName("XXX_unrecognized")
|
||||||
|
if !uf.IsValid() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
uin := uf.Bytes()
|
||||||
|
if len(uin) > 0 {
|
||||||
|
out.FieldByName("XXX_unrecognized").SetBytes(append([]byte(nil), uin...))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// mergeAny performs a merge between two values of the same type.
|
||||||
|
// viaPtr indicates whether the values were indirected through a pointer (implying proto2).
|
||||||
|
// prop is set if this is a struct field (it may be nil).
|
||||||
|
func mergeAny(out, in reflect.Value, viaPtr bool, prop *Properties) {
|
||||||
|
if in.Type() == protoMessageType {
|
||||||
|
if !in.IsNil() {
|
||||||
|
if out.IsNil() {
|
||||||
|
out.Set(reflect.ValueOf(Clone(in.Interface().(Message))))
|
||||||
|
} else {
|
||||||
|
Merge(out.Interface().(Message), in.Interface().(Message))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
switch in.Kind() {
|
||||||
|
case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64,
|
||||||
|
reflect.String, reflect.Uint32, reflect.Uint64:
|
||||||
|
if !viaPtr && isProto3Zero(in) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
out.Set(in)
|
||||||
|
case reflect.Interface:
|
||||||
|
// Probably a oneof field; copy non-nil values.
|
||||||
|
if in.IsNil() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// Allocate destination if it is not set, or set to a different type.
|
||||||
|
// Otherwise we will merge as normal.
|
||||||
|
if out.IsNil() || out.Elem().Type() != in.Elem().Type() {
|
||||||
|
out.Set(reflect.New(in.Elem().Elem().Type())) // interface -> *T -> T -> new(T)
|
||||||
|
}
|
||||||
|
mergeAny(out.Elem(), in.Elem(), false, nil)
|
||||||
|
case reflect.Map:
|
||||||
|
if in.Len() == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if out.IsNil() {
|
||||||
|
out.Set(reflect.MakeMap(in.Type()))
|
||||||
|
}
|
||||||
|
// For maps with value types of *T or []byte we need to deep copy each value.
|
||||||
|
elemKind := in.Type().Elem().Kind()
|
||||||
|
for _, key := range in.MapKeys() {
|
||||||
|
var val reflect.Value
|
||||||
|
switch elemKind {
|
||||||
|
case reflect.Ptr:
|
||||||
|
val = reflect.New(in.Type().Elem().Elem())
|
||||||
|
mergeAny(val, in.MapIndex(key), false, nil)
|
||||||
|
case reflect.Slice:
|
||||||
|
val = in.MapIndex(key)
|
||||||
|
val = reflect.ValueOf(append([]byte{}, val.Bytes()...))
|
||||||
|
default:
|
||||||
|
val = in.MapIndex(key)
|
||||||
|
}
|
||||||
|
out.SetMapIndex(key, val)
|
||||||
|
}
|
||||||
|
case reflect.Ptr:
|
||||||
|
if in.IsNil() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if out.IsNil() {
|
||||||
|
out.Set(reflect.New(in.Elem().Type()))
|
||||||
|
}
|
||||||
|
mergeAny(out.Elem(), in.Elem(), true, nil)
|
||||||
|
case reflect.Slice:
|
||||||
|
if in.IsNil() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if in.Type().Elem().Kind() == reflect.Uint8 {
|
||||||
|
// []byte is a scalar bytes field, not a repeated field.
|
||||||
|
|
||||||
|
// Edge case: if this is in a proto3 message, a zero length
|
||||||
|
// bytes field is considered the zero value, and should not
|
||||||
|
// be merged.
|
||||||
|
if prop != nil && prop.proto3 && in.Len() == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make a deep copy.
|
||||||
|
// Append to []byte{} instead of []byte(nil) so that we never end up
|
||||||
|
// with a nil result.
|
||||||
|
out.SetBytes(append([]byte{}, in.Bytes()...))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
n := in.Len()
|
||||||
|
if out.IsNil() {
|
||||||
|
out.Set(reflect.MakeSlice(in.Type(), 0, n))
|
||||||
|
}
|
||||||
|
switch in.Type().Elem().Kind() {
|
||||||
|
case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64,
|
||||||
|
reflect.String, reflect.Uint32, reflect.Uint64:
|
||||||
|
out.Set(reflect.AppendSlice(out, in))
|
||||||
|
default:
|
||||||
|
for i := 0; i < n; i++ {
|
||||||
|
x := reflect.Indirect(reflect.New(in.Type().Elem()))
|
||||||
|
mergeAny(x, in.Index(i), false, nil)
|
||||||
|
out.Set(reflect.Append(out, x))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.Struct:
|
||||||
|
mergeStruct(out, in)
|
||||||
|
default:
|
||||||
|
// unknown type, so not a protocol buffer
|
||||||
|
log.Printf("proto: don't know how to copy %v", in)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func mergeExtension(out, in map[int32]Extension) {
|
||||||
|
for extNum, eIn := range in {
|
||||||
|
eOut := Extension{desc: eIn.desc}
|
||||||
|
if eIn.value != nil {
|
||||||
|
v := reflect.New(reflect.TypeOf(eIn.value)).Elem()
|
||||||
|
mergeAny(v, reflect.ValueOf(eIn.value), false, nil)
|
||||||
|
eOut.value = v.Interface()
|
||||||
|
}
|
||||||
|
if eIn.enc != nil {
|
||||||
|
eOut.enc = make([]byte, len(eIn.enc))
|
||||||
|
copy(eOut.enc, eIn.enc)
|
||||||
|
}
|
||||||
|
|
||||||
|
out[extNum] = eOut
|
||||||
|
}
|
||||||
|
}
|
428
vendor/github.com/golang/protobuf/proto/decode.go
generated
vendored
Normal file
428
vendor/github.com/golang/protobuf/proto/decode.go
generated
vendored
Normal file
|
@ -0,0 +1,428 @@
|
||||||
|
// Go support for Protocol Buffers - Google's data interchange format
|
||||||
|
//
|
||||||
|
// Copyright 2010 The Go Authors. All rights reserved.
|
||||||
|
// https://github.com/golang/protobuf
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are
|
||||||
|
// met:
|
||||||
|
//
|
||||||
|
// * Redistributions of source code must retain the above copyright
|
||||||
|
// notice, this list of conditions and the following disclaimer.
|
||||||
|
// * Redistributions in binary form must reproduce the above
|
||||||
|
// copyright notice, this list of conditions and the following disclaimer
|
||||||
|
// in the documentation and/or other materials provided with the
|
||||||
|
// distribution.
|
||||||
|
// * Neither the name of Google Inc. nor the names of its
|
||||||
|
// contributors may be used to endorse or promote products derived from
|
||||||
|
// this software without specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
package proto
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Routines for decoding protocol buffer data to construct in-memory representations.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// errOverflow is returned when an integer is too large to be represented.
|
||||||
|
var errOverflow = errors.New("proto: integer overflow")
|
||||||
|
|
||||||
|
// ErrInternalBadWireType is returned by generated code when an incorrect
|
||||||
|
// wire type is encountered. It does not get returned to user code.
|
||||||
|
var ErrInternalBadWireType = errors.New("proto: internal error: bad wiretype for oneof")
|
||||||
|
|
||||||
|
// DecodeVarint reads a varint-encoded integer from the slice.
|
||||||
|
// It returns the integer and the number of bytes consumed, or
|
||||||
|
// zero if there is not enough.
|
||||||
|
// This is the format for the
|
||||||
|
// int32, int64, uint32, uint64, bool, and enum
|
||||||
|
// protocol buffer types.
|
||||||
|
func DecodeVarint(buf []byte) (x uint64, n int) {
|
||||||
|
for shift := uint(0); shift < 64; shift += 7 {
|
||||||
|
if n >= len(buf) {
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
b := uint64(buf[n])
|
||||||
|
n++
|
||||||
|
x |= (b & 0x7F) << shift
|
||||||
|
if (b & 0x80) == 0 {
|
||||||
|
return x, n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The number is too large to represent in a 64-bit value.
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Buffer) decodeVarintSlow() (x uint64, err error) {
|
||||||
|
i := p.index
|
||||||
|
l := len(p.buf)
|
||||||
|
|
||||||
|
for shift := uint(0); shift < 64; shift += 7 {
|
||||||
|
if i >= l {
|
||||||
|
err = io.ErrUnexpectedEOF
|
||||||
|
return
|
||||||
|
}
|
||||||
|
b := p.buf[i]
|
||||||
|
i++
|
||||||
|
x |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
p.index = i
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The number is too large to represent in a 64-bit value.
|
||||||
|
err = errOverflow
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeVarint reads a varint-encoded integer from the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// int32, int64, uint32, uint64, bool, and enum
|
||||||
|
// protocol buffer types.
|
||||||
|
func (p *Buffer) DecodeVarint() (x uint64, err error) {
|
||||||
|
i := p.index
|
||||||
|
buf := p.buf
|
||||||
|
|
||||||
|
if i >= len(buf) {
|
||||||
|
return 0, io.ErrUnexpectedEOF
|
||||||
|
} else if buf[i] < 0x80 {
|
||||||
|
p.index++
|
||||||
|
return uint64(buf[i]), nil
|
||||||
|
} else if len(buf)-i < 10 {
|
||||||
|
return p.decodeVarintSlow()
|
||||||
|
}
|
||||||
|
|
||||||
|
var b uint64
|
||||||
|
// we already checked the first byte
|
||||||
|
x = uint64(buf[i]) - 0x80
|
||||||
|
i++
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 7
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 7
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 14
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 14
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 21
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 21
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 28
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 28
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 35
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 35
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 42
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 42
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 49
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 49
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 56
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
x -= 0x80 << 56
|
||||||
|
|
||||||
|
b = uint64(buf[i])
|
||||||
|
i++
|
||||||
|
x += b << 63
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
goto done
|
||||||
|
}
|
||||||
|
// x -= 0x80 << 63 // Always zero.
|
||||||
|
|
||||||
|
return 0, errOverflow
|
||||||
|
|
||||||
|
done:
|
||||||
|
p.index = i
|
||||||
|
return x, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeFixed64 reads a 64-bit integer from the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// fixed64, sfixed64, and double protocol buffer types.
|
||||||
|
func (p *Buffer) DecodeFixed64() (x uint64, err error) {
|
||||||
|
// x, err already 0
|
||||||
|
i := p.index + 8
|
||||||
|
if i < 0 || i > len(p.buf) {
|
||||||
|
err = io.ErrUnexpectedEOF
|
||||||
|
return
|
||||||
|
}
|
||||||
|
p.index = i
|
||||||
|
|
||||||
|
x = uint64(p.buf[i-8])
|
||||||
|
x |= uint64(p.buf[i-7]) << 8
|
||||||
|
x |= uint64(p.buf[i-6]) << 16
|
||||||
|
x |= uint64(p.buf[i-5]) << 24
|
||||||
|
x |= uint64(p.buf[i-4]) << 32
|
||||||
|
x |= uint64(p.buf[i-3]) << 40
|
||||||
|
x |= uint64(p.buf[i-2]) << 48
|
||||||
|
x |= uint64(p.buf[i-1]) << 56
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeFixed32 reads a 32-bit integer from the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// fixed32, sfixed32, and float protocol buffer types.
|
||||||
|
func (p *Buffer) DecodeFixed32() (x uint64, err error) {
|
||||||
|
// x, err already 0
|
||||||
|
i := p.index + 4
|
||||||
|
if i < 0 || i > len(p.buf) {
|
||||||
|
err = io.ErrUnexpectedEOF
|
||||||
|
return
|
||||||
|
}
|
||||||
|
p.index = i
|
||||||
|
|
||||||
|
x = uint64(p.buf[i-4])
|
||||||
|
x |= uint64(p.buf[i-3]) << 8
|
||||||
|
x |= uint64(p.buf[i-2]) << 16
|
||||||
|
x |= uint64(p.buf[i-1]) << 24
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeZigzag64 reads a zigzag-encoded 64-bit integer
|
||||||
|
// from the Buffer.
|
||||||
|
// This is the format used for the sint64 protocol buffer type.
|
||||||
|
func (p *Buffer) DecodeZigzag64() (x uint64, err error) {
|
||||||
|
x, err = p.DecodeVarint()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
x = (x >> 1) ^ uint64((int64(x&1)<<63)>>63)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeZigzag32 reads a zigzag-encoded 32-bit integer
|
||||||
|
// from the Buffer.
|
||||||
|
// This is the format used for the sint32 protocol buffer type.
|
||||||
|
func (p *Buffer) DecodeZigzag32() (x uint64, err error) {
|
||||||
|
x, err = p.DecodeVarint()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
x = uint64((uint32(x) >> 1) ^ uint32((int32(x&1)<<31)>>31))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeRawBytes reads a count-delimited byte buffer from the Buffer.
|
||||||
|
// This is the format used for the bytes protocol buffer
|
||||||
|
// type and for embedded messages.
|
||||||
|
func (p *Buffer) DecodeRawBytes(alloc bool) (buf []byte, err error) {
|
||||||
|
n, err := p.DecodeVarint()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
nb := int(n)
|
||||||
|
if nb < 0 {
|
||||||
|
return nil, fmt.Errorf("proto: bad byte length %d", nb)
|
||||||
|
}
|
||||||
|
end := p.index + nb
|
||||||
|
if end < p.index || end > len(p.buf) {
|
||||||
|
return nil, io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
|
||||||
|
if !alloc {
|
||||||
|
// todo: check if can get more uses of alloc=false
|
||||||
|
buf = p.buf[p.index:end]
|
||||||
|
p.index += nb
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
buf = make([]byte, nb)
|
||||||
|
copy(buf, p.buf[p.index:])
|
||||||
|
p.index += nb
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeStringBytes reads an encoded string from the Buffer.
|
||||||
|
// This is the format used for the proto2 string type.
|
||||||
|
func (p *Buffer) DecodeStringBytes() (s string, err error) {
|
||||||
|
buf, err := p.DecodeRawBytes(false)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return string(buf), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unmarshaler is the interface representing objects that can
|
||||||
|
// unmarshal themselves. The argument points to data that may be
|
||||||
|
// overwritten, so implementations should not keep references to the
|
||||||
|
// buffer.
|
||||||
|
// Unmarshal implementations should not clear the receiver.
|
||||||
|
// Any unmarshaled data should be merged into the receiver.
|
||||||
|
// Callers of Unmarshal that do not want to retain existing data
|
||||||
|
// should Reset the receiver before calling Unmarshal.
|
||||||
|
type Unmarshaler interface {
|
||||||
|
Unmarshal([]byte) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// newUnmarshaler is the interface representing objects that can
|
||||||
|
// unmarshal themselves. The semantics are identical to Unmarshaler.
|
||||||
|
//
|
||||||
|
// This exists to support protoc-gen-go generated messages.
|
||||||
|
// The proto package will stop type-asserting to this interface in the future.
|
||||||
|
//
|
||||||
|
// DO NOT DEPEND ON THIS.
|
||||||
|
type newUnmarshaler interface {
|
||||||
|
XXX_Unmarshal([]byte) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unmarshal parses the protocol buffer representation in buf and places the
|
||||||
|
// decoded result in pb. If the struct underlying pb does not match
|
||||||
|
// the data in buf, the results can be unpredictable.
|
||||||
|
//
|
||||||
|
// Unmarshal resets pb before starting to unmarshal, so any
|
||||||
|
// existing data in pb is always removed. Use UnmarshalMerge
|
||||||
|
// to preserve and append to existing data.
|
||||||
|
func Unmarshal(buf []byte, pb Message) error {
|
||||||
|
pb.Reset()
|
||||||
|
if u, ok := pb.(newUnmarshaler); ok {
|
||||||
|
return u.XXX_Unmarshal(buf)
|
||||||
|
}
|
||||||
|
if u, ok := pb.(Unmarshaler); ok {
|
||||||
|
return u.Unmarshal(buf)
|
||||||
|
}
|
||||||
|
return NewBuffer(buf).Unmarshal(pb)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnmarshalMerge parses the protocol buffer representation in buf and
|
||||||
|
// writes the decoded result to pb. If the struct underlying pb does not match
|
||||||
|
// the data in buf, the results can be unpredictable.
|
||||||
|
//
|
||||||
|
// UnmarshalMerge merges into existing data in pb.
|
||||||
|
// Most code should use Unmarshal instead.
|
||||||
|
func UnmarshalMerge(buf []byte, pb Message) error {
|
||||||
|
if u, ok := pb.(newUnmarshaler); ok {
|
||||||
|
return u.XXX_Unmarshal(buf)
|
||||||
|
}
|
||||||
|
if u, ok := pb.(Unmarshaler); ok {
|
||||||
|
// NOTE: The history of proto have unfortunately been inconsistent
|
||||||
|
// whether Unmarshaler should or should not implicitly clear itself.
|
||||||
|
// Some implementations do, most do not.
|
||||||
|
// Thus, calling this here may or may not do what people want.
|
||||||
|
//
|
||||||
|
// See https://github.com/golang/protobuf/issues/424
|
||||||
|
return u.Unmarshal(buf)
|
||||||
|
}
|
||||||
|
return NewBuffer(buf).Unmarshal(pb)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeMessage reads a count-delimited message from the Buffer.
|
||||||
|
func (p *Buffer) DecodeMessage(pb Message) error {
|
||||||
|
enc, err := p.DecodeRawBytes(false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return NewBuffer(enc).Unmarshal(pb)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeGroup reads a tag-delimited group from the Buffer.
|
||||||
|
// StartGroup tag is already consumed. This function consumes
|
||||||
|
// EndGroup tag.
|
||||||
|
func (p *Buffer) DecodeGroup(pb Message) error {
|
||||||
|
b := p.buf[p.index:]
|
||||||
|
x, y := findEndGroup(b)
|
||||||
|
if x < 0 {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
err := Unmarshal(b[:x], pb)
|
||||||
|
p.index += y
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unmarshal parses the protocol buffer representation in the
|
||||||
|
// Buffer and places the decoded result in pb. If the struct
|
||||||
|
// underlying pb does not match the data in the buffer, the results can be
|
||||||
|
// unpredictable.
|
||||||
|
//
|
||||||
|
// Unlike proto.Unmarshal, this does not reset pb before starting to unmarshal.
|
||||||
|
func (p *Buffer) Unmarshal(pb Message) error {
|
||||||
|
// If the object can unmarshal itself, let it.
|
||||||
|
if u, ok := pb.(newUnmarshaler); ok {
|
||||||
|
err := u.XXX_Unmarshal(p.buf[p.index:])
|
||||||
|
p.index = len(p.buf)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if u, ok := pb.(Unmarshaler); ok {
|
||||||
|
// NOTE: The history of proto have unfortunately been inconsistent
|
||||||
|
// whether Unmarshaler should or should not implicitly clear itself.
|
||||||
|
// Some implementations do, most do not.
|
||||||
|
// Thus, calling this here may or may not do what people want.
|
||||||
|
//
|
||||||
|
// See https://github.com/golang/protobuf/issues/424
|
||||||
|
err := u.Unmarshal(p.buf[p.index:])
|
||||||
|
p.index = len(p.buf)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Slow workaround for messages that aren't Unmarshalers.
|
||||||
|
// This includes some hand-coded .pb.go files and
|
||||||
|
// bootstrap protos.
|
||||||
|
// TODO: fix all of those and then add Unmarshal to
|
||||||
|
// the Message interface. Then:
|
||||||
|
// The cast above and code below can be deleted.
|
||||||
|
// The old unmarshaler can be deleted.
|
||||||
|
// Clients can call Unmarshal directly (can already do that, actually).
|
||||||
|
var info InternalMessageInfo
|
||||||
|
err := info.Unmarshal(pb, p.buf[p.index:])
|
||||||
|
p.index = len(p.buf)
|
||||||
|
return err
|
||||||
|
}
|
350
vendor/github.com/golang/protobuf/proto/discard.go
generated
vendored
Normal file
350
vendor/github.com/golang/protobuf/proto/discard.go
generated
vendored
Normal file
|
@ -0,0 +1,350 @@
|
||||||
|
// Go support for Protocol Buffers - Google's data interchange format
|
||||||
|
//
|
||||||
|
// Copyright 2017 The Go Authors. All rights reserved.
|
||||||
|
// https://github.com/golang/protobuf
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are
|
||||||
|
// met:
|
||||||
|
//
|
||||||
|
// * Redistributions of source code must retain the above copyright
|
||||||
|
// notice, this list of conditions and the following disclaimer.
|
||||||
|
// * Redistributions in binary form must reproduce the above
|
||||||
|
// copyright notice, this list of conditions and the following disclaimer
|
||||||
|
// in the documentation and/or other materials provided with the
|
||||||
|
// distribution.
|
||||||
|
// * Neither the name of Google Inc. nor the names of its
|
||||||
|
// contributors may be used to endorse or promote products derived from
|
||||||
|
// this software without specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
package proto
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"sync/atomic"
|
||||||
|
)
|
||||||
|
|
||||||
|
type generatedDiscarder interface {
|
||||||
|
XXX_DiscardUnknown()
|
||||||
|
}
|
||||||
|
|
||||||
|
// DiscardUnknown recursively discards all unknown fields from this message
|
||||||
|
// and all embedded messages.
|
||||||
|
//
|
||||||
|
// When unmarshaling a message with unrecognized fields, the tags and values
|
||||||
|
// of such fields are preserved in the Message. This allows a later call to
|
||||||
|
// marshal to be able to produce a message that continues to have those
|
||||||
|
// unrecognized fields. To avoid this, DiscardUnknown is used to
|
||||||
|
// explicitly clear the unknown fields after unmarshaling.
|
||||||
|
//
|
||||||
|
// For proto2 messages, the unknown fields of message extensions are only
|
||||||
|
// discarded from messages that have been accessed via GetExtension.
|
||||||
|
func DiscardUnknown(m Message) {
|
||||||
|
if m, ok := m.(generatedDiscarder); ok {
|
||||||
|
m.XXX_DiscardUnknown()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// TODO: Dynamically populate a InternalMessageInfo for legacy messages,
|
||||||
|
// but the master branch has no implementation for InternalMessageInfo,
|
||||||
|
// so it would be more work to replicate that approach.
|
||||||
|
discardLegacy(m)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DiscardUnknown recursively discards all unknown fields.
|
||||||
|
func (a *InternalMessageInfo) DiscardUnknown(m Message) {
|
||||||
|
di := atomicLoadDiscardInfo(&a.discard)
|
||||||
|
if di == nil {
|
||||||
|
di = getDiscardInfo(reflect.TypeOf(m).Elem())
|
||||||
|
atomicStoreDiscardInfo(&a.discard, di)
|
||||||
|
}
|
||||||
|
di.discard(toPointer(&m))
|
||||||
|
}
|
||||||
|
|
||||||
|
type discardInfo struct {
|
||||||
|
typ reflect.Type
|
||||||
|
|
||||||
|
initialized int32 // 0: only typ is valid, 1: everything is valid
|
||||||
|
lock sync.Mutex
|
||||||
|
|
||||||
|
fields []discardFieldInfo
|
||||||
|
unrecognized field
|
||||||
|
}
|
||||||
|
|
||||||
|
type discardFieldInfo struct {
|
||||||
|
field field // Offset of field, guaranteed to be valid
|
||||||
|
discard func(src pointer)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
discardInfoMap = map[reflect.Type]*discardInfo{}
|
||||||
|
discardInfoLock sync.Mutex
|
||||||
|
)
|
||||||
|
|
||||||
|
func getDiscardInfo(t reflect.Type) *discardInfo {
|
||||||
|
discardInfoLock.Lock()
|
||||||
|
defer discardInfoLock.Unlock()
|
||||||
|
di := discardInfoMap[t]
|
||||||
|
if di == nil {
|
||||||
|
di = &discardInfo{typ: t}
|
||||||
|
discardInfoMap[t] = di
|
||||||
|
}
|
||||||
|
return di
|
||||||
|
}
|
||||||
|
|
||||||
|
func (di *discardInfo) discard(src pointer) {
|
||||||
|
if src.isNil() {
|
||||||
|
return // Nothing to do.
|
||||||
|
}
|
||||||
|
|
||||||
|
if atomic.LoadInt32(&di.initialized) == 0 {
|
||||||
|
di.computeDiscardInfo()
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, fi := range di.fields {
|
||||||
|
sfp := src.offset(fi.field)
|
||||||
|
fi.discard(sfp)
|
||||||
|
}
|
||||||
|
|
||||||
|
// For proto2 messages, only discard unknown fields in message extensions
|
||||||
|
// that have been accessed via GetExtension.
|
||||||
|
if em, err := extendable(src.asPointerTo(di.typ).Interface()); err == nil {
|
||||||
|
// Ignore lock since DiscardUnknown is not concurrency safe.
|
||||||
|
emm, _ := em.extensionsRead()
|
||||||
|
for _, mx := range emm {
|
||||||
|
if m, ok := mx.value.(Message); ok {
|
||||||
|
DiscardUnknown(m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if di.unrecognized.IsValid() {
|
||||||
|
*src.offset(di.unrecognized).toBytes() = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (di *discardInfo) computeDiscardInfo() {
|
||||||
|
di.lock.Lock()
|
||||||
|
defer di.lock.Unlock()
|
||||||
|
if di.initialized != 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
t := di.typ
|
||||||
|
n := t.NumField()
|
||||||
|
|
||||||
|
for i := 0; i < n; i++ {
|
||||||
|
f := t.Field(i)
|
||||||
|
if strings.HasPrefix(f.Name, "XXX_") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
dfi := discardFieldInfo{field: toField(&f)}
|
||||||
|
tf := f.Type
|
||||||
|
|
||||||
|
// Unwrap tf to get its most basic type.
|
||||||
|
var isPointer, isSlice bool
|
||||||
|
if tf.Kind() == reflect.Slice && tf.Elem().Kind() != reflect.Uint8 {
|
||||||
|
isSlice = true
|
||||||
|
tf = tf.Elem()
|
||||||
|
}
|
||||||
|
if tf.Kind() == reflect.Ptr {
|
||||||
|
isPointer = true
|
||||||
|
tf = tf.Elem()
|
||||||
|
}
|
||||||
|
if isPointer && isSlice && tf.Kind() != reflect.Struct {
|
||||||
|
panic(fmt.Sprintf("%v.%s cannot be a slice of pointers to primitive types", t, f.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
switch tf.Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
switch {
|
||||||
|
case !isPointer:
|
||||||
|
panic(fmt.Sprintf("%v.%s cannot be a direct struct value", t, f.Name))
|
||||||
|
case isSlice: // E.g., []*pb.T
|
||||||
|
di := getDiscardInfo(tf)
|
||||||
|
dfi.discard = func(src pointer) {
|
||||||
|
sps := src.getPointerSlice()
|
||||||
|
for _, sp := range sps {
|
||||||
|
if !sp.isNil() {
|
||||||
|
di.discard(sp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default: // E.g., *pb.T
|
||||||
|
di := getDiscardInfo(tf)
|
||||||
|
dfi.discard = func(src pointer) {
|
||||||
|
sp := src.getPointer()
|
||||||
|
if !sp.isNil() {
|
||||||
|
di.discard(sp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.Map:
|
||||||
|
switch {
|
||||||
|
case isPointer || isSlice:
|
||||||
|
panic(fmt.Sprintf("%v.%s cannot be a pointer to a map or a slice of map values", t, f.Name))
|
||||||
|
default: // E.g., map[K]V
|
||||||
|
if tf.Elem().Kind() == reflect.Ptr { // Proto struct (e.g., *T)
|
||||||
|
dfi.discard = func(src pointer) {
|
||||||
|
sm := src.asPointerTo(tf).Elem()
|
||||||
|
if sm.Len() == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, key := range sm.MapKeys() {
|
||||||
|
val := sm.MapIndex(key)
|
||||||
|
DiscardUnknown(val.Interface().(Message))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
dfi.discard = func(pointer) {} // Noop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.Interface:
|
||||||
|
// Must be oneof field.
|
||||||
|
switch {
|
||||||
|
case isPointer || isSlice:
|
||||||
|
panic(fmt.Sprintf("%v.%s cannot be a pointer to a interface or a slice of interface values", t, f.Name))
|
||||||
|
default: // E.g., interface{}
|
||||||
|
// TODO: Make this faster?
|
||||||
|
dfi.discard = func(src pointer) {
|
||||||
|
su := src.asPointerTo(tf).Elem()
|
||||||
|
if !su.IsNil() {
|
||||||
|
sv := su.Elem().Elem().Field(0)
|
||||||
|
if sv.Kind() == reflect.Ptr && sv.IsNil() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
switch sv.Type().Kind() {
|
||||||
|
case reflect.Ptr: // Proto struct (e.g., *T)
|
||||||
|
DiscardUnknown(sv.Interface().(Message))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
di.fields = append(di.fields, dfi)
|
||||||
|
}
|
||||||
|
|
||||||
|
di.unrecognized = invalidField
|
||||||
|
if f, ok := t.FieldByName("XXX_unrecognized"); ok {
|
||||||
|
if f.Type != reflect.TypeOf([]byte{}) {
|
||||||
|
panic("expected XXX_unrecognized to be of type []byte")
|
||||||
|
}
|
||||||
|
di.unrecognized = toField(&f)
|
||||||
|
}
|
||||||
|
|
||||||
|
atomic.StoreInt32(&di.initialized, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func discardLegacy(m Message) {
|
||||||
|
v := reflect.ValueOf(m)
|
||||||
|
if v.Kind() != reflect.Ptr || v.IsNil() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
v = v.Elem()
|
||||||
|
if v.Kind() != reflect.Struct {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
t := v.Type()
|
||||||
|
|
||||||
|
for i := 0; i < v.NumField(); i++ {
|
||||||
|
f := t.Field(i)
|
||||||
|
if strings.HasPrefix(f.Name, "XXX_") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
vf := v.Field(i)
|
||||||
|
tf := f.Type
|
||||||
|
|
||||||
|
// Unwrap tf to get its most basic type.
|
||||||
|
var isPointer, isSlice bool
|
||||||
|
if tf.Kind() == reflect.Slice && tf.Elem().Kind() != reflect.Uint8 {
|
||||||
|
isSlice = true
|
||||||
|
tf = tf.Elem()
|
||||||
|
}
|
||||||
|
if tf.Kind() == reflect.Ptr {
|
||||||
|
isPointer = true
|
||||||
|
tf = tf.Elem()
|
||||||
|
}
|
||||||
|
if isPointer && isSlice && tf.Kind() != reflect.Struct {
|
||||||
|
panic(fmt.Sprintf("%T.%s cannot be a slice of pointers to primitive types", m, f.Name))
|
||||||
|
}
|
||||||
|
|
||||||
|
switch tf.Kind() {
|
||||||
|
case reflect.Struct:
|
||||||
|
switch {
|
||||||
|
case !isPointer:
|
||||||
|
panic(fmt.Sprintf("%T.%s cannot be a direct struct value", m, f.Name))
|
||||||
|
case isSlice: // E.g., []*pb.T
|
||||||
|
for j := 0; j < vf.Len(); j++ {
|
||||||
|
discardLegacy(vf.Index(j).Interface().(Message))
|
||||||
|
}
|
||||||
|
default: // E.g., *pb.T
|
||||||
|
discardLegacy(vf.Interface().(Message))
|
||||||
|
}
|
||||||
|
case reflect.Map:
|
||||||
|
switch {
|
||||||
|
case isPointer || isSlice:
|
||||||
|
panic(fmt.Sprintf("%T.%s cannot be a pointer to a map or a slice of map values", m, f.Name))
|
||||||
|
default: // E.g., map[K]V
|
||||||
|
tv := vf.Type().Elem()
|
||||||
|
if tv.Kind() == reflect.Ptr && tv.Implements(protoMessageType) { // Proto struct (e.g., *T)
|
||||||
|
for _, key := range vf.MapKeys() {
|
||||||
|
val := vf.MapIndex(key)
|
||||||
|
discardLegacy(val.Interface().(Message))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case reflect.Interface:
|
||||||
|
// Must be oneof field.
|
||||||
|
switch {
|
||||||
|
case isPointer || isSlice:
|
||||||
|
panic(fmt.Sprintf("%T.%s cannot be a pointer to a interface or a slice of interface values", m, f.Name))
|
||||||
|
default: // E.g., test_proto.isCommunique_Union interface
|
||||||
|
if !vf.IsNil() && f.Tag.Get("protobuf_oneof") != "" {
|
||||||
|
vf = vf.Elem() // E.g., *test_proto.Communique_Msg
|
||||||
|
if !vf.IsNil() {
|
||||||
|
vf = vf.Elem() // E.g., test_proto.Communique_Msg
|
||||||
|
vf = vf.Field(0) // E.g., Proto struct (e.g., *T) or primitive value
|
||||||
|
if vf.Kind() == reflect.Ptr {
|
||||||
|
discardLegacy(vf.Interface().(Message))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if vf := v.FieldByName("XXX_unrecognized"); vf.IsValid() {
|
||||||
|
if vf.Type() != reflect.TypeOf([]byte{}) {
|
||||||
|
panic("expected XXX_unrecognized to be of type []byte")
|
||||||
|
}
|
||||||
|
vf.Set(reflect.ValueOf([]byte(nil)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// For proto2 messages, only discard unknown fields in message extensions
|
||||||
|
// that have been accessed via GetExtension.
|
||||||
|
if em, err := extendable(m); err == nil {
|
||||||
|
// Ignore lock since discardLegacy is not concurrency safe.
|
||||||
|
emm, _ := em.extensionsRead()
|
||||||
|
for _, mx := range emm {
|
||||||
|
if m, ok := mx.value.(Message); ok {
|
||||||
|
discardLegacy(m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
203
vendor/github.com/golang/protobuf/proto/encode.go
generated
vendored
Normal file
203
vendor/github.com/golang/protobuf/proto/encode.go
generated
vendored
Normal file
|
@ -0,0 +1,203 @@
|
||||||
|
// Go support for Protocol Buffers - Google's data interchange format
|
||||||
|
//
|
||||||
|
// Copyright 2010 The Go Authors. All rights reserved.
|
||||||
|
// https://github.com/golang/protobuf
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are
|
||||||
|
// met:
|
||||||
|
//
|
||||||
|
// * Redistributions of source code must retain the above copyright
|
||||||
|
// notice, this list of conditions and the following disclaimer.
|
||||||
|
// * Redistributions in binary form must reproduce the above
|
||||||
|
// copyright notice, this list of conditions and the following disclaimer
|
||||||
|
// in the documentation and/or other materials provided with the
|
||||||
|
// distribution.
|
||||||
|
// * Neither the name of Google Inc. nor the names of its
|
||||||
|
// contributors may be used to endorse or promote products derived from
|
||||||
|
// this software without specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
package proto
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Routines for encoding data into the wire format for protocol buffers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// errRepeatedHasNil is the error returned if Marshal is called with
|
||||||
|
// a struct with a repeated field containing a nil element.
|
||||||
|
errRepeatedHasNil = errors.New("proto: repeated field has nil element")
|
||||||
|
|
||||||
|
// errOneofHasNil is the error returned if Marshal is called with
|
||||||
|
// a struct with a oneof field containing a nil element.
|
||||||
|
errOneofHasNil = errors.New("proto: oneof field has nil value")
|
||||||
|
|
||||||
|
// ErrNil is the error returned if Marshal is called with nil.
|
||||||
|
ErrNil = errors.New("proto: Marshal called with nil")
|
||||||
|
|
||||||
|
// ErrTooLarge is the error returned if Marshal is called with a
|
||||||
|
// message that encodes to >2GB.
|
||||||
|
ErrTooLarge = errors.New("proto: message encodes to over 2 GB")
|
||||||
|
)
|
||||||
|
|
||||||
|
// The fundamental encoders that put bytes on the wire.
|
||||||
|
// Those that take integer types all accept uint64 and are
|
||||||
|
// therefore of type valueEncoder.
|
||||||
|
|
||||||
|
const maxVarintBytes = 10 // maximum length of a varint
|
||||||
|
|
||||||
|
// EncodeVarint returns the varint encoding of x.
|
||||||
|
// This is the format for the
|
||||||
|
// int32, int64, uint32, uint64, bool, and enum
|
||||||
|
// protocol buffer types.
|
||||||
|
// Not used by the package itself, but helpful to clients
|
||||||
|
// wishing to use the same encoding.
|
||||||
|
func EncodeVarint(x uint64) []byte {
|
||||||
|
var buf [maxVarintBytes]byte
|
||||||
|
var n int
|
||||||
|
for n = 0; x > 127; n++ {
|
||||||
|
buf[n] = 0x80 | uint8(x&0x7F)
|
||||||
|
x >>= 7
|
||||||
|
}
|
||||||
|
buf[n] = uint8(x)
|
||||||
|
n++
|
||||||
|
return buf[0:n]
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeVarint writes a varint-encoded integer to the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// int32, int64, uint32, uint64, bool, and enum
|
||||||
|
// protocol buffer types.
|
||||||
|
func (p *Buffer) EncodeVarint(x uint64) error {
|
||||||
|
for x >= 1<<7 {
|
||||||
|
p.buf = append(p.buf, uint8(x&0x7f|0x80))
|
||||||
|
x >>= 7
|
||||||
|
}
|
||||||
|
p.buf = append(p.buf, uint8(x))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SizeVarint returns the varint encoding size of an integer.
|
||||||
|
func SizeVarint(x uint64) int {
|
||||||
|
switch {
|
||||||
|
case x < 1<<7:
|
||||||
|
return 1
|
||||||
|
case x < 1<<14:
|
||||||
|
return 2
|
||||||
|
case x < 1<<21:
|
||||||
|
return 3
|
||||||
|
case x < 1<<28:
|
||||||
|
return 4
|
||||||
|
case x < 1<<35:
|
||||||
|
return 5
|
||||||
|
case x < 1<<42:
|
||||||
|
return 6
|
||||||
|
case x < 1<<49:
|
||||||
|
return 7
|
||||||
|
case x < 1<<56:
|
||||||
|
return 8
|
||||||
|
case x < 1<<63:
|
||||||
|
return 9
|
||||||
|
}
|
||||||
|
return 10
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeFixed64 writes a 64-bit integer to the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// fixed64, sfixed64, and double protocol buffer types.
|
||||||
|
func (p *Buffer) EncodeFixed64(x uint64) error {
|
||||||
|
p.buf = append(p.buf,
|
||||||
|
uint8(x),
|
||||||
|
uint8(x>>8),
|
||||||
|
uint8(x>>16),
|
||||||
|
uint8(x>>24),
|
||||||
|
uint8(x>>32),
|
||||||
|
uint8(x>>40),
|
||||||
|
uint8(x>>48),
|
||||||
|
uint8(x>>56))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeFixed32 writes a 32-bit integer to the Buffer.
|
||||||
|
// This is the format for the
|
||||||
|
// fixed32, sfixed32, and float protocol buffer types.
|
||||||
|
func (p *Buffer) EncodeFixed32(x uint64) error {
|
||||||
|
p.buf = append(p.buf,
|
||||||
|
uint8(x),
|
||||||
|
uint8(x>>8),
|
||||||
|
uint8(x>>16),
|
||||||
|
uint8(x>>24))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeZigzag64 writes a zigzag-encoded 64-bit integer
|
||||||
|
// to the Buffer.
|
||||||
|
// This is the format used for the sint64 protocol buffer type.
|
||||||
|
func (p *Buffer) EncodeZigzag64(x uint64) error {
|
||||||
|
// use signed number to get arithmetic right shift.
|
||||||
|
return p.EncodeVarint(uint64((x << 1) ^ uint64((int64(x) >> 63))))
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeZigzag32 writes a zigzag-encoded 32-bit integer
|
||||||
|
// to the Buffer.
|
||||||
|
// This is the format used for the sint32 protocol buffer type.
|
||||||
|
func (p *Buffer) EncodeZigzag32(x uint64) error {
|
||||||
|
// use signed number to get arithmetic right shift.
|
||||||
|
return p.EncodeVarint(uint64((uint32(x) << 1) ^ uint32((int32(x) >> 31))))
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeRawBytes writes a count-delimited byte buffer to the Buffer.
|
||||||
|
// This is the format used for the bytes protocol buffer
|
||||||
|
// type and for embedded messages.
|
||||||
|
func (p *Buffer) EncodeRawBytes(b []byte) error {
|
||||||
|
p.EncodeVarint(uint64(len(b)))
|
||||||
|
p.buf = append(p.buf, b...)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeStringBytes writes an encoded string to the Buffer.
|
||||||
|
// This is the format used for the proto2 string type.
|
||||||
|
func (p *Buffer) EncodeStringBytes(s string) error {
|
||||||
|
p.EncodeVarint(uint64(len(s)))
|
||||||
|
p.buf = append(p.buf, s...)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Marshaler is the interface representing objects that can marshal themselves.
|
||||||
|
type Marshaler interface {
|
||||||
|
Marshal() ([]byte, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeMessage writes the protocol buffer to the Buffer,
|
||||||
|
// prefixed by a varint-encoded length.
|
||||||
|
func (p *Buffer) EncodeMessage(pb Message) error {
|
||||||
|
siz := Size(pb)
|
||||||
|
p.EncodeVarint(uint64(siz))
|
||||||
|
return p.Marshal(pb)
|
||||||
|
}
|
||||||
|
|
||||||
|
// All protocol buffer fields are nillable, but be careful.
|
||||||
|
func isNil(v reflect.Value) bool {
|
||||||
|
switch v.Kind() {
|
||||||
|
case reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice:
|
||||||
|
return v.IsNil()
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
300
vendor/github.com/golang/protobuf/proto/equal.go
generated
vendored
Normal file
300
vendor/github.com/golang/protobuf/proto/equal.go
generated
vendored
Normal file
|
@ -0,0 +1,300 @@
|
||||||
|
// Go support for Protocol Buffers - Google's data interchange format
|
||||||
|
//
|
||||||
|
// Copyright 2011 The Go Authors. All rights reserved.
|
||||||
|
// https://github.com/golang/protobuf
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are
|
||||||
|
// met:
|
||||||
|
//
|
||||||
|
// * Redistributions of source code must retain the above copyright
|
||||||
|
// notice, this list of conditions and the following disclaimer.
|
||||||
|
// * Redistributions in binary form must reproduce the above
|
||||||
|
// copyright notice, this list of conditions and the following disclaimer
|
||||||
|
// in the documentation and/or other materials provided with the
|
||||||
|
// distribution.
|
||||||
|
// * Neither the name of Google Inc. nor the names of its
|
||||||
|
// contributors may be used to endorse or promote products derived from
|
||||||
|
// this software without specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
// Protocol buffer comparison.
|
||||||
|
|
||||||
|
package proto
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"log"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
/*
|
||||||
|
Equal returns true iff protocol buffers a and b are equal.
|
||||||
|
The arguments must both be pointers to protocol buffer structs.
|
||||||
|
|
||||||
|
Equality is defined in this way:
|
||||||
|
- Two messages are equal iff they are the same type,
|
||||||
|
corresponding fields are equal, unknown field sets
|
||||||
|
are equal, and extensions sets are equal.
|
||||||
|
- Two set scalar fields are equal iff their values are equal.
|
||||||
|
If the fields are of a floating-point type, remember that
|
||||||
|
NaN != x for all x, including NaN. If the message is defined
|
||||||
|
in a proto3 .proto file, fields are not "set"; specifically,
|
||||||
|
zero length proto3 "bytes" fields are equal (nil == {}).
|
||||||
|
- Two repeated fields are equal iff their lengths are the same,
|
||||||
|
and their corresponding elements are equal. Note a "bytes" field,
|
||||||
|
although represented by []byte, is not a repeated field and the
|
||||||
|
rule for the scalar fields described above applies.
|
||||||
|
- Two unset fields are equal.
|
||||||
|
- Two unknown field sets are equal if their current
|
||||||
|
encoded state is equal.
|
||||||
|
- Two extension sets are equal iff they have corresponding
|
||||||
|
elements that are pairwise equal.
|
||||||
|
- Two map fields are equal iff their lengths are the same,
|
||||||
|
and they contain the same set of elements. Zero-length map
|
||||||
|
fields are equal.
|
||||||
|
- Every other combination of things are not equal.
|
||||||
|
|
||||||
|
The return value is undefined if a and b are not protocol buffers.
|
||||||
|
*/
|
||||||
|
func Equal(a, b Message) bool {
|
||||||
|
if a == nil || b == nil {
|
||||||
|
return a == b
|
||||||
|
}
|
||||||
|
v1, v2 := reflect.ValueOf(a), reflect.ValueOf(b)
|
||||||
|
if v1.Type() != v2.Type() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if v1.Kind() == reflect.Ptr {
|
||||||
|
if v1.IsNil() {
|
||||||
|
return v2.IsNil()
|
||||||
|
}
|
||||||
|
if v2.IsNil() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
v1, v2 = v1.Elem(), v2.Elem()
|
||||||
|
}
|
||||||
|
if v1.Kind() != reflect.Struct {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return equalStruct(v1, v2)
|
||||||
|
}
|
||||||
|
|
||||||
|
// v1 and v2 are known to have the same type.
|
||||||
|
func equalStruct(v1, v2 reflect.Value) bool {
|
||||||
|
sprop := GetProperties(v1.Type())
|
||||||
|
for i := 0; i < v1.NumField(); i++ {
|
||||||
|
f := v1.Type().Field(i)
|
||||||
|
if strings.HasPrefix(f.Name, "XXX_") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
f1, f2 := v1.Field(i), v2.Field(i)
|
||||||
|
if f.Type.Kind() == reflect.Ptr {
|
||||||
|
if n1, n2 := f1.IsNil(), f2.IsNil(); n1 && n2 {
|
||||||
|
// both unset
|
||||||
|
continue
|
||||||
|
} else if n1 != n2 {
|
||||||
|
// set/unset mismatch
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
f1, f2 = f1.Elem(), f2.Elem()
|
||||||
|
}
|
||||||
|
if !equalAny(f1, f2, sprop.Prop[i]) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if em1 := v1.FieldByName("XXX_InternalExtensions"); em1.IsValid() {
|
||||||
|
em2 := v2.FieldByName("XXX_InternalExtensions")
|
||||||
|
if !equalExtensions(v1.Type(), em1.Interface().(XXX_InternalExtensions), em2.Interface().(XXX_InternalExtensions)) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if em1 := v1.FieldByName("XXX_extensions"); em1.IsValid() {
|
||||||
|
em2 := v2.FieldByName("XXX_extensions")
|
||||||
|
if !equalExtMap(v1.Type(), em1.Interface().(map[int32]Extension), em2.Interface().(map[int32]Extension)) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uf := v1.FieldByName("XXX_unrecognized")
|
||||||
|
if !uf.IsValid() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
u1 := uf.Bytes()
|
||||||
|
u2 := v2.FieldByName("XXX_unrecognized").Bytes()
|
||||||
|
return bytes.Equal(u1, u2)
|
||||||
|
}
|
||||||
|
|
||||||
|
// v1 and v2 are known to have the same type.
|
||||||
|
// prop may be nil.
|
||||||
|
func equalAny(v1, v2 reflect.Value, prop *Properties) bool {
|
||||||
|
if v1.Type() == protoMessageType {
|
||||||
|
m1, _ := v1.Interface().(Message)
|
||||||
|
m2, _ := v2.Interface().(Message)
|
||||||
|
return Equal(m1, m2)
|
||||||
|
}
|
||||||
|
switch v1.Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
return v1.Bool() == v2.Bool()
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
return v1.Float() == v2.Float()
|
||||||
|
case reflect.Int32, reflect.Int64:
|
||||||
|
return v1.Int() == v2.Int()
|
||||||
|
case reflect.Interface:
|
||||||
|
// Probably a oneof field; compare the inner values.
|
||||||
|
n1, n2 := v1.IsNil(), v2.IsNil()
|
||||||
|
if n1 || n2 {
|
||||||
|
return n1 == n2
|
||||||
|
}
|
||||||
|
e1, e2 := v1.Elem(), v2.Elem()
|
||||||
|
if e1.Type() != e2.Type() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return equalAny(e1, e2, nil)
|
||||||
|
case reflect.Map:
|
||||||
|
if v1.Len() != v2.Len() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for _, key := range v1.MapKeys() {
|
||||||
|
val2 := v2.MapIndex(key)
|
||||||
|
if !val2.IsValid() {
|
||||||
|
// This key was not found in the second map.
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if !equalAny(v1.MapIndex(key), val2, nil) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
case reflect.Ptr:
|
||||||
|
// Maps may have nil values in them, so check for nil.
|
||||||
|
if v1.IsNil() && v2.IsNil() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if v1.IsNil() != v2.IsNil() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return equalAny(v1.Elem(), v2.Elem(), prop)
|
||||||
|
case reflect.Slice:
|
||||||
|
if v1.Type().Elem().Kind() == reflect.Uint8 {
|
||||||
|
// short circuit: []byte
|
||||||
|
|
||||||
|
// Edge case: if this is in a proto3 message, a zero length
|
||||||
|
// bytes field is considered the zero value.
|
||||||
|
if prop != nil && prop.proto3 && v1.Len() == 0 && v2.Len() == 0 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if v1.IsNil() != v2.IsNil() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return bytes.Equal(v1.Interface().([]byte), v2.Interface().([]byte))
|
||||||
|
}
|
||||||
|
|
||||||
|
if v1.Len() != v2.Len() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for i := 0; i < v1.Len(); i++ {
|
||||||
|
if !equalAny(v1.Index(i), v2.Index(i), prop) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
case reflect.String:
|
||||||
|
return v1.Interface().(string) == v2.Interface().(string)
|
||||||
|
case reflect.Struct:
|
||||||
|
return equalStruct(v1, v2)
|
||||||
|
case reflect.Uint32, reflect.Uint64:
|
||||||
|
return v1.Uint() == v2.Uint()
|
||||||
|
}
|
||||||
|
|
||||||
|
// unknown type, so not a protocol buffer
|
||||||
|
log.Printf("proto: don't know how to compare %v", v1)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// base is the struct type that the extensions are based on.
|
||||||
|
// x1 and x2 are InternalExtensions.
|
||||||
|
func equalExtensions(base reflect.Type, x1, x2 XXX_InternalExtensions) bool {
|
||||||
|
em1, _ := x1.extensionsRead()
|
||||||
|
em2, _ := x2.extensionsRead()
|
||||||
|
return equalExtMap(base, em1, em2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func equalExtMap(base reflect.Type, em1, em2 map[int32]Extension) bool {
|
||||||
|
if len(em1) != len(em2) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for extNum, e1 := range em1 {
|
||||||
|
e2, ok := em2[extNum]
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
m1, m2 := e1.value, e2.value
|
||||||
|
|
||||||
|
if m1 == nil && m2 == nil {
|
||||||
|
// Both have only encoded form.
|
||||||
|
if bytes.Equal(e1.enc, e2.enc) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// The bytes are different, but the extensions might still be
|
||||||
|
// equal. We need to decode them to compare.
|
||||||
|
}
|
||||||
|
|
||||||
|
if m1 != nil && m2 != nil {
|
||||||
|
// Both are unencoded.
|
||||||
|
if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2), nil) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// At least one is encoded. To do a semantically correct comparison
|
||||||
|
// we need to unmarshal them first.
|
||||||
|
var desc *ExtensionDesc
|
||||||
|
if m := extensionMaps[base]; m != nil {
|
||||||
|
desc = m[extNum]
|
||||||
|
}
|
||||||
|
if desc == nil {
|
||||||
|
// If both have only encoded form and the bytes are the same,
|
||||||
|
// it is handled above. We get here when the bytes are different.
|
||||||
|
// We don't know how to decode it, so just compare them as byte
|
||||||
|
// slices.
|
||||||
|
log.Printf("proto: don't know how to compare extension %d of %v", extNum, base)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
if m1 == nil {
|
||||||
|
m1, err = decodeExtension(e1.enc, desc)
|
||||||
|
}
|
||||||
|
if m2 == nil && err == nil {
|
||||||
|
m2, err = decodeExtension(e2.enc, desc)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
// The encoded form is invalid.
|
||||||
|
log.Printf("proto: badly encoded extension %d of %v: %v", extNum, base, err)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2), nil) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue