mirror of
https://github.com/prometheus/node_exporter.git
synced 2024-12-31 16:37:31 -08:00
Merge pull request #1782 from prometheus/superq/drop_vendor
Remove vendor directory
This commit is contained in:
commit
a4f45e823f
|
@ -17,6 +17,7 @@ jobs:
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- prometheus/setup_environment
|
- prometheus/setup_environment
|
||||||
|
- run: go mod download
|
||||||
- run: make
|
- run: make
|
||||||
- prometheus/store_artifact:
|
- prometheus/store_artifact:
|
||||||
file: node_exporter
|
file: node_exporter
|
||||||
|
|
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -34,3 +34,5 @@ dependencies-stamp
|
||||||
|
|
||||||
# Test files extracted from ttar
|
# Test files extracted from ttar
|
||||||
collector/fixtures/sys/
|
collector/fixtures/sys/
|
||||||
|
|
||||||
|
/vendor
|
||||||
|
|
|
@ -1,6 +1,3 @@
|
||||||
run:
|
|
||||||
modules-download-mode: vendor
|
|
||||||
|
|
||||||
linters:
|
linters:
|
||||||
enable:
|
enable:
|
||||||
- golint
|
- golint
|
||||||
|
|
|
@ -8,7 +8,7 @@ repository:
|
||||||
build:
|
build:
|
||||||
binaries:
|
binaries:
|
||||||
- name: node_exporter
|
- name: node_exporter
|
||||||
flags: -mod=vendor -a -tags 'netgo static_build'
|
flags: -a -tags 'netgo static_build'
|
||||||
ldflags: |
|
ldflags: |
|
||||||
-X github.com/prometheus/common/version.Version={{.Version}}
|
-X github.com/prometheus/common/version.Version={{.Version}}
|
||||||
-X github.com/prometheus/common/version.Revision={{.Revision}}
|
-X github.com/prometheus/common/version.Revision={{.Revision}}
|
||||||
|
|
|
@ -7,7 +7,7 @@ repository:
|
||||||
build:
|
build:
|
||||||
binaries:
|
binaries:
|
||||||
- name: node_exporter
|
- name: node_exporter
|
||||||
flags: -mod=vendor -a -tags 'netgo static_build'
|
flags: -a -tags 'netgo static_build'
|
||||||
ldflags: |
|
ldflags: |
|
||||||
-X github.com/prometheus/common/version.Version={{.Version}}
|
-X github.com/prometheus/common/version.Version={{.Version}}
|
||||||
-X github.com/prometheus/common/version.Revision={{.Revision}}
|
-X github.com/prometheus/common/version.Revision={{.Revision}}
|
||||||
|
|
27
vendor/github.com/alecthomas/template/LICENSE
generated
vendored
27
vendor/github.com/alecthomas/template/LICENSE
generated
vendored
|
@ -1,27 +0,0 @@
|
||||||
Copyright (c) 2012 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.
|
|
25
vendor/github.com/alecthomas/template/README.md
generated
vendored
25
vendor/github.com/alecthomas/template/README.md
generated
vendored
|
@ -1,25 +0,0 @@
|
||||||
# Go's `text/template` package with newline elision
|
|
||||||
|
|
||||||
This is a fork of Go 1.4's [text/template](http://golang.org/pkg/text/template/) package with one addition: a backslash immediately after a closing delimiter will delete all subsequent newlines until a non-newline.
|
|
||||||
|
|
||||||
eg.
|
|
||||||
|
|
||||||
```
|
|
||||||
{{if true}}\
|
|
||||||
hello
|
|
||||||
{{end}}\
|
|
||||||
```
|
|
||||||
|
|
||||||
Will result in:
|
|
||||||
|
|
||||||
```
|
|
||||||
hello\n
|
|
||||||
```
|
|
||||||
|
|
||||||
Rather than:
|
|
||||||
|
|
||||||
```
|
|
||||||
\n
|
|
||||||
hello\n
|
|
||||||
\n
|
|
||||||
```
|
|
406
vendor/github.com/alecthomas/template/doc.go
generated
vendored
406
vendor/github.com/alecthomas/template/doc.go
generated
vendored
|
@ -1,406 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
/*
|
|
||||||
Package template implements data-driven templates for generating textual output.
|
|
||||||
|
|
||||||
To generate HTML output, see package html/template, which has the same interface
|
|
||||||
as this package but automatically secures HTML output against certain attacks.
|
|
||||||
|
|
||||||
Templates are executed by applying them to a data structure. Annotations in the
|
|
||||||
template refer to elements of the data structure (typically a field of a struct
|
|
||||||
or a key in a map) to control execution and derive values to be displayed.
|
|
||||||
Execution of the template walks the structure and sets the cursor, represented
|
|
||||||
by a period '.' and called "dot", to the value at the current location in the
|
|
||||||
structure as execution proceeds.
|
|
||||||
|
|
||||||
The input text for a template is UTF-8-encoded text in any format.
|
|
||||||
"Actions"--data evaluations or control structures--are delimited by
|
|
||||||
"{{" and "}}"; all text outside actions is copied to the output unchanged.
|
|
||||||
Actions may not span newlines, although comments can.
|
|
||||||
|
|
||||||
Once parsed, a template may be executed safely in parallel.
|
|
||||||
|
|
||||||
Here is a trivial example that prints "17 items are made of wool".
|
|
||||||
|
|
||||||
type Inventory struct {
|
|
||||||
Material string
|
|
||||||
Count uint
|
|
||||||
}
|
|
||||||
sweaters := Inventory{"wool", 17}
|
|
||||||
tmpl, err := template.New("test").Parse("{{.Count}} items are made of {{.Material}}")
|
|
||||||
if err != nil { panic(err) }
|
|
||||||
err = tmpl.Execute(os.Stdout, sweaters)
|
|
||||||
if err != nil { panic(err) }
|
|
||||||
|
|
||||||
More intricate examples appear below.
|
|
||||||
|
|
||||||
Actions
|
|
||||||
|
|
||||||
Here is the list of actions. "Arguments" and "pipelines" are evaluations of
|
|
||||||
data, defined in detail below.
|
|
||||||
|
|
||||||
*/
|
|
||||||
// {{/* a comment */}}
|
|
||||||
// A comment; discarded. May contain newlines.
|
|
||||||
// Comments do not nest and must start and end at the
|
|
||||||
// delimiters, as shown here.
|
|
||||||
/*
|
|
||||||
|
|
||||||
{{pipeline}}
|
|
||||||
The default textual representation of the value of the pipeline
|
|
||||||
is copied to the output.
|
|
||||||
|
|
||||||
{{if pipeline}} T1 {{end}}
|
|
||||||
If the value of the pipeline is empty, no output is generated;
|
|
||||||
otherwise, T1 is executed. The empty values are false, 0, any
|
|
||||||
nil pointer or interface value, and any array, slice, map, or
|
|
||||||
string of length zero.
|
|
||||||
Dot is unaffected.
|
|
||||||
|
|
||||||
{{if pipeline}} T1 {{else}} T0 {{end}}
|
|
||||||
If the value of the pipeline is empty, T0 is executed;
|
|
||||||
otherwise, T1 is executed. Dot is unaffected.
|
|
||||||
|
|
||||||
{{if pipeline}} T1 {{else if pipeline}} T0 {{end}}
|
|
||||||
To simplify the appearance of if-else chains, the else action
|
|
||||||
of an if may include another if directly; the effect is exactly
|
|
||||||
the same as writing
|
|
||||||
{{if pipeline}} T1 {{else}}{{if pipeline}} T0 {{end}}{{end}}
|
|
||||||
|
|
||||||
{{range pipeline}} T1 {{end}}
|
|
||||||
The value of the pipeline must be an array, slice, map, or channel.
|
|
||||||
If the value of the pipeline has length zero, nothing is output;
|
|
||||||
otherwise, dot is set to the successive elements of the array,
|
|
||||||
slice, or map and T1 is executed. If the value is a map and the
|
|
||||||
keys are of basic type with a defined order ("comparable"), the
|
|
||||||
elements will be visited in sorted key order.
|
|
||||||
|
|
||||||
{{range pipeline}} T1 {{else}} T0 {{end}}
|
|
||||||
The value of the pipeline must be an array, slice, map, or channel.
|
|
||||||
If the value of the pipeline has length zero, dot is unaffected and
|
|
||||||
T0 is executed; otherwise, dot is set to the successive elements
|
|
||||||
of the array, slice, or map and T1 is executed.
|
|
||||||
|
|
||||||
{{template "name"}}
|
|
||||||
The template with the specified name is executed with nil data.
|
|
||||||
|
|
||||||
{{template "name" pipeline}}
|
|
||||||
The template with the specified name is executed with dot set
|
|
||||||
to the value of the pipeline.
|
|
||||||
|
|
||||||
{{with pipeline}} T1 {{end}}
|
|
||||||
If the value of the pipeline is empty, no output is generated;
|
|
||||||
otherwise, dot is set to the value of the pipeline and T1 is
|
|
||||||
executed.
|
|
||||||
|
|
||||||
{{with pipeline}} T1 {{else}} T0 {{end}}
|
|
||||||
If the value of the pipeline is empty, dot is unaffected and T0
|
|
||||||
is executed; otherwise, dot is set to the value of the pipeline
|
|
||||||
and T1 is executed.
|
|
||||||
|
|
||||||
Arguments
|
|
||||||
|
|
||||||
An argument is a simple value, denoted by one of the following.
|
|
||||||
|
|
||||||
- A boolean, string, character, integer, floating-point, imaginary
|
|
||||||
or complex constant in Go syntax. These behave like Go's untyped
|
|
||||||
constants, although raw strings may not span newlines.
|
|
||||||
- The keyword nil, representing an untyped Go nil.
|
|
||||||
- The character '.' (period):
|
|
||||||
.
|
|
||||||
The result is the value of dot.
|
|
||||||
- A variable name, which is a (possibly empty) alphanumeric string
|
|
||||||
preceded by a dollar sign, such as
|
|
||||||
$piOver2
|
|
||||||
or
|
|
||||||
$
|
|
||||||
The result is the value of the variable.
|
|
||||||
Variables are described below.
|
|
||||||
- The name of a field of the data, which must be a struct, preceded
|
|
||||||
by a period, such as
|
|
||||||
.Field
|
|
||||||
The result is the value of the field. Field invocations may be
|
|
||||||
chained:
|
|
||||||
.Field1.Field2
|
|
||||||
Fields can also be evaluated on variables, including chaining:
|
|
||||||
$x.Field1.Field2
|
|
||||||
- The name of a key of the data, which must be a map, preceded
|
|
||||||
by a period, such as
|
|
||||||
.Key
|
|
||||||
The result is the map element value indexed by the key.
|
|
||||||
Key invocations may be chained and combined with fields to any
|
|
||||||
depth:
|
|
||||||
.Field1.Key1.Field2.Key2
|
|
||||||
Although the key must be an alphanumeric identifier, unlike with
|
|
||||||
field names they do not need to start with an upper case letter.
|
|
||||||
Keys can also be evaluated on variables, including chaining:
|
|
||||||
$x.key1.key2
|
|
||||||
- The name of a niladic method of the data, preceded by a period,
|
|
||||||
such as
|
|
||||||
.Method
|
|
||||||
The result is the value of invoking the method with dot as the
|
|
||||||
receiver, dot.Method(). Such a method must have one return value (of
|
|
||||||
any type) or two return values, the second of which is an error.
|
|
||||||
If it has two and the returned error is non-nil, execution terminates
|
|
||||||
and an error is returned to the caller as the value of Execute.
|
|
||||||
Method invocations may be chained and combined with fields and keys
|
|
||||||
to any depth:
|
|
||||||
.Field1.Key1.Method1.Field2.Key2.Method2
|
|
||||||
Methods can also be evaluated on variables, including chaining:
|
|
||||||
$x.Method1.Field
|
|
||||||
- The name of a niladic function, such as
|
|
||||||
fun
|
|
||||||
The result is the value of invoking the function, fun(). The return
|
|
||||||
types and values behave as in methods. Functions and function
|
|
||||||
names are described below.
|
|
||||||
- A parenthesized instance of one the above, for grouping. The result
|
|
||||||
may be accessed by a field or map key invocation.
|
|
||||||
print (.F1 arg1) (.F2 arg2)
|
|
||||||
(.StructValuedMethod "arg").Field
|
|
||||||
|
|
||||||
Arguments may evaluate to any type; if they are pointers the implementation
|
|
||||||
automatically indirects to the base type when required.
|
|
||||||
If an evaluation yields a function value, such as a function-valued
|
|
||||||
field of a struct, the function is not invoked automatically, but it
|
|
||||||
can be used as a truth value for an if action and the like. To invoke
|
|
||||||
it, use the call function, defined below.
|
|
||||||
|
|
||||||
A pipeline is a possibly chained sequence of "commands". A command is a simple
|
|
||||||
value (argument) or a function or method call, possibly with multiple arguments:
|
|
||||||
|
|
||||||
Argument
|
|
||||||
The result is the value of evaluating the argument.
|
|
||||||
.Method [Argument...]
|
|
||||||
The method can be alone or the last element of a chain but,
|
|
||||||
unlike methods in the middle of a chain, it can take arguments.
|
|
||||||
The result is the value of calling the method with the
|
|
||||||
arguments:
|
|
||||||
dot.Method(Argument1, etc.)
|
|
||||||
functionName [Argument...]
|
|
||||||
The result is the value of calling the function associated
|
|
||||||
with the name:
|
|
||||||
function(Argument1, etc.)
|
|
||||||
Functions and function names are described below.
|
|
||||||
|
|
||||||
Pipelines
|
|
||||||
|
|
||||||
A pipeline may be "chained" by separating a sequence of commands with pipeline
|
|
||||||
characters '|'. In a chained pipeline, the result of the each command is
|
|
||||||
passed as the last argument of the following command. The output of the final
|
|
||||||
command in the pipeline is the value of the pipeline.
|
|
||||||
|
|
||||||
The output of a command will be either one value or two values, the second of
|
|
||||||
which has type error. If that second value is present and evaluates to
|
|
||||||
non-nil, execution terminates and the error is returned to the caller of
|
|
||||||
Execute.
|
|
||||||
|
|
||||||
Variables
|
|
||||||
|
|
||||||
A pipeline inside an action may initialize a variable to capture the result.
|
|
||||||
The initialization has syntax
|
|
||||||
|
|
||||||
$variable := pipeline
|
|
||||||
|
|
||||||
where $variable is the name of the variable. An action that declares a
|
|
||||||
variable produces no output.
|
|
||||||
|
|
||||||
If a "range" action initializes a variable, the variable is set to the
|
|
||||||
successive elements of the iteration. Also, a "range" may declare two
|
|
||||||
variables, separated by a comma:
|
|
||||||
|
|
||||||
range $index, $element := pipeline
|
|
||||||
|
|
||||||
in which case $index and $element are set to the successive values of the
|
|
||||||
array/slice index or map key and element, respectively. Note that if there is
|
|
||||||
only one variable, it is assigned the element; this is opposite to the
|
|
||||||
convention in Go range clauses.
|
|
||||||
|
|
||||||
A variable's scope extends to the "end" action of the control structure ("if",
|
|
||||||
"with", or "range") in which it is declared, or to the end of the template if
|
|
||||||
there is no such control structure. A template invocation does not inherit
|
|
||||||
variables from the point of its invocation.
|
|
||||||
|
|
||||||
When execution begins, $ is set to the data argument passed to Execute, that is,
|
|
||||||
to the starting value of dot.
|
|
||||||
|
|
||||||
Examples
|
|
||||||
|
|
||||||
Here are some example one-line templates demonstrating pipelines and variables.
|
|
||||||
All produce the quoted word "output":
|
|
||||||
|
|
||||||
{{"\"output\""}}
|
|
||||||
A string constant.
|
|
||||||
{{`"output"`}}
|
|
||||||
A raw string constant.
|
|
||||||
{{printf "%q" "output"}}
|
|
||||||
A function call.
|
|
||||||
{{"output" | printf "%q"}}
|
|
||||||
A function call whose final argument comes from the previous
|
|
||||||
command.
|
|
||||||
{{printf "%q" (print "out" "put")}}
|
|
||||||
A parenthesized argument.
|
|
||||||
{{"put" | printf "%s%s" "out" | printf "%q"}}
|
|
||||||
A more elaborate call.
|
|
||||||
{{"output" | printf "%s" | printf "%q"}}
|
|
||||||
A longer chain.
|
|
||||||
{{with "output"}}{{printf "%q" .}}{{end}}
|
|
||||||
A with action using dot.
|
|
||||||
{{with $x := "output" | printf "%q"}}{{$x}}{{end}}
|
|
||||||
A with action that creates and uses a variable.
|
|
||||||
{{with $x := "output"}}{{printf "%q" $x}}{{end}}
|
|
||||||
A with action that uses the variable in another action.
|
|
||||||
{{with $x := "output"}}{{$x | printf "%q"}}{{end}}
|
|
||||||
The same, but pipelined.
|
|
||||||
|
|
||||||
Functions
|
|
||||||
|
|
||||||
During execution functions are found in two function maps: first in the
|
|
||||||
template, then in the global function map. By default, no functions are defined
|
|
||||||
in the template but the Funcs method can be used to add them.
|
|
||||||
|
|
||||||
Predefined global functions are named as follows.
|
|
||||||
|
|
||||||
and
|
|
||||||
Returns the boolean AND of its arguments by returning the
|
|
||||||
first empty argument or the last argument, that is,
|
|
||||||
"and x y" behaves as "if x then y else x". All the
|
|
||||||
arguments are evaluated.
|
|
||||||
call
|
|
||||||
Returns the result of calling the first argument, which
|
|
||||||
must be a function, with the remaining arguments as parameters.
|
|
||||||
Thus "call .X.Y 1 2" is, in Go notation, dot.X.Y(1, 2) where
|
|
||||||
Y is a func-valued field, map entry, or the like.
|
|
||||||
The first argument must be the result of an evaluation
|
|
||||||
that yields a value of function type (as distinct from
|
|
||||||
a predefined function such as print). The function must
|
|
||||||
return either one or two result values, the second of which
|
|
||||||
is of type error. If the arguments don't match the function
|
|
||||||
or the returned error value is non-nil, execution stops.
|
|
||||||
html
|
|
||||||
Returns the escaped HTML equivalent of the textual
|
|
||||||
representation of its arguments.
|
|
||||||
index
|
|
||||||
Returns the result of indexing its first argument by the
|
|
||||||
following arguments. Thus "index x 1 2 3" is, in Go syntax,
|
|
||||||
x[1][2][3]. Each indexed item must be a map, slice, or array.
|
|
||||||
js
|
|
||||||
Returns the escaped JavaScript equivalent of the textual
|
|
||||||
representation of its arguments.
|
|
||||||
len
|
|
||||||
Returns the integer length of its argument.
|
|
||||||
not
|
|
||||||
Returns the boolean negation of its single argument.
|
|
||||||
or
|
|
||||||
Returns the boolean OR of its arguments by returning the
|
|
||||||
first non-empty argument or the last argument, that is,
|
|
||||||
"or x y" behaves as "if x then x else y". All the
|
|
||||||
arguments are evaluated.
|
|
||||||
print
|
|
||||||
An alias for fmt.Sprint
|
|
||||||
printf
|
|
||||||
An alias for fmt.Sprintf
|
|
||||||
println
|
|
||||||
An alias for fmt.Sprintln
|
|
||||||
urlquery
|
|
||||||
Returns the escaped value of the textual representation of
|
|
||||||
its arguments in a form suitable for embedding in a URL query.
|
|
||||||
|
|
||||||
The boolean functions take any zero value to be false and a non-zero
|
|
||||||
value to be true.
|
|
||||||
|
|
||||||
There is also a set of binary comparison operators defined as
|
|
||||||
functions:
|
|
||||||
|
|
||||||
eq
|
|
||||||
Returns the boolean truth of arg1 == arg2
|
|
||||||
ne
|
|
||||||
Returns the boolean truth of arg1 != arg2
|
|
||||||
lt
|
|
||||||
Returns the boolean truth of arg1 < arg2
|
|
||||||
le
|
|
||||||
Returns the boolean truth of arg1 <= arg2
|
|
||||||
gt
|
|
||||||
Returns the boolean truth of arg1 > arg2
|
|
||||||
ge
|
|
||||||
Returns the boolean truth of arg1 >= arg2
|
|
||||||
|
|
||||||
For simpler multi-way equality tests, eq (only) accepts two or more
|
|
||||||
arguments and compares the second and subsequent to the first,
|
|
||||||
returning in effect
|
|
||||||
|
|
||||||
arg1==arg2 || arg1==arg3 || arg1==arg4 ...
|
|
||||||
|
|
||||||
(Unlike with || in Go, however, eq is a function call and all the
|
|
||||||
arguments will be evaluated.)
|
|
||||||
|
|
||||||
The comparison functions work on basic types only (or named basic
|
|
||||||
types, such as "type Celsius float32"). They implement the Go rules
|
|
||||||
for comparison of values, except that size and exact type are
|
|
||||||
ignored, so any integer value, signed or unsigned, may be compared
|
|
||||||
with any other integer value. (The arithmetic value is compared,
|
|
||||||
not the bit pattern, so all negative integers are less than all
|
|
||||||
unsigned integers.) However, as usual, one may not compare an int
|
|
||||||
with a float32 and so on.
|
|
||||||
|
|
||||||
Associated templates
|
|
||||||
|
|
||||||
Each template is named by a string specified when it is created. Also, each
|
|
||||||
template is associated with zero or more other templates that it may invoke by
|
|
||||||
name; such associations are transitive and form a name space of templates.
|
|
||||||
|
|
||||||
A template may use a template invocation to instantiate another associated
|
|
||||||
template; see the explanation of the "template" action above. The name must be
|
|
||||||
that of a template associated with the template that contains the invocation.
|
|
||||||
|
|
||||||
Nested template definitions
|
|
||||||
|
|
||||||
When parsing a template, another template may be defined and associated with the
|
|
||||||
template being parsed. Template definitions must appear at the top level of the
|
|
||||||
template, much like global variables in a Go program.
|
|
||||||
|
|
||||||
The syntax of such definitions is to surround each template declaration with a
|
|
||||||
"define" and "end" action.
|
|
||||||
|
|
||||||
The define action names the template being created by providing a string
|
|
||||||
constant. Here is a simple example:
|
|
||||||
|
|
||||||
`{{define "T1"}}ONE{{end}}
|
|
||||||
{{define "T2"}}TWO{{end}}
|
|
||||||
{{define "T3"}}{{template "T1"}} {{template "T2"}}{{end}}
|
|
||||||
{{template "T3"}}`
|
|
||||||
|
|
||||||
This defines two templates, T1 and T2, and a third T3 that invokes the other two
|
|
||||||
when it is executed. Finally it invokes T3. If executed this template will
|
|
||||||
produce the text
|
|
||||||
|
|
||||||
ONE TWO
|
|
||||||
|
|
||||||
By construction, a template may reside in only one association. If it's
|
|
||||||
necessary to have a template addressable from multiple associations, the
|
|
||||||
template definition must be parsed multiple times to create distinct *Template
|
|
||||||
values, or must be copied with the Clone or AddParseTree method.
|
|
||||||
|
|
||||||
Parse may be called multiple times to assemble the various associated templates;
|
|
||||||
see the ParseFiles and ParseGlob functions and methods for simple ways to parse
|
|
||||||
related templates stored in files.
|
|
||||||
|
|
||||||
A template may be executed directly or through ExecuteTemplate, which executes
|
|
||||||
an associated template identified by name. To invoke our example above, we
|
|
||||||
might write,
|
|
||||||
|
|
||||||
err := tmpl.Execute(os.Stdout, "no data needed")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("execution failed: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
or to invoke a particular template explicitly by name,
|
|
||||||
|
|
||||||
err := tmpl.ExecuteTemplate(os.Stdout, "T2", "no data needed")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("execution failed: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
*/
|
|
||||||
package template
|
|
845
vendor/github.com/alecthomas/template/exec.go
generated
vendored
845
vendor/github.com/alecthomas/template/exec.go
generated
vendored
|
@ -1,845 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package template
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"reflect"
|
|
||||||
"runtime"
|
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/alecthomas/template/parse"
|
|
||||||
)
|
|
||||||
|
|
||||||
// state represents the state of an execution. It's not part of the
|
|
||||||
// template so that multiple executions of the same template
|
|
||||||
// can execute in parallel.
|
|
||||||
type state struct {
|
|
||||||
tmpl *Template
|
|
||||||
wr io.Writer
|
|
||||||
node parse.Node // current node, for errors
|
|
||||||
vars []variable // push-down stack of variable values.
|
|
||||||
}
|
|
||||||
|
|
||||||
// variable holds the dynamic value of a variable such as $, $x etc.
|
|
||||||
type variable struct {
|
|
||||||
name string
|
|
||||||
value reflect.Value
|
|
||||||
}
|
|
||||||
|
|
||||||
// push pushes a new variable on the stack.
|
|
||||||
func (s *state) push(name string, value reflect.Value) {
|
|
||||||
s.vars = append(s.vars, variable{name, value})
|
|
||||||
}
|
|
||||||
|
|
||||||
// mark returns the length of the variable stack.
|
|
||||||
func (s *state) mark() int {
|
|
||||||
return len(s.vars)
|
|
||||||
}
|
|
||||||
|
|
||||||
// pop pops the variable stack up to the mark.
|
|
||||||
func (s *state) pop(mark int) {
|
|
||||||
s.vars = s.vars[0:mark]
|
|
||||||
}
|
|
||||||
|
|
||||||
// setVar overwrites the top-nth variable on the stack. Used by range iterations.
|
|
||||||
func (s *state) setVar(n int, value reflect.Value) {
|
|
||||||
s.vars[len(s.vars)-n].value = value
|
|
||||||
}
|
|
||||||
|
|
||||||
// varValue returns the value of the named variable.
|
|
||||||
func (s *state) varValue(name string) reflect.Value {
|
|
||||||
for i := s.mark() - 1; i >= 0; i-- {
|
|
||||||
if s.vars[i].name == name {
|
|
||||||
return s.vars[i].value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
s.errorf("undefined variable: %s", name)
|
|
||||||
return zero
|
|
||||||
}
|
|
||||||
|
|
||||||
var zero reflect.Value
|
|
||||||
|
|
||||||
// at marks the state to be on node n, for error reporting.
|
|
||||||
func (s *state) at(node parse.Node) {
|
|
||||||
s.node = node
|
|
||||||
}
|
|
||||||
|
|
||||||
// doublePercent returns the string with %'s replaced by %%, if necessary,
|
|
||||||
// so it can be used safely inside a Printf format string.
|
|
||||||
func doublePercent(str string) string {
|
|
||||||
if strings.Contains(str, "%") {
|
|
||||||
str = strings.Replace(str, "%", "%%", -1)
|
|
||||||
}
|
|
||||||
return str
|
|
||||||
}
|
|
||||||
|
|
||||||
// errorf formats the error and terminates processing.
|
|
||||||
func (s *state) errorf(format string, args ...interface{}) {
|
|
||||||
name := doublePercent(s.tmpl.Name())
|
|
||||||
if s.node == nil {
|
|
||||||
format = fmt.Sprintf("template: %s: %s", name, format)
|
|
||||||
} else {
|
|
||||||
location, context := s.tmpl.ErrorContext(s.node)
|
|
||||||
format = fmt.Sprintf("template: %s: executing %q at <%s>: %s", location, name, doublePercent(context), format)
|
|
||||||
}
|
|
||||||
panic(fmt.Errorf(format, args...))
|
|
||||||
}
|
|
||||||
|
|
||||||
// errRecover is the handler that turns panics into returns from the top
|
|
||||||
// level of Parse.
|
|
||||||
func errRecover(errp *error) {
|
|
||||||
e := recover()
|
|
||||||
if e != nil {
|
|
||||||
switch err := e.(type) {
|
|
||||||
case runtime.Error:
|
|
||||||
panic(e)
|
|
||||||
case error:
|
|
||||||
*errp = err
|
|
||||||
default:
|
|
||||||
panic(e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ExecuteTemplate applies the template associated with t that has the given name
|
|
||||||
// to the specified data object and writes the output to wr.
|
|
||||||
// If an error occurs executing the template or writing its output,
|
|
||||||
// execution stops, but partial results may already have been written to
|
|
||||||
// the output writer.
|
|
||||||
// A template may be executed safely in parallel.
|
|
||||||
func (t *Template) ExecuteTemplate(wr io.Writer, name string, data interface{}) error {
|
|
||||||
tmpl := t.tmpl[name]
|
|
||||||
if tmpl == nil {
|
|
||||||
return fmt.Errorf("template: no template %q associated with template %q", name, t.name)
|
|
||||||
}
|
|
||||||
return tmpl.Execute(wr, data)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Execute applies a parsed template to the specified data object,
|
|
||||||
// and writes the output to wr.
|
|
||||||
// If an error occurs executing the template or writing its output,
|
|
||||||
// execution stops, but partial results may already have been written to
|
|
||||||
// the output writer.
|
|
||||||
// A template may be executed safely in parallel.
|
|
||||||
func (t *Template) Execute(wr io.Writer, data interface{}) (err error) {
|
|
||||||
defer errRecover(&err)
|
|
||||||
value := reflect.ValueOf(data)
|
|
||||||
state := &state{
|
|
||||||
tmpl: t,
|
|
||||||
wr: wr,
|
|
||||||
vars: []variable{{"$", value}},
|
|
||||||
}
|
|
||||||
t.init()
|
|
||||||
if t.Tree == nil || t.Root == nil {
|
|
||||||
var b bytes.Buffer
|
|
||||||
for name, tmpl := range t.tmpl {
|
|
||||||
if tmpl.Tree == nil || tmpl.Root == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if b.Len() > 0 {
|
|
||||||
b.WriteString(", ")
|
|
||||||
}
|
|
||||||
fmt.Fprintf(&b, "%q", name)
|
|
||||||
}
|
|
||||||
var s string
|
|
||||||
if b.Len() > 0 {
|
|
||||||
s = "; defined templates are: " + b.String()
|
|
||||||
}
|
|
||||||
state.errorf("%q is an incomplete or empty template%s", t.Name(), s)
|
|
||||||
}
|
|
||||||
state.walk(value, t.Root)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Walk functions step through the major pieces of the template structure,
|
|
||||||
// generating output as they go.
|
|
||||||
func (s *state) walk(dot reflect.Value, node parse.Node) {
|
|
||||||
s.at(node)
|
|
||||||
switch node := node.(type) {
|
|
||||||
case *parse.ActionNode:
|
|
||||||
// Do not pop variables so they persist until next end.
|
|
||||||
// Also, if the action declares variables, don't print the result.
|
|
||||||
val := s.evalPipeline(dot, node.Pipe)
|
|
||||||
if len(node.Pipe.Decl) == 0 {
|
|
||||||
s.printValue(node, val)
|
|
||||||
}
|
|
||||||
case *parse.IfNode:
|
|
||||||
s.walkIfOrWith(parse.NodeIf, dot, node.Pipe, node.List, node.ElseList)
|
|
||||||
case *parse.ListNode:
|
|
||||||
for _, node := range node.Nodes {
|
|
||||||
s.walk(dot, node)
|
|
||||||
}
|
|
||||||
case *parse.RangeNode:
|
|
||||||
s.walkRange(dot, node)
|
|
||||||
case *parse.TemplateNode:
|
|
||||||
s.walkTemplate(dot, node)
|
|
||||||
case *parse.TextNode:
|
|
||||||
if _, err := s.wr.Write(node.Text); err != nil {
|
|
||||||
s.errorf("%s", err)
|
|
||||||
}
|
|
||||||
case *parse.WithNode:
|
|
||||||
s.walkIfOrWith(parse.NodeWith, dot, node.Pipe, node.List, node.ElseList)
|
|
||||||
default:
|
|
||||||
s.errorf("unknown node: %s", node)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// walkIfOrWith walks an 'if' or 'with' node. The two control structures
|
|
||||||
// are identical in behavior except that 'with' sets dot.
|
|
||||||
func (s *state) walkIfOrWith(typ parse.NodeType, dot reflect.Value, pipe *parse.PipeNode, list, elseList *parse.ListNode) {
|
|
||||||
defer s.pop(s.mark())
|
|
||||||
val := s.evalPipeline(dot, pipe)
|
|
||||||
truth, ok := isTrue(val)
|
|
||||||
if !ok {
|
|
||||||
s.errorf("if/with can't use %v", val)
|
|
||||||
}
|
|
||||||
if truth {
|
|
||||||
if typ == parse.NodeWith {
|
|
||||||
s.walk(val, list)
|
|
||||||
} else {
|
|
||||||
s.walk(dot, list)
|
|
||||||
}
|
|
||||||
} else if elseList != nil {
|
|
||||||
s.walk(dot, elseList)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// isTrue reports whether the value is 'true', in the sense of not the zero of its type,
|
|
||||||
// and whether the value has a meaningful truth value.
|
|
||||||
func isTrue(val reflect.Value) (truth, ok bool) {
|
|
||||||
if !val.IsValid() {
|
|
||||||
// Something like var x interface{}, never set. It's a form of nil.
|
|
||||||
return false, true
|
|
||||||
}
|
|
||||||
switch val.Kind() {
|
|
||||||
case reflect.Array, reflect.Map, reflect.Slice, reflect.String:
|
|
||||||
truth = val.Len() > 0
|
|
||||||
case reflect.Bool:
|
|
||||||
truth = val.Bool()
|
|
||||||
case reflect.Complex64, reflect.Complex128:
|
|
||||||
truth = val.Complex() != 0
|
|
||||||
case reflect.Chan, reflect.Func, reflect.Ptr, reflect.Interface:
|
|
||||||
truth = !val.IsNil()
|
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
||||||
truth = val.Int() != 0
|
|
||||||
case reflect.Float32, reflect.Float64:
|
|
||||||
truth = val.Float() != 0
|
|
||||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
||||||
truth = val.Uint() != 0
|
|
||||||
case reflect.Struct:
|
|
||||||
truth = true // Struct values are always true.
|
|
||||||
default:
|
|
||||||
return
|
|
||||||
}
|
|
||||||
return truth, true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) walkRange(dot reflect.Value, r *parse.RangeNode) {
|
|
||||||
s.at(r)
|
|
||||||
defer s.pop(s.mark())
|
|
||||||
val, _ := indirect(s.evalPipeline(dot, r.Pipe))
|
|
||||||
// mark top of stack before any variables in the body are pushed.
|
|
||||||
mark := s.mark()
|
|
||||||
oneIteration := func(index, elem reflect.Value) {
|
|
||||||
// Set top var (lexically the second if there are two) to the element.
|
|
||||||
if len(r.Pipe.Decl) > 0 {
|
|
||||||
s.setVar(1, elem)
|
|
||||||
}
|
|
||||||
// Set next var (lexically the first if there are two) to the index.
|
|
||||||
if len(r.Pipe.Decl) > 1 {
|
|
||||||
s.setVar(2, index)
|
|
||||||
}
|
|
||||||
s.walk(elem, r.List)
|
|
||||||
s.pop(mark)
|
|
||||||
}
|
|
||||||
switch val.Kind() {
|
|
||||||
case reflect.Array, reflect.Slice:
|
|
||||||
if val.Len() == 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
for i := 0; i < val.Len(); i++ {
|
|
||||||
oneIteration(reflect.ValueOf(i), val.Index(i))
|
|
||||||
}
|
|
||||||
return
|
|
||||||
case reflect.Map:
|
|
||||||
if val.Len() == 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
for _, key := range sortKeys(val.MapKeys()) {
|
|
||||||
oneIteration(key, val.MapIndex(key))
|
|
||||||
}
|
|
||||||
return
|
|
||||||
case reflect.Chan:
|
|
||||||
if val.IsNil() {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
i := 0
|
|
||||||
for ; ; i++ {
|
|
||||||
elem, ok := val.Recv()
|
|
||||||
if !ok {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
oneIteration(reflect.ValueOf(i), elem)
|
|
||||||
}
|
|
||||||
if i == 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
return
|
|
||||||
case reflect.Invalid:
|
|
||||||
break // An invalid value is likely a nil map, etc. and acts like an empty map.
|
|
||||||
default:
|
|
||||||
s.errorf("range can't iterate over %v", val)
|
|
||||||
}
|
|
||||||
if r.ElseList != nil {
|
|
||||||
s.walk(dot, r.ElseList)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) walkTemplate(dot reflect.Value, t *parse.TemplateNode) {
|
|
||||||
s.at(t)
|
|
||||||
tmpl := s.tmpl.tmpl[t.Name]
|
|
||||||
if tmpl == nil {
|
|
||||||
s.errorf("template %q not defined", t.Name)
|
|
||||||
}
|
|
||||||
// Variables declared by the pipeline persist.
|
|
||||||
dot = s.evalPipeline(dot, t.Pipe)
|
|
||||||
newState := *s
|
|
||||||
newState.tmpl = tmpl
|
|
||||||
// No dynamic scoping: template invocations inherit no variables.
|
|
||||||
newState.vars = []variable{{"$", dot}}
|
|
||||||
newState.walk(dot, tmpl.Root)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Eval functions evaluate pipelines, commands, and their elements and extract
|
|
||||||
// values from the data structure by examining fields, calling methods, and so on.
|
|
||||||
// The printing of those values happens only through walk functions.
|
|
||||||
|
|
||||||
// evalPipeline returns the value acquired by evaluating a pipeline. If the
|
|
||||||
// pipeline has a variable declaration, the variable will be pushed on the
|
|
||||||
// stack. Callers should therefore pop the stack after they are finished
|
|
||||||
// executing commands depending on the pipeline value.
|
|
||||||
func (s *state) evalPipeline(dot reflect.Value, pipe *parse.PipeNode) (value reflect.Value) {
|
|
||||||
if pipe == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s.at(pipe)
|
|
||||||
for _, cmd := range pipe.Cmds {
|
|
||||||
value = s.evalCommand(dot, cmd, value) // previous value is this one's final arg.
|
|
||||||
// If the object has type interface{}, dig down one level to the thing inside.
|
|
||||||
if value.Kind() == reflect.Interface && value.Type().NumMethod() == 0 {
|
|
||||||
value = reflect.ValueOf(value.Interface()) // lovely!
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for _, variable := range pipe.Decl {
|
|
||||||
s.push(variable.Ident[0], value)
|
|
||||||
}
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) notAFunction(args []parse.Node, final reflect.Value) {
|
|
||||||
if len(args) > 1 || final.IsValid() {
|
|
||||||
s.errorf("can't give argument to non-function %s", args[0])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalCommand(dot reflect.Value, cmd *parse.CommandNode, final reflect.Value) reflect.Value {
|
|
||||||
firstWord := cmd.Args[0]
|
|
||||||
switch n := firstWord.(type) {
|
|
||||||
case *parse.FieldNode:
|
|
||||||
return s.evalFieldNode(dot, n, cmd.Args, final)
|
|
||||||
case *parse.ChainNode:
|
|
||||||
return s.evalChainNode(dot, n, cmd.Args, final)
|
|
||||||
case *parse.IdentifierNode:
|
|
||||||
// Must be a function.
|
|
||||||
return s.evalFunction(dot, n, cmd, cmd.Args, final)
|
|
||||||
case *parse.PipeNode:
|
|
||||||
// Parenthesized pipeline. The arguments are all inside the pipeline; final is ignored.
|
|
||||||
return s.evalPipeline(dot, n)
|
|
||||||
case *parse.VariableNode:
|
|
||||||
return s.evalVariableNode(dot, n, cmd.Args, final)
|
|
||||||
}
|
|
||||||
s.at(firstWord)
|
|
||||||
s.notAFunction(cmd.Args, final)
|
|
||||||
switch word := firstWord.(type) {
|
|
||||||
case *parse.BoolNode:
|
|
||||||
return reflect.ValueOf(word.True)
|
|
||||||
case *parse.DotNode:
|
|
||||||
return dot
|
|
||||||
case *parse.NilNode:
|
|
||||||
s.errorf("nil is not a command")
|
|
||||||
case *parse.NumberNode:
|
|
||||||
return s.idealConstant(word)
|
|
||||||
case *parse.StringNode:
|
|
||||||
return reflect.ValueOf(word.Text)
|
|
||||||
}
|
|
||||||
s.errorf("can't evaluate command %q", firstWord)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
// idealConstant is called to return the value of a number in a context where
|
|
||||||
// we don't know the type. In that case, the syntax of the number tells us
|
|
||||||
// its type, and we use Go rules to resolve. Note there is no such thing as
|
|
||||||
// a uint ideal constant in this situation - the value must be of int type.
|
|
||||||
func (s *state) idealConstant(constant *parse.NumberNode) reflect.Value {
|
|
||||||
// These are ideal constants but we don't know the type
|
|
||||||
// and we have no context. (If it was a method argument,
|
|
||||||
// we'd know what we need.) The syntax guides us to some extent.
|
|
||||||
s.at(constant)
|
|
||||||
switch {
|
|
||||||
case constant.IsComplex:
|
|
||||||
return reflect.ValueOf(constant.Complex128) // incontrovertible.
|
|
||||||
case constant.IsFloat && !isHexConstant(constant.Text) && strings.IndexAny(constant.Text, ".eE") >= 0:
|
|
||||||
return reflect.ValueOf(constant.Float64)
|
|
||||||
case constant.IsInt:
|
|
||||||
n := int(constant.Int64)
|
|
||||||
if int64(n) != constant.Int64 {
|
|
||||||
s.errorf("%s overflows int", constant.Text)
|
|
||||||
}
|
|
||||||
return reflect.ValueOf(n)
|
|
||||||
case constant.IsUint:
|
|
||||||
s.errorf("%s overflows int", constant.Text)
|
|
||||||
}
|
|
||||||
return zero
|
|
||||||
}
|
|
||||||
|
|
||||||
func isHexConstant(s string) bool {
|
|
||||||
return len(s) > 2 && s[0] == '0' && (s[1] == 'x' || s[1] == 'X')
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalFieldNode(dot reflect.Value, field *parse.FieldNode, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
s.at(field)
|
|
||||||
return s.evalFieldChain(dot, dot, field, field.Ident, args, final)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalChainNode(dot reflect.Value, chain *parse.ChainNode, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
s.at(chain)
|
|
||||||
// (pipe).Field1.Field2 has pipe as .Node, fields as .Field. Eval the pipeline, then the fields.
|
|
||||||
pipe := s.evalArg(dot, nil, chain.Node)
|
|
||||||
if len(chain.Field) == 0 {
|
|
||||||
s.errorf("internal error: no fields in evalChainNode")
|
|
||||||
}
|
|
||||||
return s.evalFieldChain(dot, pipe, chain, chain.Field, args, final)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalVariableNode(dot reflect.Value, variable *parse.VariableNode, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
// $x.Field has $x as the first ident, Field as the second. Eval the var, then the fields.
|
|
||||||
s.at(variable)
|
|
||||||
value := s.varValue(variable.Ident[0])
|
|
||||||
if len(variable.Ident) == 1 {
|
|
||||||
s.notAFunction(args, final)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
return s.evalFieldChain(dot, value, variable, variable.Ident[1:], args, final)
|
|
||||||
}
|
|
||||||
|
|
||||||
// evalFieldChain evaluates .X.Y.Z possibly followed by arguments.
|
|
||||||
// dot is the environment in which to evaluate arguments, while
|
|
||||||
// receiver is the value being walked along the chain.
|
|
||||||
func (s *state) evalFieldChain(dot, receiver reflect.Value, node parse.Node, ident []string, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
n := len(ident)
|
|
||||||
for i := 0; i < n-1; i++ {
|
|
||||||
receiver = s.evalField(dot, ident[i], node, nil, zero, receiver)
|
|
||||||
}
|
|
||||||
// Now if it's a method, it gets the arguments.
|
|
||||||
return s.evalField(dot, ident[n-1], node, args, final, receiver)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalFunction(dot reflect.Value, node *parse.IdentifierNode, cmd parse.Node, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
s.at(node)
|
|
||||||
name := node.Ident
|
|
||||||
function, ok := findFunction(name, s.tmpl)
|
|
||||||
if !ok {
|
|
||||||
s.errorf("%q is not a defined function", name)
|
|
||||||
}
|
|
||||||
return s.evalCall(dot, function, cmd, name, args, final)
|
|
||||||
}
|
|
||||||
|
|
||||||
// evalField evaluates an expression like (.Field) or (.Field arg1 arg2).
|
|
||||||
// The 'final' argument represents the return value from the preceding
|
|
||||||
// value of the pipeline, if any.
|
|
||||||
func (s *state) evalField(dot reflect.Value, fieldName string, node parse.Node, args []parse.Node, final, receiver reflect.Value) reflect.Value {
|
|
||||||
if !receiver.IsValid() {
|
|
||||||
return zero
|
|
||||||
}
|
|
||||||
typ := receiver.Type()
|
|
||||||
receiver, _ = indirect(receiver)
|
|
||||||
// Unless it's an interface, need to get to a value of type *T to guarantee
|
|
||||||
// we see all methods of T and *T.
|
|
||||||
ptr := receiver
|
|
||||||
if ptr.Kind() != reflect.Interface && ptr.CanAddr() {
|
|
||||||
ptr = ptr.Addr()
|
|
||||||
}
|
|
||||||
if method := ptr.MethodByName(fieldName); method.IsValid() {
|
|
||||||
return s.evalCall(dot, method, node, fieldName, args, final)
|
|
||||||
}
|
|
||||||
hasArgs := len(args) > 1 || final.IsValid()
|
|
||||||
// It's not a method; must be a field of a struct or an element of a map. The receiver must not be nil.
|
|
||||||
receiver, isNil := indirect(receiver)
|
|
||||||
if isNil {
|
|
||||||
s.errorf("nil pointer evaluating %s.%s", typ, fieldName)
|
|
||||||
}
|
|
||||||
switch receiver.Kind() {
|
|
||||||
case reflect.Struct:
|
|
||||||
tField, ok := receiver.Type().FieldByName(fieldName)
|
|
||||||
if ok {
|
|
||||||
field := receiver.FieldByIndex(tField.Index)
|
|
||||||
if tField.PkgPath != "" { // field is unexported
|
|
||||||
s.errorf("%s is an unexported field of struct type %s", fieldName, typ)
|
|
||||||
}
|
|
||||||
// If it's a function, we must call it.
|
|
||||||
if hasArgs {
|
|
||||||
s.errorf("%s has arguments but cannot be invoked as function", fieldName)
|
|
||||||
}
|
|
||||||
return field
|
|
||||||
}
|
|
||||||
s.errorf("%s is not a field of struct type %s", fieldName, typ)
|
|
||||||
case reflect.Map:
|
|
||||||
// If it's a map, attempt to use the field name as a key.
|
|
||||||
nameVal := reflect.ValueOf(fieldName)
|
|
||||||
if nameVal.Type().AssignableTo(receiver.Type().Key()) {
|
|
||||||
if hasArgs {
|
|
||||||
s.errorf("%s is not a method but has arguments", fieldName)
|
|
||||||
}
|
|
||||||
return receiver.MapIndex(nameVal)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
s.errorf("can't evaluate field %s in type %s", fieldName, typ)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
errorType = reflect.TypeOf((*error)(nil)).Elem()
|
|
||||||
fmtStringerType = reflect.TypeOf((*fmt.Stringer)(nil)).Elem()
|
|
||||||
)
|
|
||||||
|
|
||||||
// evalCall executes a function or method call. If it's a method, fun already has the receiver bound, so
|
|
||||||
// it looks just like a function call. The arg list, if non-nil, includes (in the manner of the shell), arg[0]
|
|
||||||
// as the function itself.
|
|
||||||
func (s *state) evalCall(dot, fun reflect.Value, node parse.Node, name string, args []parse.Node, final reflect.Value) reflect.Value {
|
|
||||||
if args != nil {
|
|
||||||
args = args[1:] // Zeroth arg is function name/node; not passed to function.
|
|
||||||
}
|
|
||||||
typ := fun.Type()
|
|
||||||
numIn := len(args)
|
|
||||||
if final.IsValid() {
|
|
||||||
numIn++
|
|
||||||
}
|
|
||||||
numFixed := len(args)
|
|
||||||
if typ.IsVariadic() {
|
|
||||||
numFixed = typ.NumIn() - 1 // last arg is the variadic one.
|
|
||||||
if numIn < numFixed {
|
|
||||||
s.errorf("wrong number of args for %s: want at least %d got %d", name, typ.NumIn()-1, len(args))
|
|
||||||
}
|
|
||||||
} else if numIn < typ.NumIn()-1 || !typ.IsVariadic() && numIn != typ.NumIn() {
|
|
||||||
s.errorf("wrong number of args for %s: want %d got %d", name, typ.NumIn(), len(args))
|
|
||||||
}
|
|
||||||
if !goodFunc(typ) {
|
|
||||||
// TODO: This could still be a confusing error; maybe goodFunc should provide info.
|
|
||||||
s.errorf("can't call method/function %q with %d results", name, typ.NumOut())
|
|
||||||
}
|
|
||||||
// Build the arg list.
|
|
||||||
argv := make([]reflect.Value, numIn)
|
|
||||||
// Args must be evaluated. Fixed args first.
|
|
||||||
i := 0
|
|
||||||
for ; i < numFixed && i < len(args); i++ {
|
|
||||||
argv[i] = s.evalArg(dot, typ.In(i), args[i])
|
|
||||||
}
|
|
||||||
// Now the ... args.
|
|
||||||
if typ.IsVariadic() {
|
|
||||||
argType := typ.In(typ.NumIn() - 1).Elem() // Argument is a slice.
|
|
||||||
for ; i < len(args); i++ {
|
|
||||||
argv[i] = s.evalArg(dot, argType, args[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Add final value if necessary.
|
|
||||||
if final.IsValid() {
|
|
||||||
t := typ.In(typ.NumIn() - 1)
|
|
||||||
if typ.IsVariadic() {
|
|
||||||
t = t.Elem()
|
|
||||||
}
|
|
||||||
argv[i] = s.validateType(final, t)
|
|
||||||
}
|
|
||||||
result := fun.Call(argv)
|
|
||||||
// If we have an error that is not nil, stop execution and return that error to the caller.
|
|
||||||
if len(result) == 2 && !result[1].IsNil() {
|
|
||||||
s.at(node)
|
|
||||||
s.errorf("error calling %s: %s", name, result[1].Interface().(error))
|
|
||||||
}
|
|
||||||
return result[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
// canBeNil reports whether an untyped nil can be assigned to the type. See reflect.Zero.
|
|
||||||
func canBeNil(typ reflect.Type) bool {
|
|
||||||
switch typ.Kind() {
|
|
||||||
case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice:
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// validateType guarantees that the value is valid and assignable to the type.
|
|
||||||
func (s *state) validateType(value reflect.Value, typ reflect.Type) reflect.Value {
|
|
||||||
if !value.IsValid() {
|
|
||||||
if typ == nil || canBeNil(typ) {
|
|
||||||
// An untyped nil interface{}. Accept as a proper nil value.
|
|
||||||
return reflect.Zero(typ)
|
|
||||||
}
|
|
||||||
s.errorf("invalid value; expected %s", typ)
|
|
||||||
}
|
|
||||||
if typ != nil && !value.Type().AssignableTo(typ) {
|
|
||||||
if value.Kind() == reflect.Interface && !value.IsNil() {
|
|
||||||
value = value.Elem()
|
|
||||||
if value.Type().AssignableTo(typ) {
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
// fallthrough
|
|
||||||
}
|
|
||||||
// Does one dereference or indirection work? We could do more, as we
|
|
||||||
// do with method receivers, but that gets messy and method receivers
|
|
||||||
// are much more constrained, so it makes more sense there than here.
|
|
||||||
// Besides, one is almost always all you need.
|
|
||||||
switch {
|
|
||||||
case value.Kind() == reflect.Ptr && value.Type().Elem().AssignableTo(typ):
|
|
||||||
value = value.Elem()
|
|
||||||
if !value.IsValid() {
|
|
||||||
s.errorf("dereference of nil pointer of type %s", typ)
|
|
||||||
}
|
|
||||||
case reflect.PtrTo(value.Type()).AssignableTo(typ) && value.CanAddr():
|
|
||||||
value = value.Addr()
|
|
||||||
default:
|
|
||||||
s.errorf("wrong type for value; expected %s; got %s", typ, value.Type())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalArg(dot reflect.Value, typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
switch arg := n.(type) {
|
|
||||||
case *parse.DotNode:
|
|
||||||
return s.validateType(dot, typ)
|
|
||||||
case *parse.NilNode:
|
|
||||||
if canBeNil(typ) {
|
|
||||||
return reflect.Zero(typ)
|
|
||||||
}
|
|
||||||
s.errorf("cannot assign nil to %s", typ)
|
|
||||||
case *parse.FieldNode:
|
|
||||||
return s.validateType(s.evalFieldNode(dot, arg, []parse.Node{n}, zero), typ)
|
|
||||||
case *parse.VariableNode:
|
|
||||||
return s.validateType(s.evalVariableNode(dot, arg, nil, zero), typ)
|
|
||||||
case *parse.PipeNode:
|
|
||||||
return s.validateType(s.evalPipeline(dot, arg), typ)
|
|
||||||
case *parse.IdentifierNode:
|
|
||||||
return s.evalFunction(dot, arg, arg, nil, zero)
|
|
||||||
case *parse.ChainNode:
|
|
||||||
return s.validateType(s.evalChainNode(dot, arg, nil, zero), typ)
|
|
||||||
}
|
|
||||||
switch typ.Kind() {
|
|
||||||
case reflect.Bool:
|
|
||||||
return s.evalBool(typ, n)
|
|
||||||
case reflect.Complex64, reflect.Complex128:
|
|
||||||
return s.evalComplex(typ, n)
|
|
||||||
case reflect.Float32, reflect.Float64:
|
|
||||||
return s.evalFloat(typ, n)
|
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
||||||
return s.evalInteger(typ, n)
|
|
||||||
case reflect.Interface:
|
|
||||||
if typ.NumMethod() == 0 {
|
|
||||||
return s.evalEmptyInterface(dot, n)
|
|
||||||
}
|
|
||||||
case reflect.String:
|
|
||||||
return s.evalString(typ, n)
|
|
||||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
||||||
return s.evalUnsignedInteger(typ, n)
|
|
||||||
}
|
|
||||||
s.errorf("can't handle %s for arg of type %s", n, typ)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalBool(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
if n, ok := n.(*parse.BoolNode); ok {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetBool(n.True)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected bool; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalString(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
if n, ok := n.(*parse.StringNode); ok {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetString(n.Text)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected string; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalInteger(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
if n, ok := n.(*parse.NumberNode); ok && n.IsInt {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetInt(n.Int64)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected integer; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalUnsignedInteger(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
if n, ok := n.(*parse.NumberNode); ok && n.IsUint {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetUint(n.Uint64)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected unsigned integer; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalFloat(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
if n, ok := n.(*parse.NumberNode); ok && n.IsFloat {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetFloat(n.Float64)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected float; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalComplex(typ reflect.Type, n parse.Node) reflect.Value {
|
|
||||||
if n, ok := n.(*parse.NumberNode); ok && n.IsComplex {
|
|
||||||
value := reflect.New(typ).Elem()
|
|
||||||
value.SetComplex(n.Complex128)
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
s.errorf("expected complex; found %s", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *state) evalEmptyInterface(dot reflect.Value, n parse.Node) reflect.Value {
|
|
||||||
s.at(n)
|
|
||||||
switch n := n.(type) {
|
|
||||||
case *parse.BoolNode:
|
|
||||||
return reflect.ValueOf(n.True)
|
|
||||||
case *parse.DotNode:
|
|
||||||
return dot
|
|
||||||
case *parse.FieldNode:
|
|
||||||
return s.evalFieldNode(dot, n, nil, zero)
|
|
||||||
case *parse.IdentifierNode:
|
|
||||||
return s.evalFunction(dot, n, n, nil, zero)
|
|
||||||
case *parse.NilNode:
|
|
||||||
// NilNode is handled in evalArg, the only place that calls here.
|
|
||||||
s.errorf("evalEmptyInterface: nil (can't happen)")
|
|
||||||
case *parse.NumberNode:
|
|
||||||
return s.idealConstant(n)
|
|
||||||
case *parse.StringNode:
|
|
||||||
return reflect.ValueOf(n.Text)
|
|
||||||
case *parse.VariableNode:
|
|
||||||
return s.evalVariableNode(dot, n, nil, zero)
|
|
||||||
case *parse.PipeNode:
|
|
||||||
return s.evalPipeline(dot, n)
|
|
||||||
}
|
|
||||||
s.errorf("can't handle assignment of %s to empty interface argument", n)
|
|
||||||
panic("not reached")
|
|
||||||
}
|
|
||||||
|
|
||||||
// indirect returns the item at the end of indirection, and a bool to indicate if it's nil.
|
|
||||||
// We indirect through pointers and empty interfaces (only) because
|
|
||||||
// non-empty interfaces have methods we might need.
|
|
||||||
func indirect(v reflect.Value) (rv reflect.Value, isNil bool) {
|
|
||||||
for ; v.Kind() == reflect.Ptr || v.Kind() == reflect.Interface; v = v.Elem() {
|
|
||||||
if v.IsNil() {
|
|
||||||
return v, true
|
|
||||||
}
|
|
||||||
if v.Kind() == reflect.Interface && v.NumMethod() > 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return v, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// printValue writes the textual representation of the value to the output of
|
|
||||||
// the template.
|
|
||||||
func (s *state) printValue(n parse.Node, v reflect.Value) {
|
|
||||||
s.at(n)
|
|
||||||
iface, ok := printableValue(v)
|
|
||||||
if !ok {
|
|
||||||
s.errorf("can't print %s of type %s", n, v.Type())
|
|
||||||
}
|
|
||||||
fmt.Fprint(s.wr, iface)
|
|
||||||
}
|
|
||||||
|
|
||||||
// printableValue returns the, possibly indirected, interface value inside v that
|
|
||||||
// is best for a call to formatted printer.
|
|
||||||
func printableValue(v reflect.Value) (interface{}, bool) {
|
|
||||||
if v.Kind() == reflect.Ptr {
|
|
||||||
v, _ = indirect(v) // fmt.Fprint handles nil.
|
|
||||||
}
|
|
||||||
if !v.IsValid() {
|
|
||||||
return "<no value>", true
|
|
||||||
}
|
|
||||||
|
|
||||||
if !v.Type().Implements(errorType) && !v.Type().Implements(fmtStringerType) {
|
|
||||||
if v.CanAddr() && (reflect.PtrTo(v.Type()).Implements(errorType) || reflect.PtrTo(v.Type()).Implements(fmtStringerType)) {
|
|
||||||
v = v.Addr()
|
|
||||||
} else {
|
|
||||||
switch v.Kind() {
|
|
||||||
case reflect.Chan, reflect.Func:
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return v.Interface(), true
|
|
||||||
}
|
|
||||||
|
|
||||||
// Types to help sort the keys in a map for reproducible output.
|
|
||||||
|
|
||||||
type rvs []reflect.Value
|
|
||||||
|
|
||||||
func (x rvs) Len() int { return len(x) }
|
|
||||||
func (x rvs) Swap(i, j int) { x[i], x[j] = x[j], x[i] }
|
|
||||||
|
|
||||||
type rvInts struct{ rvs }
|
|
||||||
|
|
||||||
func (x rvInts) Less(i, j int) bool { return x.rvs[i].Int() < x.rvs[j].Int() }
|
|
||||||
|
|
||||||
type rvUints struct{ rvs }
|
|
||||||
|
|
||||||
func (x rvUints) Less(i, j int) bool { return x.rvs[i].Uint() < x.rvs[j].Uint() }
|
|
||||||
|
|
||||||
type rvFloats struct{ rvs }
|
|
||||||
|
|
||||||
func (x rvFloats) Less(i, j int) bool { return x.rvs[i].Float() < x.rvs[j].Float() }
|
|
||||||
|
|
||||||
type rvStrings struct{ rvs }
|
|
||||||
|
|
||||||
func (x rvStrings) Less(i, j int) bool { return x.rvs[i].String() < x.rvs[j].String() }
|
|
||||||
|
|
||||||
// sortKeys sorts (if it can) the slice of reflect.Values, which is a slice of map keys.
|
|
||||||
func sortKeys(v []reflect.Value) []reflect.Value {
|
|
||||||
if len(v) <= 1 {
|
|
||||||
return v
|
|
||||||
}
|
|
||||||
switch v[0].Kind() {
|
|
||||||
case reflect.Float32, reflect.Float64:
|
|
||||||
sort.Sort(rvFloats{v})
|
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
||||||
sort.Sort(rvInts{v})
|
|
||||||
case reflect.String:
|
|
||||||
sort.Sort(rvStrings{v})
|
|
||||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
||||||
sort.Sort(rvUints{v})
|
|
||||||
}
|
|
||||||
return v
|
|
||||||
}
|
|
598
vendor/github.com/alecthomas/template/funcs.go
generated
vendored
598
vendor/github.com/alecthomas/template/funcs.go
generated
vendored
|
@ -1,598 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package template
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"net/url"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"unicode"
|
|
||||||
"unicode/utf8"
|
|
||||||
)
|
|
||||||
|
|
||||||
// FuncMap is the type of the map defining the mapping from names to functions.
|
|
||||||
// Each function must have either a single return value, or two return values of
|
|
||||||
// which the second has type error. In that case, if the second (error)
|
|
||||||
// return value evaluates to non-nil during execution, execution terminates and
|
|
||||||
// Execute returns that error.
|
|
||||||
type FuncMap map[string]interface{}
|
|
||||||
|
|
||||||
var builtins = FuncMap{
|
|
||||||
"and": and,
|
|
||||||
"call": call,
|
|
||||||
"html": HTMLEscaper,
|
|
||||||
"index": index,
|
|
||||||
"js": JSEscaper,
|
|
||||||
"len": length,
|
|
||||||
"not": not,
|
|
||||||
"or": or,
|
|
||||||
"print": fmt.Sprint,
|
|
||||||
"printf": fmt.Sprintf,
|
|
||||||
"println": fmt.Sprintln,
|
|
||||||
"urlquery": URLQueryEscaper,
|
|
||||||
|
|
||||||
// Comparisons
|
|
||||||
"eq": eq, // ==
|
|
||||||
"ge": ge, // >=
|
|
||||||
"gt": gt, // >
|
|
||||||
"le": le, // <=
|
|
||||||
"lt": lt, // <
|
|
||||||
"ne": ne, // !=
|
|
||||||
}
|
|
||||||
|
|
||||||
var builtinFuncs = createValueFuncs(builtins)
|
|
||||||
|
|
||||||
// createValueFuncs turns a FuncMap into a map[string]reflect.Value
|
|
||||||
func createValueFuncs(funcMap FuncMap) map[string]reflect.Value {
|
|
||||||
m := make(map[string]reflect.Value)
|
|
||||||
addValueFuncs(m, funcMap)
|
|
||||||
return m
|
|
||||||
}
|
|
||||||
|
|
||||||
// addValueFuncs adds to values the functions in funcs, converting them to reflect.Values.
|
|
||||||
func addValueFuncs(out map[string]reflect.Value, in FuncMap) {
|
|
||||||
for name, fn := range in {
|
|
||||||
v := reflect.ValueOf(fn)
|
|
||||||
if v.Kind() != reflect.Func {
|
|
||||||
panic("value for " + name + " not a function")
|
|
||||||
}
|
|
||||||
if !goodFunc(v.Type()) {
|
|
||||||
panic(fmt.Errorf("can't install method/function %q with %d results", name, v.Type().NumOut()))
|
|
||||||
}
|
|
||||||
out[name] = v
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// addFuncs adds to values the functions in funcs. It does no checking of the input -
|
|
||||||
// call addValueFuncs first.
|
|
||||||
func addFuncs(out, in FuncMap) {
|
|
||||||
for name, fn := range in {
|
|
||||||
out[name] = fn
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// goodFunc checks that the function or method has the right result signature.
|
|
||||||
func goodFunc(typ reflect.Type) bool {
|
|
||||||
// We allow functions with 1 result or 2 results where the second is an error.
|
|
||||||
switch {
|
|
||||||
case typ.NumOut() == 1:
|
|
||||||
return true
|
|
||||||
case typ.NumOut() == 2 && typ.Out(1) == errorType:
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// findFunction looks for a function in the template, and global map.
|
|
||||||
func findFunction(name string, tmpl *Template) (reflect.Value, bool) {
|
|
||||||
if tmpl != nil && tmpl.common != nil {
|
|
||||||
if fn := tmpl.execFuncs[name]; fn.IsValid() {
|
|
||||||
return fn, true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if fn := builtinFuncs[name]; fn.IsValid() {
|
|
||||||
return fn, true
|
|
||||||
}
|
|
||||||
return reflect.Value{}, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Indexing.
|
|
||||||
|
|
||||||
// index returns the result of indexing its first argument by the following
|
|
||||||
// arguments. Thus "index x 1 2 3" is, in Go syntax, x[1][2][3]. Each
|
|
||||||
// indexed item must be a map, slice, or array.
|
|
||||||
func index(item interface{}, indices ...interface{}) (interface{}, error) {
|
|
||||||
v := reflect.ValueOf(item)
|
|
||||||
for _, i := range indices {
|
|
||||||
index := reflect.ValueOf(i)
|
|
||||||
var isNil bool
|
|
||||||
if v, isNil = indirect(v); isNil {
|
|
||||||
return nil, fmt.Errorf("index of nil pointer")
|
|
||||||
}
|
|
||||||
switch v.Kind() {
|
|
||||||
case reflect.Array, reflect.Slice, reflect.String:
|
|
||||||
var x int64
|
|
||||||
switch index.Kind() {
|
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
||||||
x = index.Int()
|
|
||||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
||||||
x = int64(index.Uint())
|
|
||||||
default:
|
|
||||||
return nil, fmt.Errorf("cannot index slice/array with type %s", index.Type())
|
|
||||||
}
|
|
||||||
if x < 0 || x >= int64(v.Len()) {
|
|
||||||
return nil, fmt.Errorf("index out of range: %d", x)
|
|
||||||
}
|
|
||||||
v = v.Index(int(x))
|
|
||||||
case reflect.Map:
|
|
||||||
if !index.IsValid() {
|
|
||||||
index = reflect.Zero(v.Type().Key())
|
|
||||||
}
|
|
||||||
if !index.Type().AssignableTo(v.Type().Key()) {
|
|
||||||
return nil, fmt.Errorf("%s is not index type for %s", index.Type(), v.Type())
|
|
||||||
}
|
|
||||||
if x := v.MapIndex(index); x.IsValid() {
|
|
||||||
v = x
|
|
||||||
} else {
|
|
||||||
v = reflect.Zero(v.Type().Elem())
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return nil, fmt.Errorf("can't index item of type %s", v.Type())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return v.Interface(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length
|
|
||||||
|
|
||||||
// length returns the length of the item, with an error if it has no defined length.
|
|
||||||
func length(item interface{}) (int, error) {
|
|
||||||
v, isNil := indirect(reflect.ValueOf(item))
|
|
||||||
if isNil {
|
|
||||||
return 0, fmt.Errorf("len of nil pointer")
|
|
||||||
}
|
|
||||||
switch v.Kind() {
|
|
||||||
case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice, reflect.String:
|
|
||||||
return v.Len(), nil
|
|
||||||
}
|
|
||||||
return 0, fmt.Errorf("len of type %s", v.Type())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function invocation
|
|
||||||
|
|
||||||
// call returns the result of evaluating the first argument as a function.
|
|
||||||
// The function must return 1 result, or 2 results, the second of which is an error.
|
|
||||||
func call(fn interface{}, args ...interface{}) (interface{}, error) {
|
|
||||||
v := reflect.ValueOf(fn)
|
|
||||||
typ := v.Type()
|
|
||||||
if typ.Kind() != reflect.Func {
|
|
||||||
return nil, fmt.Errorf("non-function of type %s", typ)
|
|
||||||
}
|
|
||||||
if !goodFunc(typ) {
|
|
||||||
return nil, fmt.Errorf("function called with %d args; should be 1 or 2", typ.NumOut())
|
|
||||||
}
|
|
||||||
numIn := typ.NumIn()
|
|
||||||
var dddType reflect.Type
|
|
||||||
if typ.IsVariadic() {
|
|
||||||
if len(args) < numIn-1 {
|
|
||||||
return nil, fmt.Errorf("wrong number of args: got %d want at least %d", len(args), numIn-1)
|
|
||||||
}
|
|
||||||
dddType = typ.In(numIn - 1).Elem()
|
|
||||||
} else {
|
|
||||||
if len(args) != numIn {
|
|
||||||
return nil, fmt.Errorf("wrong number of args: got %d want %d", len(args), numIn)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
argv := make([]reflect.Value, len(args))
|
|
||||||
for i, arg := range args {
|
|
||||||
value := reflect.ValueOf(arg)
|
|
||||||
// Compute the expected type. Clumsy because of variadics.
|
|
||||||
var argType reflect.Type
|
|
||||||
if !typ.IsVariadic() || i < numIn-1 {
|
|
||||||
argType = typ.In(i)
|
|
||||||
} else {
|
|
||||||
argType = dddType
|
|
||||||
}
|
|
||||||
if !value.IsValid() && canBeNil(argType) {
|
|
||||||
value = reflect.Zero(argType)
|
|
||||||
}
|
|
||||||
if !value.Type().AssignableTo(argType) {
|
|
||||||
return nil, fmt.Errorf("arg %d has type %s; should be %s", i, value.Type(), argType)
|
|
||||||
}
|
|
||||||
argv[i] = value
|
|
||||||
}
|
|
||||||
result := v.Call(argv)
|
|
||||||
if len(result) == 2 && !result[1].IsNil() {
|
|
||||||
return result[0].Interface(), result[1].Interface().(error)
|
|
||||||
}
|
|
||||||
return result[0].Interface(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Boolean logic.
|
|
||||||
|
|
||||||
func truth(a interface{}) bool {
|
|
||||||
t, _ := isTrue(reflect.ValueOf(a))
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
// and computes the Boolean AND of its arguments, returning
|
|
||||||
// the first false argument it encounters, or the last argument.
|
|
||||||
func and(arg0 interface{}, args ...interface{}) interface{} {
|
|
||||||
if !truth(arg0) {
|
|
||||||
return arg0
|
|
||||||
}
|
|
||||||
for i := range args {
|
|
||||||
arg0 = args[i]
|
|
||||||
if !truth(arg0) {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return arg0
|
|
||||||
}
|
|
||||||
|
|
||||||
// or computes the Boolean OR of its arguments, returning
|
|
||||||
// the first true argument it encounters, or the last argument.
|
|
||||||
func or(arg0 interface{}, args ...interface{}) interface{} {
|
|
||||||
if truth(arg0) {
|
|
||||||
return arg0
|
|
||||||
}
|
|
||||||
for i := range args {
|
|
||||||
arg0 = args[i]
|
|
||||||
if truth(arg0) {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return arg0
|
|
||||||
}
|
|
||||||
|
|
||||||
// not returns the Boolean negation of its argument.
|
|
||||||
func not(arg interface{}) (truth bool) {
|
|
||||||
truth, _ = isTrue(reflect.ValueOf(arg))
|
|
||||||
return !truth
|
|
||||||
}
|
|
||||||
|
|
||||||
// Comparison.
|
|
||||||
|
|
||||||
// TODO: Perhaps allow comparison between signed and unsigned integers.
|
|
||||||
|
|
||||||
var (
|
|
||||||
errBadComparisonType = errors.New("invalid type for comparison")
|
|
||||||
errBadComparison = errors.New("incompatible types for comparison")
|
|
||||||
errNoComparison = errors.New("missing argument for comparison")
|
|
||||||
)
|
|
||||||
|
|
||||||
type kind int
|
|
||||||
|
|
||||||
const (
|
|
||||||
invalidKind kind = iota
|
|
||||||
boolKind
|
|
||||||
complexKind
|
|
||||||
intKind
|
|
||||||
floatKind
|
|
||||||
integerKind
|
|
||||||
stringKind
|
|
||||||
uintKind
|
|
||||||
)
|
|
||||||
|
|
||||||
func basicKind(v reflect.Value) (kind, error) {
|
|
||||||
switch v.Kind() {
|
|
||||||
case reflect.Bool:
|
|
||||||
return boolKind, nil
|
|
||||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
||||||
return intKind, nil
|
|
||||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
||||||
return uintKind, nil
|
|
||||||
case reflect.Float32, reflect.Float64:
|
|
||||||
return floatKind, nil
|
|
||||||
case reflect.Complex64, reflect.Complex128:
|
|
||||||
return complexKind, nil
|
|
||||||
case reflect.String:
|
|
||||||
return stringKind, nil
|
|
||||||
}
|
|
||||||
return invalidKind, errBadComparisonType
|
|
||||||
}
|
|
||||||
|
|
||||||
// eq evaluates the comparison a == b || a == c || ...
|
|
||||||
func eq(arg1 interface{}, arg2 ...interface{}) (bool, error) {
|
|
||||||
v1 := reflect.ValueOf(arg1)
|
|
||||||
k1, err := basicKind(v1)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
if len(arg2) == 0 {
|
|
||||||
return false, errNoComparison
|
|
||||||
}
|
|
||||||
for _, arg := range arg2 {
|
|
||||||
v2 := reflect.ValueOf(arg)
|
|
||||||
k2, err := basicKind(v2)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
truth := false
|
|
||||||
if k1 != k2 {
|
|
||||||
// Special case: Can compare integer values regardless of type's sign.
|
|
||||||
switch {
|
|
||||||
case k1 == intKind && k2 == uintKind:
|
|
||||||
truth = v1.Int() >= 0 && uint64(v1.Int()) == v2.Uint()
|
|
||||||
case k1 == uintKind && k2 == intKind:
|
|
||||||
truth = v2.Int() >= 0 && v1.Uint() == uint64(v2.Int())
|
|
||||||
default:
|
|
||||||
return false, errBadComparison
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
switch k1 {
|
|
||||||
case boolKind:
|
|
||||||
truth = v1.Bool() == v2.Bool()
|
|
||||||
case complexKind:
|
|
||||||
truth = v1.Complex() == v2.Complex()
|
|
||||||
case floatKind:
|
|
||||||
truth = v1.Float() == v2.Float()
|
|
||||||
case intKind:
|
|
||||||
truth = v1.Int() == v2.Int()
|
|
||||||
case stringKind:
|
|
||||||
truth = v1.String() == v2.String()
|
|
||||||
case uintKind:
|
|
||||||
truth = v1.Uint() == v2.Uint()
|
|
||||||
default:
|
|
||||||
panic("invalid kind")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if truth {
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ne evaluates the comparison a != b.
|
|
||||||
func ne(arg1, arg2 interface{}) (bool, error) {
|
|
||||||
// != is the inverse of ==.
|
|
||||||
equal, err := eq(arg1, arg2)
|
|
||||||
return !equal, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// lt evaluates the comparison a < b.
|
|
||||||
func lt(arg1, arg2 interface{}) (bool, error) {
|
|
||||||
v1 := reflect.ValueOf(arg1)
|
|
||||||
k1, err := basicKind(v1)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
v2 := reflect.ValueOf(arg2)
|
|
||||||
k2, err := basicKind(v2)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
truth := false
|
|
||||||
if k1 != k2 {
|
|
||||||
// Special case: Can compare integer values regardless of type's sign.
|
|
||||||
switch {
|
|
||||||
case k1 == intKind && k2 == uintKind:
|
|
||||||
truth = v1.Int() < 0 || uint64(v1.Int()) < v2.Uint()
|
|
||||||
case k1 == uintKind && k2 == intKind:
|
|
||||||
truth = v2.Int() >= 0 && v1.Uint() < uint64(v2.Int())
|
|
||||||
default:
|
|
||||||
return false, errBadComparison
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
switch k1 {
|
|
||||||
case boolKind, complexKind:
|
|
||||||
return false, errBadComparisonType
|
|
||||||
case floatKind:
|
|
||||||
truth = v1.Float() < v2.Float()
|
|
||||||
case intKind:
|
|
||||||
truth = v1.Int() < v2.Int()
|
|
||||||
case stringKind:
|
|
||||||
truth = v1.String() < v2.String()
|
|
||||||
case uintKind:
|
|
||||||
truth = v1.Uint() < v2.Uint()
|
|
||||||
default:
|
|
||||||
panic("invalid kind")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return truth, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// le evaluates the comparison <= b.
|
|
||||||
func le(arg1, arg2 interface{}) (bool, error) {
|
|
||||||
// <= is < or ==.
|
|
||||||
lessThan, err := lt(arg1, arg2)
|
|
||||||
if lessThan || err != nil {
|
|
||||||
return lessThan, err
|
|
||||||
}
|
|
||||||
return eq(arg1, arg2)
|
|
||||||
}
|
|
||||||
|
|
||||||
// gt evaluates the comparison a > b.
|
|
||||||
func gt(arg1, arg2 interface{}) (bool, error) {
|
|
||||||
// > is the inverse of <=.
|
|
||||||
lessOrEqual, err := le(arg1, arg2)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
return !lessOrEqual, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ge evaluates the comparison a >= b.
|
|
||||||
func ge(arg1, arg2 interface{}) (bool, error) {
|
|
||||||
// >= is the inverse of <.
|
|
||||||
lessThan, err := lt(arg1, arg2)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
return !lessThan, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTML escaping.
|
|
||||||
|
|
||||||
var (
|
|
||||||
htmlQuot = []byte(""") // shorter than """
|
|
||||||
htmlApos = []byte("'") // shorter than "'" and apos was not in HTML until HTML5
|
|
||||||
htmlAmp = []byte("&")
|
|
||||||
htmlLt = []byte("<")
|
|
||||||
htmlGt = []byte(">")
|
|
||||||
)
|
|
||||||
|
|
||||||
// HTMLEscape writes to w the escaped HTML equivalent of the plain text data b.
|
|
||||||
func HTMLEscape(w io.Writer, b []byte) {
|
|
||||||
last := 0
|
|
||||||
for i, c := range b {
|
|
||||||
var html []byte
|
|
||||||
switch c {
|
|
||||||
case '"':
|
|
||||||
html = htmlQuot
|
|
||||||
case '\'':
|
|
||||||
html = htmlApos
|
|
||||||
case '&':
|
|
||||||
html = htmlAmp
|
|
||||||
case '<':
|
|
||||||
html = htmlLt
|
|
||||||
case '>':
|
|
||||||
html = htmlGt
|
|
||||||
default:
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
w.Write(b[last:i])
|
|
||||||
w.Write(html)
|
|
||||||
last = i + 1
|
|
||||||
}
|
|
||||||
w.Write(b[last:])
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTMLEscapeString returns the escaped HTML equivalent of the plain text data s.
|
|
||||||
func HTMLEscapeString(s string) string {
|
|
||||||
// Avoid allocation if we can.
|
|
||||||
if strings.IndexAny(s, `'"&<>`) < 0 {
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
var b bytes.Buffer
|
|
||||||
HTMLEscape(&b, []byte(s))
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTMLEscaper returns the escaped HTML equivalent of the textual
|
|
||||||
// representation of its arguments.
|
|
||||||
func HTMLEscaper(args ...interface{}) string {
|
|
||||||
return HTMLEscapeString(evalArgs(args))
|
|
||||||
}
|
|
||||||
|
|
||||||
// JavaScript escaping.
|
|
||||||
|
|
||||||
var (
|
|
||||||
jsLowUni = []byte(`\u00`)
|
|
||||||
hex = []byte("0123456789ABCDEF")
|
|
||||||
|
|
||||||
jsBackslash = []byte(`\\`)
|
|
||||||
jsApos = []byte(`\'`)
|
|
||||||
jsQuot = []byte(`\"`)
|
|
||||||
jsLt = []byte(`\x3C`)
|
|
||||||
jsGt = []byte(`\x3E`)
|
|
||||||
)
|
|
||||||
|
|
||||||
// JSEscape writes to w the escaped JavaScript equivalent of the plain text data b.
|
|
||||||
func JSEscape(w io.Writer, b []byte) {
|
|
||||||
last := 0
|
|
||||||
for i := 0; i < len(b); i++ {
|
|
||||||
c := b[i]
|
|
||||||
|
|
||||||
if !jsIsSpecial(rune(c)) {
|
|
||||||
// fast path: nothing to do
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
w.Write(b[last:i])
|
|
||||||
|
|
||||||
if c < utf8.RuneSelf {
|
|
||||||
// Quotes, slashes and angle brackets get quoted.
|
|
||||||
// Control characters get written as \u00XX.
|
|
||||||
switch c {
|
|
||||||
case '\\':
|
|
||||||
w.Write(jsBackslash)
|
|
||||||
case '\'':
|
|
||||||
w.Write(jsApos)
|
|
||||||
case '"':
|
|
||||||
w.Write(jsQuot)
|
|
||||||
case '<':
|
|
||||||
w.Write(jsLt)
|
|
||||||
case '>':
|
|
||||||
w.Write(jsGt)
|
|
||||||
default:
|
|
||||||
w.Write(jsLowUni)
|
|
||||||
t, b := c>>4, c&0x0f
|
|
||||||
w.Write(hex[t : t+1])
|
|
||||||
w.Write(hex[b : b+1])
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Unicode rune.
|
|
||||||
r, size := utf8.DecodeRune(b[i:])
|
|
||||||
if unicode.IsPrint(r) {
|
|
||||||
w.Write(b[i : i+size])
|
|
||||||
} else {
|
|
||||||
fmt.Fprintf(w, "\\u%04X", r)
|
|
||||||
}
|
|
||||||
i += size - 1
|
|
||||||
}
|
|
||||||
last = i + 1
|
|
||||||
}
|
|
||||||
w.Write(b[last:])
|
|
||||||
}
|
|
||||||
|
|
||||||
// JSEscapeString returns the escaped JavaScript equivalent of the plain text data s.
|
|
||||||
func JSEscapeString(s string) string {
|
|
||||||
// Avoid allocation if we can.
|
|
||||||
if strings.IndexFunc(s, jsIsSpecial) < 0 {
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
var b bytes.Buffer
|
|
||||||
JSEscape(&b, []byte(s))
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
func jsIsSpecial(r rune) bool {
|
|
||||||
switch r {
|
|
||||||
case '\\', '\'', '"', '<', '>':
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return r < ' ' || utf8.RuneSelf <= r
|
|
||||||
}
|
|
||||||
|
|
||||||
// JSEscaper returns the escaped JavaScript equivalent of the textual
|
|
||||||
// representation of its arguments.
|
|
||||||
func JSEscaper(args ...interface{}) string {
|
|
||||||
return JSEscapeString(evalArgs(args))
|
|
||||||
}
|
|
||||||
|
|
||||||
// URLQueryEscaper returns the escaped value of the textual representation of
|
|
||||||
// its arguments in a form suitable for embedding in a URL query.
|
|
||||||
func URLQueryEscaper(args ...interface{}) string {
|
|
||||||
return url.QueryEscape(evalArgs(args))
|
|
||||||
}
|
|
||||||
|
|
||||||
// evalArgs formats the list of arguments into a string. It is therefore equivalent to
|
|
||||||
// fmt.Sprint(args...)
|
|
||||||
// except that each argument is indirected (if a pointer), as required,
|
|
||||||
// using the same rules as the default string evaluation during template
|
|
||||||
// execution.
|
|
||||||
func evalArgs(args []interface{}) string {
|
|
||||||
ok := false
|
|
||||||
var s string
|
|
||||||
// Fast path for simple common case.
|
|
||||||
if len(args) == 1 {
|
|
||||||
s, ok = args[0].(string)
|
|
||||||
}
|
|
||||||
if !ok {
|
|
||||||
for i, arg := range args {
|
|
||||||
a, ok := printableValue(reflect.ValueOf(arg))
|
|
||||||
if ok {
|
|
||||||
args[i] = a
|
|
||||||
} // else left fmt do its thing
|
|
||||||
}
|
|
||||||
s = fmt.Sprint(args...)
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
1
vendor/github.com/alecthomas/template/go.mod
generated
vendored
1
vendor/github.com/alecthomas/template/go.mod
generated
vendored
|
@ -1 +0,0 @@
|
||||||
module github.com/alecthomas/template
|
|
108
vendor/github.com/alecthomas/template/helper.go
generated
vendored
108
vendor/github.com/alecthomas/template/helper.go
generated
vendored
|
@ -1,108 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
// Helper functions to make constructing templates easier.
|
|
||||||
|
|
||||||
package template
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"path/filepath"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Functions and methods to parse templates.
|
|
||||||
|
|
||||||
// Must is a helper that wraps a call to a function returning (*Template, error)
|
|
||||||
// and panics if the error is non-nil. It is intended for use in variable
|
|
||||||
// initializations such as
|
|
||||||
// var t = template.Must(template.New("name").Parse("text"))
|
|
||||||
func Must(t *Template, err error) *Template {
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParseFiles creates a new Template and parses the template definitions from
|
|
||||||
// the named files. The returned template's name will have the (base) name and
|
|
||||||
// (parsed) contents of the first file. There must be at least one file.
|
|
||||||
// If an error occurs, parsing stops and the returned *Template is nil.
|
|
||||||
func ParseFiles(filenames ...string) (*Template, error) {
|
|
||||||
return parseFiles(nil, filenames...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParseFiles parses the named files and associates the resulting templates with
|
|
||||||
// t. If an error occurs, parsing stops and the returned template is nil;
|
|
||||||
// otherwise it is t. There must be at least one file.
|
|
||||||
func (t *Template) ParseFiles(filenames ...string) (*Template, error) {
|
|
||||||
return parseFiles(t, filenames...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseFiles is the helper for the method and function. If the argument
|
|
||||||
// template is nil, it is created from the first file.
|
|
||||||
func parseFiles(t *Template, filenames ...string) (*Template, error) {
|
|
||||||
if len(filenames) == 0 {
|
|
||||||
// Not really a problem, but be consistent.
|
|
||||||
return nil, fmt.Errorf("template: no files named in call to ParseFiles")
|
|
||||||
}
|
|
||||||
for _, filename := range filenames {
|
|
||||||
b, err := ioutil.ReadFile(filename)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
s := string(b)
|
|
||||||
name := filepath.Base(filename)
|
|
||||||
// First template becomes return value if not already defined,
|
|
||||||
// and we use that one for subsequent New calls to associate
|
|
||||||
// all the templates together. Also, if this file has the same name
|
|
||||||
// as t, this file becomes the contents of t, so
|
|
||||||
// t, err := New(name).Funcs(xxx).ParseFiles(name)
|
|
||||||
// works. Otherwise we create a new template associated with t.
|
|
||||||
var tmpl *Template
|
|
||||||
if t == nil {
|
|
||||||
t = New(name)
|
|
||||||
}
|
|
||||||
if name == t.Name() {
|
|
||||||
tmpl = t
|
|
||||||
} else {
|
|
||||||
tmpl = t.New(name)
|
|
||||||
}
|
|
||||||
_, err = tmpl.Parse(s)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return t, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParseGlob creates a new Template and parses the template definitions from the
|
|
||||||
// files identified by the pattern, which must match at least one file. The
|
|
||||||
// returned template will have the (base) name and (parsed) contents of the
|
|
||||||
// first file matched by the pattern. ParseGlob is equivalent to calling
|
|
||||||
// ParseFiles with the list of files matched by the pattern.
|
|
||||||
func ParseGlob(pattern string) (*Template, error) {
|
|
||||||
return parseGlob(nil, pattern)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParseGlob parses the template definitions in the files identified by the
|
|
||||||
// pattern and associates the resulting templates with t. The pattern is
|
|
||||||
// processed by filepath.Glob and must match at least one file. ParseGlob is
|
|
||||||
// equivalent to calling t.ParseFiles with the list of files matched by the
|
|
||||||
// pattern.
|
|
||||||
func (t *Template) ParseGlob(pattern string) (*Template, error) {
|
|
||||||
return parseGlob(t, pattern)
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseGlob is the implementation of the function and method ParseGlob.
|
|
||||||
func parseGlob(t *Template, pattern string) (*Template, error) {
|
|
||||||
filenames, err := filepath.Glob(pattern)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if len(filenames) == 0 {
|
|
||||||
return nil, fmt.Errorf("template: pattern matches no files: %#q", pattern)
|
|
||||||
}
|
|
||||||
return parseFiles(t, filenames...)
|
|
||||||
}
|
|
556
vendor/github.com/alecthomas/template/parse/lex.go
generated
vendored
556
vendor/github.com/alecthomas/template/parse/lex.go
generated
vendored
|
@ -1,556 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package parse
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
"unicode"
|
|
||||||
"unicode/utf8"
|
|
||||||
)
|
|
||||||
|
|
||||||
// item represents a token or text string returned from the scanner.
|
|
||||||
type item struct {
|
|
||||||
typ itemType // The type of this item.
|
|
||||||
pos Pos // The starting position, in bytes, of this item in the input string.
|
|
||||||
val string // The value of this item.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i item) String() string {
|
|
||||||
switch {
|
|
||||||
case i.typ == itemEOF:
|
|
||||||
return "EOF"
|
|
||||||
case i.typ == itemError:
|
|
||||||
return i.val
|
|
||||||
case i.typ > itemKeyword:
|
|
||||||
return fmt.Sprintf("<%s>", i.val)
|
|
||||||
case len(i.val) > 10:
|
|
||||||
return fmt.Sprintf("%.10q...", i.val)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("%q", i.val)
|
|
||||||
}
|
|
||||||
|
|
||||||
// itemType identifies the type of lex items.
|
|
||||||
type itemType int
|
|
||||||
|
|
||||||
const (
|
|
||||||
itemError itemType = iota // error occurred; value is text of error
|
|
||||||
itemBool // boolean constant
|
|
||||||
itemChar // printable ASCII character; grab bag for comma etc.
|
|
||||||
itemCharConstant // character constant
|
|
||||||
itemComplex // complex constant (1+2i); imaginary is just a number
|
|
||||||
itemColonEquals // colon-equals (':=') introducing a declaration
|
|
||||||
itemEOF
|
|
||||||
itemField // alphanumeric identifier starting with '.'
|
|
||||||
itemIdentifier // alphanumeric identifier not starting with '.'
|
|
||||||
itemLeftDelim // left action delimiter
|
|
||||||
itemLeftParen // '(' inside action
|
|
||||||
itemNumber // simple number, including imaginary
|
|
||||||
itemPipe // pipe symbol
|
|
||||||
itemRawString // raw quoted string (includes quotes)
|
|
||||||
itemRightDelim // right action delimiter
|
|
||||||
itemElideNewline // elide newline after right delim
|
|
||||||
itemRightParen // ')' inside action
|
|
||||||
itemSpace // run of spaces separating arguments
|
|
||||||
itemString // quoted string (includes quotes)
|
|
||||||
itemText // plain text
|
|
||||||
itemVariable // variable starting with '$', such as '$' or '$1' or '$hello'
|
|
||||||
// Keywords appear after all the rest.
|
|
||||||
itemKeyword // used only to delimit the keywords
|
|
||||||
itemDot // the cursor, spelled '.'
|
|
||||||
itemDefine // define keyword
|
|
||||||
itemElse // else keyword
|
|
||||||
itemEnd // end keyword
|
|
||||||
itemIf // if keyword
|
|
||||||
itemNil // the untyped nil constant, easiest to treat as a keyword
|
|
||||||
itemRange // range keyword
|
|
||||||
itemTemplate // template keyword
|
|
||||||
itemWith // with keyword
|
|
||||||
)
|
|
||||||
|
|
||||||
var key = map[string]itemType{
|
|
||||||
".": itemDot,
|
|
||||||
"define": itemDefine,
|
|
||||||
"else": itemElse,
|
|
||||||
"end": itemEnd,
|
|
||||||
"if": itemIf,
|
|
||||||
"range": itemRange,
|
|
||||||
"nil": itemNil,
|
|
||||||
"template": itemTemplate,
|
|
||||||
"with": itemWith,
|
|
||||||
}
|
|
||||||
|
|
||||||
const eof = -1
|
|
||||||
|
|
||||||
// stateFn represents the state of the scanner as a function that returns the next state.
|
|
||||||
type stateFn func(*lexer) stateFn
|
|
||||||
|
|
||||||
// lexer holds the state of the scanner.
|
|
||||||
type lexer struct {
|
|
||||||
name string // the name of the input; used only for error reports
|
|
||||||
input string // the string being scanned
|
|
||||||
leftDelim string // start of action
|
|
||||||
rightDelim string // end of action
|
|
||||||
state stateFn // the next lexing function to enter
|
|
||||||
pos Pos // current position in the input
|
|
||||||
start Pos // start position of this item
|
|
||||||
width Pos // width of last rune read from input
|
|
||||||
lastPos Pos // position of most recent item returned by nextItem
|
|
||||||
items chan item // channel of scanned items
|
|
||||||
parenDepth int // nesting depth of ( ) exprs
|
|
||||||
}
|
|
||||||
|
|
||||||
// next returns the next rune in the input.
|
|
||||||
func (l *lexer) next() rune {
|
|
||||||
if int(l.pos) >= len(l.input) {
|
|
||||||
l.width = 0
|
|
||||||
return eof
|
|
||||||
}
|
|
||||||
r, w := utf8.DecodeRuneInString(l.input[l.pos:])
|
|
||||||
l.width = Pos(w)
|
|
||||||
l.pos += l.width
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
// peek returns but does not consume the next rune in the input.
|
|
||||||
func (l *lexer) peek() rune {
|
|
||||||
r := l.next()
|
|
||||||
l.backup()
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
// backup steps back one rune. Can only be called once per call of next.
|
|
||||||
func (l *lexer) backup() {
|
|
||||||
l.pos -= l.width
|
|
||||||
}
|
|
||||||
|
|
||||||
// emit passes an item back to the client.
|
|
||||||
func (l *lexer) emit(t itemType) {
|
|
||||||
l.items <- item{t, l.start, l.input[l.start:l.pos]}
|
|
||||||
l.start = l.pos
|
|
||||||
}
|
|
||||||
|
|
||||||
// ignore skips over the pending input before this point.
|
|
||||||
func (l *lexer) ignore() {
|
|
||||||
l.start = l.pos
|
|
||||||
}
|
|
||||||
|
|
||||||
// accept consumes the next rune if it's from the valid set.
|
|
||||||
func (l *lexer) accept(valid string) bool {
|
|
||||||
if strings.IndexRune(valid, l.next()) >= 0 {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
l.backup()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// acceptRun consumes a run of runes from the valid set.
|
|
||||||
func (l *lexer) acceptRun(valid string) {
|
|
||||||
for strings.IndexRune(valid, l.next()) >= 0 {
|
|
||||||
}
|
|
||||||
l.backup()
|
|
||||||
}
|
|
||||||
|
|
||||||
// lineNumber reports which line we're on, based on the position of
|
|
||||||
// the previous item returned by nextItem. Doing it this way
|
|
||||||
// means we don't have to worry about peek double counting.
|
|
||||||
func (l *lexer) lineNumber() int {
|
|
||||||
return 1 + strings.Count(l.input[:l.lastPos], "\n")
|
|
||||||
}
|
|
||||||
|
|
||||||
// errorf returns an error token and terminates the scan by passing
|
|
||||||
// back a nil pointer that will be the next state, terminating l.nextItem.
|
|
||||||
func (l *lexer) errorf(format string, args ...interface{}) stateFn {
|
|
||||||
l.items <- item{itemError, l.start, fmt.Sprintf(format, args...)}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// nextItem returns the next item from the input.
|
|
||||||
func (l *lexer) nextItem() item {
|
|
||||||
item := <-l.items
|
|
||||||
l.lastPos = item.pos
|
|
||||||
return item
|
|
||||||
}
|
|
||||||
|
|
||||||
// lex creates a new scanner for the input string.
|
|
||||||
func lex(name, input, left, right string) *lexer {
|
|
||||||
if left == "" {
|
|
||||||
left = leftDelim
|
|
||||||
}
|
|
||||||
if right == "" {
|
|
||||||
right = rightDelim
|
|
||||||
}
|
|
||||||
l := &lexer{
|
|
||||||
name: name,
|
|
||||||
input: input,
|
|
||||||
leftDelim: left,
|
|
||||||
rightDelim: right,
|
|
||||||
items: make(chan item),
|
|
||||||
}
|
|
||||||
go l.run()
|
|
||||||
return l
|
|
||||||
}
|
|
||||||
|
|
||||||
// run runs the state machine for the lexer.
|
|
||||||
func (l *lexer) run() {
|
|
||||||
for l.state = lexText; l.state != nil; {
|
|
||||||
l.state = l.state(l)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// state functions
|
|
||||||
|
|
||||||
const (
|
|
||||||
leftDelim = "{{"
|
|
||||||
rightDelim = "}}"
|
|
||||||
leftComment = "/*"
|
|
||||||
rightComment = "*/"
|
|
||||||
)
|
|
||||||
|
|
||||||
// lexText scans until an opening action delimiter, "{{".
|
|
||||||
func lexText(l *lexer) stateFn {
|
|
||||||
for {
|
|
||||||
if strings.HasPrefix(l.input[l.pos:], l.leftDelim) {
|
|
||||||
if l.pos > l.start {
|
|
||||||
l.emit(itemText)
|
|
||||||
}
|
|
||||||
return lexLeftDelim
|
|
||||||
}
|
|
||||||
if l.next() == eof {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Correctly reached EOF.
|
|
||||||
if l.pos > l.start {
|
|
||||||
l.emit(itemText)
|
|
||||||
}
|
|
||||||
l.emit(itemEOF)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexLeftDelim scans the left delimiter, which is known to be present.
|
|
||||||
func lexLeftDelim(l *lexer) stateFn {
|
|
||||||
l.pos += Pos(len(l.leftDelim))
|
|
||||||
if strings.HasPrefix(l.input[l.pos:], leftComment) {
|
|
||||||
return lexComment
|
|
||||||
}
|
|
||||||
l.emit(itemLeftDelim)
|
|
||||||
l.parenDepth = 0
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexComment scans a comment. The left comment marker is known to be present.
|
|
||||||
func lexComment(l *lexer) stateFn {
|
|
||||||
l.pos += Pos(len(leftComment))
|
|
||||||
i := strings.Index(l.input[l.pos:], rightComment)
|
|
||||||
if i < 0 {
|
|
||||||
return l.errorf("unclosed comment")
|
|
||||||
}
|
|
||||||
l.pos += Pos(i + len(rightComment))
|
|
||||||
if !strings.HasPrefix(l.input[l.pos:], l.rightDelim) {
|
|
||||||
return l.errorf("comment ends before closing delimiter")
|
|
||||||
|
|
||||||
}
|
|
||||||
l.pos += Pos(len(l.rightDelim))
|
|
||||||
l.ignore()
|
|
||||||
return lexText
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexRightDelim scans the right delimiter, which is known to be present.
|
|
||||||
func lexRightDelim(l *lexer) stateFn {
|
|
||||||
l.pos += Pos(len(l.rightDelim))
|
|
||||||
l.emit(itemRightDelim)
|
|
||||||
if l.peek() == '\\' {
|
|
||||||
l.pos++
|
|
||||||
l.emit(itemElideNewline)
|
|
||||||
}
|
|
||||||
return lexText
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexInsideAction scans the elements inside action delimiters.
|
|
||||||
func lexInsideAction(l *lexer) stateFn {
|
|
||||||
// Either number, quoted string, or identifier.
|
|
||||||
// Spaces separate arguments; runs of spaces turn into itemSpace.
|
|
||||||
// Pipe symbols separate and are emitted.
|
|
||||||
if strings.HasPrefix(l.input[l.pos:], l.rightDelim+"\\") || strings.HasPrefix(l.input[l.pos:], l.rightDelim) {
|
|
||||||
if l.parenDepth == 0 {
|
|
||||||
return lexRightDelim
|
|
||||||
}
|
|
||||||
return l.errorf("unclosed left paren")
|
|
||||||
}
|
|
||||||
switch r := l.next(); {
|
|
||||||
case r == eof || isEndOfLine(r):
|
|
||||||
return l.errorf("unclosed action")
|
|
||||||
case isSpace(r):
|
|
||||||
return lexSpace
|
|
||||||
case r == ':':
|
|
||||||
if l.next() != '=' {
|
|
||||||
return l.errorf("expected :=")
|
|
||||||
}
|
|
||||||
l.emit(itemColonEquals)
|
|
||||||
case r == '|':
|
|
||||||
l.emit(itemPipe)
|
|
||||||
case r == '"':
|
|
||||||
return lexQuote
|
|
||||||
case r == '`':
|
|
||||||
return lexRawQuote
|
|
||||||
case r == '$':
|
|
||||||
return lexVariable
|
|
||||||
case r == '\'':
|
|
||||||
return lexChar
|
|
||||||
case r == '.':
|
|
||||||
// special look-ahead for ".field" so we don't break l.backup().
|
|
||||||
if l.pos < Pos(len(l.input)) {
|
|
||||||
r := l.input[l.pos]
|
|
||||||
if r < '0' || '9' < r {
|
|
||||||
return lexField
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fallthrough // '.' can start a number.
|
|
||||||
case r == '+' || r == '-' || ('0' <= r && r <= '9'):
|
|
||||||
l.backup()
|
|
||||||
return lexNumber
|
|
||||||
case isAlphaNumeric(r):
|
|
||||||
l.backup()
|
|
||||||
return lexIdentifier
|
|
||||||
case r == '(':
|
|
||||||
l.emit(itemLeftParen)
|
|
||||||
l.parenDepth++
|
|
||||||
return lexInsideAction
|
|
||||||
case r == ')':
|
|
||||||
l.emit(itemRightParen)
|
|
||||||
l.parenDepth--
|
|
||||||
if l.parenDepth < 0 {
|
|
||||||
return l.errorf("unexpected right paren %#U", r)
|
|
||||||
}
|
|
||||||
return lexInsideAction
|
|
||||||
case r <= unicode.MaxASCII && unicode.IsPrint(r):
|
|
||||||
l.emit(itemChar)
|
|
||||||
return lexInsideAction
|
|
||||||
default:
|
|
||||||
return l.errorf("unrecognized character in action: %#U", r)
|
|
||||||
}
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexSpace scans a run of space characters.
|
|
||||||
// One space has already been seen.
|
|
||||||
func lexSpace(l *lexer) stateFn {
|
|
||||||
for isSpace(l.peek()) {
|
|
||||||
l.next()
|
|
||||||
}
|
|
||||||
l.emit(itemSpace)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexIdentifier scans an alphanumeric.
|
|
||||||
func lexIdentifier(l *lexer) stateFn {
|
|
||||||
Loop:
|
|
||||||
for {
|
|
||||||
switch r := l.next(); {
|
|
||||||
case isAlphaNumeric(r):
|
|
||||||
// absorb.
|
|
||||||
default:
|
|
||||||
l.backup()
|
|
||||||
word := l.input[l.start:l.pos]
|
|
||||||
if !l.atTerminator() {
|
|
||||||
return l.errorf("bad character %#U", r)
|
|
||||||
}
|
|
||||||
switch {
|
|
||||||
case key[word] > itemKeyword:
|
|
||||||
l.emit(key[word])
|
|
||||||
case word[0] == '.':
|
|
||||||
l.emit(itemField)
|
|
||||||
case word == "true", word == "false":
|
|
||||||
l.emit(itemBool)
|
|
||||||
default:
|
|
||||||
l.emit(itemIdentifier)
|
|
||||||
}
|
|
||||||
break Loop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexField scans a field: .Alphanumeric.
|
|
||||||
// The . has been scanned.
|
|
||||||
func lexField(l *lexer) stateFn {
|
|
||||||
return lexFieldOrVariable(l, itemField)
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexVariable scans a Variable: $Alphanumeric.
|
|
||||||
// The $ has been scanned.
|
|
||||||
func lexVariable(l *lexer) stateFn {
|
|
||||||
if l.atTerminator() { // Nothing interesting follows -> "$".
|
|
||||||
l.emit(itemVariable)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
return lexFieldOrVariable(l, itemVariable)
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexVariable scans a field or variable: [.$]Alphanumeric.
|
|
||||||
// The . or $ has been scanned.
|
|
||||||
func lexFieldOrVariable(l *lexer, typ itemType) stateFn {
|
|
||||||
if l.atTerminator() { // Nothing interesting follows -> "." or "$".
|
|
||||||
if typ == itemVariable {
|
|
||||||
l.emit(itemVariable)
|
|
||||||
} else {
|
|
||||||
l.emit(itemDot)
|
|
||||||
}
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
var r rune
|
|
||||||
for {
|
|
||||||
r = l.next()
|
|
||||||
if !isAlphaNumeric(r) {
|
|
||||||
l.backup()
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !l.atTerminator() {
|
|
||||||
return l.errorf("bad character %#U", r)
|
|
||||||
}
|
|
||||||
l.emit(typ)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// atTerminator reports whether the input is at valid termination character to
|
|
||||||
// appear after an identifier. Breaks .X.Y into two pieces. Also catches cases
|
|
||||||
// like "$x+2" not being acceptable without a space, in case we decide one
|
|
||||||
// day to implement arithmetic.
|
|
||||||
func (l *lexer) atTerminator() bool {
|
|
||||||
r := l.peek()
|
|
||||||
if isSpace(r) || isEndOfLine(r) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
switch r {
|
|
||||||
case eof, '.', ',', '|', ':', ')', '(':
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
// Does r start the delimiter? This can be ambiguous (with delim=="//", $x/2 will
|
|
||||||
// succeed but should fail) but only in extremely rare cases caused by willfully
|
|
||||||
// bad choice of delimiter.
|
|
||||||
if rd, _ := utf8.DecodeRuneInString(l.rightDelim); rd == r {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexChar scans a character constant. The initial quote is already
|
|
||||||
// scanned. Syntax checking is done by the parser.
|
|
||||||
func lexChar(l *lexer) stateFn {
|
|
||||||
Loop:
|
|
||||||
for {
|
|
||||||
switch l.next() {
|
|
||||||
case '\\':
|
|
||||||
if r := l.next(); r != eof && r != '\n' {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
fallthrough
|
|
||||||
case eof, '\n':
|
|
||||||
return l.errorf("unterminated character constant")
|
|
||||||
case '\'':
|
|
||||||
break Loop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
l.emit(itemCharConstant)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexNumber scans a number: decimal, octal, hex, float, or imaginary. This
|
|
||||||
// isn't a perfect number scanner - for instance it accepts "." and "0x0.2"
|
|
||||||
// and "089" - but when it's wrong the input is invalid and the parser (via
|
|
||||||
// strconv) will notice.
|
|
||||||
func lexNumber(l *lexer) stateFn {
|
|
||||||
if !l.scanNumber() {
|
|
||||||
return l.errorf("bad number syntax: %q", l.input[l.start:l.pos])
|
|
||||||
}
|
|
||||||
if sign := l.peek(); sign == '+' || sign == '-' {
|
|
||||||
// Complex: 1+2i. No spaces, must end in 'i'.
|
|
||||||
if !l.scanNumber() || l.input[l.pos-1] != 'i' {
|
|
||||||
return l.errorf("bad number syntax: %q", l.input[l.start:l.pos])
|
|
||||||
}
|
|
||||||
l.emit(itemComplex)
|
|
||||||
} else {
|
|
||||||
l.emit(itemNumber)
|
|
||||||
}
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *lexer) scanNumber() bool {
|
|
||||||
// Optional leading sign.
|
|
||||||
l.accept("+-")
|
|
||||||
// Is it hex?
|
|
||||||
digits := "0123456789"
|
|
||||||
if l.accept("0") && l.accept("xX") {
|
|
||||||
digits = "0123456789abcdefABCDEF"
|
|
||||||
}
|
|
||||||
l.acceptRun(digits)
|
|
||||||
if l.accept(".") {
|
|
||||||
l.acceptRun(digits)
|
|
||||||
}
|
|
||||||
if l.accept("eE") {
|
|
||||||
l.accept("+-")
|
|
||||||
l.acceptRun("0123456789")
|
|
||||||
}
|
|
||||||
// Is it imaginary?
|
|
||||||
l.accept("i")
|
|
||||||
// Next thing mustn't be alphanumeric.
|
|
||||||
if isAlphaNumeric(l.peek()) {
|
|
||||||
l.next()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexQuote scans a quoted string.
|
|
||||||
func lexQuote(l *lexer) stateFn {
|
|
||||||
Loop:
|
|
||||||
for {
|
|
||||||
switch l.next() {
|
|
||||||
case '\\':
|
|
||||||
if r := l.next(); r != eof && r != '\n' {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
fallthrough
|
|
||||||
case eof, '\n':
|
|
||||||
return l.errorf("unterminated quoted string")
|
|
||||||
case '"':
|
|
||||||
break Loop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
l.emit(itemString)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// lexRawQuote scans a raw quoted string.
|
|
||||||
func lexRawQuote(l *lexer) stateFn {
|
|
||||||
Loop:
|
|
||||||
for {
|
|
||||||
switch l.next() {
|
|
||||||
case eof, '\n':
|
|
||||||
return l.errorf("unterminated raw quoted string")
|
|
||||||
case '`':
|
|
||||||
break Loop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
l.emit(itemRawString)
|
|
||||||
return lexInsideAction
|
|
||||||
}
|
|
||||||
|
|
||||||
// isSpace reports whether r is a space character.
|
|
||||||
func isSpace(r rune) bool {
|
|
||||||
return r == ' ' || r == '\t'
|
|
||||||
}
|
|
||||||
|
|
||||||
// isEndOfLine reports whether r is an end-of-line character.
|
|
||||||
func isEndOfLine(r rune) bool {
|
|
||||||
return r == '\r' || r == '\n'
|
|
||||||
}
|
|
||||||
|
|
||||||
// isAlphaNumeric reports whether r is an alphabetic, digit, or underscore.
|
|
||||||
func isAlphaNumeric(r rune) bool {
|
|
||||||
return r == '_' || unicode.IsLetter(r) || unicode.IsDigit(r)
|
|
||||||
}
|
|
834
vendor/github.com/alecthomas/template/parse/node.go
generated
vendored
834
vendor/github.com/alecthomas/template/parse/node.go
generated
vendored
|
@ -1,834 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
// Parse nodes.
|
|
||||||
|
|
||||||
package parse
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var textFormat = "%s" // Changed to "%q" in tests for better error messages.
|
|
||||||
|
|
||||||
// A Node is an element in the parse tree. The interface is trivial.
|
|
||||||
// The interface contains an unexported method so that only
|
|
||||||
// types local to this package can satisfy it.
|
|
||||||
type Node interface {
|
|
||||||
Type() NodeType
|
|
||||||
String() string
|
|
||||||
// Copy does a deep copy of the Node and all its components.
|
|
||||||
// To avoid type assertions, some XxxNodes also have specialized
|
|
||||||
// CopyXxx methods that return *XxxNode.
|
|
||||||
Copy() Node
|
|
||||||
Position() Pos // byte position of start of node in full original input string
|
|
||||||
// tree returns the containing *Tree.
|
|
||||||
// It is unexported so all implementations of Node are in this package.
|
|
||||||
tree() *Tree
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeType identifies the type of a parse tree node.
|
|
||||||
type NodeType int
|
|
||||||
|
|
||||||
// Pos represents a byte position in the original input text from which
|
|
||||||
// this template was parsed.
|
|
||||||
type Pos int
|
|
||||||
|
|
||||||
func (p Pos) Position() Pos {
|
|
||||||
return p
|
|
||||||
}
|
|
||||||
|
|
||||||
// Type returns itself and provides an easy default implementation
|
|
||||||
// for embedding in a Node. Embedded in all non-trivial Nodes.
|
|
||||||
func (t NodeType) Type() NodeType {
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
NodeText NodeType = iota // Plain text.
|
|
||||||
NodeAction // A non-control action such as a field evaluation.
|
|
||||||
NodeBool // A boolean constant.
|
|
||||||
NodeChain // A sequence of field accesses.
|
|
||||||
NodeCommand // An element of a pipeline.
|
|
||||||
NodeDot // The cursor, dot.
|
|
||||||
nodeElse // An else action. Not added to tree.
|
|
||||||
nodeEnd // An end action. Not added to tree.
|
|
||||||
NodeField // A field or method name.
|
|
||||||
NodeIdentifier // An identifier; always a function name.
|
|
||||||
NodeIf // An if action.
|
|
||||||
NodeList // A list of Nodes.
|
|
||||||
NodeNil // An untyped nil constant.
|
|
||||||
NodeNumber // A numerical constant.
|
|
||||||
NodePipe // A pipeline of commands.
|
|
||||||
NodeRange // A range action.
|
|
||||||
NodeString // A string constant.
|
|
||||||
NodeTemplate // A template invocation action.
|
|
||||||
NodeVariable // A $ variable.
|
|
||||||
NodeWith // A with action.
|
|
||||||
)
|
|
||||||
|
|
||||||
// Nodes.
|
|
||||||
|
|
||||||
// ListNode holds a sequence of nodes.
|
|
||||||
type ListNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Nodes []Node // The element nodes in lexical order.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newList(pos Pos) *ListNode {
|
|
||||||
return &ListNode{tr: t, NodeType: NodeList, Pos: pos}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *ListNode) append(n Node) {
|
|
||||||
l.Nodes = append(l.Nodes, n)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *ListNode) tree() *Tree {
|
|
||||||
return l.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *ListNode) String() string {
|
|
||||||
b := new(bytes.Buffer)
|
|
||||||
for _, n := range l.Nodes {
|
|
||||||
fmt.Fprint(b, n)
|
|
||||||
}
|
|
||||||
return b.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *ListNode) CopyList() *ListNode {
|
|
||||||
if l == nil {
|
|
||||||
return l
|
|
||||||
}
|
|
||||||
n := l.tr.newList(l.Pos)
|
|
||||||
for _, elem := range l.Nodes {
|
|
||||||
n.append(elem.Copy())
|
|
||||||
}
|
|
||||||
return n
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *ListNode) Copy() Node {
|
|
||||||
return l.CopyList()
|
|
||||||
}
|
|
||||||
|
|
||||||
// TextNode holds plain text.
|
|
||||||
type TextNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Text []byte // The text; may span newlines.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newText(pos Pos, text string) *TextNode {
|
|
||||||
return &TextNode{tr: t, NodeType: NodeText, Pos: pos, Text: []byte(text)}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TextNode) String() string {
|
|
||||||
return fmt.Sprintf(textFormat, t.Text)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TextNode) tree() *Tree {
|
|
||||||
return t.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TextNode) Copy() Node {
|
|
||||||
return &TextNode{tr: t.tr, NodeType: NodeText, Pos: t.Pos, Text: append([]byte{}, t.Text...)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PipeNode holds a pipeline with optional declaration
|
|
||||||
type PipeNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Line int // The line number in the input (deprecated; kept for compatibility)
|
|
||||||
Decl []*VariableNode // Variable declarations in lexical order.
|
|
||||||
Cmds []*CommandNode // The commands in lexical order.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newPipeline(pos Pos, line int, decl []*VariableNode) *PipeNode {
|
|
||||||
return &PipeNode{tr: t, NodeType: NodePipe, Pos: pos, Line: line, Decl: decl}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *PipeNode) append(command *CommandNode) {
|
|
||||||
p.Cmds = append(p.Cmds, command)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *PipeNode) String() string {
|
|
||||||
s := ""
|
|
||||||
if len(p.Decl) > 0 {
|
|
||||||
for i, v := range p.Decl {
|
|
||||||
if i > 0 {
|
|
||||||
s += ", "
|
|
||||||
}
|
|
||||||
s += v.String()
|
|
||||||
}
|
|
||||||
s += " := "
|
|
||||||
}
|
|
||||||
for i, c := range p.Cmds {
|
|
||||||
if i > 0 {
|
|
||||||
s += " | "
|
|
||||||
}
|
|
||||||
s += c.String()
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *PipeNode) tree() *Tree {
|
|
||||||
return p.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *PipeNode) CopyPipe() *PipeNode {
|
|
||||||
if p == nil {
|
|
||||||
return p
|
|
||||||
}
|
|
||||||
var decl []*VariableNode
|
|
||||||
for _, d := range p.Decl {
|
|
||||||
decl = append(decl, d.Copy().(*VariableNode))
|
|
||||||
}
|
|
||||||
n := p.tr.newPipeline(p.Pos, p.Line, decl)
|
|
||||||
for _, c := range p.Cmds {
|
|
||||||
n.append(c.Copy().(*CommandNode))
|
|
||||||
}
|
|
||||||
return n
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *PipeNode) Copy() Node {
|
|
||||||
return p.CopyPipe()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ActionNode holds an action (something bounded by delimiters).
|
|
||||||
// Control actions have their own nodes; ActionNode represents simple
|
|
||||||
// ones such as field evaluations and parenthesized pipelines.
|
|
||||||
type ActionNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Line int // The line number in the input (deprecated; kept for compatibility)
|
|
||||||
Pipe *PipeNode // The pipeline in the action.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newAction(pos Pos, line int, pipe *PipeNode) *ActionNode {
|
|
||||||
return &ActionNode{tr: t, NodeType: NodeAction, Pos: pos, Line: line, Pipe: pipe}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *ActionNode) String() string {
|
|
||||||
return fmt.Sprintf("{{%s}}", a.Pipe)
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *ActionNode) tree() *Tree {
|
|
||||||
return a.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *ActionNode) Copy() Node {
|
|
||||||
return a.tr.newAction(a.Pos, a.Line, a.Pipe.CopyPipe())
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// CommandNode holds a command (a pipeline inside an evaluating action).
|
|
||||||
type CommandNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Args []Node // Arguments in lexical order: Identifier, field, or constant.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newCommand(pos Pos) *CommandNode {
|
|
||||||
return &CommandNode{tr: t, NodeType: NodeCommand, Pos: pos}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *CommandNode) append(arg Node) {
|
|
||||||
c.Args = append(c.Args, arg)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *CommandNode) String() string {
|
|
||||||
s := ""
|
|
||||||
for i, arg := range c.Args {
|
|
||||||
if i > 0 {
|
|
||||||
s += " "
|
|
||||||
}
|
|
||||||
if arg, ok := arg.(*PipeNode); ok {
|
|
||||||
s += "(" + arg.String() + ")"
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
s += arg.String()
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *CommandNode) tree() *Tree {
|
|
||||||
return c.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *CommandNode) Copy() Node {
|
|
||||||
if c == nil {
|
|
||||||
return c
|
|
||||||
}
|
|
||||||
n := c.tr.newCommand(c.Pos)
|
|
||||||
for _, c := range c.Args {
|
|
||||||
n.append(c.Copy())
|
|
||||||
}
|
|
||||||
return n
|
|
||||||
}
|
|
||||||
|
|
||||||
// IdentifierNode holds an identifier.
|
|
||||||
type IdentifierNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Ident string // The identifier's name.
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewIdentifier returns a new IdentifierNode with the given identifier name.
|
|
||||||
func NewIdentifier(ident string) *IdentifierNode {
|
|
||||||
return &IdentifierNode{NodeType: NodeIdentifier, Ident: ident}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetPos sets the position. NewIdentifier is a public method so we can't modify its signature.
|
|
||||||
// Chained for convenience.
|
|
||||||
// TODO: fix one day?
|
|
||||||
func (i *IdentifierNode) SetPos(pos Pos) *IdentifierNode {
|
|
||||||
i.Pos = pos
|
|
||||||
return i
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetTree sets the parent tree for the node. NewIdentifier is a public method so we can't modify its signature.
|
|
||||||
// Chained for convenience.
|
|
||||||
// TODO: fix one day?
|
|
||||||
func (i *IdentifierNode) SetTree(t *Tree) *IdentifierNode {
|
|
||||||
i.tr = t
|
|
||||||
return i
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *IdentifierNode) String() string {
|
|
||||||
return i.Ident
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *IdentifierNode) tree() *Tree {
|
|
||||||
return i.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *IdentifierNode) Copy() Node {
|
|
||||||
return NewIdentifier(i.Ident).SetTree(i.tr).SetPos(i.Pos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// VariableNode holds a list of variable names, possibly with chained field
|
|
||||||
// accesses. The dollar sign is part of the (first) name.
|
|
||||||
type VariableNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Ident []string // Variable name and fields in lexical order.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newVariable(pos Pos, ident string) *VariableNode {
|
|
||||||
return &VariableNode{tr: t, NodeType: NodeVariable, Pos: pos, Ident: strings.Split(ident, ".")}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *VariableNode) String() string {
|
|
||||||
s := ""
|
|
||||||
for i, id := range v.Ident {
|
|
||||||
if i > 0 {
|
|
||||||
s += "."
|
|
||||||
}
|
|
||||||
s += id
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *VariableNode) tree() *Tree {
|
|
||||||
return v.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *VariableNode) Copy() Node {
|
|
||||||
return &VariableNode{tr: v.tr, NodeType: NodeVariable, Pos: v.Pos, Ident: append([]string{}, v.Ident...)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// DotNode holds the special identifier '.'.
|
|
||||||
type DotNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newDot(pos Pos) *DotNode {
|
|
||||||
return &DotNode{tr: t, NodeType: NodeDot, Pos: pos}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *DotNode) Type() NodeType {
|
|
||||||
// Override method on embedded NodeType for API compatibility.
|
|
||||||
// TODO: Not really a problem; could change API without effect but
|
|
||||||
// api tool complains.
|
|
||||||
return NodeDot
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *DotNode) String() string {
|
|
||||||
return "."
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *DotNode) tree() *Tree {
|
|
||||||
return d.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *DotNode) Copy() Node {
|
|
||||||
return d.tr.newDot(d.Pos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NilNode holds the special identifier 'nil' representing an untyped nil constant.
|
|
||||||
type NilNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newNil(pos Pos) *NilNode {
|
|
||||||
return &NilNode{tr: t, NodeType: NodeNil, Pos: pos}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NilNode) Type() NodeType {
|
|
||||||
// Override method on embedded NodeType for API compatibility.
|
|
||||||
// TODO: Not really a problem; could change API without effect but
|
|
||||||
// api tool complains.
|
|
||||||
return NodeNil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NilNode) String() string {
|
|
||||||
return "nil"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NilNode) tree() *Tree {
|
|
||||||
return n.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NilNode) Copy() Node {
|
|
||||||
return n.tr.newNil(n.Pos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// FieldNode holds a field (identifier starting with '.').
|
|
||||||
// The names may be chained ('.x.y').
|
|
||||||
// The period is dropped from each ident.
|
|
||||||
type FieldNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Ident []string // The identifiers in lexical order.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newField(pos Pos, ident string) *FieldNode {
|
|
||||||
return &FieldNode{tr: t, NodeType: NodeField, Pos: pos, Ident: strings.Split(ident[1:], ".")} // [1:] to drop leading period
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *FieldNode) String() string {
|
|
||||||
s := ""
|
|
||||||
for _, id := range f.Ident {
|
|
||||||
s += "." + id
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *FieldNode) tree() *Tree {
|
|
||||||
return f.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *FieldNode) Copy() Node {
|
|
||||||
return &FieldNode{tr: f.tr, NodeType: NodeField, Pos: f.Pos, Ident: append([]string{}, f.Ident...)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ChainNode holds a term followed by a chain of field accesses (identifier starting with '.').
|
|
||||||
// The names may be chained ('.x.y').
|
|
||||||
// The periods are dropped from each ident.
|
|
||||||
type ChainNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Node Node
|
|
||||||
Field []string // The identifiers in lexical order.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newChain(pos Pos, node Node) *ChainNode {
|
|
||||||
return &ChainNode{tr: t, NodeType: NodeChain, Pos: pos, Node: node}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add adds the named field (which should start with a period) to the end of the chain.
|
|
||||||
func (c *ChainNode) Add(field string) {
|
|
||||||
if len(field) == 0 || field[0] != '.' {
|
|
||||||
panic("no dot in field")
|
|
||||||
}
|
|
||||||
field = field[1:] // Remove leading dot.
|
|
||||||
if field == "" {
|
|
||||||
panic("empty field")
|
|
||||||
}
|
|
||||||
c.Field = append(c.Field, field)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *ChainNode) String() string {
|
|
||||||
s := c.Node.String()
|
|
||||||
if _, ok := c.Node.(*PipeNode); ok {
|
|
||||||
s = "(" + s + ")"
|
|
||||||
}
|
|
||||||
for _, field := range c.Field {
|
|
||||||
s += "." + field
|
|
||||||
}
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *ChainNode) tree() *Tree {
|
|
||||||
return c.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *ChainNode) Copy() Node {
|
|
||||||
return &ChainNode{tr: c.tr, NodeType: NodeChain, Pos: c.Pos, Node: c.Node, Field: append([]string{}, c.Field...)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// BoolNode holds a boolean constant.
|
|
||||||
type BoolNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
True bool // The value of the boolean constant.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newBool(pos Pos, true bool) *BoolNode {
|
|
||||||
return &BoolNode{tr: t, NodeType: NodeBool, Pos: pos, True: true}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BoolNode) String() string {
|
|
||||||
if b.True {
|
|
||||||
return "true"
|
|
||||||
}
|
|
||||||
return "false"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BoolNode) tree() *Tree {
|
|
||||||
return b.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BoolNode) Copy() Node {
|
|
||||||
return b.tr.newBool(b.Pos, b.True)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NumberNode holds a number: signed or unsigned integer, float, or complex.
|
|
||||||
// The value is parsed and stored under all the types that can represent the value.
|
|
||||||
// This simulates in a small amount of code the behavior of Go's ideal constants.
|
|
||||||
type NumberNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
IsInt bool // Number has an integral value.
|
|
||||||
IsUint bool // Number has an unsigned integral value.
|
|
||||||
IsFloat bool // Number has a floating-point value.
|
|
||||||
IsComplex bool // Number is complex.
|
|
||||||
Int64 int64 // The signed integer value.
|
|
||||||
Uint64 uint64 // The unsigned integer value.
|
|
||||||
Float64 float64 // The floating-point value.
|
|
||||||
Complex128 complex128 // The complex value.
|
|
||||||
Text string // The original textual representation from the input.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newNumber(pos Pos, text string, typ itemType) (*NumberNode, error) {
|
|
||||||
n := &NumberNode{tr: t, NodeType: NodeNumber, Pos: pos, Text: text}
|
|
||||||
switch typ {
|
|
||||||
case itemCharConstant:
|
|
||||||
rune, _, tail, err := strconv.UnquoteChar(text[1:], text[0])
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if tail != "'" {
|
|
||||||
return nil, fmt.Errorf("malformed character constant: %s", text)
|
|
||||||
}
|
|
||||||
n.Int64 = int64(rune)
|
|
||||||
n.IsInt = true
|
|
||||||
n.Uint64 = uint64(rune)
|
|
||||||
n.IsUint = true
|
|
||||||
n.Float64 = float64(rune) // odd but those are the rules.
|
|
||||||
n.IsFloat = true
|
|
||||||
return n, nil
|
|
||||||
case itemComplex:
|
|
||||||
// fmt.Sscan can parse the pair, so let it do the work.
|
|
||||||
if _, err := fmt.Sscan(text, &n.Complex128); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
n.IsComplex = true
|
|
||||||
n.simplifyComplex()
|
|
||||||
return n, nil
|
|
||||||
}
|
|
||||||
// Imaginary constants can only be complex unless they are zero.
|
|
||||||
if len(text) > 0 && text[len(text)-1] == 'i' {
|
|
||||||
f, err := strconv.ParseFloat(text[:len(text)-1], 64)
|
|
||||||
if err == nil {
|
|
||||||
n.IsComplex = true
|
|
||||||
n.Complex128 = complex(0, f)
|
|
||||||
n.simplifyComplex()
|
|
||||||
return n, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Do integer test first so we get 0x123 etc.
|
|
||||||
u, err := strconv.ParseUint(text, 0, 64) // will fail for -0; fixed below.
|
|
||||||
if err == nil {
|
|
||||||
n.IsUint = true
|
|
||||||
n.Uint64 = u
|
|
||||||
}
|
|
||||||
i, err := strconv.ParseInt(text, 0, 64)
|
|
||||||
if err == nil {
|
|
||||||
n.IsInt = true
|
|
||||||
n.Int64 = i
|
|
||||||
if i == 0 {
|
|
||||||
n.IsUint = true // in case of -0.
|
|
||||||
n.Uint64 = u
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If an integer extraction succeeded, promote the float.
|
|
||||||
if n.IsInt {
|
|
||||||
n.IsFloat = true
|
|
||||||
n.Float64 = float64(n.Int64)
|
|
||||||
} else if n.IsUint {
|
|
||||||
n.IsFloat = true
|
|
||||||
n.Float64 = float64(n.Uint64)
|
|
||||||
} else {
|
|
||||||
f, err := strconv.ParseFloat(text, 64)
|
|
||||||
if err == nil {
|
|
||||||
n.IsFloat = true
|
|
||||||
n.Float64 = f
|
|
||||||
// If a floating-point extraction succeeded, extract the int if needed.
|
|
||||||
if !n.IsInt && float64(int64(f)) == f {
|
|
||||||
n.IsInt = true
|
|
||||||
n.Int64 = int64(f)
|
|
||||||
}
|
|
||||||
if !n.IsUint && float64(uint64(f)) == f {
|
|
||||||
n.IsUint = true
|
|
||||||
n.Uint64 = uint64(f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !n.IsInt && !n.IsUint && !n.IsFloat {
|
|
||||||
return nil, fmt.Errorf("illegal number syntax: %q", text)
|
|
||||||
}
|
|
||||||
return n, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// simplifyComplex pulls out any other types that are represented by the complex number.
|
|
||||||
// These all require that the imaginary part be zero.
|
|
||||||
func (n *NumberNode) simplifyComplex() {
|
|
||||||
n.IsFloat = imag(n.Complex128) == 0
|
|
||||||
if n.IsFloat {
|
|
||||||
n.Float64 = real(n.Complex128)
|
|
||||||
n.IsInt = float64(int64(n.Float64)) == n.Float64
|
|
||||||
if n.IsInt {
|
|
||||||
n.Int64 = int64(n.Float64)
|
|
||||||
}
|
|
||||||
n.IsUint = float64(uint64(n.Float64)) == n.Float64
|
|
||||||
if n.IsUint {
|
|
||||||
n.Uint64 = uint64(n.Float64)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NumberNode) String() string {
|
|
||||||
return n.Text
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NumberNode) tree() *Tree {
|
|
||||||
return n.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *NumberNode) Copy() Node {
|
|
||||||
nn := new(NumberNode)
|
|
||||||
*nn = *n // Easy, fast, correct.
|
|
||||||
return nn
|
|
||||||
}
|
|
||||||
|
|
||||||
// StringNode holds a string constant. The value has been "unquoted".
|
|
||||||
type StringNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Quoted string // The original text of the string, with quotes.
|
|
||||||
Text string // The string, after quote processing.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newString(pos Pos, orig, text string) *StringNode {
|
|
||||||
return &StringNode{tr: t, NodeType: NodeString, Pos: pos, Quoted: orig, Text: text}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *StringNode) String() string {
|
|
||||||
return s.Quoted
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *StringNode) tree() *Tree {
|
|
||||||
return s.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *StringNode) Copy() Node {
|
|
||||||
return s.tr.newString(s.Pos, s.Quoted, s.Text)
|
|
||||||
}
|
|
||||||
|
|
||||||
// endNode represents an {{end}} action.
|
|
||||||
// It does not appear in the final parse tree.
|
|
||||||
type endNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newEnd(pos Pos) *endNode {
|
|
||||||
return &endNode{tr: t, NodeType: nodeEnd, Pos: pos}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *endNode) String() string {
|
|
||||||
return "{{end}}"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *endNode) tree() *Tree {
|
|
||||||
return e.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *endNode) Copy() Node {
|
|
||||||
return e.tr.newEnd(e.Pos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// elseNode represents an {{else}} action. Does not appear in the final tree.
|
|
||||||
type elseNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Line int // The line number in the input (deprecated; kept for compatibility)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newElse(pos Pos, line int) *elseNode {
|
|
||||||
return &elseNode{tr: t, NodeType: nodeElse, Pos: pos, Line: line}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *elseNode) Type() NodeType {
|
|
||||||
return nodeElse
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *elseNode) String() string {
|
|
||||||
return "{{else}}"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *elseNode) tree() *Tree {
|
|
||||||
return e.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *elseNode) Copy() Node {
|
|
||||||
return e.tr.newElse(e.Pos, e.Line)
|
|
||||||
}
|
|
||||||
|
|
||||||
// BranchNode is the common representation of if, range, and with.
|
|
||||||
type BranchNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Line int // The line number in the input (deprecated; kept for compatibility)
|
|
||||||
Pipe *PipeNode // The pipeline to be evaluated.
|
|
||||||
List *ListNode // What to execute if the value is non-empty.
|
|
||||||
ElseList *ListNode // What to execute if the value is empty (nil if absent).
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BranchNode) String() string {
|
|
||||||
name := ""
|
|
||||||
switch b.NodeType {
|
|
||||||
case NodeIf:
|
|
||||||
name = "if"
|
|
||||||
case NodeRange:
|
|
||||||
name = "range"
|
|
||||||
case NodeWith:
|
|
||||||
name = "with"
|
|
||||||
default:
|
|
||||||
panic("unknown branch type")
|
|
||||||
}
|
|
||||||
if b.ElseList != nil {
|
|
||||||
return fmt.Sprintf("{{%s %s}}%s{{else}}%s{{end}}", name, b.Pipe, b.List, b.ElseList)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("{{%s %s}}%s{{end}}", name, b.Pipe, b.List)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BranchNode) tree() *Tree {
|
|
||||||
return b.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b *BranchNode) Copy() Node {
|
|
||||||
switch b.NodeType {
|
|
||||||
case NodeIf:
|
|
||||||
return b.tr.newIf(b.Pos, b.Line, b.Pipe, b.List, b.ElseList)
|
|
||||||
case NodeRange:
|
|
||||||
return b.tr.newRange(b.Pos, b.Line, b.Pipe, b.List, b.ElseList)
|
|
||||||
case NodeWith:
|
|
||||||
return b.tr.newWith(b.Pos, b.Line, b.Pipe, b.List, b.ElseList)
|
|
||||||
default:
|
|
||||||
panic("unknown branch type")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// IfNode represents an {{if}} action and its commands.
|
|
||||||
type IfNode struct {
|
|
||||||
BranchNode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newIf(pos Pos, line int, pipe *PipeNode, list, elseList *ListNode) *IfNode {
|
|
||||||
return &IfNode{BranchNode{tr: t, NodeType: NodeIf, Pos: pos, Line: line, Pipe: pipe, List: list, ElseList: elseList}}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *IfNode) Copy() Node {
|
|
||||||
return i.tr.newIf(i.Pos, i.Line, i.Pipe.CopyPipe(), i.List.CopyList(), i.ElseList.CopyList())
|
|
||||||
}
|
|
||||||
|
|
||||||
// RangeNode represents a {{range}} action and its commands.
|
|
||||||
type RangeNode struct {
|
|
||||||
BranchNode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newRange(pos Pos, line int, pipe *PipeNode, list, elseList *ListNode) *RangeNode {
|
|
||||||
return &RangeNode{BranchNode{tr: t, NodeType: NodeRange, Pos: pos, Line: line, Pipe: pipe, List: list, ElseList: elseList}}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *RangeNode) Copy() Node {
|
|
||||||
return r.tr.newRange(r.Pos, r.Line, r.Pipe.CopyPipe(), r.List.CopyList(), r.ElseList.CopyList())
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithNode represents a {{with}} action and its commands.
|
|
||||||
type WithNode struct {
|
|
||||||
BranchNode
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newWith(pos Pos, line int, pipe *PipeNode, list, elseList *ListNode) *WithNode {
|
|
||||||
return &WithNode{BranchNode{tr: t, NodeType: NodeWith, Pos: pos, Line: line, Pipe: pipe, List: list, ElseList: elseList}}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *WithNode) Copy() Node {
|
|
||||||
return w.tr.newWith(w.Pos, w.Line, w.Pipe.CopyPipe(), w.List.CopyList(), w.ElseList.CopyList())
|
|
||||||
}
|
|
||||||
|
|
||||||
// TemplateNode represents a {{template}} action.
|
|
||||||
type TemplateNode struct {
|
|
||||||
NodeType
|
|
||||||
Pos
|
|
||||||
tr *Tree
|
|
||||||
Line int // The line number in the input (deprecated; kept for compatibility)
|
|
||||||
Name string // The name of the template (unquoted).
|
|
||||||
Pipe *PipeNode // The command to evaluate as dot for the template.
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) newTemplate(pos Pos, line int, name string, pipe *PipeNode) *TemplateNode {
|
|
||||||
return &TemplateNode{tr: t, NodeType: NodeTemplate, Pos: pos, Line: line, Name: name, Pipe: pipe}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TemplateNode) String() string {
|
|
||||||
if t.Pipe == nil {
|
|
||||||
return fmt.Sprintf("{{template %q}}", t.Name)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("{{template %q %s}}", t.Name, t.Pipe)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TemplateNode) tree() *Tree {
|
|
||||||
return t.tr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *TemplateNode) Copy() Node {
|
|
||||||
return t.tr.newTemplate(t.Pos, t.Line, t.Name, t.Pipe.CopyPipe())
|
|
||||||
}
|
|
700
vendor/github.com/alecthomas/template/parse/parse.go
generated
vendored
700
vendor/github.com/alecthomas/template/parse/parse.go
generated
vendored
|
@ -1,700 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
// Package parse builds parse trees for templates as defined by text/template
|
|
||||||
// and html/template. Clients should use those packages to construct templates
|
|
||||||
// rather than this one, which provides shared internal data structures not
|
|
||||||
// intended for general use.
|
|
||||||
package parse
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"runtime"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Tree is the representation of a single parsed template.
|
|
||||||
type Tree struct {
|
|
||||||
Name string // name of the template represented by the tree.
|
|
||||||
ParseName string // name of the top-level template during parsing, for error messages.
|
|
||||||
Root *ListNode // top-level root of the tree.
|
|
||||||
text string // text parsed to create the template (or its parent)
|
|
||||||
// Parsing only; cleared after parse.
|
|
||||||
funcs []map[string]interface{}
|
|
||||||
lex *lexer
|
|
||||||
token [3]item // three-token lookahead for parser.
|
|
||||||
peekCount int
|
|
||||||
vars []string // variables defined at the moment.
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy returns a copy of the Tree. Any parsing state is discarded.
|
|
||||||
func (t *Tree) Copy() *Tree {
|
|
||||||
if t == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return &Tree{
|
|
||||||
Name: t.Name,
|
|
||||||
ParseName: t.ParseName,
|
|
||||||
Root: t.Root.CopyList(),
|
|
||||||
text: t.text,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse returns a map from template name to parse.Tree, created by parsing the
|
|
||||||
// templates described in the argument string. The top-level template will be
|
|
||||||
// given the specified name. If an error is encountered, parsing stops and an
|
|
||||||
// empty map is returned with the error.
|
|
||||||
func Parse(name, text, leftDelim, rightDelim string, funcs ...map[string]interface{}) (treeSet map[string]*Tree, err error) {
|
|
||||||
treeSet = make(map[string]*Tree)
|
|
||||||
t := New(name)
|
|
||||||
t.text = text
|
|
||||||
_, err = t.Parse(text, leftDelim, rightDelim, treeSet, funcs...)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// next returns the next token.
|
|
||||||
func (t *Tree) next() item {
|
|
||||||
if t.peekCount > 0 {
|
|
||||||
t.peekCount--
|
|
||||||
} else {
|
|
||||||
t.token[0] = t.lex.nextItem()
|
|
||||||
}
|
|
||||||
return t.token[t.peekCount]
|
|
||||||
}
|
|
||||||
|
|
||||||
// backup backs the input stream up one token.
|
|
||||||
func (t *Tree) backup() {
|
|
||||||
t.peekCount++
|
|
||||||
}
|
|
||||||
|
|
||||||
// backup2 backs the input stream up two tokens.
|
|
||||||
// The zeroth token is already there.
|
|
||||||
func (t *Tree) backup2(t1 item) {
|
|
||||||
t.token[1] = t1
|
|
||||||
t.peekCount = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
// backup3 backs the input stream up three tokens
|
|
||||||
// The zeroth token is already there.
|
|
||||||
func (t *Tree) backup3(t2, t1 item) { // Reverse order: we're pushing back.
|
|
||||||
t.token[1] = t1
|
|
||||||
t.token[2] = t2
|
|
||||||
t.peekCount = 3
|
|
||||||
}
|
|
||||||
|
|
||||||
// peek returns but does not consume the next token.
|
|
||||||
func (t *Tree) peek() item {
|
|
||||||
if t.peekCount > 0 {
|
|
||||||
return t.token[t.peekCount-1]
|
|
||||||
}
|
|
||||||
t.peekCount = 1
|
|
||||||
t.token[0] = t.lex.nextItem()
|
|
||||||
return t.token[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
// nextNonSpace returns the next non-space token.
|
|
||||||
func (t *Tree) nextNonSpace() (token item) {
|
|
||||||
for {
|
|
||||||
token = t.next()
|
|
||||||
if token.typ != itemSpace {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return token
|
|
||||||
}
|
|
||||||
|
|
||||||
// peekNonSpace returns but does not consume the next non-space token.
|
|
||||||
func (t *Tree) peekNonSpace() (token item) {
|
|
||||||
for {
|
|
||||||
token = t.next()
|
|
||||||
if token.typ != itemSpace {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
t.backup()
|
|
||||||
return token
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parsing.
|
|
||||||
|
|
||||||
// New allocates a new parse tree with the given name.
|
|
||||||
func New(name string, funcs ...map[string]interface{}) *Tree {
|
|
||||||
return &Tree{
|
|
||||||
Name: name,
|
|
||||||
funcs: funcs,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrorContext returns a textual representation of the location of the node in the input text.
|
|
||||||
// The receiver is only used when the node does not have a pointer to the tree inside,
|
|
||||||
// which can occur in old code.
|
|
||||||
func (t *Tree) ErrorContext(n Node) (location, context string) {
|
|
||||||
pos := int(n.Position())
|
|
||||||
tree := n.tree()
|
|
||||||
if tree == nil {
|
|
||||||
tree = t
|
|
||||||
}
|
|
||||||
text := tree.text[:pos]
|
|
||||||
byteNum := strings.LastIndex(text, "\n")
|
|
||||||
if byteNum == -1 {
|
|
||||||
byteNum = pos // On first line.
|
|
||||||
} else {
|
|
||||||
byteNum++ // After the newline.
|
|
||||||
byteNum = pos - byteNum
|
|
||||||
}
|
|
||||||
lineNum := 1 + strings.Count(text, "\n")
|
|
||||||
context = n.String()
|
|
||||||
if len(context) > 20 {
|
|
||||||
context = fmt.Sprintf("%.20s...", context)
|
|
||||||
}
|
|
||||||
return fmt.Sprintf("%s:%d:%d", tree.ParseName, lineNum, byteNum), context
|
|
||||||
}
|
|
||||||
|
|
||||||
// errorf formats the error and terminates processing.
|
|
||||||
func (t *Tree) errorf(format string, args ...interface{}) {
|
|
||||||
t.Root = nil
|
|
||||||
format = fmt.Sprintf("template: %s:%d: %s", t.ParseName, t.lex.lineNumber(), format)
|
|
||||||
panic(fmt.Errorf(format, args...))
|
|
||||||
}
|
|
||||||
|
|
||||||
// error terminates processing.
|
|
||||||
func (t *Tree) error(err error) {
|
|
||||||
t.errorf("%s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// expect consumes the next token and guarantees it has the required type.
|
|
||||||
func (t *Tree) expect(expected itemType, context string) item {
|
|
||||||
token := t.nextNonSpace()
|
|
||||||
if token.typ != expected {
|
|
||||||
t.unexpected(token, context)
|
|
||||||
}
|
|
||||||
return token
|
|
||||||
}
|
|
||||||
|
|
||||||
// expectOneOf consumes the next token and guarantees it has one of the required types.
|
|
||||||
func (t *Tree) expectOneOf(expected1, expected2 itemType, context string) item {
|
|
||||||
token := t.nextNonSpace()
|
|
||||||
if token.typ != expected1 && token.typ != expected2 {
|
|
||||||
t.unexpected(token, context)
|
|
||||||
}
|
|
||||||
return token
|
|
||||||
}
|
|
||||||
|
|
||||||
// unexpected complains about the token and terminates processing.
|
|
||||||
func (t *Tree) unexpected(token item, context string) {
|
|
||||||
t.errorf("unexpected %s in %s", token, context)
|
|
||||||
}
|
|
||||||
|
|
||||||
// recover is the handler that turns panics into returns from the top level of Parse.
|
|
||||||
func (t *Tree) recover(errp *error) {
|
|
||||||
e := recover()
|
|
||||||
if e != nil {
|
|
||||||
if _, ok := e.(runtime.Error); ok {
|
|
||||||
panic(e)
|
|
||||||
}
|
|
||||||
if t != nil {
|
|
||||||
t.stopParse()
|
|
||||||
}
|
|
||||||
*errp = e.(error)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// startParse initializes the parser, using the lexer.
|
|
||||||
func (t *Tree) startParse(funcs []map[string]interface{}, lex *lexer) {
|
|
||||||
t.Root = nil
|
|
||||||
t.lex = lex
|
|
||||||
t.vars = []string{"$"}
|
|
||||||
t.funcs = funcs
|
|
||||||
}
|
|
||||||
|
|
||||||
// stopParse terminates parsing.
|
|
||||||
func (t *Tree) stopParse() {
|
|
||||||
t.lex = nil
|
|
||||||
t.vars = nil
|
|
||||||
t.funcs = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse parses the template definition string to construct a representation of
|
|
||||||
// the template for execution. If either action delimiter string is empty, the
|
|
||||||
// default ("{{" or "}}") is used. Embedded template definitions are added to
|
|
||||||
// the treeSet map.
|
|
||||||
func (t *Tree) Parse(text, leftDelim, rightDelim string, treeSet map[string]*Tree, funcs ...map[string]interface{}) (tree *Tree, err error) {
|
|
||||||
defer t.recover(&err)
|
|
||||||
t.ParseName = t.Name
|
|
||||||
t.startParse(funcs, lex(t.Name, text, leftDelim, rightDelim))
|
|
||||||
t.text = text
|
|
||||||
t.parse(treeSet)
|
|
||||||
t.add(treeSet)
|
|
||||||
t.stopParse()
|
|
||||||
return t, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// add adds tree to the treeSet.
|
|
||||||
func (t *Tree) add(treeSet map[string]*Tree) {
|
|
||||||
tree := treeSet[t.Name]
|
|
||||||
if tree == nil || IsEmptyTree(tree.Root) {
|
|
||||||
treeSet[t.Name] = t
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !IsEmptyTree(t.Root) {
|
|
||||||
t.errorf("template: multiple definition of template %q", t.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsEmptyTree reports whether this tree (node) is empty of everything but space.
|
|
||||||
func IsEmptyTree(n Node) bool {
|
|
||||||
switch n := n.(type) {
|
|
||||||
case nil:
|
|
||||||
return true
|
|
||||||
case *ActionNode:
|
|
||||||
case *IfNode:
|
|
||||||
case *ListNode:
|
|
||||||
for _, node := range n.Nodes {
|
|
||||||
if !IsEmptyTree(node) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
case *RangeNode:
|
|
||||||
case *TemplateNode:
|
|
||||||
case *TextNode:
|
|
||||||
return len(bytes.TrimSpace(n.Text)) == 0
|
|
||||||
case *WithNode:
|
|
||||||
default:
|
|
||||||
panic("unknown node: " + n.String())
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// parse is the top-level parser for a template, essentially the same
|
|
||||||
// as itemList except it also parses {{define}} actions.
|
|
||||||
// It runs to EOF.
|
|
||||||
func (t *Tree) parse(treeSet map[string]*Tree) (next Node) {
|
|
||||||
t.Root = t.newList(t.peek().pos)
|
|
||||||
for t.peek().typ != itemEOF {
|
|
||||||
if t.peek().typ == itemLeftDelim {
|
|
||||||
delim := t.next()
|
|
||||||
if t.nextNonSpace().typ == itemDefine {
|
|
||||||
newT := New("definition") // name will be updated once we know it.
|
|
||||||
newT.text = t.text
|
|
||||||
newT.ParseName = t.ParseName
|
|
||||||
newT.startParse(t.funcs, t.lex)
|
|
||||||
newT.parseDefinition(treeSet)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
t.backup2(delim)
|
|
||||||
}
|
|
||||||
n := t.textOrAction()
|
|
||||||
if n.Type() == nodeEnd {
|
|
||||||
t.errorf("unexpected %s", n)
|
|
||||||
}
|
|
||||||
t.Root.append(n)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseDefinition parses a {{define}} ... {{end}} template definition and
|
|
||||||
// installs the definition in the treeSet map. The "define" keyword has already
|
|
||||||
// been scanned.
|
|
||||||
func (t *Tree) parseDefinition(treeSet map[string]*Tree) {
|
|
||||||
const context = "define clause"
|
|
||||||
name := t.expectOneOf(itemString, itemRawString, context)
|
|
||||||
var err error
|
|
||||||
t.Name, err = strconv.Unquote(name.val)
|
|
||||||
if err != nil {
|
|
||||||
t.error(err)
|
|
||||||
}
|
|
||||||
t.expect(itemRightDelim, context)
|
|
||||||
var end Node
|
|
||||||
t.Root, end = t.itemList()
|
|
||||||
if end.Type() != nodeEnd {
|
|
||||||
t.errorf("unexpected %s in %s", end, context)
|
|
||||||
}
|
|
||||||
t.add(treeSet)
|
|
||||||
t.stopParse()
|
|
||||||
}
|
|
||||||
|
|
||||||
// itemList:
|
|
||||||
// textOrAction*
|
|
||||||
// Terminates at {{end}} or {{else}}, returned separately.
|
|
||||||
func (t *Tree) itemList() (list *ListNode, next Node) {
|
|
||||||
list = t.newList(t.peekNonSpace().pos)
|
|
||||||
for t.peekNonSpace().typ != itemEOF {
|
|
||||||
n := t.textOrAction()
|
|
||||||
switch n.Type() {
|
|
||||||
case nodeEnd, nodeElse:
|
|
||||||
return list, n
|
|
||||||
}
|
|
||||||
list.append(n)
|
|
||||||
}
|
|
||||||
t.errorf("unexpected EOF")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// textOrAction:
|
|
||||||
// text | action
|
|
||||||
func (t *Tree) textOrAction() Node {
|
|
||||||
switch token := t.nextNonSpace(); token.typ {
|
|
||||||
case itemElideNewline:
|
|
||||||
return t.elideNewline()
|
|
||||||
case itemText:
|
|
||||||
return t.newText(token.pos, token.val)
|
|
||||||
case itemLeftDelim:
|
|
||||||
return t.action()
|
|
||||||
default:
|
|
||||||
t.unexpected(token, "input")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// elideNewline:
|
|
||||||
// Remove newlines trailing rightDelim if \\ is present.
|
|
||||||
func (t *Tree) elideNewline() Node {
|
|
||||||
token := t.peek()
|
|
||||||
if token.typ != itemText {
|
|
||||||
t.unexpected(token, "input")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
t.next()
|
|
||||||
stripped := strings.TrimLeft(token.val, "\n\r")
|
|
||||||
diff := len(token.val) - len(stripped)
|
|
||||||
if diff > 0 {
|
|
||||||
// This is a bit nasty. We mutate the token in-place to remove
|
|
||||||
// preceding newlines.
|
|
||||||
token.pos += Pos(diff)
|
|
||||||
token.val = stripped
|
|
||||||
}
|
|
||||||
return t.newText(token.pos, token.val)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Action:
|
|
||||||
// control
|
|
||||||
// command ("|" command)*
|
|
||||||
// Left delim is past. Now get actions.
|
|
||||||
// First word could be a keyword such as range.
|
|
||||||
func (t *Tree) action() (n Node) {
|
|
||||||
switch token := t.nextNonSpace(); token.typ {
|
|
||||||
case itemElse:
|
|
||||||
return t.elseControl()
|
|
||||||
case itemEnd:
|
|
||||||
return t.endControl()
|
|
||||||
case itemIf:
|
|
||||||
return t.ifControl()
|
|
||||||
case itemRange:
|
|
||||||
return t.rangeControl()
|
|
||||||
case itemTemplate:
|
|
||||||
return t.templateControl()
|
|
||||||
case itemWith:
|
|
||||||
return t.withControl()
|
|
||||||
}
|
|
||||||
t.backup()
|
|
||||||
// Do not pop variables; they persist until "end".
|
|
||||||
return t.newAction(t.peek().pos, t.lex.lineNumber(), t.pipeline("command"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Pipeline:
|
|
||||||
// declarations? command ('|' command)*
|
|
||||||
func (t *Tree) pipeline(context string) (pipe *PipeNode) {
|
|
||||||
var decl []*VariableNode
|
|
||||||
pos := t.peekNonSpace().pos
|
|
||||||
// Are there declarations?
|
|
||||||
for {
|
|
||||||
if v := t.peekNonSpace(); v.typ == itemVariable {
|
|
||||||
t.next()
|
|
||||||
// Since space is a token, we need 3-token look-ahead here in the worst case:
|
|
||||||
// in "$x foo" we need to read "foo" (as opposed to ":=") to know that $x is an
|
|
||||||
// argument variable rather than a declaration. So remember the token
|
|
||||||
// adjacent to the variable so we can push it back if necessary.
|
|
||||||
tokenAfterVariable := t.peek()
|
|
||||||
if next := t.peekNonSpace(); next.typ == itemColonEquals || (next.typ == itemChar && next.val == ",") {
|
|
||||||
t.nextNonSpace()
|
|
||||||
variable := t.newVariable(v.pos, v.val)
|
|
||||||
decl = append(decl, variable)
|
|
||||||
t.vars = append(t.vars, v.val)
|
|
||||||
if next.typ == itemChar && next.val == "," {
|
|
||||||
if context == "range" && len(decl) < 2 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
t.errorf("too many declarations in %s", context)
|
|
||||||
}
|
|
||||||
} else if tokenAfterVariable.typ == itemSpace {
|
|
||||||
t.backup3(v, tokenAfterVariable)
|
|
||||||
} else {
|
|
||||||
t.backup2(v)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
pipe = t.newPipeline(pos, t.lex.lineNumber(), decl)
|
|
||||||
for {
|
|
||||||
switch token := t.nextNonSpace(); token.typ {
|
|
||||||
case itemRightDelim, itemRightParen:
|
|
||||||
if len(pipe.Cmds) == 0 {
|
|
||||||
t.errorf("missing value for %s", context)
|
|
||||||
}
|
|
||||||
if token.typ == itemRightParen {
|
|
||||||
t.backup()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
case itemBool, itemCharConstant, itemComplex, itemDot, itemField, itemIdentifier,
|
|
||||||
itemNumber, itemNil, itemRawString, itemString, itemVariable, itemLeftParen:
|
|
||||||
t.backup()
|
|
||||||
pipe.append(t.command())
|
|
||||||
default:
|
|
||||||
t.unexpected(token, context)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Tree) parseControl(allowElseIf bool, context string) (pos Pos, line int, pipe *PipeNode, list, elseList *ListNode) {
|
|
||||||
defer t.popVars(len(t.vars))
|
|
||||||
line = t.lex.lineNumber()
|
|
||||||
pipe = t.pipeline(context)
|
|
||||||
var next Node
|
|
||||||
list, next = t.itemList()
|
|
||||||
switch next.Type() {
|
|
||||||
case nodeEnd: //done
|
|
||||||
case nodeElse:
|
|
||||||
if allowElseIf {
|
|
||||||
// Special case for "else if". If the "else" is followed immediately by an "if",
|
|
||||||
// the elseControl will have left the "if" token pending. Treat
|
|
||||||
// {{if a}}_{{else if b}}_{{end}}
|
|
||||||
// as
|
|
||||||
// {{if a}}_{{else}}{{if b}}_{{end}}{{end}}.
|
|
||||||
// To do this, parse the if as usual and stop at it {{end}}; the subsequent{{end}}
|
|
||||||
// is assumed. This technique works even for long if-else-if chains.
|
|
||||||
// TODO: Should we allow else-if in with and range?
|
|
||||||
if t.peek().typ == itemIf {
|
|
||||||
t.next() // Consume the "if" token.
|
|
||||||
elseList = t.newList(next.Position())
|
|
||||||
elseList.append(t.ifControl())
|
|
||||||
// Do not consume the next item - only one {{end}} required.
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
elseList, next = t.itemList()
|
|
||||||
if next.Type() != nodeEnd {
|
|
||||||
t.errorf("expected end; found %s", next)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return pipe.Position(), line, pipe, list, elseList
|
|
||||||
}
|
|
||||||
|
|
||||||
// If:
|
|
||||||
// {{if pipeline}} itemList {{end}}
|
|
||||||
// {{if pipeline}} itemList {{else}} itemList {{end}}
|
|
||||||
// If keyword is past.
|
|
||||||
func (t *Tree) ifControl() Node {
|
|
||||||
return t.newIf(t.parseControl(true, "if"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Range:
|
|
||||||
// {{range pipeline}} itemList {{end}}
|
|
||||||
// {{range pipeline}} itemList {{else}} itemList {{end}}
|
|
||||||
// Range keyword is past.
|
|
||||||
func (t *Tree) rangeControl() Node {
|
|
||||||
return t.newRange(t.parseControl(false, "range"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// With:
|
|
||||||
// {{with pipeline}} itemList {{end}}
|
|
||||||
// {{with pipeline}} itemList {{else}} itemList {{end}}
|
|
||||||
// If keyword is past.
|
|
||||||
func (t *Tree) withControl() Node {
|
|
||||||
return t.newWith(t.parseControl(false, "with"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// End:
|
|
||||||
// {{end}}
|
|
||||||
// End keyword is past.
|
|
||||||
func (t *Tree) endControl() Node {
|
|
||||||
return t.newEnd(t.expect(itemRightDelim, "end").pos)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Else:
|
|
||||||
// {{else}}
|
|
||||||
// Else keyword is past.
|
|
||||||
func (t *Tree) elseControl() Node {
|
|
||||||
// Special case for "else if".
|
|
||||||
peek := t.peekNonSpace()
|
|
||||||
if peek.typ == itemIf {
|
|
||||||
// We see "{{else if ... " but in effect rewrite it to {{else}}{{if ... ".
|
|
||||||
return t.newElse(peek.pos, t.lex.lineNumber())
|
|
||||||
}
|
|
||||||
return t.newElse(t.expect(itemRightDelim, "else").pos, t.lex.lineNumber())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Template:
|
|
||||||
// {{template stringValue pipeline}}
|
|
||||||
// Template keyword is past. The name must be something that can evaluate
|
|
||||||
// to a string.
|
|
||||||
func (t *Tree) templateControl() Node {
|
|
||||||
var name string
|
|
||||||
token := t.nextNonSpace()
|
|
||||||
switch token.typ {
|
|
||||||
case itemString, itemRawString:
|
|
||||||
s, err := strconv.Unquote(token.val)
|
|
||||||
if err != nil {
|
|
||||||
t.error(err)
|
|
||||||
}
|
|
||||||
name = s
|
|
||||||
default:
|
|
||||||
t.unexpected(token, "template invocation")
|
|
||||||
}
|
|
||||||
var pipe *PipeNode
|
|
||||||
if t.nextNonSpace().typ != itemRightDelim {
|
|
||||||
t.backup()
|
|
||||||
// Do not pop variables; they persist until "end".
|
|
||||||
pipe = t.pipeline("template")
|
|
||||||
}
|
|
||||||
return t.newTemplate(token.pos, t.lex.lineNumber(), name, pipe)
|
|
||||||
}
|
|
||||||
|
|
||||||
// command:
|
|
||||||
// operand (space operand)*
|
|
||||||
// space-separated arguments up to a pipeline character or right delimiter.
|
|
||||||
// we consume the pipe character but leave the right delim to terminate the action.
|
|
||||||
func (t *Tree) command() *CommandNode {
|
|
||||||
cmd := t.newCommand(t.peekNonSpace().pos)
|
|
||||||
for {
|
|
||||||
t.peekNonSpace() // skip leading spaces.
|
|
||||||
operand := t.operand()
|
|
||||||
if operand != nil {
|
|
||||||
cmd.append(operand)
|
|
||||||
}
|
|
||||||
switch token := t.next(); token.typ {
|
|
||||||
case itemSpace:
|
|
||||||
continue
|
|
||||||
case itemError:
|
|
||||||
t.errorf("%s", token.val)
|
|
||||||
case itemRightDelim, itemRightParen:
|
|
||||||
t.backup()
|
|
||||||
case itemPipe:
|
|
||||||
default:
|
|
||||||
t.errorf("unexpected %s in operand; missing space?", token)
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if len(cmd.Args) == 0 {
|
|
||||||
t.errorf("empty command")
|
|
||||||
}
|
|
||||||
return cmd
|
|
||||||
}
|
|
||||||
|
|
||||||
// operand:
|
|
||||||
// term .Field*
|
|
||||||
// An operand is a space-separated component of a command,
|
|
||||||
// a term possibly followed by field accesses.
|
|
||||||
// A nil return means the next item is not an operand.
|
|
||||||
func (t *Tree) operand() Node {
|
|
||||||
node := t.term()
|
|
||||||
if node == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if t.peek().typ == itemField {
|
|
||||||
chain := t.newChain(t.peek().pos, node)
|
|
||||||
for t.peek().typ == itemField {
|
|
||||||
chain.Add(t.next().val)
|
|
||||||
}
|
|
||||||
// Compatibility with original API: If the term is of type NodeField
|
|
||||||
// or NodeVariable, just put more fields on the original.
|
|
||||||
// Otherwise, keep the Chain node.
|
|
||||||
// TODO: Switch to Chains always when we can.
|
|
||||||
switch node.Type() {
|
|
||||||
case NodeField:
|
|
||||||
node = t.newField(chain.Position(), chain.String())
|
|
||||||
case NodeVariable:
|
|
||||||
node = t.newVariable(chain.Position(), chain.String())
|
|
||||||
default:
|
|
||||||
node = chain
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return node
|
|
||||||
}
|
|
||||||
|
|
||||||
// term:
|
|
||||||
// literal (number, string, nil, boolean)
|
|
||||||
// function (identifier)
|
|
||||||
// .
|
|
||||||
// .Field
|
|
||||||
// $
|
|
||||||
// '(' pipeline ')'
|
|
||||||
// A term is a simple "expression".
|
|
||||||
// A nil return means the next item is not a term.
|
|
||||||
func (t *Tree) term() Node {
|
|
||||||
switch token := t.nextNonSpace(); token.typ {
|
|
||||||
case itemError:
|
|
||||||
t.errorf("%s", token.val)
|
|
||||||
case itemIdentifier:
|
|
||||||
if !t.hasFunction(token.val) {
|
|
||||||
t.errorf("function %q not defined", token.val)
|
|
||||||
}
|
|
||||||
return NewIdentifier(token.val).SetTree(t).SetPos(token.pos)
|
|
||||||
case itemDot:
|
|
||||||
return t.newDot(token.pos)
|
|
||||||
case itemNil:
|
|
||||||
return t.newNil(token.pos)
|
|
||||||
case itemVariable:
|
|
||||||
return t.useVar(token.pos, token.val)
|
|
||||||
case itemField:
|
|
||||||
return t.newField(token.pos, token.val)
|
|
||||||
case itemBool:
|
|
||||||
return t.newBool(token.pos, token.val == "true")
|
|
||||||
case itemCharConstant, itemComplex, itemNumber:
|
|
||||||
number, err := t.newNumber(token.pos, token.val, token.typ)
|
|
||||||
if err != nil {
|
|
||||||
t.error(err)
|
|
||||||
}
|
|
||||||
return number
|
|
||||||
case itemLeftParen:
|
|
||||||
pipe := t.pipeline("parenthesized pipeline")
|
|
||||||
if token := t.next(); token.typ != itemRightParen {
|
|
||||||
t.errorf("unclosed right paren: unexpected %s", token)
|
|
||||||
}
|
|
||||||
return pipe
|
|
||||||
case itemString, itemRawString:
|
|
||||||
s, err := strconv.Unquote(token.val)
|
|
||||||
if err != nil {
|
|
||||||
t.error(err)
|
|
||||||
}
|
|
||||||
return t.newString(token.pos, token.val, s)
|
|
||||||
}
|
|
||||||
t.backup()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// hasFunction reports if a function name exists in the Tree's maps.
|
|
||||||
func (t *Tree) hasFunction(name string) bool {
|
|
||||||
for _, funcMap := range t.funcs {
|
|
||||||
if funcMap == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if funcMap[name] != nil {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// popVars trims the variable list to the specified length
|
|
||||||
func (t *Tree) popVars(n int) {
|
|
||||||
t.vars = t.vars[:n]
|
|
||||||
}
|
|
||||||
|
|
||||||
// useVar returns a node for a variable reference. It errors if the
|
|
||||||
// variable is not defined.
|
|
||||||
func (t *Tree) useVar(pos Pos, name string) Node {
|
|
||||||
v := t.newVariable(pos, name)
|
|
||||||
for _, varName := range t.vars {
|
|
||||||
if varName == v.Ident[0] {
|
|
||||||
return v
|
|
||||||
}
|
|
||||||
}
|
|
||||||
t.errorf("undefined variable %q", v.Ident[0])
|
|
||||||
return nil
|
|
||||||
}
|
|
218
vendor/github.com/alecthomas/template/template.go
generated
vendored
218
vendor/github.com/alecthomas/template/template.go
generated
vendored
|
@ -1,218 +0,0 @@
|
||||||
// Copyright 2011 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package template
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
|
|
||||||
"github.com/alecthomas/template/parse"
|
|
||||||
)
|
|
||||||
|
|
||||||
// common holds the information shared by related templates.
|
|
||||||
type common struct {
|
|
||||||
tmpl map[string]*Template
|
|
||||||
// We use two maps, one for parsing and one for execution.
|
|
||||||
// This separation makes the API cleaner since it doesn't
|
|
||||||
// expose reflection to the client.
|
|
||||||
parseFuncs FuncMap
|
|
||||||
execFuncs map[string]reflect.Value
|
|
||||||
}
|
|
||||||
|
|
||||||
// Template is the representation of a parsed template. The *parse.Tree
|
|
||||||
// field is exported only for use by html/template and should be treated
|
|
||||||
// as unexported by all other clients.
|
|
||||||
type Template struct {
|
|
||||||
name string
|
|
||||||
*parse.Tree
|
|
||||||
*common
|
|
||||||
leftDelim string
|
|
||||||
rightDelim string
|
|
||||||
}
|
|
||||||
|
|
||||||
// New allocates a new template with the given name.
|
|
||||||
func New(name string) *Template {
|
|
||||||
return &Template{
|
|
||||||
name: name,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Name returns the name of the template.
|
|
||||||
func (t *Template) Name() string {
|
|
||||||
return t.name
|
|
||||||
}
|
|
||||||
|
|
||||||
// New allocates a new template associated with the given one and with the same
|
|
||||||
// delimiters. The association, which is transitive, allows one template to
|
|
||||||
// invoke another with a {{template}} action.
|
|
||||||
func (t *Template) New(name string) *Template {
|
|
||||||
t.init()
|
|
||||||
return &Template{
|
|
||||||
name: name,
|
|
||||||
common: t.common,
|
|
||||||
leftDelim: t.leftDelim,
|
|
||||||
rightDelim: t.rightDelim,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Template) init() {
|
|
||||||
if t.common == nil {
|
|
||||||
t.common = new(common)
|
|
||||||
t.tmpl = make(map[string]*Template)
|
|
||||||
t.parseFuncs = make(FuncMap)
|
|
||||||
t.execFuncs = make(map[string]reflect.Value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clone returns a duplicate of the template, including all associated
|
|
||||||
// templates. The actual representation is not copied, but the name space of
|
|
||||||
// associated templates is, so further calls to Parse in the copy will add
|
|
||||||
// templates to the copy but not to the original. Clone can be used to prepare
|
|
||||||
// common templates and use them with variant definitions for other templates
|
|
||||||
// by adding the variants after the clone is made.
|
|
||||||
func (t *Template) Clone() (*Template, error) {
|
|
||||||
nt := t.copy(nil)
|
|
||||||
nt.init()
|
|
||||||
nt.tmpl[t.name] = nt
|
|
||||||
for k, v := range t.tmpl {
|
|
||||||
if k == t.name { // Already installed.
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// The associated templates share nt's common structure.
|
|
||||||
tmpl := v.copy(nt.common)
|
|
||||||
nt.tmpl[k] = tmpl
|
|
||||||
}
|
|
||||||
for k, v := range t.parseFuncs {
|
|
||||||
nt.parseFuncs[k] = v
|
|
||||||
}
|
|
||||||
for k, v := range t.execFuncs {
|
|
||||||
nt.execFuncs[k] = v
|
|
||||||
}
|
|
||||||
return nt, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// copy returns a shallow copy of t, with common set to the argument.
|
|
||||||
func (t *Template) copy(c *common) *Template {
|
|
||||||
nt := New(t.name)
|
|
||||||
nt.Tree = t.Tree
|
|
||||||
nt.common = c
|
|
||||||
nt.leftDelim = t.leftDelim
|
|
||||||
nt.rightDelim = t.rightDelim
|
|
||||||
return nt
|
|
||||||
}
|
|
||||||
|
|
||||||
// AddParseTree creates a new template with the name and parse tree
|
|
||||||
// and associates it with t.
|
|
||||||
func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error) {
|
|
||||||
if t.common != nil && t.tmpl[name] != nil {
|
|
||||||
return nil, fmt.Errorf("template: redefinition of template %q", name)
|
|
||||||
}
|
|
||||||
nt := t.New(name)
|
|
||||||
nt.Tree = tree
|
|
||||||
t.tmpl[name] = nt
|
|
||||||
return nt, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Templates returns a slice of the templates associated with t, including t
|
|
||||||
// itself.
|
|
||||||
func (t *Template) Templates() []*Template {
|
|
||||||
if t.common == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
// Return a slice so we don't expose the map.
|
|
||||||
m := make([]*Template, 0, len(t.tmpl))
|
|
||||||
for _, v := range t.tmpl {
|
|
||||||
m = append(m, v)
|
|
||||||
}
|
|
||||||
return m
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delims sets the action delimiters to the specified strings, to be used in
|
|
||||||
// subsequent calls to Parse, ParseFiles, or ParseGlob. Nested template
|
|
||||||
// definitions will inherit the settings. An empty delimiter stands for the
|
|
||||||
// corresponding default: {{ or }}.
|
|
||||||
// The return value is the template, so calls can be chained.
|
|
||||||
func (t *Template) Delims(left, right string) *Template {
|
|
||||||
t.leftDelim = left
|
|
||||||
t.rightDelim = right
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
// Funcs adds the elements of the argument map to the template's function map.
|
|
||||||
// It panics if a value in the map is not a function with appropriate return
|
|
||||||
// type. However, it is legal to overwrite elements of the map. The return
|
|
||||||
// value is the template, so calls can be chained.
|
|
||||||
func (t *Template) Funcs(funcMap FuncMap) *Template {
|
|
||||||
t.init()
|
|
||||||
addValueFuncs(t.execFuncs, funcMap)
|
|
||||||
addFuncs(t.parseFuncs, funcMap)
|
|
||||||
return t
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lookup returns the template with the given name that is associated with t,
|
|
||||||
// or nil if there is no such template.
|
|
||||||
func (t *Template) Lookup(name string) *Template {
|
|
||||||
if t.common == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return t.tmpl[name]
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse parses a string into a template. Nested template definitions will be
|
|
||||||
// associated with the top-level template t. Parse may be called multiple times
|
|
||||||
// to parse definitions of templates to associate with t. It is an error if a
|
|
||||||
// resulting template is non-empty (contains content other than template
|
|
||||||
// definitions) and would replace a non-empty template with the same name.
|
|
||||||
// (In multiple calls to Parse with the same receiver template, only one call
|
|
||||||
// can contain text other than space, comments, and template definitions.)
|
|
||||||
func (t *Template) Parse(text string) (*Template, error) {
|
|
||||||
t.init()
|
|
||||||
trees, err := parse.Parse(t.name, text, t.leftDelim, t.rightDelim, t.parseFuncs, builtins)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
// Add the newly parsed trees, including the one for t, into our common structure.
|
|
||||||
for name, tree := range trees {
|
|
||||||
// If the name we parsed is the name of this template, overwrite this template.
|
|
||||||
// The associate method checks it's not a redefinition.
|
|
||||||
tmpl := t
|
|
||||||
if name != t.name {
|
|
||||||
tmpl = t.New(name)
|
|
||||||
}
|
|
||||||
// Even if t == tmpl, we need to install it in the common.tmpl map.
|
|
||||||
if replace, err := t.associate(tmpl, tree); err != nil {
|
|
||||||
return nil, err
|
|
||||||
} else if replace {
|
|
||||||
tmpl.Tree = tree
|
|
||||||
}
|
|
||||||
tmpl.leftDelim = t.leftDelim
|
|
||||||
tmpl.rightDelim = t.rightDelim
|
|
||||||
}
|
|
||||||
return t, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// associate installs the new template into the group of templates associated
|
|
||||||
// with t. It is an error to reuse a name except to overwrite an empty
|
|
||||||
// template. The two are already known to share the common structure.
|
|
||||||
// The boolean return value reports wither to store this tree as t.Tree.
|
|
||||||
func (t *Template) associate(new *Template, tree *parse.Tree) (bool, error) {
|
|
||||||
if new.common != t.common {
|
|
||||||
panic("internal error: associate not common")
|
|
||||||
}
|
|
||||||
name := new.name
|
|
||||||
if old := t.tmpl[name]; old != nil {
|
|
||||||
oldIsEmpty := parse.IsEmptyTree(old.Root)
|
|
||||||
newIsEmpty := parse.IsEmptyTree(tree.Root)
|
|
||||||
if newIsEmpty {
|
|
||||||
// Whether old is empty or not, new is empty; no reason to replace old.
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
if !oldIsEmpty {
|
|
||||||
return false, fmt.Errorf("template: redefinition of template %q", name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
t.tmpl[name] = new
|
|
||||||
return true, nil
|
|
||||||
}
|
|
19
vendor/github.com/alecthomas/units/COPYING
generated
vendored
19
vendor/github.com/alecthomas/units/COPYING
generated
vendored
|
@ -1,19 +0,0 @@
|
||||||
Copyright (C) 2014 Alec Thomas
|
|
||||||
|
|
||||||
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.
|
|
11
vendor/github.com/alecthomas/units/README.md
generated
vendored
11
vendor/github.com/alecthomas/units/README.md
generated
vendored
|
@ -1,11 +0,0 @@
|
||||||
# Units - Helpful unit multipliers and functions for Go
|
|
||||||
|
|
||||||
The goal of this package is to have functionality similar to the [time](http://golang.org/pkg/time/) package.
|
|
||||||
|
|
||||||
It allows for code like this:
|
|
||||||
|
|
||||||
```go
|
|
||||||
n, err := ParseBase2Bytes("1KB")
|
|
||||||
// n == 1024
|
|
||||||
n = units.Mebibyte * 512
|
|
||||||
```
|
|
85
vendor/github.com/alecthomas/units/bytes.go
generated
vendored
85
vendor/github.com/alecthomas/units/bytes.go
generated
vendored
|
@ -1,85 +0,0 @@
|
||||||
package units
|
|
||||||
|
|
||||||
// Base2Bytes is the old non-SI power-of-2 byte scale (1024 bytes in a kilobyte,
|
|
||||||
// etc.).
|
|
||||||
type Base2Bytes int64
|
|
||||||
|
|
||||||
// Base-2 byte units.
|
|
||||||
const (
|
|
||||||
Kibibyte Base2Bytes = 1024
|
|
||||||
KiB = Kibibyte
|
|
||||||
Mebibyte = Kibibyte * 1024
|
|
||||||
MiB = Mebibyte
|
|
||||||
Gibibyte = Mebibyte * 1024
|
|
||||||
GiB = Gibibyte
|
|
||||||
Tebibyte = Gibibyte * 1024
|
|
||||||
TiB = Tebibyte
|
|
||||||
Pebibyte = Tebibyte * 1024
|
|
||||||
PiB = Pebibyte
|
|
||||||
Exbibyte = Pebibyte * 1024
|
|
||||||
EiB = Exbibyte
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
bytesUnitMap = MakeUnitMap("iB", "B", 1024)
|
|
||||||
oldBytesUnitMap = MakeUnitMap("B", "B", 1024)
|
|
||||||
)
|
|
||||||
|
|
||||||
// ParseBase2Bytes supports both iB and B in base-2 multipliers. That is, KB
|
|
||||||
// and KiB are both 1024.
|
|
||||||
// However "kB", which is the correct SI spelling of 1000 Bytes, is rejected.
|
|
||||||
func ParseBase2Bytes(s string) (Base2Bytes, error) {
|
|
||||||
n, err := ParseUnit(s, bytesUnitMap)
|
|
||||||
if err != nil {
|
|
||||||
n, err = ParseUnit(s, oldBytesUnitMap)
|
|
||||||
}
|
|
||||||
return Base2Bytes(n), err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (b Base2Bytes) String() string {
|
|
||||||
return ToString(int64(b), 1024, "iB", "B")
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
metricBytesUnitMap = MakeUnitMap("B", "B", 1000)
|
|
||||||
)
|
|
||||||
|
|
||||||
// MetricBytes are SI byte units (1000 bytes in a kilobyte).
|
|
||||||
type MetricBytes SI
|
|
||||||
|
|
||||||
// SI base-10 byte units.
|
|
||||||
const (
|
|
||||||
Kilobyte MetricBytes = 1000
|
|
||||||
KB = Kilobyte
|
|
||||||
Megabyte = Kilobyte * 1000
|
|
||||||
MB = Megabyte
|
|
||||||
Gigabyte = Megabyte * 1000
|
|
||||||
GB = Gigabyte
|
|
||||||
Terabyte = Gigabyte * 1000
|
|
||||||
TB = Terabyte
|
|
||||||
Petabyte = Terabyte * 1000
|
|
||||||
PB = Petabyte
|
|
||||||
Exabyte = Petabyte * 1000
|
|
||||||
EB = Exabyte
|
|
||||||
)
|
|
||||||
|
|
||||||
// ParseMetricBytes parses base-10 metric byte units. That is, KB is 1000 bytes.
|
|
||||||
func ParseMetricBytes(s string) (MetricBytes, error) {
|
|
||||||
n, err := ParseUnit(s, metricBytesUnitMap)
|
|
||||||
return MetricBytes(n), err
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: represents 1000B as uppercase "KB", while SI standard requires "kB".
|
|
||||||
func (m MetricBytes) String() string {
|
|
||||||
return ToString(int64(m), 1000, "B", "B")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ParseStrictBytes supports both iB and B suffixes for base 2 and metric,
|
|
||||||
// respectively. That is, KiB represents 1024 and kB, KB represent 1000.
|
|
||||||
func ParseStrictBytes(s string) (int64, error) {
|
|
||||||
n, err := ParseUnit(s, bytesUnitMap)
|
|
||||||
if err != nil {
|
|
||||||
n, err = ParseUnit(s, metricBytesUnitMap)
|
|
||||||
}
|
|
||||||
return int64(n), err
|
|
||||||
}
|
|
13
vendor/github.com/alecthomas/units/doc.go
generated
vendored
13
vendor/github.com/alecthomas/units/doc.go
generated
vendored
|
@ -1,13 +0,0 @@
|
||||||
// Package units provides helpful unit multipliers and functions for Go.
|
|
||||||
//
|
|
||||||
// The goal of this package is to have functionality similar to the time [1] package.
|
|
||||||
//
|
|
||||||
//
|
|
||||||
// [1] http://golang.org/pkg/time/
|
|
||||||
//
|
|
||||||
// It allows for code like this:
|
|
||||||
//
|
|
||||||
// n, err := ParseBase2Bytes("1KB")
|
|
||||||
// // n == 1024
|
|
||||||
// n = units.Mebibyte * 512
|
|
||||||
package units
|
|
3
vendor/github.com/alecthomas/units/go.mod
generated
vendored
3
vendor/github.com/alecthomas/units/go.mod
generated
vendored
|
@ -1,3 +0,0 @@
|
||||||
module github.com/alecthomas/units
|
|
||||||
|
|
||||||
require github.com/stretchr/testify v1.4.0
|
|
11
vendor/github.com/alecthomas/units/go.sum
generated
vendored
11
vendor/github.com/alecthomas/units/go.sum
generated
vendored
|
@ -1,11 +0,0 @@
|
||||||
github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
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/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
|
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw=
|
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
50
vendor/github.com/alecthomas/units/si.go
generated
vendored
50
vendor/github.com/alecthomas/units/si.go
generated
vendored
|
@ -1,50 +0,0 @@
|
||||||
package units
|
|
||||||
|
|
||||||
// SI units.
|
|
||||||
type SI int64
|
|
||||||
|
|
||||||
// SI unit multiples.
|
|
||||||
const (
|
|
||||||
Kilo SI = 1000
|
|
||||||
Mega = Kilo * 1000
|
|
||||||
Giga = Mega * 1000
|
|
||||||
Tera = Giga * 1000
|
|
||||||
Peta = Tera * 1000
|
|
||||||
Exa = Peta * 1000
|
|
||||||
)
|
|
||||||
|
|
||||||
func MakeUnitMap(suffix, shortSuffix string, scale int64) map[string]float64 {
|
|
||||||
res := map[string]float64{
|
|
||||||
shortSuffix: 1,
|
|
||||||
// see below for "k" / "K"
|
|
||||||
"M" + suffix: float64(scale * scale),
|
|
||||||
"G" + suffix: float64(scale * scale * scale),
|
|
||||||
"T" + suffix: float64(scale * scale * scale * scale),
|
|
||||||
"P" + suffix: float64(scale * scale * scale * scale * scale),
|
|
||||||
"E" + suffix: float64(scale * scale * scale * scale * scale * scale),
|
|
||||||
}
|
|
||||||
|
|
||||||
// Standard SI prefixes use lowercase "k" for kilo = 1000.
|
|
||||||
// For compatibility, and to be fool-proof, we accept both "k" and "K" in metric mode.
|
|
||||||
//
|
|
||||||
// However, official binary prefixes are always capitalized - "KiB" -
|
|
||||||
// and we specifically never parse "kB" as 1024B because:
|
|
||||||
//
|
|
||||||
// (1) people pedantic enough to use lowercase according to SI unlikely to abuse "k" to mean 1024 :-)
|
|
||||||
//
|
|
||||||
// (2) Use of capital K for 1024 was an informal tradition predating IEC prefixes:
|
|
||||||
// "The binary meaning of the kilobyte for 1024 bytes typically uses the symbol KB, with an
|
|
||||||
// uppercase letter K."
|
|
||||||
// -- https://en.wikipedia.org/wiki/Kilobyte#Base_2_(1024_bytes)
|
|
||||||
// "Capitalization of the letter K became the de facto standard for binary notation, although this
|
|
||||||
// could not be extended to higher powers, and use of the lowercase k did persist.[13][14][15]"
|
|
||||||
// -- https://en.wikipedia.org/wiki/Binary_prefix#History
|
|
||||||
// See also the extensive https://en.wikipedia.org/wiki/Timeline_of_binary_prefixes.
|
|
||||||
if scale == 1024 {
|
|
||||||
res["K"+suffix] = float64(scale)
|
|
||||||
} else {
|
|
||||||
res["k"+suffix] = float64(scale)
|
|
||||||
res["K"+suffix] = float64(scale)
|
|
||||||
}
|
|
||||||
return res
|
|
||||||
}
|
|
138
vendor/github.com/alecthomas/units/util.go
generated
vendored
138
vendor/github.com/alecthomas/units/util.go
generated
vendored
|
@ -1,138 +0,0 @@
|
||||||
package units
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
siUnits = []string{"", "K", "M", "G", "T", "P", "E"}
|
|
||||||
)
|
|
||||||
|
|
||||||
func ToString(n int64, scale int64, suffix, baseSuffix string) string {
|
|
||||||
mn := len(siUnits)
|
|
||||||
out := make([]string, mn)
|
|
||||||
for i, m := range siUnits {
|
|
||||||
if n%scale != 0 || i == 0 && n == 0 {
|
|
||||||
s := suffix
|
|
||||||
if i == 0 {
|
|
||||||
s = baseSuffix
|
|
||||||
}
|
|
||||||
out[mn-1-i] = fmt.Sprintf("%d%s%s", n%scale, m, s)
|
|
||||||
}
|
|
||||||
n /= scale
|
|
||||||
if n == 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return strings.Join(out, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Below code ripped straight from http://golang.org/src/pkg/time/format.go?s=33392:33438#L1123
|
|
||||||
var errLeadingInt = errors.New("units: bad [0-9]*") // never printed
|
|
||||||
|
|
||||||
// leadingInt consumes the leading [0-9]* from s.
|
|
||||||
func leadingInt(s string) (x int64, rem string, err error) {
|
|
||||||
i := 0
|
|
||||||
for ; i < len(s); i++ {
|
|
||||||
c := s[i]
|
|
||||||
if c < '0' || c > '9' {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if x >= (1<<63-10)/10 {
|
|
||||||
// overflow
|
|
||||||
return 0, "", errLeadingInt
|
|
||||||
}
|
|
||||||
x = x*10 + int64(c) - '0'
|
|
||||||
}
|
|
||||||
return x, s[i:], nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseUnit(s string, unitMap map[string]float64) (int64, error) {
|
|
||||||
// [-+]?([0-9]*(\.[0-9]*)?[a-z]+)+
|
|
||||||
orig := s
|
|
||||||
f := float64(0)
|
|
||||||
neg := false
|
|
||||||
|
|
||||||
// Consume [-+]?
|
|
||||||
if s != "" {
|
|
||||||
c := s[0]
|
|
||||||
if c == '-' || c == '+' {
|
|
||||||
neg = c == '-'
|
|
||||||
s = s[1:]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Special case: if all that is left is "0", this is zero.
|
|
||||||
if s == "0" {
|
|
||||||
return 0, nil
|
|
||||||
}
|
|
||||||
if s == "" {
|
|
||||||
return 0, errors.New("units: invalid " + orig)
|
|
||||||
}
|
|
||||||
for s != "" {
|
|
||||||
g := float64(0) // this element of the sequence
|
|
||||||
|
|
||||||
var x int64
|
|
||||||
var err error
|
|
||||||
|
|
||||||
// The next character must be [0-9.]
|
|
||||||
if !(s[0] == '.' || ('0' <= s[0] && s[0] <= '9')) {
|
|
||||||
return 0, errors.New("units: invalid " + orig)
|
|
||||||
}
|
|
||||||
// Consume [0-9]*
|
|
||||||
pl := len(s)
|
|
||||||
x, s, err = leadingInt(s)
|
|
||||||
if err != nil {
|
|
||||||
return 0, errors.New("units: invalid " + orig)
|
|
||||||
}
|
|
||||||
g = float64(x)
|
|
||||||
pre := pl != len(s) // whether we consumed anything before a period
|
|
||||||
|
|
||||||
// Consume (\.[0-9]*)?
|
|
||||||
post := false
|
|
||||||
if s != "" && s[0] == '.' {
|
|
||||||
s = s[1:]
|
|
||||||
pl := len(s)
|
|
||||||
x, s, err = leadingInt(s)
|
|
||||||
if err != nil {
|
|
||||||
return 0, errors.New("units: invalid " + orig)
|
|
||||||
}
|
|
||||||
scale := 1.0
|
|
||||||
for n := pl - len(s); n > 0; n-- {
|
|
||||||
scale *= 10
|
|
||||||
}
|
|
||||||
g += float64(x) / scale
|
|
||||||
post = pl != len(s)
|
|
||||||
}
|
|
||||||
if !pre && !post {
|
|
||||||
// no digits (e.g. ".s" or "-.s")
|
|
||||||
return 0, errors.New("units: invalid " + orig)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Consume unit.
|
|
||||||
i := 0
|
|
||||||
for ; i < len(s); i++ {
|
|
||||||
c := s[i]
|
|
||||||
if c == '.' || ('0' <= c && c <= '9') {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
u := s[:i]
|
|
||||||
s = s[i:]
|
|
||||||
unit, ok := unitMap[u]
|
|
||||||
if !ok {
|
|
||||||
return 0, errors.New("units: unknown unit " + u + " in " + orig)
|
|
||||||
}
|
|
||||||
|
|
||||||
f += g * unit
|
|
||||||
}
|
|
||||||
|
|
||||||
if neg {
|
|
||||||
f = -f
|
|
||||||
}
|
|
||||||
if f < float64(-1<<63) || f > float64(1<<63-1) {
|
|
||||||
return 0, errors.New("units: overflow parsing unit")
|
|
||||||
}
|
|
||||||
return int64(f), nil
|
|
||||||
}
|
|
14
vendor/github.com/beevik/ntp/.travis.yml
generated
vendored
14
vendor/github.com/beevik/ntp/.travis.yml
generated
vendored
|
@ -1,14 +0,0 @@
|
||||||
language: go
|
|
||||||
sudo: false
|
|
||||||
|
|
||||||
go:
|
|
||||||
- 1.9.x
|
|
||||||
- 1.12.x
|
|
||||||
- tip
|
|
||||||
|
|
||||||
matrix:
|
|
||||||
allow_failures:
|
|
||||||
- go: tip
|
|
||||||
|
|
||||||
script:
|
|
||||||
- go test -v ./...
|
|
7
vendor/github.com/beevik/ntp/CONTRIBUTORS
generated
vendored
7
vendor/github.com/beevik/ntp/CONTRIBUTORS
generated
vendored
|
@ -1,7 +0,0 @@
|
||||||
Brett Vickers (beevik)
|
|
||||||
Mikhail Salosin (AlphaB)
|
|
||||||
Anton Tolchanov (knyar)
|
|
||||||
Christopher Batey (chbatey)
|
|
||||||
Meng Zhuo (mengzhuo)
|
|
||||||
Leonid Evdokimov (darkk)
|
|
||||||
Ask Bjørn Hansen (abh)
|
|
24
vendor/github.com/beevik/ntp/LICENSE
generated
vendored
24
vendor/github.com/beevik/ntp/LICENSE
generated
vendored
|
@ -1,24 +0,0 @@
|
||||||
Copyright 2015-2017 Brett Vickers. All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
|
||||||
modification, are permitted provided that the following conditions
|
|
||||||
are met:
|
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright
|
|
||||||
notice, this list of conditions and the following disclaimer.
|
|
||||||
|
|
||||||
2. 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.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY COPYRIGHT HOLDER ``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 COPYRIGHT HOLDER 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.
|
|
72
vendor/github.com/beevik/ntp/README.md
generated
vendored
72
vendor/github.com/beevik/ntp/README.md
generated
vendored
|
@ -1,72 +0,0 @@
|
||||||
[![Build Status](https://travis-ci.org/beevik/ntp.svg?branch=master)](https://travis-ci.org/beevik/ntp)
|
|
||||||
[![GoDoc](https://godoc.org/github.com/beevik/ntp?status.svg)](https://godoc.org/github.com/beevik/ntp)
|
|
||||||
|
|
||||||
ntp
|
|
||||||
===
|
|
||||||
|
|
||||||
The ntp package is an implementation of a Simple NTP (SNTP) client based on
|
|
||||||
[RFC5905](https://tools.ietf.org/html/rfc5905). It allows you to connect to
|
|
||||||
a remote NTP server and request information about the current time.
|
|
||||||
|
|
||||||
|
|
||||||
## Querying the current time
|
|
||||||
|
|
||||||
If all you care about is the current time according to a remote NTP server,
|
|
||||||
simply use the `Time` function:
|
|
||||||
```go
|
|
||||||
time, err := ntp.Time("0.beevik-ntp.pool.ntp.org")
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Querying time metadata
|
|
||||||
|
|
||||||
To obtain the current time as well as some additional metadata about the time,
|
|
||||||
use the [`Query`](https://godoc.org/github.com/beevik/ntp#Query) function:
|
|
||||||
```go
|
|
||||||
response, err := ntp.Query("0.beevik-ntp.pool.ntp.org")
|
|
||||||
time := time.Now().Add(response.ClockOffset)
|
|
||||||
```
|
|
||||||
|
|
||||||
Alternatively, use the [`QueryWithOptions`](https://godoc.org/github.com/beevik/ntp#QueryWithOptions)
|
|
||||||
function if you want to change the default behavior used by the `Query`
|
|
||||||
function:
|
|
||||||
```go
|
|
||||||
options := ntp.QueryOptions{ Timeout: 30*time.Second, TTL: 5 }
|
|
||||||
response, err := ntp.QueryWithOptions("0.beevik-ntp.pool.ntp.org", options)
|
|
||||||
time := time.Now().Add(response.ClockOffset)
|
|
||||||
```
|
|
||||||
|
|
||||||
The [`Response`](https://godoc.org/github.com/beevik/ntp#Response) structure
|
|
||||||
returned by `Query` includes the following information:
|
|
||||||
* `Time`: The time the server transmitted its response, according to its own clock.
|
|
||||||
* `ClockOffset`: The estimated offset of the local system clock relative to the server's clock. For a more accurate time reading, you may add this offset to any subsequent system clock reading.
|
|
||||||
* `RTT`: An estimate of the round-trip-time delay between the client and the server.
|
|
||||||
* `Precision`: The precision of the server's clock reading.
|
|
||||||
* `Stratum`: The server's stratum, which indicates the number of hops from the server to the reference clock. A stratum 1 server is directly attached to the reference clock. If the stratum is zero, the server has responded with the "kiss of death".
|
|
||||||
* `ReferenceID`: A unique identifier for the consulted reference clock.
|
|
||||||
* `ReferenceTime`: The time at which the server last updated its local clock setting.
|
|
||||||
* `RootDelay`: The server's aggregate round-trip-time delay to the stratum 1 server.
|
|
||||||
* `RootDispersion`: The server's estimated maximum measurement error relative to the reference clock.
|
|
||||||
* `RootDistance`: An estimate of the root synchronization distance between the client and the stratum 1 server.
|
|
||||||
* `Leap`: The leap second indicator, indicating whether a second should be added to or removed from the current month's last minute.
|
|
||||||
* `MinError`: A lower bound on the clock error between the client and the server.
|
|
||||||
* `KissCode`: A 4-character string describing the reason for a "kiss of death" response (stratum=0).
|
|
||||||
* `Poll`: The maximum polling interval between successive messages to the server.
|
|
||||||
|
|
||||||
The `Response` structure's [`Validate`](https://godoc.org/github.com/beevik/ntp#Response.Validate)
|
|
||||||
method performs additional sanity checks to determine whether the response is
|
|
||||||
suitable for time synchronization purposes.
|
|
||||||
```go
|
|
||||||
err := response.Validate()
|
|
||||||
if err == nil {
|
|
||||||
// response data is suitable for synchronization purposes
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Using the NTP pool
|
|
||||||
|
|
||||||
The NTP pool is a shared resource used by people all over the world.
|
|
||||||
To prevent it from becoming overloaded, please avoid querying the standard
|
|
||||||
`pool.ntp.org` zone names in your applications. Instead, consider requesting
|
|
||||||
your own [vendor zone](http://www.pool.ntp.org/en/vendors.html) or [joining
|
|
||||||
the pool](http://www.pool.ntp.org/join.html).
|
|
64
vendor/github.com/beevik/ntp/RELEASE_NOTES.md
generated
vendored
64
vendor/github.com/beevik/ntp/RELEASE_NOTES.md
generated
vendored
|
@ -1,64 +0,0 @@
|
||||||
Release v0.3.0
|
|
||||||
==============
|
|
||||||
|
|
||||||
There have been no breaking changes or further deprecations since the
|
|
||||||
previous release.
|
|
||||||
|
|
||||||
**Changes**
|
|
||||||
|
|
||||||
* Fixed a bug in the calculation of NTP timestamps.
|
|
||||||
|
|
||||||
Release v0.2.0
|
|
||||||
==============
|
|
||||||
|
|
||||||
There are no breaking changes or further deprecations in this release.
|
|
||||||
|
|
||||||
**Changes**
|
|
||||||
|
|
||||||
* Added `KissCode` to the `Response` structure.
|
|
||||||
|
|
||||||
|
|
||||||
Release v0.1.1
|
|
||||||
==============
|
|
||||||
|
|
||||||
**Breaking changes**
|
|
||||||
|
|
||||||
* Removed the `MaxStratum` constant.
|
|
||||||
|
|
||||||
**Deprecations**
|
|
||||||
|
|
||||||
* Officially deprecated the `TimeV` function.
|
|
||||||
|
|
||||||
**Internal changes**
|
|
||||||
|
|
||||||
* Removed `minDispersion` from the `RootDistance` calculation, since the value
|
|
||||||
was arbitrary.
|
|
||||||
* Moved some validation into main code path so that invalid `TransmitTime` and
|
|
||||||
`mode` responses trigger an error even when `Response.Validate` is not
|
|
||||||
called.
|
|
||||||
|
|
||||||
|
|
||||||
Release v0.1.0
|
|
||||||
==============
|
|
||||||
|
|
||||||
This is the initial release of the `ntp` package. Currently it supports the following features:
|
|
||||||
* `Time()` to query the current time according to a remote NTP server.
|
|
||||||
* `Query()` to query multiple pieces of time-related information from a remote NTP server.
|
|
||||||
* `QueryWithOptions()`, which is like `Query()` but with the ability to override default query options.
|
|
||||||
|
|
||||||
Time-related information returned by the `Query` functions includes:
|
|
||||||
* `Time`: the time the server transmitted its response, according to the server's clock.
|
|
||||||
* `ClockOffset`: the estimated offset of the client's clock relative to the server's clock. You may apply this offset to any local system clock reading once the query is complete.
|
|
||||||
* `RTT`: an estimate of the round-trip-time delay between the client and the server.
|
|
||||||
* `Precision`: the precision of the server's clock reading.
|
|
||||||
* `Stratum`: the "stratum" level of the server, where 1 indicates a server directly connected to a reference clock, and values greater than 1 indicating the number of hops from the reference clock.
|
|
||||||
* `ReferenceID`: A unique identifier for the NTP server that was contacted.
|
|
||||||
* `ReferenceTime`: The time at which the server last updated its local clock setting.
|
|
||||||
* `RootDelay`: The server's round-trip delay to the reference clock.
|
|
||||||
* `RootDispersion`: The server's total dispersion to the referenced clock.
|
|
||||||
* `RootDistance`: An estimate of the root synchronization distance.
|
|
||||||
* `Leap`: The leap second indicator.
|
|
||||||
* `MinError`: A lower bound on the clock error between the client and the server.
|
|
||||||
* `Poll`: the maximum polling interval between successive messages on the server.
|
|
||||||
|
|
||||||
The `Response` structure returned by the `Query` functions also contains a `Response.Validate()` function that returns an error if any of the fields returned by the server are invalid.
|
|
573
vendor/github.com/beevik/ntp/ntp.go
generated
vendored
573
vendor/github.com/beevik/ntp/ntp.go
generated
vendored
|
@ -1,573 +0,0 @@
|
||||||
// Copyright 2015-2017 Brett Vickers.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
// Package ntp provides an implementation of a Simple NTP (SNTP) client
|
|
||||||
// capable of querying the current time from a remote NTP server. See
|
|
||||||
// RFC5905 (https://tools.ietf.org/html/rfc5905) for more details.
|
|
||||||
//
|
|
||||||
// This approach grew out of a go-nuts post by Michael Hofmann:
|
|
||||||
// https://groups.google.com/forum/?fromgroups#!topic/golang-nuts/FlcdMU5fkLQ
|
|
||||||
package ntp
|
|
||||||
|
|
||||||
import (
|
|
||||||
"crypto/rand"
|
|
||||||
"encoding/binary"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"golang.org/x/net/ipv4"
|
|
||||||
)
|
|
||||||
|
|
||||||
// The LeapIndicator is used to warn if a leap second should be inserted
|
|
||||||
// or deleted in the last minute of the current month.
|
|
||||||
type LeapIndicator uint8
|
|
||||||
|
|
||||||
const (
|
|
||||||
// LeapNoWarning indicates no impending leap second.
|
|
||||||
LeapNoWarning LeapIndicator = 0
|
|
||||||
|
|
||||||
// LeapAddSecond indicates the last minute of the day has 61 seconds.
|
|
||||||
LeapAddSecond = 1
|
|
||||||
|
|
||||||
// LeapDelSecond indicates the last minute of the day has 59 seconds.
|
|
||||||
LeapDelSecond = 2
|
|
||||||
|
|
||||||
// LeapNotInSync indicates an unsynchronized leap second.
|
|
||||||
LeapNotInSync = 3
|
|
||||||
)
|
|
||||||
|
|
||||||
// Internal constants
|
|
||||||
const (
|
|
||||||
defaultNtpVersion = 4
|
|
||||||
nanoPerSec = 1000000000
|
|
||||||
maxStratum = 16
|
|
||||||
defaultTimeout = 5 * time.Second
|
|
||||||
maxPollInterval = (1 << 17) * time.Second
|
|
||||||
maxDispersion = 16 * time.Second
|
|
||||||
)
|
|
||||||
|
|
||||||
// Internal variables
|
|
||||||
var (
|
|
||||||
ntpEpoch = time.Date(1900, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
||||||
)
|
|
||||||
|
|
||||||
type mode uint8
|
|
||||||
|
|
||||||
// NTP modes. This package uses only client mode.
|
|
||||||
const (
|
|
||||||
reserved mode = 0 + iota
|
|
||||||
symmetricActive
|
|
||||||
symmetricPassive
|
|
||||||
client
|
|
||||||
server
|
|
||||||
broadcast
|
|
||||||
controlMessage
|
|
||||||
reservedPrivate
|
|
||||||
)
|
|
||||||
|
|
||||||
// An ntpTime is a 64-bit fixed-point (Q32.32) representation of the number of
|
|
||||||
// seconds elapsed.
|
|
||||||
type ntpTime uint64
|
|
||||||
|
|
||||||
// Duration interprets the fixed-point ntpTime as a number of elapsed seconds
|
|
||||||
// and returns the corresponding time.Duration value.
|
|
||||||
func (t ntpTime) Duration() time.Duration {
|
|
||||||
sec := (t >> 32) * nanoPerSec
|
|
||||||
frac := (t & 0xffffffff) * nanoPerSec
|
|
||||||
nsec := frac >> 32
|
|
||||||
if uint32(frac) >= 0x80000000 {
|
|
||||||
nsec++
|
|
||||||
}
|
|
||||||
return time.Duration(sec + nsec)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Time interprets the fixed-point ntpTime as an absolute time and returns
|
|
||||||
// the corresponding time.Time value.
|
|
||||||
func (t ntpTime) Time() time.Time {
|
|
||||||
return ntpEpoch.Add(t.Duration())
|
|
||||||
}
|
|
||||||
|
|
||||||
// toNtpTime converts the time.Time value t into its 64-bit fixed-point
|
|
||||||
// ntpTime representation.
|
|
||||||
func toNtpTime(t time.Time) ntpTime {
|
|
||||||
nsec := uint64(t.Sub(ntpEpoch))
|
|
||||||
sec := nsec / nanoPerSec
|
|
||||||
nsec = uint64(nsec-sec*nanoPerSec) << 32
|
|
||||||
frac := uint64(nsec / nanoPerSec)
|
|
||||||
if nsec%nanoPerSec >= nanoPerSec/2 {
|
|
||||||
frac++
|
|
||||||
}
|
|
||||||
return ntpTime(sec<<32 | frac)
|
|
||||||
}
|
|
||||||
|
|
||||||
// An ntpTimeShort is a 32-bit fixed-point (Q16.16) representation of the
|
|
||||||
// number of seconds elapsed.
|
|
||||||
type ntpTimeShort uint32
|
|
||||||
|
|
||||||
// Duration interprets the fixed-point ntpTimeShort as a number of elapsed
|
|
||||||
// seconds and returns the corresponding time.Duration value.
|
|
||||||
func (t ntpTimeShort) Duration() time.Duration {
|
|
||||||
sec := uint64(t>>16) * nanoPerSec
|
|
||||||
frac := uint64(t&0xffff) * nanoPerSec
|
|
||||||
nsec := frac >> 16
|
|
||||||
if uint16(frac) >= 0x8000 {
|
|
||||||
nsec++
|
|
||||||
}
|
|
||||||
return time.Duration(sec + nsec)
|
|
||||||
}
|
|
||||||
|
|
||||||
// msg is an internal representation of an NTP packet.
|
|
||||||
type msg struct {
|
|
||||||
LiVnMode uint8 // Leap Indicator (2) + Version (3) + Mode (3)
|
|
||||||
Stratum uint8
|
|
||||||
Poll int8
|
|
||||||
Precision int8
|
|
||||||
RootDelay ntpTimeShort
|
|
||||||
RootDispersion ntpTimeShort
|
|
||||||
ReferenceID uint32
|
|
||||||
ReferenceTime ntpTime
|
|
||||||
OriginTime ntpTime
|
|
||||||
ReceiveTime ntpTime
|
|
||||||
TransmitTime ntpTime
|
|
||||||
}
|
|
||||||
|
|
||||||
// setVersion sets the NTP protocol version on the message.
|
|
||||||
func (m *msg) setVersion(v int) {
|
|
||||||
m.LiVnMode = (m.LiVnMode & 0xc7) | uint8(v)<<3
|
|
||||||
}
|
|
||||||
|
|
||||||
// setMode sets the NTP protocol mode on the message.
|
|
||||||
func (m *msg) setMode(md mode) {
|
|
||||||
m.LiVnMode = (m.LiVnMode & 0xf8) | uint8(md)
|
|
||||||
}
|
|
||||||
|
|
||||||
// setLeap modifies the leap indicator on the message.
|
|
||||||
func (m *msg) setLeap(li LeapIndicator) {
|
|
||||||
m.LiVnMode = (m.LiVnMode & 0x3f) | uint8(li)<<6
|
|
||||||
}
|
|
||||||
|
|
||||||
// getVersion returns the version value in the message.
|
|
||||||
func (m *msg) getVersion() int {
|
|
||||||
return int((m.LiVnMode >> 3) & 0x07)
|
|
||||||
}
|
|
||||||
|
|
||||||
// getMode returns the mode value in the message.
|
|
||||||
func (m *msg) getMode() mode {
|
|
||||||
return mode(m.LiVnMode & 0x07)
|
|
||||||
}
|
|
||||||
|
|
||||||
// getLeap returns the leap indicator on the message.
|
|
||||||
func (m *msg) getLeap() LeapIndicator {
|
|
||||||
return LeapIndicator((m.LiVnMode >> 6) & 0x03)
|
|
||||||
}
|
|
||||||
|
|
||||||
// QueryOptions contains the list of configurable options that may be used
|
|
||||||
// with the QueryWithOptions function.
|
|
||||||
type QueryOptions struct {
|
|
||||||
Timeout time.Duration // defaults to 5 seconds
|
|
||||||
Version int // NTP protocol version, defaults to 4
|
|
||||||
LocalAddress string // IP address to use for the client address
|
|
||||||
Port int // Server port, defaults to 123
|
|
||||||
TTL int // IP TTL to use, defaults to system default
|
|
||||||
}
|
|
||||||
|
|
||||||
// A Response contains time data, some of which is returned by the NTP server
|
|
||||||
// and some of which is calculated by the client.
|
|
||||||
type Response struct {
|
|
||||||
// Time is the transmit time reported by the server just before it
|
|
||||||
// responded to the client's NTP query.
|
|
||||||
Time time.Time
|
|
||||||
|
|
||||||
// ClockOffset is the estimated offset of the client clock relative to
|
|
||||||
// the server. Add this to the client's system clock time to obtain a
|
|
||||||
// more accurate time.
|
|
||||||
ClockOffset time.Duration
|
|
||||||
|
|
||||||
// RTT is the measured round-trip-time delay estimate between the client
|
|
||||||
// and the server.
|
|
||||||
RTT time.Duration
|
|
||||||
|
|
||||||
// Precision is the reported precision of the server's clock.
|
|
||||||
Precision time.Duration
|
|
||||||
|
|
||||||
// Stratum is the "stratum level" of the server. The smaller the number,
|
|
||||||
// the closer the server is to the reference clock. Stratum 1 servers are
|
|
||||||
// attached directly to the reference clock. A stratum value of 0
|
|
||||||
// indicates the "kiss of death," which typically occurs when the client
|
|
||||||
// issues too many requests to the server in a short period of time.
|
|
||||||
Stratum uint8
|
|
||||||
|
|
||||||
// ReferenceID is a 32-bit identifier identifying the server or
|
|
||||||
// reference clock.
|
|
||||||
ReferenceID uint32
|
|
||||||
|
|
||||||
// ReferenceTime is the time when the server's system clock was last
|
|
||||||
// set or corrected.
|
|
||||||
ReferenceTime time.Time
|
|
||||||
|
|
||||||
// RootDelay is the server's estimated aggregate round-trip-time delay to
|
|
||||||
// the stratum 1 server.
|
|
||||||
RootDelay time.Duration
|
|
||||||
|
|
||||||
// RootDispersion is the server's estimated maximum measurement error
|
|
||||||
// relative to the stratum 1 server.
|
|
||||||
RootDispersion time.Duration
|
|
||||||
|
|
||||||
// RootDistance is an estimate of the total synchronization distance
|
|
||||||
// between the client and the stratum 1 server.
|
|
||||||
RootDistance time.Duration
|
|
||||||
|
|
||||||
// Leap indicates whether a leap second should be added or removed from
|
|
||||||
// the current month's last minute.
|
|
||||||
Leap LeapIndicator
|
|
||||||
|
|
||||||
// MinError is a lower bound on the error between the client and server
|
|
||||||
// clocks. When the client and server are not synchronized to the same
|
|
||||||
// clock, the reported timestamps may appear to violate the principle of
|
|
||||||
// causality. In other words, the NTP server's response may indicate
|
|
||||||
// that a message was received before it was sent. In such cases, the
|
|
||||||
// minimum error may be useful.
|
|
||||||
MinError time.Duration
|
|
||||||
|
|
||||||
// KissCode is a 4-character string describing the reason for a
|
|
||||||
// "kiss of death" response (stratum = 0). For a list of standard kiss
|
|
||||||
// codes, see https://tools.ietf.org/html/rfc5905#section-7.4.
|
|
||||||
KissCode string
|
|
||||||
|
|
||||||
// Poll is the maximum interval between successive NTP polling messages.
|
|
||||||
// It is not relevant for simple NTP clients like this one.
|
|
||||||
Poll time.Duration
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validate checks if the response is valid for the purposes of time
|
|
||||||
// synchronization.
|
|
||||||
func (r *Response) Validate() error {
|
|
||||||
// Handle invalid stratum values.
|
|
||||||
if r.Stratum == 0 {
|
|
||||||
return fmt.Errorf("kiss of death received: %s", r.KissCode)
|
|
||||||
}
|
|
||||||
if r.Stratum >= maxStratum {
|
|
||||||
return errors.New("invalid stratum in response")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle invalid leap second indicator.
|
|
||||||
if r.Leap == LeapNotInSync {
|
|
||||||
return errors.New("invalid leap second")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Estimate the "freshness" of the time. If it exceeds the maximum
|
|
||||||
// polling interval (~36 hours), then it cannot be considered "fresh".
|
|
||||||
freshness := r.Time.Sub(r.ReferenceTime)
|
|
||||||
if freshness > maxPollInterval {
|
|
||||||
return errors.New("server clock not fresh")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate the peer synchronization distance, lambda:
|
|
||||||
// lambda := RootDelay/2 + RootDispersion
|
|
||||||
// If this value exceeds MAXDISP (16s), then the time is not suitable
|
|
||||||
// for synchronization purposes.
|
|
||||||
// https://tools.ietf.org/html/rfc5905#appendix-A.5.1.1.
|
|
||||||
lambda := r.RootDelay/2 + r.RootDispersion
|
|
||||||
if lambda > maxDispersion {
|
|
||||||
return errors.New("invalid dispersion")
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the server's transmit time is before its reference time, the
|
|
||||||
// response is invalid.
|
|
||||||
if r.Time.Before(r.ReferenceTime) {
|
|
||||||
return errors.New("invalid time reported")
|
|
||||||
}
|
|
||||||
|
|
||||||
// nil means the response is valid.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Query returns a response from the remote NTP server host. It contains
|
|
||||||
// the time at which the server transmitted the response as well as other
|
|
||||||
// useful information about the time and the remote server.
|
|
||||||
func Query(host string) (*Response, error) {
|
|
||||||
return QueryWithOptions(host, QueryOptions{})
|
|
||||||
}
|
|
||||||
|
|
||||||
// QueryWithOptions performs the same function as Query but allows for the
|
|
||||||
// customization of several query options.
|
|
||||||
func QueryWithOptions(host string, opt QueryOptions) (*Response, error) {
|
|
||||||
m, now, err := getTime(host, opt)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return parseTime(m, now), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// TimeV returns the current time using information from a remote NTP server.
|
|
||||||
// On error, it returns the local system time. The version may be 2, 3, or 4.
|
|
||||||
//
|
|
||||||
// Deprecated: TimeV is deprecated. Use QueryWithOptions instead.
|
|
||||||
func TimeV(host string, version int) (time.Time, error) {
|
|
||||||
m, recvTime, err := getTime(host, QueryOptions{Version: version})
|
|
||||||
if err != nil {
|
|
||||||
return time.Now(), err
|
|
||||||
}
|
|
||||||
|
|
||||||
r := parseTime(m, recvTime)
|
|
||||||
err = r.Validate()
|
|
||||||
if err != nil {
|
|
||||||
return time.Now(), err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use the clock offset to calculate the time.
|
|
||||||
return time.Now().Add(r.ClockOffset), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Time returns the current time using information from a remote NTP server.
|
|
||||||
// It uses version 4 of the NTP protocol. On error, it returns the local
|
|
||||||
// system time.
|
|
||||||
func Time(host string) (time.Time, error) {
|
|
||||||
return TimeV(host, defaultNtpVersion)
|
|
||||||
}
|
|
||||||
|
|
||||||
// getTime performs the NTP server query and returns the response message
|
|
||||||
// along with the local system time it was received.
|
|
||||||
func getTime(host string, opt QueryOptions) (*msg, ntpTime, error) {
|
|
||||||
if opt.Version == 0 {
|
|
||||||
opt.Version = defaultNtpVersion
|
|
||||||
}
|
|
||||||
if opt.Version < 2 || opt.Version > 4 {
|
|
||||||
return nil, 0, errors.New("invalid protocol version requested")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Resolve the remote NTP server address.
|
|
||||||
raddr, err := net.ResolveUDPAddr("udp", net.JoinHostPort(host, "123"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Resolve the local address if specified as an option.
|
|
||||||
var laddr *net.UDPAddr
|
|
||||||
if opt.LocalAddress != "" {
|
|
||||||
laddr, err = net.ResolveUDPAddr("udp", net.JoinHostPort(opt.LocalAddress, "0"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Override the port if requested.
|
|
||||||
if opt.Port != 0 {
|
|
||||||
raddr.Port = opt.Port
|
|
||||||
}
|
|
||||||
|
|
||||||
// Prepare a "connection" to the remote server.
|
|
||||||
con, err := net.DialUDP("udp", laddr, raddr)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
defer con.Close()
|
|
||||||
|
|
||||||
// Set a TTL for the packet if requested.
|
|
||||||
if opt.TTL != 0 {
|
|
||||||
ipcon := ipv4.NewConn(con)
|
|
||||||
err = ipcon.SetTTL(opt.TTL)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set a timeout on the connection.
|
|
||||||
if opt.Timeout == 0 {
|
|
||||||
opt.Timeout = defaultTimeout
|
|
||||||
}
|
|
||||||
con.SetDeadline(time.Now().Add(opt.Timeout))
|
|
||||||
|
|
||||||
// Allocate a message to hold the response.
|
|
||||||
recvMsg := new(msg)
|
|
||||||
|
|
||||||
// Allocate a message to hold the query.
|
|
||||||
xmitMsg := new(msg)
|
|
||||||
xmitMsg.setMode(client)
|
|
||||||
xmitMsg.setVersion(opt.Version)
|
|
||||||
xmitMsg.setLeap(LeapNotInSync)
|
|
||||||
|
|
||||||
// To ensure privacy and prevent spoofing, try to use a random 64-bit
|
|
||||||
// value for the TransmitTime. If crypto/rand couldn't generate a
|
|
||||||
// random value, fall back to using the system clock. Keep track of
|
|
||||||
// when the messsage was actually transmitted.
|
|
||||||
bits := make([]byte, 8)
|
|
||||||
_, err = rand.Read(bits)
|
|
||||||
var xmitTime time.Time
|
|
||||||
if err == nil {
|
|
||||||
xmitMsg.TransmitTime = ntpTime(binary.BigEndian.Uint64(bits))
|
|
||||||
xmitTime = time.Now()
|
|
||||||
} else {
|
|
||||||
xmitTime = time.Now()
|
|
||||||
xmitMsg.TransmitTime = toNtpTime(xmitTime)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Transmit the query.
|
|
||||||
err = binary.Write(con, binary.BigEndian, xmitMsg)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Receive the response.
|
|
||||||
err = binary.Read(con, binary.BigEndian, recvMsg)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Keep track of the time the response was received.
|
|
||||||
delta := time.Since(xmitTime)
|
|
||||||
if delta < 0 {
|
|
||||||
// The local system may have had its clock adjusted since it
|
|
||||||
// sent the query. In go 1.9 and later, time.Since ensures
|
|
||||||
// that a monotonic clock is used, so delta can never be less
|
|
||||||
// than zero. In versions before 1.9, a monotonic clock is
|
|
||||||
// not used, so we have to check.
|
|
||||||
return nil, 0, errors.New("client clock ticked backwards")
|
|
||||||
}
|
|
||||||
recvTime := toNtpTime(xmitTime.Add(delta))
|
|
||||||
|
|
||||||
// Check for invalid fields.
|
|
||||||
if recvMsg.getMode() != server {
|
|
||||||
return nil, 0, errors.New("invalid mode in response")
|
|
||||||
}
|
|
||||||
if recvMsg.TransmitTime == ntpTime(0) {
|
|
||||||
return nil, 0, errors.New("invalid transmit time in response")
|
|
||||||
}
|
|
||||||
if recvMsg.OriginTime != xmitMsg.TransmitTime {
|
|
||||||
return nil, 0, errors.New("server response mismatch")
|
|
||||||
}
|
|
||||||
if recvMsg.ReceiveTime > recvMsg.TransmitTime {
|
|
||||||
return nil, 0, errors.New("server clock ticked backwards")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Correct the received message's origin time using the actual
|
|
||||||
// transmit time.
|
|
||||||
recvMsg.OriginTime = toNtpTime(xmitTime)
|
|
||||||
|
|
||||||
return recvMsg, recvTime, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseTime parses the NTP packet along with the packet receive time to
|
|
||||||
// generate a Response record.
|
|
||||||
func parseTime(m *msg, recvTime ntpTime) *Response {
|
|
||||||
r := &Response{
|
|
||||||
Time: m.TransmitTime.Time(),
|
|
||||||
ClockOffset: offset(m.OriginTime, m.ReceiveTime, m.TransmitTime, recvTime),
|
|
||||||
RTT: rtt(m.OriginTime, m.ReceiveTime, m.TransmitTime, recvTime),
|
|
||||||
Precision: toInterval(m.Precision),
|
|
||||||
Stratum: m.Stratum,
|
|
||||||
ReferenceID: m.ReferenceID,
|
|
||||||
ReferenceTime: m.ReferenceTime.Time(),
|
|
||||||
RootDelay: m.RootDelay.Duration(),
|
|
||||||
RootDispersion: m.RootDispersion.Duration(),
|
|
||||||
Leap: m.getLeap(),
|
|
||||||
MinError: minError(m.OriginTime, m.ReceiveTime, m.TransmitTime, recvTime),
|
|
||||||
Poll: toInterval(m.Poll),
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate values depending on other calculated values
|
|
||||||
r.RootDistance = rootDistance(r.RTT, r.RootDelay, r.RootDispersion)
|
|
||||||
|
|
||||||
// If a kiss of death was received, interpret the reference ID as
|
|
||||||
// a kiss code.
|
|
||||||
if r.Stratum == 0 {
|
|
||||||
r.KissCode = kissCode(r.ReferenceID)
|
|
||||||
}
|
|
||||||
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
// The following helper functions calculate additional metadata about the
|
|
||||||
// timestamps received from an NTP server. The timestamps returned by
|
|
||||||
// the server are given the following variable names:
|
|
||||||
//
|
|
||||||
// org = Origin Timestamp (client send time)
|
|
||||||
// rec = Receive Timestamp (server receive time)
|
|
||||||
// xmt = Transmit Timestamp (server reply time)
|
|
||||||
// dst = Destination Timestamp (client receive time)
|
|
||||||
|
|
||||||
func rtt(org, rec, xmt, dst ntpTime) time.Duration {
|
|
||||||
// round trip delay time
|
|
||||||
// rtt = (dst-org) - (xmt-rec)
|
|
||||||
a := dst.Time().Sub(org.Time())
|
|
||||||
b := xmt.Time().Sub(rec.Time())
|
|
||||||
rtt := a - b
|
|
||||||
if rtt < 0 {
|
|
||||||
rtt = 0
|
|
||||||
}
|
|
||||||
return rtt
|
|
||||||
}
|
|
||||||
|
|
||||||
func offset(org, rec, xmt, dst ntpTime) time.Duration {
|
|
||||||
// local clock offset
|
|
||||||
// offset = ((rec-org) + (xmt-dst)) / 2
|
|
||||||
a := rec.Time().Sub(org.Time())
|
|
||||||
b := xmt.Time().Sub(dst.Time())
|
|
||||||
return (a + b) / time.Duration(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
func minError(org, rec, xmt, dst ntpTime) time.Duration {
|
|
||||||
// Each NTP response contains two pairs of send/receive timestamps.
|
|
||||||
// When either pair indicates a "causality violation", we calculate the
|
|
||||||
// error as the difference in time between them. The minimum error is
|
|
||||||
// the greater of the two causality violations.
|
|
||||||
var error0, error1 ntpTime
|
|
||||||
if org >= rec {
|
|
||||||
error0 = org - rec
|
|
||||||
}
|
|
||||||
if xmt >= dst {
|
|
||||||
error1 = xmt - dst
|
|
||||||
}
|
|
||||||
if error0 > error1 {
|
|
||||||
return error0.Duration()
|
|
||||||
}
|
|
||||||
return error1.Duration()
|
|
||||||
}
|
|
||||||
|
|
||||||
func rootDistance(rtt, rootDelay, rootDisp time.Duration) time.Duration {
|
|
||||||
// The root distance is:
|
|
||||||
// the maximum error due to all causes of the local clock
|
|
||||||
// relative to the primary server. It is defined as half the
|
|
||||||
// total delay plus total dispersion plus peer jitter.
|
|
||||||
// (https://tools.ietf.org/html/rfc5905#appendix-A.5.5.2)
|
|
||||||
//
|
|
||||||
// In the reference implementation, it is calculated as follows:
|
|
||||||
// rootDist = max(MINDISP, rootDelay + rtt)/2 + rootDisp
|
|
||||||
// + peerDisp + PHI * (uptime - peerUptime)
|
|
||||||
// + peerJitter
|
|
||||||
// For an SNTP client which sends only a single packet, most of these
|
|
||||||
// terms are irrelevant and become 0.
|
|
||||||
totalDelay := rtt + rootDelay
|
|
||||||
return totalDelay/2 + rootDisp
|
|
||||||
}
|
|
||||||
|
|
||||||
func toInterval(t int8) time.Duration {
|
|
||||||
switch {
|
|
||||||
case t > 0:
|
|
||||||
return time.Duration(uint64(time.Second) << uint(t))
|
|
||||||
case t < 0:
|
|
||||||
return time.Duration(uint64(time.Second) >> uint(-t))
|
|
||||||
default:
|
|
||||||
return time.Second
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func kissCode(id uint32) string {
|
|
||||||
isPrintable := func(ch byte) bool { return ch >= 32 && ch <= 126 }
|
|
||||||
|
|
||||||
b := []byte{
|
|
||||||
byte(id >> 24),
|
|
||||||
byte(id >> 16),
|
|
||||||
byte(id >> 8),
|
|
||||||
byte(id),
|
|
||||||
}
|
|
||||||
for _, ch := range b {
|
|
||||||
if !isPrintable(ch) {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return string(b)
|
|
||||||
}
|
|
20
vendor/github.com/beorn7/perks/LICENSE
generated
vendored
20
vendor/github.com/beorn7/perks/LICENSE
generated
vendored
|
@ -1,20 +0,0 @@
|
||||||
Copyright (C) 2013 Blake Mizerany
|
|
||||||
|
|
||||||
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.
|
|
2388
vendor/github.com/beorn7/perks/quantile/exampledata.txt
generated
vendored
2388
vendor/github.com/beorn7/perks/quantile/exampledata.txt
generated
vendored
File diff suppressed because it is too large
Load diff
316
vendor/github.com/beorn7/perks/quantile/stream.go
generated
vendored
316
vendor/github.com/beorn7/perks/quantile/stream.go
generated
vendored
|
@ -1,316 +0,0 @@
|
||||||
// Package quantile computes approximate quantiles over an unbounded data
|
|
||||||
// stream within low memory and CPU bounds.
|
|
||||||
//
|
|
||||||
// A small amount of accuracy is traded to achieve the above properties.
|
|
||||||
//
|
|
||||||
// Multiple streams can be merged before calling Query to generate a single set
|
|
||||||
// of results. This is meaningful when the streams represent the same type of
|
|
||||||
// data. See Merge and Samples.
|
|
||||||
//
|
|
||||||
// For more detailed information about the algorithm used, see:
|
|
||||||
//
|
|
||||||
// Effective Computation of Biased Quantiles over Data Streams
|
|
||||||
//
|
|
||||||
// http://www.cs.rutgers.edu/~muthu/bquant.pdf
|
|
||||||
package quantile
|
|
||||||
|
|
||||||
import (
|
|
||||||
"math"
|
|
||||||
"sort"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Sample holds an observed value and meta information for compression. JSON
|
|
||||||
// tags have been added for convenience.
|
|
||||||
type Sample struct {
|
|
||||||
Value float64 `json:",string"`
|
|
||||||
Width float64 `json:",string"`
|
|
||||||
Delta float64 `json:",string"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Samples represents a slice of samples. It implements sort.Interface.
|
|
||||||
type Samples []Sample
|
|
||||||
|
|
||||||
func (a Samples) Len() int { return len(a) }
|
|
||||||
func (a Samples) Less(i, j int) bool { return a[i].Value < a[j].Value }
|
|
||||||
func (a Samples) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
||||||
|
|
||||||
type invariant func(s *stream, r float64) float64
|
|
||||||
|
|
||||||
// NewLowBiased returns an initialized Stream for low-biased quantiles
|
|
||||||
// (e.g. 0.01, 0.1, 0.5) where the needed quantiles are not known a priori, but
|
|
||||||
// error guarantees can still be given even for the lower ranks of the data
|
|
||||||
// distribution.
|
|
||||||
//
|
|
||||||
// The provided epsilon is a relative error, i.e. the true quantile of a value
|
|
||||||
// returned by a query is guaranteed to be within (1±Epsilon)*Quantile.
|
|
||||||
//
|
|
||||||
// See http://www.cs.rutgers.edu/~muthu/bquant.pdf for time, space, and error
|
|
||||||
// properties.
|
|
||||||
func NewLowBiased(epsilon float64) *Stream {
|
|
||||||
ƒ := func(s *stream, r float64) float64 {
|
|
||||||
return 2 * epsilon * r
|
|
||||||
}
|
|
||||||
return newStream(ƒ)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewHighBiased returns an initialized Stream for high-biased quantiles
|
|
||||||
// (e.g. 0.01, 0.1, 0.5) where the needed quantiles are not known a priori, but
|
|
||||||
// error guarantees can still be given even for the higher ranks of the data
|
|
||||||
// distribution.
|
|
||||||
//
|
|
||||||
// The provided epsilon is a relative error, i.e. the true quantile of a value
|
|
||||||
// returned by a query is guaranteed to be within 1-(1±Epsilon)*(1-Quantile).
|
|
||||||
//
|
|
||||||
// See http://www.cs.rutgers.edu/~muthu/bquant.pdf for time, space, and error
|
|
||||||
// properties.
|
|
||||||
func NewHighBiased(epsilon float64) *Stream {
|
|
||||||
ƒ := func(s *stream, r float64) float64 {
|
|
||||||
return 2 * epsilon * (s.n - r)
|
|
||||||
}
|
|
||||||
return newStream(ƒ)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewTargeted returns an initialized Stream concerned with a particular set of
|
|
||||||
// quantile values that are supplied a priori. Knowing these a priori reduces
|
|
||||||
// space and computation time. The targets map maps the desired quantiles to
|
|
||||||
// their absolute errors, i.e. the true quantile of a value returned by a query
|
|
||||||
// is guaranteed to be within (Quantile±Epsilon).
|
|
||||||
//
|
|
||||||
// See http://www.cs.rutgers.edu/~muthu/bquant.pdf for time, space, and error properties.
|
|
||||||
func NewTargeted(targetMap map[float64]float64) *Stream {
|
|
||||||
// Convert map to slice to avoid slow iterations on a map.
|
|
||||||
// ƒ is called on the hot path, so converting the map to a slice
|
|
||||||
// beforehand results in significant CPU savings.
|
|
||||||
targets := targetMapToSlice(targetMap)
|
|
||||||
|
|
||||||
ƒ := func(s *stream, r float64) float64 {
|
|
||||||
var m = math.MaxFloat64
|
|
||||||
var f float64
|
|
||||||
for _, t := range targets {
|
|
||||||
if t.quantile*s.n <= r {
|
|
||||||
f = (2 * t.epsilon * r) / t.quantile
|
|
||||||
} else {
|
|
||||||
f = (2 * t.epsilon * (s.n - r)) / (1 - t.quantile)
|
|
||||||
}
|
|
||||||
if f < m {
|
|
||||||
m = f
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return m
|
|
||||||
}
|
|
||||||
return newStream(ƒ)
|
|
||||||
}
|
|
||||||
|
|
||||||
type target struct {
|
|
||||||
quantile float64
|
|
||||||
epsilon float64
|
|
||||||
}
|
|
||||||
|
|
||||||
func targetMapToSlice(targetMap map[float64]float64) []target {
|
|
||||||
targets := make([]target, 0, len(targetMap))
|
|
||||||
|
|
||||||
for quantile, epsilon := range targetMap {
|
|
||||||
t := target{
|
|
||||||
quantile: quantile,
|
|
||||||
epsilon: epsilon,
|
|
||||||
}
|
|
||||||
targets = append(targets, t)
|
|
||||||
}
|
|
||||||
|
|
||||||
return targets
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stream computes quantiles for a stream of float64s. It is not thread-safe by
|
|
||||||
// design. Take care when using across multiple goroutines.
|
|
||||||
type Stream struct {
|
|
||||||
*stream
|
|
||||||
b Samples
|
|
||||||
sorted bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func newStream(ƒ invariant) *Stream {
|
|
||||||
x := &stream{ƒ: ƒ}
|
|
||||||
return &Stream{x, make(Samples, 0, 500), true}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Insert inserts v into the stream.
|
|
||||||
func (s *Stream) Insert(v float64) {
|
|
||||||
s.insert(Sample{Value: v, Width: 1})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Stream) insert(sample Sample) {
|
|
||||||
s.b = append(s.b, sample)
|
|
||||||
s.sorted = false
|
|
||||||
if len(s.b) == cap(s.b) {
|
|
||||||
s.flush()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Query returns the computed qth percentiles value. If s was created with
|
|
||||||
// NewTargeted, and q is not in the set of quantiles provided a priori, Query
|
|
||||||
// will return an unspecified result.
|
|
||||||
func (s *Stream) Query(q float64) float64 {
|
|
||||||
if !s.flushed() {
|
|
||||||
// Fast path when there hasn't been enough data for a flush;
|
|
||||||
// this also yields better accuracy for small sets of data.
|
|
||||||
l := len(s.b)
|
|
||||||
if l == 0 {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
i := int(math.Ceil(float64(l) * q))
|
|
||||||
if i > 0 {
|
|
||||||
i -= 1
|
|
||||||
}
|
|
||||||
s.maybeSort()
|
|
||||||
return s.b[i].Value
|
|
||||||
}
|
|
||||||
s.flush()
|
|
||||||
return s.stream.query(q)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Merge merges samples into the underlying streams samples. This is handy when
|
|
||||||
// merging multiple streams from separate threads, database shards, etc.
|
|
||||||
//
|
|
||||||
// ATTENTION: This method is broken and does not yield correct results. The
|
|
||||||
// underlying algorithm is not capable of merging streams correctly.
|
|
||||||
func (s *Stream) Merge(samples Samples) {
|
|
||||||
sort.Sort(samples)
|
|
||||||
s.stream.merge(samples)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reset reinitializes and clears the list reusing the samples buffer memory.
|
|
||||||
func (s *Stream) Reset() {
|
|
||||||
s.stream.reset()
|
|
||||||
s.b = s.b[:0]
|
|
||||||
}
|
|
||||||
|
|
||||||
// Samples returns stream samples held by s.
|
|
||||||
func (s *Stream) Samples() Samples {
|
|
||||||
if !s.flushed() {
|
|
||||||
return s.b
|
|
||||||
}
|
|
||||||
s.flush()
|
|
||||||
return s.stream.samples()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Count returns the total number of samples observed in the stream
|
|
||||||
// since initialization.
|
|
||||||
func (s *Stream) Count() int {
|
|
||||||
return len(s.b) + s.stream.count()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Stream) flush() {
|
|
||||||
s.maybeSort()
|
|
||||||
s.stream.merge(s.b)
|
|
||||||
s.b = s.b[:0]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Stream) maybeSort() {
|
|
||||||
if !s.sorted {
|
|
||||||
s.sorted = true
|
|
||||||
sort.Sort(s.b)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Stream) flushed() bool {
|
|
||||||
return len(s.stream.l) > 0
|
|
||||||
}
|
|
||||||
|
|
||||||
type stream struct {
|
|
||||||
n float64
|
|
||||||
l []Sample
|
|
||||||
ƒ invariant
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) reset() {
|
|
||||||
s.l = s.l[:0]
|
|
||||||
s.n = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) insert(v float64) {
|
|
||||||
s.merge(Samples{{v, 1, 0}})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) merge(samples Samples) {
|
|
||||||
// TODO(beorn7): This tries to merge not only individual samples, but
|
|
||||||
// whole summaries. The paper doesn't mention merging summaries at
|
|
||||||
// all. Unittests show that the merging is inaccurate. Find out how to
|
|
||||||
// do merges properly.
|
|
||||||
var r float64
|
|
||||||
i := 0
|
|
||||||
for _, sample := range samples {
|
|
||||||
for ; i < len(s.l); i++ {
|
|
||||||
c := s.l[i]
|
|
||||||
if c.Value > sample.Value {
|
|
||||||
// Insert at position i.
|
|
||||||
s.l = append(s.l, Sample{})
|
|
||||||
copy(s.l[i+1:], s.l[i:])
|
|
||||||
s.l[i] = Sample{
|
|
||||||
sample.Value,
|
|
||||||
sample.Width,
|
|
||||||
math.Max(sample.Delta, math.Floor(s.ƒ(s, r))-1),
|
|
||||||
// TODO(beorn7): How to calculate delta correctly?
|
|
||||||
}
|
|
||||||
i++
|
|
||||||
goto inserted
|
|
||||||
}
|
|
||||||
r += c.Width
|
|
||||||
}
|
|
||||||
s.l = append(s.l, Sample{sample.Value, sample.Width, 0})
|
|
||||||
i++
|
|
||||||
inserted:
|
|
||||||
s.n += sample.Width
|
|
||||||
r += sample.Width
|
|
||||||
}
|
|
||||||
s.compress()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) count() int {
|
|
||||||
return int(s.n)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) query(q float64) float64 {
|
|
||||||
t := math.Ceil(q * s.n)
|
|
||||||
t += math.Ceil(s.ƒ(s, t) / 2)
|
|
||||||
p := s.l[0]
|
|
||||||
var r float64
|
|
||||||
for _, c := range s.l[1:] {
|
|
||||||
r += p.Width
|
|
||||||
if r+c.Width+c.Delta > t {
|
|
||||||
return p.Value
|
|
||||||
}
|
|
||||||
p = c
|
|
||||||
}
|
|
||||||
return p.Value
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) compress() {
|
|
||||||
if len(s.l) < 2 {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
x := s.l[len(s.l)-1]
|
|
||||||
xi := len(s.l) - 1
|
|
||||||
r := s.n - 1 - x.Width
|
|
||||||
|
|
||||||
for i := len(s.l) - 2; i >= 0; i-- {
|
|
||||||
c := s.l[i]
|
|
||||||
if c.Width+x.Width+x.Delta <= s.ƒ(s, r) {
|
|
||||||
x.Width += c.Width
|
|
||||||
s.l[xi] = x
|
|
||||||
// Remove element at i.
|
|
||||||
copy(s.l[i:], s.l[i+1:])
|
|
||||||
s.l = s.l[:len(s.l)-1]
|
|
||||||
xi -= 1
|
|
||||||
} else {
|
|
||||||
x = c
|
|
||||||
xi = i
|
|
||||||
}
|
|
||||||
r -= c.Width
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stream) samples() Samples {
|
|
||||||
samples := make(Samples, len(s.l))
|
|
||||||
copy(samples, s.l)
|
|
||||||
return samples
|
|
||||||
}
|
|
8
vendor/github.com/cespare/xxhash/v2/.travis.yml
generated
vendored
8
vendor/github.com/cespare/xxhash/v2/.travis.yml
generated
vendored
|
@ -1,8 +0,0 @@
|
||||||
language: go
|
|
||||||
go:
|
|
||||||
- "1.x"
|
|
||||||
- master
|
|
||||||
env:
|
|
||||||
- TAGS=""
|
|
||||||
- TAGS="-tags purego"
|
|
||||||
script: go test $TAGS -v ./...
|
|
22
vendor/github.com/cespare/xxhash/v2/LICENSE.txt
generated
vendored
22
vendor/github.com/cespare/xxhash/v2/LICENSE.txt
generated
vendored
|
@ -1,22 +0,0 @@
|
||||||
Copyright (c) 2016 Caleb Spare
|
|
||||||
|
|
||||||
MIT License
|
|
||||||
|
|
||||||
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.
|
|
67
vendor/github.com/cespare/xxhash/v2/README.md
generated
vendored
67
vendor/github.com/cespare/xxhash/v2/README.md
generated
vendored
|
@ -1,67 +0,0 @@
|
||||||
# xxhash
|
|
||||||
|
|
||||||
[![GoDoc](https://godoc.org/github.com/cespare/xxhash?status.svg)](https://godoc.org/github.com/cespare/xxhash)
|
|
||||||
[![Build Status](https://travis-ci.org/cespare/xxhash.svg?branch=master)](https://travis-ci.org/cespare/xxhash)
|
|
||||||
|
|
||||||
xxhash is a Go implementation of the 64-bit
|
|
||||||
[xxHash](http://cyan4973.github.io/xxHash/) algorithm, XXH64. This is a
|
|
||||||
high-quality hashing algorithm that is much faster than anything in the Go
|
|
||||||
standard library.
|
|
||||||
|
|
||||||
This package provides a straightforward API:
|
|
||||||
|
|
||||||
```
|
|
||||||
func Sum64(b []byte) uint64
|
|
||||||
func Sum64String(s string) uint64
|
|
||||||
type Digest struct{ ... }
|
|
||||||
func New() *Digest
|
|
||||||
```
|
|
||||||
|
|
||||||
The `Digest` type implements hash.Hash64. Its key methods are:
|
|
||||||
|
|
||||||
```
|
|
||||||
func (*Digest) Write([]byte) (int, error)
|
|
||||||
func (*Digest) WriteString(string) (int, error)
|
|
||||||
func (*Digest) Sum64() uint64
|
|
||||||
```
|
|
||||||
|
|
||||||
This implementation provides a fast pure-Go implementation and an even faster
|
|
||||||
assembly implementation for amd64.
|
|
||||||
|
|
||||||
## Compatibility
|
|
||||||
|
|
||||||
This package is in a module and the latest code is in version 2 of the module.
|
|
||||||
You need a version of Go with at least "minimal module compatibility" to use
|
|
||||||
github.com/cespare/xxhash/v2:
|
|
||||||
|
|
||||||
* 1.9.7+ for Go 1.9
|
|
||||||
* 1.10.3+ for Go 1.10
|
|
||||||
* Go 1.11 or later
|
|
||||||
|
|
||||||
I recommend using the latest release of Go.
|
|
||||||
|
|
||||||
## Benchmarks
|
|
||||||
|
|
||||||
Here are some quick benchmarks comparing the pure-Go and assembly
|
|
||||||
implementations of Sum64.
|
|
||||||
|
|
||||||
| input size | purego | asm |
|
|
||||||
| --- | --- | --- |
|
|
||||||
| 5 B | 979.66 MB/s | 1291.17 MB/s |
|
|
||||||
| 100 B | 7475.26 MB/s | 7973.40 MB/s |
|
|
||||||
| 4 KB | 17573.46 MB/s | 17602.65 MB/s |
|
|
||||||
| 10 MB | 17131.46 MB/s | 17142.16 MB/s |
|
|
||||||
|
|
||||||
These numbers were generated on Ubuntu 18.04 with an Intel i7-8700K CPU using
|
|
||||||
the following commands under Go 1.11.2:
|
|
||||||
|
|
||||||
```
|
|
||||||
$ go test -tags purego -benchtime 10s -bench '/xxhash,direct,bytes'
|
|
||||||
$ go test -benchtime 10s -bench '/xxhash,direct,bytes'
|
|
||||||
```
|
|
||||||
|
|
||||||
## Projects using this package
|
|
||||||
|
|
||||||
- [InfluxDB](https://github.com/influxdata/influxdb)
|
|
||||||
- [Prometheus](https://github.com/prometheus/prometheus)
|
|
||||||
- [FreeCache](https://github.com/coocood/freecache)
|
|
3
vendor/github.com/cespare/xxhash/v2/go.mod
generated
vendored
3
vendor/github.com/cespare/xxhash/v2/go.mod
generated
vendored
|
@ -1,3 +0,0 @@
|
||||||
module github.com/cespare/xxhash/v2
|
|
||||||
|
|
||||||
go 1.11
|
|
0
vendor/github.com/cespare/xxhash/v2/go.sum
generated
vendored
0
vendor/github.com/cespare/xxhash/v2/go.sum
generated
vendored
236
vendor/github.com/cespare/xxhash/v2/xxhash.go
generated
vendored
236
vendor/github.com/cespare/xxhash/v2/xxhash.go
generated
vendored
|
@ -1,236 +0,0 @@
|
||||||
// Package xxhash implements the 64-bit variant of xxHash (XXH64) as described
|
|
||||||
// at http://cyan4973.github.io/xxHash/.
|
|
||||||
package xxhash
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/binary"
|
|
||||||
"errors"
|
|
||||||
"math/bits"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
prime1 uint64 = 11400714785074694791
|
|
||||||
prime2 uint64 = 14029467366897019727
|
|
||||||
prime3 uint64 = 1609587929392839161
|
|
||||||
prime4 uint64 = 9650029242287828579
|
|
||||||
prime5 uint64 = 2870177450012600261
|
|
||||||
)
|
|
||||||
|
|
||||||
// NOTE(caleb): I'm using both consts and vars of the primes. Using consts where
|
|
||||||
// possible in the Go code is worth a small (but measurable) performance boost
|
|
||||||
// by avoiding some MOVQs. Vars are needed for the asm and also are useful for
|
|
||||||
// convenience in the Go code in a few places where we need to intentionally
|
|
||||||
// avoid constant arithmetic (e.g., v1 := prime1 + prime2 fails because the
|
|
||||||
// result overflows a uint64).
|
|
||||||
var (
|
|
||||||
prime1v = prime1
|
|
||||||
prime2v = prime2
|
|
||||||
prime3v = prime3
|
|
||||||
prime4v = prime4
|
|
||||||
prime5v = prime5
|
|
||||||
)
|
|
||||||
|
|
||||||
// Digest implements hash.Hash64.
|
|
||||||
type Digest struct {
|
|
||||||
v1 uint64
|
|
||||||
v2 uint64
|
|
||||||
v3 uint64
|
|
||||||
v4 uint64
|
|
||||||
total uint64
|
|
||||||
mem [32]byte
|
|
||||||
n int // how much of mem is used
|
|
||||||
}
|
|
||||||
|
|
||||||
// New creates a new Digest that computes the 64-bit xxHash algorithm.
|
|
||||||
func New() *Digest {
|
|
||||||
var d Digest
|
|
||||||
d.Reset()
|
|
||||||
return &d
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reset clears the Digest's state so that it can be reused.
|
|
||||||
func (d *Digest) Reset() {
|
|
||||||
d.v1 = prime1v + prime2
|
|
||||||
d.v2 = prime2
|
|
||||||
d.v3 = 0
|
|
||||||
d.v4 = -prime1v
|
|
||||||
d.total = 0
|
|
||||||
d.n = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// Size always returns 8 bytes.
|
|
||||||
func (d *Digest) Size() int { return 8 }
|
|
||||||
|
|
||||||
// BlockSize always returns 32 bytes.
|
|
||||||
func (d *Digest) BlockSize() int { return 32 }
|
|
||||||
|
|
||||||
// Write adds more data to d. It always returns len(b), nil.
|
|
||||||
func (d *Digest) Write(b []byte) (n int, err error) {
|
|
||||||
n = len(b)
|
|
||||||
d.total += uint64(n)
|
|
||||||
|
|
||||||
if d.n+n < 32 {
|
|
||||||
// This new data doesn't even fill the current block.
|
|
||||||
copy(d.mem[d.n:], b)
|
|
||||||
d.n += n
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if d.n > 0 {
|
|
||||||
// Finish off the partial block.
|
|
||||||
copy(d.mem[d.n:], b)
|
|
||||||
d.v1 = round(d.v1, u64(d.mem[0:8]))
|
|
||||||
d.v2 = round(d.v2, u64(d.mem[8:16]))
|
|
||||||
d.v3 = round(d.v3, u64(d.mem[16:24]))
|
|
||||||
d.v4 = round(d.v4, u64(d.mem[24:32]))
|
|
||||||
b = b[32-d.n:]
|
|
||||||
d.n = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(b) >= 32 {
|
|
||||||
// One or more full blocks left.
|
|
||||||
nw := writeBlocks(d, b)
|
|
||||||
b = b[nw:]
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store any remaining partial block.
|
|
||||||
copy(d.mem[:], b)
|
|
||||||
d.n = len(b)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sum appends the current hash to b and returns the resulting slice.
|
|
||||||
func (d *Digest) Sum(b []byte) []byte {
|
|
||||||
s := d.Sum64()
|
|
||||||
return append(
|
|
||||||
b,
|
|
||||||
byte(s>>56),
|
|
||||||
byte(s>>48),
|
|
||||||
byte(s>>40),
|
|
||||||
byte(s>>32),
|
|
||||||
byte(s>>24),
|
|
||||||
byte(s>>16),
|
|
||||||
byte(s>>8),
|
|
||||||
byte(s),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sum64 returns the current hash.
|
|
||||||
func (d *Digest) Sum64() uint64 {
|
|
||||||
var h uint64
|
|
||||||
|
|
||||||
if d.total >= 32 {
|
|
||||||
v1, v2, v3, v4 := d.v1, d.v2, d.v3, d.v4
|
|
||||||
h = rol1(v1) + rol7(v2) + rol12(v3) + rol18(v4)
|
|
||||||
h = mergeRound(h, v1)
|
|
||||||
h = mergeRound(h, v2)
|
|
||||||
h = mergeRound(h, v3)
|
|
||||||
h = mergeRound(h, v4)
|
|
||||||
} else {
|
|
||||||
h = d.v3 + prime5
|
|
||||||
}
|
|
||||||
|
|
||||||
h += d.total
|
|
||||||
|
|
||||||
i, end := 0, d.n
|
|
||||||
for ; i+8 <= end; i += 8 {
|
|
||||||
k1 := round(0, u64(d.mem[i:i+8]))
|
|
||||||
h ^= k1
|
|
||||||
h = rol27(h)*prime1 + prime4
|
|
||||||
}
|
|
||||||
if i+4 <= end {
|
|
||||||
h ^= uint64(u32(d.mem[i:i+4])) * prime1
|
|
||||||
h = rol23(h)*prime2 + prime3
|
|
||||||
i += 4
|
|
||||||
}
|
|
||||||
for i < end {
|
|
||||||
h ^= uint64(d.mem[i]) * prime5
|
|
||||||
h = rol11(h) * prime1
|
|
||||||
i++
|
|
||||||
}
|
|
||||||
|
|
||||||
h ^= h >> 33
|
|
||||||
h *= prime2
|
|
||||||
h ^= h >> 29
|
|
||||||
h *= prime3
|
|
||||||
h ^= h >> 32
|
|
||||||
|
|
||||||
return h
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
magic = "xxh\x06"
|
|
||||||
marshaledSize = len(magic) + 8*5 + 32
|
|
||||||
)
|
|
||||||
|
|
||||||
// MarshalBinary implements the encoding.BinaryMarshaler interface.
|
|
||||||
func (d *Digest) MarshalBinary() ([]byte, error) {
|
|
||||||
b := make([]byte, 0, marshaledSize)
|
|
||||||
b = append(b, magic...)
|
|
||||||
b = appendUint64(b, d.v1)
|
|
||||||
b = appendUint64(b, d.v2)
|
|
||||||
b = appendUint64(b, d.v3)
|
|
||||||
b = appendUint64(b, d.v4)
|
|
||||||
b = appendUint64(b, d.total)
|
|
||||||
b = append(b, d.mem[:d.n]...)
|
|
||||||
b = b[:len(b)+len(d.mem)-d.n]
|
|
||||||
return b, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// UnmarshalBinary implements the encoding.BinaryUnmarshaler interface.
|
|
||||||
func (d *Digest) UnmarshalBinary(b []byte) error {
|
|
||||||
if len(b) < len(magic) || string(b[:len(magic)]) != magic {
|
|
||||||
return errors.New("xxhash: invalid hash state identifier")
|
|
||||||
}
|
|
||||||
if len(b) != marshaledSize {
|
|
||||||
return errors.New("xxhash: invalid hash state size")
|
|
||||||
}
|
|
||||||
b = b[len(magic):]
|
|
||||||
b, d.v1 = consumeUint64(b)
|
|
||||||
b, d.v2 = consumeUint64(b)
|
|
||||||
b, d.v3 = consumeUint64(b)
|
|
||||||
b, d.v4 = consumeUint64(b)
|
|
||||||
b, d.total = consumeUint64(b)
|
|
||||||
copy(d.mem[:], b)
|
|
||||||
b = b[len(d.mem):]
|
|
||||||
d.n = int(d.total % uint64(len(d.mem)))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func appendUint64(b []byte, x uint64) []byte {
|
|
||||||
var a [8]byte
|
|
||||||
binary.LittleEndian.PutUint64(a[:], x)
|
|
||||||
return append(b, a[:]...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func consumeUint64(b []byte) ([]byte, uint64) {
|
|
||||||
x := u64(b)
|
|
||||||
return b[8:], x
|
|
||||||
}
|
|
||||||
|
|
||||||
func u64(b []byte) uint64 { return binary.LittleEndian.Uint64(b) }
|
|
||||||
func u32(b []byte) uint32 { return binary.LittleEndian.Uint32(b) }
|
|
||||||
|
|
||||||
func round(acc, input uint64) uint64 {
|
|
||||||
acc += input * prime2
|
|
||||||
acc = rol31(acc)
|
|
||||||
acc *= prime1
|
|
||||||
return acc
|
|
||||||
}
|
|
||||||
|
|
||||||
func mergeRound(acc, val uint64) uint64 {
|
|
||||||
val = round(0, val)
|
|
||||||
acc ^= val
|
|
||||||
acc = acc*prime1 + prime4
|
|
||||||
return acc
|
|
||||||
}
|
|
||||||
|
|
||||||
func rol1(x uint64) uint64 { return bits.RotateLeft64(x, 1) }
|
|
||||||
func rol7(x uint64) uint64 { return bits.RotateLeft64(x, 7) }
|
|
||||||
func rol11(x uint64) uint64 { return bits.RotateLeft64(x, 11) }
|
|
||||||
func rol12(x uint64) uint64 { return bits.RotateLeft64(x, 12) }
|
|
||||||
func rol18(x uint64) uint64 { return bits.RotateLeft64(x, 18) }
|
|
||||||
func rol23(x uint64) uint64 { return bits.RotateLeft64(x, 23) }
|
|
||||||
func rol27(x uint64) uint64 { return bits.RotateLeft64(x, 27) }
|
|
||||||
func rol31(x uint64) uint64 { return bits.RotateLeft64(x, 31) }
|
|
13
vendor/github.com/cespare/xxhash/v2/xxhash_amd64.go
generated
vendored
13
vendor/github.com/cespare/xxhash/v2/xxhash_amd64.go
generated
vendored
|
@ -1,13 +0,0 @@
|
||||||
// +build !appengine
|
|
||||||
// +build gc
|
|
||||||
// +build !purego
|
|
||||||
|
|
||||||
package xxhash
|
|
||||||
|
|
||||||
// Sum64 computes the 64-bit xxHash digest of b.
|
|
||||||
//
|
|
||||||
//go:noescape
|
|
||||||
func Sum64(b []byte) uint64
|
|
||||||
|
|
||||||
//go:noescape
|
|
||||||
func writeBlocks(d *Digest, b []byte) int
|
|
215
vendor/github.com/cespare/xxhash/v2/xxhash_amd64.s
generated
vendored
215
vendor/github.com/cespare/xxhash/v2/xxhash_amd64.s
generated
vendored
|
@ -1,215 +0,0 @@
|
||||||
// +build !appengine
|
|
||||||
// +build gc
|
|
||||||
// +build !purego
|
|
||||||
|
|
||||||
#include "textflag.h"
|
|
||||||
|
|
||||||
// Register allocation:
|
|
||||||
// AX h
|
|
||||||
// CX pointer to advance through b
|
|
||||||
// DX n
|
|
||||||
// BX loop end
|
|
||||||
// R8 v1, k1
|
|
||||||
// R9 v2
|
|
||||||
// R10 v3
|
|
||||||
// R11 v4
|
|
||||||
// R12 tmp
|
|
||||||
// R13 prime1v
|
|
||||||
// R14 prime2v
|
|
||||||
// R15 prime4v
|
|
||||||
|
|
||||||
// round reads from and advances the buffer pointer in CX.
|
|
||||||
// It assumes that R13 has prime1v and R14 has prime2v.
|
|
||||||
#define round(r) \
|
|
||||||
MOVQ (CX), R12 \
|
|
||||||
ADDQ $8, CX \
|
|
||||||
IMULQ R14, R12 \
|
|
||||||
ADDQ R12, r \
|
|
||||||
ROLQ $31, r \
|
|
||||||
IMULQ R13, r
|
|
||||||
|
|
||||||
// mergeRound applies a merge round on the two registers acc and val.
|
|
||||||
// It assumes that R13 has prime1v, R14 has prime2v, and R15 has prime4v.
|
|
||||||
#define mergeRound(acc, val) \
|
|
||||||
IMULQ R14, val \
|
|
||||||
ROLQ $31, val \
|
|
||||||
IMULQ R13, val \
|
|
||||||
XORQ val, acc \
|
|
||||||
IMULQ R13, acc \
|
|
||||||
ADDQ R15, acc
|
|
||||||
|
|
||||||
// func Sum64(b []byte) uint64
|
|
||||||
TEXT ·Sum64(SB), NOSPLIT, $0-32
|
|
||||||
// Load fixed primes.
|
|
||||||
MOVQ ·prime1v(SB), R13
|
|
||||||
MOVQ ·prime2v(SB), R14
|
|
||||||
MOVQ ·prime4v(SB), R15
|
|
||||||
|
|
||||||
// Load slice.
|
|
||||||
MOVQ b_base+0(FP), CX
|
|
||||||
MOVQ b_len+8(FP), DX
|
|
||||||
LEAQ (CX)(DX*1), BX
|
|
||||||
|
|
||||||
// The first loop limit will be len(b)-32.
|
|
||||||
SUBQ $32, BX
|
|
||||||
|
|
||||||
// Check whether we have at least one block.
|
|
||||||
CMPQ DX, $32
|
|
||||||
JLT noBlocks
|
|
||||||
|
|
||||||
// Set up initial state (v1, v2, v3, v4).
|
|
||||||
MOVQ R13, R8
|
|
||||||
ADDQ R14, R8
|
|
||||||
MOVQ R14, R9
|
|
||||||
XORQ R10, R10
|
|
||||||
XORQ R11, R11
|
|
||||||
SUBQ R13, R11
|
|
||||||
|
|
||||||
// Loop until CX > BX.
|
|
||||||
blockLoop:
|
|
||||||
round(R8)
|
|
||||||
round(R9)
|
|
||||||
round(R10)
|
|
||||||
round(R11)
|
|
||||||
|
|
||||||
CMPQ CX, BX
|
|
||||||
JLE blockLoop
|
|
||||||
|
|
||||||
MOVQ R8, AX
|
|
||||||
ROLQ $1, AX
|
|
||||||
MOVQ R9, R12
|
|
||||||
ROLQ $7, R12
|
|
||||||
ADDQ R12, AX
|
|
||||||
MOVQ R10, R12
|
|
||||||
ROLQ $12, R12
|
|
||||||
ADDQ R12, AX
|
|
||||||
MOVQ R11, R12
|
|
||||||
ROLQ $18, R12
|
|
||||||
ADDQ R12, AX
|
|
||||||
|
|
||||||
mergeRound(AX, R8)
|
|
||||||
mergeRound(AX, R9)
|
|
||||||
mergeRound(AX, R10)
|
|
||||||
mergeRound(AX, R11)
|
|
||||||
|
|
||||||
JMP afterBlocks
|
|
||||||
|
|
||||||
noBlocks:
|
|
||||||
MOVQ ·prime5v(SB), AX
|
|
||||||
|
|
||||||
afterBlocks:
|
|
||||||
ADDQ DX, AX
|
|
||||||
|
|
||||||
// Right now BX has len(b)-32, and we want to loop until CX > len(b)-8.
|
|
||||||
ADDQ $24, BX
|
|
||||||
|
|
||||||
CMPQ CX, BX
|
|
||||||
JG fourByte
|
|
||||||
|
|
||||||
wordLoop:
|
|
||||||
// Calculate k1.
|
|
||||||
MOVQ (CX), R8
|
|
||||||
ADDQ $8, CX
|
|
||||||
IMULQ R14, R8
|
|
||||||
ROLQ $31, R8
|
|
||||||
IMULQ R13, R8
|
|
||||||
|
|
||||||
XORQ R8, AX
|
|
||||||
ROLQ $27, AX
|
|
||||||
IMULQ R13, AX
|
|
||||||
ADDQ R15, AX
|
|
||||||
|
|
||||||
CMPQ CX, BX
|
|
||||||
JLE wordLoop
|
|
||||||
|
|
||||||
fourByte:
|
|
||||||
ADDQ $4, BX
|
|
||||||
CMPQ CX, BX
|
|
||||||
JG singles
|
|
||||||
|
|
||||||
MOVL (CX), R8
|
|
||||||
ADDQ $4, CX
|
|
||||||
IMULQ R13, R8
|
|
||||||
XORQ R8, AX
|
|
||||||
|
|
||||||
ROLQ $23, AX
|
|
||||||
IMULQ R14, AX
|
|
||||||
ADDQ ·prime3v(SB), AX
|
|
||||||
|
|
||||||
singles:
|
|
||||||
ADDQ $4, BX
|
|
||||||
CMPQ CX, BX
|
|
||||||
JGE finalize
|
|
||||||
|
|
||||||
singlesLoop:
|
|
||||||
MOVBQZX (CX), R12
|
|
||||||
ADDQ $1, CX
|
|
||||||
IMULQ ·prime5v(SB), R12
|
|
||||||
XORQ R12, AX
|
|
||||||
|
|
||||||
ROLQ $11, AX
|
|
||||||
IMULQ R13, AX
|
|
||||||
|
|
||||||
CMPQ CX, BX
|
|
||||||
JL singlesLoop
|
|
||||||
|
|
||||||
finalize:
|
|
||||||
MOVQ AX, R12
|
|
||||||
SHRQ $33, R12
|
|
||||||
XORQ R12, AX
|
|
||||||
IMULQ R14, AX
|
|
||||||
MOVQ AX, R12
|
|
||||||
SHRQ $29, R12
|
|
||||||
XORQ R12, AX
|
|
||||||
IMULQ ·prime3v(SB), AX
|
|
||||||
MOVQ AX, R12
|
|
||||||
SHRQ $32, R12
|
|
||||||
XORQ R12, AX
|
|
||||||
|
|
||||||
MOVQ AX, ret+24(FP)
|
|
||||||
RET
|
|
||||||
|
|
||||||
// writeBlocks uses the same registers as above except that it uses AX to store
|
|
||||||
// the d pointer.
|
|
||||||
|
|
||||||
// func writeBlocks(d *Digest, b []byte) int
|
|
||||||
TEXT ·writeBlocks(SB), NOSPLIT, $0-40
|
|
||||||
// Load fixed primes needed for round.
|
|
||||||
MOVQ ·prime1v(SB), R13
|
|
||||||
MOVQ ·prime2v(SB), R14
|
|
||||||
|
|
||||||
// Load slice.
|
|
||||||
MOVQ b_base+8(FP), CX
|
|
||||||
MOVQ b_len+16(FP), DX
|
|
||||||
LEAQ (CX)(DX*1), BX
|
|
||||||
SUBQ $32, BX
|
|
||||||
|
|
||||||
// Load vN from d.
|
|
||||||
MOVQ d+0(FP), AX
|
|
||||||
MOVQ 0(AX), R8 // v1
|
|
||||||
MOVQ 8(AX), R9 // v2
|
|
||||||
MOVQ 16(AX), R10 // v3
|
|
||||||
MOVQ 24(AX), R11 // v4
|
|
||||||
|
|
||||||
// We don't need to check the loop condition here; this function is
|
|
||||||
// always called with at least one block of data to process.
|
|
||||||
blockLoop:
|
|
||||||
round(R8)
|
|
||||||
round(R9)
|
|
||||||
round(R10)
|
|
||||||
round(R11)
|
|
||||||
|
|
||||||
CMPQ CX, BX
|
|
||||||
JLE blockLoop
|
|
||||||
|
|
||||||
// Copy vN back to d.
|
|
||||||
MOVQ R8, 0(AX)
|
|
||||||
MOVQ R9, 8(AX)
|
|
||||||
MOVQ R10, 16(AX)
|
|
||||||
MOVQ R11, 24(AX)
|
|
||||||
|
|
||||||
// The number of bytes written is CX minus the old base pointer.
|
|
||||||
SUBQ b_base+8(FP), CX
|
|
||||||
MOVQ CX, ret+32(FP)
|
|
||||||
|
|
||||||
RET
|
|
76
vendor/github.com/cespare/xxhash/v2/xxhash_other.go
generated
vendored
76
vendor/github.com/cespare/xxhash/v2/xxhash_other.go
generated
vendored
|
@ -1,76 +0,0 @@
|
||||||
// +build !amd64 appengine !gc purego
|
|
||||||
|
|
||||||
package xxhash
|
|
||||||
|
|
||||||
// Sum64 computes the 64-bit xxHash digest of b.
|
|
||||||
func Sum64(b []byte) uint64 {
|
|
||||||
// A simpler version would be
|
|
||||||
// d := New()
|
|
||||||
// d.Write(b)
|
|
||||||
// return d.Sum64()
|
|
||||||
// but this is faster, particularly for small inputs.
|
|
||||||
|
|
||||||
n := len(b)
|
|
||||||
var h uint64
|
|
||||||
|
|
||||||
if n >= 32 {
|
|
||||||
v1 := prime1v + prime2
|
|
||||||
v2 := prime2
|
|
||||||
v3 := uint64(0)
|
|
||||||
v4 := -prime1v
|
|
||||||
for len(b) >= 32 {
|
|
||||||
v1 = round(v1, u64(b[0:8:len(b)]))
|
|
||||||
v2 = round(v2, u64(b[8:16:len(b)]))
|
|
||||||
v3 = round(v3, u64(b[16:24:len(b)]))
|
|
||||||
v4 = round(v4, u64(b[24:32:len(b)]))
|
|
||||||
b = b[32:len(b):len(b)]
|
|
||||||
}
|
|
||||||
h = rol1(v1) + rol7(v2) + rol12(v3) + rol18(v4)
|
|
||||||
h = mergeRound(h, v1)
|
|
||||||
h = mergeRound(h, v2)
|
|
||||||
h = mergeRound(h, v3)
|
|
||||||
h = mergeRound(h, v4)
|
|
||||||
} else {
|
|
||||||
h = prime5
|
|
||||||
}
|
|
||||||
|
|
||||||
h += uint64(n)
|
|
||||||
|
|
||||||
i, end := 0, len(b)
|
|
||||||
for ; i+8 <= end; i += 8 {
|
|
||||||
k1 := round(0, u64(b[i:i+8:len(b)]))
|
|
||||||
h ^= k1
|
|
||||||
h = rol27(h)*prime1 + prime4
|
|
||||||
}
|
|
||||||
if i+4 <= end {
|
|
||||||
h ^= uint64(u32(b[i:i+4:len(b)])) * prime1
|
|
||||||
h = rol23(h)*prime2 + prime3
|
|
||||||
i += 4
|
|
||||||
}
|
|
||||||
for ; i < end; i++ {
|
|
||||||
h ^= uint64(b[i]) * prime5
|
|
||||||
h = rol11(h) * prime1
|
|
||||||
}
|
|
||||||
|
|
||||||
h ^= h >> 33
|
|
||||||
h *= prime2
|
|
||||||
h ^= h >> 29
|
|
||||||
h *= prime3
|
|
||||||
h ^= h >> 32
|
|
||||||
|
|
||||||
return h
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeBlocks(d *Digest, b []byte) int {
|
|
||||||
v1, v2, v3, v4 := d.v1, d.v2, d.v3, d.v4
|
|
||||||
n := len(b)
|
|
||||||
for len(b) >= 32 {
|
|
||||||
v1 = round(v1, u64(b[0:8:len(b)]))
|
|
||||||
v2 = round(v2, u64(b[8:16:len(b)]))
|
|
||||||
v3 = round(v3, u64(b[16:24:len(b)]))
|
|
||||||
v4 = round(v4, u64(b[24:32:len(b)]))
|
|
||||||
b = b[32:len(b):len(b)]
|
|
||||||
}
|
|
||||||
d.v1, d.v2, d.v3, d.v4 = v1, v2, v3, v4
|
|
||||||
return n - len(b)
|
|
||||||
}
|
|
15
vendor/github.com/cespare/xxhash/v2/xxhash_safe.go
generated
vendored
15
vendor/github.com/cespare/xxhash/v2/xxhash_safe.go
generated
vendored
|
@ -1,15 +0,0 @@
|
||||||
// +build appengine
|
|
||||||
|
|
||||||
// This file contains the safe implementations of otherwise unsafe-using code.
|
|
||||||
|
|
||||||
package xxhash
|
|
||||||
|
|
||||||
// Sum64String computes the 64-bit xxHash digest of s.
|
|
||||||
func Sum64String(s string) uint64 {
|
|
||||||
return Sum64([]byte(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
// WriteString adds more data to d. It always returns len(s), nil.
|
|
||||||
func (d *Digest) WriteString(s string) (n int, err error) {
|
|
||||||
return d.Write([]byte(s))
|
|
||||||
}
|
|
46
vendor/github.com/cespare/xxhash/v2/xxhash_unsafe.go
generated
vendored
46
vendor/github.com/cespare/xxhash/v2/xxhash_unsafe.go
generated
vendored
|
@ -1,46 +0,0 @@
|
||||||
// +build !appengine
|
|
||||||
|
|
||||||
// This file encapsulates usage of unsafe.
|
|
||||||
// xxhash_safe.go contains the safe implementations.
|
|
||||||
|
|
||||||
package xxhash
|
|
||||||
|
|
||||||
import (
|
|
||||||
"reflect"
|
|
||||||
"unsafe"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Notes:
|
|
||||||
//
|
|
||||||
// See https://groups.google.com/d/msg/golang-nuts/dcjzJy-bSpw/tcZYBzQqAQAJ
|
|
||||||
// for some discussion about these unsafe conversions.
|
|
||||||
//
|
|
||||||
// In the future it's possible that compiler optimizations will make these
|
|
||||||
// unsafe operations unnecessary: https://golang.org/issue/2205.
|
|
||||||
//
|
|
||||||
// Both of these wrapper functions still incur function call overhead since they
|
|
||||||
// will not be inlined. We could write Go/asm copies of Sum64 and Digest.Write
|
|
||||||
// for strings to squeeze out a bit more speed. Mid-stack inlining should
|
|
||||||
// eventually fix this.
|
|
||||||
|
|
||||||
// Sum64String computes the 64-bit xxHash digest of s.
|
|
||||||
// It may be faster than Sum64([]byte(s)) by avoiding a copy.
|
|
||||||
func Sum64String(s string) uint64 {
|
|
||||||
var b []byte
|
|
||||||
bh := (*reflect.SliceHeader)(unsafe.Pointer(&b))
|
|
||||||
bh.Data = (*reflect.StringHeader)(unsafe.Pointer(&s)).Data
|
|
||||||
bh.Len = len(s)
|
|
||||||
bh.Cap = len(s)
|
|
||||||
return Sum64(b)
|
|
||||||
}
|
|
||||||
|
|
||||||
// WriteString adds more data to d. It always returns len(s), nil.
|
|
||||||
// It may be faster than Write([]byte(s)) by avoiding a copy.
|
|
||||||
func (d *Digest) WriteString(s string) (n int, err error) {
|
|
||||||
var b []byte
|
|
||||||
bh := (*reflect.SliceHeader)(unsafe.Pointer(&b))
|
|
||||||
bh.Data = (*reflect.StringHeader)(unsafe.Pointer(&s)).Data
|
|
||||||
bh.Len = len(s)
|
|
||||||
bh.Cap = len(s)
|
|
||||||
return d.Write(b)
|
|
||||||
}
|
|
191
vendor/github.com/coreos/go-systemd/LICENSE
generated
vendored
191
vendor/github.com/coreos/go-systemd/LICENSE
generated
vendored
|
@ -1,191 +0,0 @@
|
||||||
Apache License
|
|
||||||
Version 2.0, January 2004
|
|
||||||
http://www.apache.org/licenses/
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
||||||
|
|
||||||
1. Definitions.
|
|
||||||
|
|
||||||
"License" shall mean the terms and conditions for use, reproduction, and
|
|
||||||
distribution as defined by Sections 1 through 9 of this document.
|
|
||||||
|
|
||||||
"Licensor" shall mean the copyright owner or entity authorized by the copyright
|
|
||||||
owner that is granting the License.
|
|
||||||
|
|
||||||
"Legal Entity" shall mean the union of the acting entity and all other entities
|
|
||||||
that control, are controlled by, or are under common control with that entity.
|
|
||||||
For the purposes of this definition, "control" means (i) the power, direct or
|
|
||||||
indirect, to cause the direction or management of such entity, whether by
|
|
||||||
contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
||||||
|
|
||||||
"You" (or "Your") shall mean an individual or Legal Entity exercising
|
|
||||||
permissions granted by this License.
|
|
||||||
|
|
||||||
"Source" form shall mean the preferred form for making modifications, including
|
|
||||||
but not limited to software source code, documentation source, and configuration
|
|
||||||
files.
|
|
||||||
|
|
||||||
"Object" form shall mean any form resulting from mechanical transformation or
|
|
||||||
translation of a Source form, including but not limited to compiled object code,
|
|
||||||
generated documentation, and conversions to other media types.
|
|
||||||
|
|
||||||
"Work" shall mean the work of authorship, whether in Source or Object form, made
|
|
||||||
available under the License, as indicated by a copyright notice that is included
|
|
||||||
in or attached to the work (an example is provided in the Appendix below).
|
|
||||||
|
|
||||||
"Derivative Works" shall mean any work, whether in Source or Object form, that
|
|
||||||
is based on (or derived from) the Work and for which the editorial revisions,
|
|
||||||
annotations, elaborations, or other modifications represent, as a whole, an
|
|
||||||
original work of authorship. For the purposes of this License, Derivative Works
|
|
||||||
shall not include works that remain separable from, or merely link (or bind by
|
|
||||||
name) to the interfaces of, the Work and Derivative Works thereof.
|
|
||||||
|
|
||||||
"Contribution" shall mean any work of authorship, including the original version
|
|
||||||
of the Work and any modifications or additions to that Work or Derivative Works
|
|
||||||
thereof, that is intentionally submitted to Licensor for inclusion in the Work
|
|
||||||
by the copyright owner or by an individual or Legal Entity authorized to submit
|
|
||||||
on behalf of the copyright owner. For the purposes of this definition,
|
|
||||||
"submitted" means any form of electronic, verbal, or written communication sent
|
|
||||||
to the Licensor or its representatives, including but not limited to
|
|
||||||
communication on electronic mailing lists, source code control systems, and
|
|
||||||
issue tracking systems that are managed by, or on behalf of, the Licensor for
|
|
||||||
the purpose of discussing and improving the Work, but excluding communication
|
|
||||||
that is conspicuously marked or otherwise designated in writing by the copyright
|
|
||||||
owner as "Not a Contribution."
|
|
||||||
|
|
||||||
"Contributor" shall mean Licensor and any individual or Legal Entity on behalf
|
|
||||||
of whom a Contribution has been received by Licensor and subsequently
|
|
||||||
incorporated within the Work.
|
|
||||||
|
|
||||||
2. Grant of Copyright License.
|
|
||||||
|
|
||||||
Subject to the terms and conditions of this License, each Contributor hereby
|
|
||||||
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
|
||||||
irrevocable copyright license to reproduce, prepare Derivative Works of,
|
|
||||||
publicly display, publicly perform, sublicense, and distribute the Work and such
|
|
||||||
Derivative Works in Source or Object form.
|
|
||||||
|
|
||||||
3. Grant of Patent License.
|
|
||||||
|
|
||||||
Subject to the terms and conditions of this License, each Contributor hereby
|
|
||||||
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
|
||||||
irrevocable (except as stated in this section) patent license to make, have
|
|
||||||
made, use, offer to sell, sell, import, and otherwise transfer the Work, where
|
|
||||||
such license applies only to those patent claims licensable by such Contributor
|
|
||||||
that are necessarily infringed by their Contribution(s) alone or by combination
|
|
||||||
of their Contribution(s) with the Work to which such Contribution(s) was
|
|
||||||
submitted. If You institute patent litigation against any entity (including a
|
|
||||||
cross-claim or counterclaim in a lawsuit) alleging that the Work or a
|
|
||||||
Contribution incorporated within the Work constitutes direct or contributory
|
|
||||||
patent infringement, then any patent licenses granted to You under this License
|
|
||||||
for that Work shall terminate as of the date such litigation is filed.
|
|
||||||
|
|
||||||
4. Redistribution.
|
|
||||||
|
|
||||||
You may reproduce and distribute copies of the Work or Derivative Works thereof
|
|
||||||
in any medium, with or without modifications, and in Source or Object form,
|
|
||||||
provided that You meet the following conditions:
|
|
||||||
|
|
||||||
You must give any other recipients of the Work or Derivative Works a copy of
|
|
||||||
this License; and
|
|
||||||
You must cause any modified files to carry prominent notices stating that You
|
|
||||||
changed the files; and
|
|
||||||
You must retain, in the Source form of any Derivative Works that You distribute,
|
|
||||||
all copyright, patent, trademark, and attribution notices from the Source form
|
|
||||||
of the Work, excluding those notices that do not pertain to any part of the
|
|
||||||
Derivative Works; and
|
|
||||||
If the Work includes a "NOTICE" text file as part of its distribution, then any
|
|
||||||
Derivative Works that You distribute must include a readable copy of the
|
|
||||||
attribution notices contained within such NOTICE file, excluding those notices
|
|
||||||
that do not pertain to any part of the Derivative Works, in at least one of the
|
|
||||||
following places: within a NOTICE text file distributed as part of the
|
|
||||||
Derivative Works; within the Source form or documentation, if provided along
|
|
||||||
with the Derivative Works; or, within a display generated by the Derivative
|
|
||||||
Works, if and wherever such third-party notices normally appear. The contents of
|
|
||||||
the NOTICE file are for informational purposes only and do not modify the
|
|
||||||
License. You may add Your own attribution notices within Derivative Works that
|
|
||||||
You distribute, alongside or as an addendum to the NOTICE text from the Work,
|
|
||||||
provided that such additional attribution notices cannot be construed as
|
|
||||||
modifying the License.
|
|
||||||
You may add Your own copyright statement to Your modifications and may provide
|
|
||||||
additional or different license terms and conditions for use, reproduction, or
|
|
||||||
distribution of Your modifications, or for any such Derivative Works as a whole,
|
|
||||||
provided Your use, reproduction, and distribution of the Work otherwise complies
|
|
||||||
with the conditions stated in this License.
|
|
||||||
|
|
||||||
5. Submission of Contributions.
|
|
||||||
|
|
||||||
Unless You explicitly state otherwise, any Contribution intentionally submitted
|
|
||||||
for inclusion in the Work by You to the Licensor shall be under the terms and
|
|
||||||
conditions of this License, without any additional terms or conditions.
|
|
||||||
Notwithstanding the above, nothing herein shall supersede or modify the terms of
|
|
||||||
any separate license agreement you may have executed with Licensor regarding
|
|
||||||
such Contributions.
|
|
||||||
|
|
||||||
6. Trademarks.
|
|
||||||
|
|
||||||
This License does not grant permission to use the trade names, trademarks,
|
|
||||||
service marks, or product names of the Licensor, except as required for
|
|
||||||
reasonable and customary use in describing the origin of the Work and
|
|
||||||
reproducing the content of the NOTICE file.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty.
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, Licensor provides the
|
|
||||||
Work (and each Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
|
|
||||||
including, without limitation, any warranties or conditions of TITLE,
|
|
||||||
NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are
|
|
||||||
solely responsible for determining the appropriateness of using or
|
|
||||||
redistributing the Work and assume any risks associated with Your exercise of
|
|
||||||
permissions under this License.
|
|
||||||
|
|
||||||
8. Limitation of Liability.
|
|
||||||
|
|
||||||
In no event and under no legal theory, whether in tort (including negligence),
|
|
||||||
contract, or otherwise, unless required by applicable law (such as deliberate
|
|
||||||
and grossly negligent acts) or agreed to in writing, shall any Contributor be
|
|
||||||
liable to You for damages, including any direct, indirect, special, incidental,
|
|
||||||
or consequential damages of any character arising as a result of this License or
|
|
||||||
out of the use or inability to use the Work (including but not limited to
|
|
||||||
damages for loss of goodwill, work stoppage, computer failure or malfunction, or
|
|
||||||
any and all other commercial damages or losses), even if such Contributor has
|
|
||||||
been advised of the possibility of such damages.
|
|
||||||
|
|
||||||
9. Accepting Warranty or Additional Liability.
|
|
||||||
|
|
||||||
While redistributing the Work or Derivative Works thereof, You may choose to
|
|
||||||
offer, and charge a fee for, acceptance of support, warranty, indemnity, or
|
|
||||||
other liability obligations and/or rights consistent with this License. However,
|
|
||||||
in accepting such obligations, You may act only on Your own behalf and on Your
|
|
||||||
sole responsibility, not on behalf of any other Contributor, and only if You
|
|
||||||
agree to indemnify, defend, and hold each Contributor harmless for any liability
|
|
||||||
incurred by, or claims asserted against, such Contributor by reason of your
|
|
||||||
accepting any such warranty or additional liability.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following boilerplate
|
|
||||||
notice, with the fields enclosed by brackets "[]" replaced with your own
|
|
||||||
identifying information. (Don't include the brackets!) The text should be
|
|
||||||
enclosed in the appropriate comment syntax for the file format. We also
|
|
||||||
recommend that a file or class name and description of purpose be included on
|
|
||||||
the same "printed page" as the copyright notice for easier identification within
|
|
||||||
third-party archives.
|
|
||||||
|
|
||||||
Copyright [yyyy] [name of copyright owner]
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
5
vendor/github.com/coreos/go-systemd/NOTICE
generated
vendored
5
vendor/github.com/coreos/go-systemd/NOTICE
generated
vendored
|
@ -1,5 +0,0 @@
|
||||||
CoreOS Project
|
|
||||||
Copyright 2018 CoreOS, Inc
|
|
||||||
|
|
||||||
This product includes software developed at CoreOS, Inc.
|
|
||||||
(http://www.coreos.com/).
|
|
240
vendor/github.com/coreos/go-systemd/dbus/dbus.go
generated
vendored
240
vendor/github.com/coreos/go-systemd/dbus/dbus.go
generated
vendored
|
@ -1,240 +0,0 @@
|
||||||
// Copyright 2015 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
// Integration with the systemd D-Bus API. See http://www.freedesktop.org/wiki/Software/systemd/dbus/
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/hex"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
|
|
||||||
"github.com/godbus/dbus"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
alpha = `abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ`
|
|
||||||
num = `0123456789`
|
|
||||||
alphanum = alpha + num
|
|
||||||
signalBuffer = 100
|
|
||||||
)
|
|
||||||
|
|
||||||
// needsEscape checks whether a byte in a potential dbus ObjectPath needs to be escaped
|
|
||||||
func needsEscape(i int, b byte) bool {
|
|
||||||
// Escape everything that is not a-z-A-Z-0-9
|
|
||||||
// Also escape 0-9 if it's the first character
|
|
||||||
return strings.IndexByte(alphanum, b) == -1 ||
|
|
||||||
(i == 0 && strings.IndexByte(num, b) != -1)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PathBusEscape sanitizes a constituent string of a dbus ObjectPath using the
|
|
||||||
// rules that systemd uses for serializing special characters.
|
|
||||||
func PathBusEscape(path string) string {
|
|
||||||
// Special case the empty string
|
|
||||||
if len(path) == 0 {
|
|
||||||
return "_"
|
|
||||||
}
|
|
||||||
n := []byte{}
|
|
||||||
for i := 0; i < len(path); i++ {
|
|
||||||
c := path[i]
|
|
||||||
if needsEscape(i, c) {
|
|
||||||
e := fmt.Sprintf("_%x", c)
|
|
||||||
n = append(n, []byte(e)...)
|
|
||||||
} else {
|
|
||||||
n = append(n, c)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return string(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
// pathBusUnescape is the inverse of PathBusEscape.
|
|
||||||
func pathBusUnescape(path string) string {
|
|
||||||
if path == "_" {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
n := []byte{}
|
|
||||||
for i := 0; i < len(path); i++ {
|
|
||||||
c := path[i]
|
|
||||||
if c == '_' && i+2 < len(path) {
|
|
||||||
res, err := hex.DecodeString(path[i+1 : i+3])
|
|
||||||
if err == nil {
|
|
||||||
n = append(n, res...)
|
|
||||||
}
|
|
||||||
i += 2
|
|
||||||
} else {
|
|
||||||
n = append(n, c)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return string(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Conn is a connection to systemd's dbus endpoint.
|
|
||||||
type Conn struct {
|
|
||||||
// sysconn/sysobj are only used to call dbus methods
|
|
||||||
sysconn *dbus.Conn
|
|
||||||
sysobj dbus.BusObject
|
|
||||||
|
|
||||||
// sigconn/sigobj are only used to receive dbus signals
|
|
||||||
sigconn *dbus.Conn
|
|
||||||
sigobj dbus.BusObject
|
|
||||||
|
|
||||||
jobListener struct {
|
|
||||||
jobs map[dbus.ObjectPath]chan<- string
|
|
||||||
sync.Mutex
|
|
||||||
}
|
|
||||||
subStateSubscriber struct {
|
|
||||||
updateCh chan<- *SubStateUpdate
|
|
||||||
errCh chan<- error
|
|
||||||
sync.Mutex
|
|
||||||
ignore map[dbus.ObjectPath]int64
|
|
||||||
cleanIgnore int64
|
|
||||||
}
|
|
||||||
propertiesSubscriber struct {
|
|
||||||
updateCh chan<- *PropertiesUpdate
|
|
||||||
errCh chan<- error
|
|
||||||
sync.Mutex
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// New establishes a connection to any available bus and authenticates.
|
|
||||||
// Callers should call Close() when done with the connection.
|
|
||||||
func New() (*Conn, error) {
|
|
||||||
conn, err := NewSystemConnection()
|
|
||||||
if err != nil && os.Geteuid() == 0 {
|
|
||||||
return NewSystemdConnection()
|
|
||||||
}
|
|
||||||
return conn, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSystemConnection establishes a connection to the system bus and authenticates.
|
|
||||||
// Callers should call Close() when done with the connection
|
|
||||||
func NewSystemConnection() (*Conn, error) {
|
|
||||||
return NewConnection(func() (*dbus.Conn, error) {
|
|
||||||
return dbusAuthHelloConnection(dbus.SystemBusPrivate)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewUserConnection establishes a connection to the session bus and
|
|
||||||
// authenticates. This can be used to connect to systemd user instances.
|
|
||||||
// Callers should call Close() when done with the connection.
|
|
||||||
func NewUserConnection() (*Conn, error) {
|
|
||||||
return NewConnection(func() (*dbus.Conn, error) {
|
|
||||||
return dbusAuthHelloConnection(dbus.SessionBusPrivate)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSystemdConnection establishes a private, direct connection to systemd.
|
|
||||||
// This can be used for communicating with systemd without a dbus daemon.
|
|
||||||
// Callers should call Close() when done with the connection.
|
|
||||||
func NewSystemdConnection() (*Conn, error) {
|
|
||||||
return NewConnection(func() (*dbus.Conn, error) {
|
|
||||||
// We skip Hello when talking directly to systemd.
|
|
||||||
return dbusAuthConnection(func(opts ...dbus.ConnOption) (*dbus.Conn, error) {
|
|
||||||
return dbus.Dial("unix:path=/run/systemd/private")
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close closes an established connection
|
|
||||||
func (c *Conn) Close() {
|
|
||||||
c.sysconn.Close()
|
|
||||||
c.sigconn.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewConnection establishes a connection to a bus using a caller-supplied function.
|
|
||||||
// This allows connecting to remote buses through a user-supplied mechanism.
|
|
||||||
// The supplied function may be called multiple times, and should return independent connections.
|
|
||||||
// The returned connection must be fully initialised: the org.freedesktop.DBus.Hello call must have succeeded,
|
|
||||||
// and any authentication should be handled by the function.
|
|
||||||
func NewConnection(dialBus func() (*dbus.Conn, error)) (*Conn, error) {
|
|
||||||
sysconn, err := dialBus()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
sigconn, err := dialBus()
|
|
||||||
if err != nil {
|
|
||||||
sysconn.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
c := &Conn{
|
|
||||||
sysconn: sysconn,
|
|
||||||
sysobj: systemdObject(sysconn),
|
|
||||||
sigconn: sigconn,
|
|
||||||
sigobj: systemdObject(sigconn),
|
|
||||||
}
|
|
||||||
|
|
||||||
c.subStateSubscriber.ignore = make(map[dbus.ObjectPath]int64)
|
|
||||||
c.jobListener.jobs = make(map[dbus.ObjectPath]chan<- string)
|
|
||||||
|
|
||||||
// Setup the listeners on jobs so that we can get completions
|
|
||||||
c.sigconn.BusObject().Call("org.freedesktop.DBus.AddMatch", 0,
|
|
||||||
"type='signal', interface='org.freedesktop.systemd1.Manager', member='JobRemoved'")
|
|
||||||
|
|
||||||
c.dispatch()
|
|
||||||
return c, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetManagerProperty returns the value of a property on the org.freedesktop.systemd1.Manager
|
|
||||||
// interface. The value is returned in its string representation, as defined at
|
|
||||||
// https://developer.gnome.org/glib/unstable/gvariant-text.html
|
|
||||||
func (c *Conn) GetManagerProperty(prop string) (string, error) {
|
|
||||||
variant, err := c.sysobj.GetProperty("org.freedesktop.systemd1.Manager." + prop)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return variant.String(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func dbusAuthConnection(createBus func(opts ...dbus.ConnOption) (*dbus.Conn, error)) (*dbus.Conn, error) {
|
|
||||||
conn, err := createBus()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Only use EXTERNAL method, and hardcode the uid (not username)
|
|
||||||
// to avoid a username lookup (which requires a dynamically linked
|
|
||||||
// libc)
|
|
||||||
methods := []dbus.Auth{dbus.AuthExternal(strconv.Itoa(os.Getuid()))}
|
|
||||||
|
|
||||||
err = conn.Auth(methods)
|
|
||||||
if err != nil {
|
|
||||||
conn.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func dbusAuthHelloConnection(createBus func(opts ...dbus.ConnOption) (*dbus.Conn, error)) (*dbus.Conn, error) {
|
|
||||||
conn, err := dbusAuthConnection(createBus)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = conn.Hello(); err != nil {
|
|
||||||
conn.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func systemdObject(conn *dbus.Conn) dbus.BusObject {
|
|
||||||
return conn.Object("org.freedesktop.systemd1", dbus.ObjectPath("/org/freedesktop/systemd1"))
|
|
||||||
}
|
|
600
vendor/github.com/coreos/go-systemd/dbus/methods.go
generated
vendored
600
vendor/github.com/coreos/go-systemd/dbus/methods.go
generated
vendored
|
@ -1,600 +0,0 @@
|
||||||
// Copyright 2015, 2018 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"path"
|
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"github.com/godbus/dbus"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (c *Conn) jobComplete(signal *dbus.Signal) {
|
|
||||||
var id uint32
|
|
||||||
var job dbus.ObjectPath
|
|
||||||
var unit string
|
|
||||||
var result string
|
|
||||||
dbus.Store(signal.Body, &id, &job, &unit, &result)
|
|
||||||
c.jobListener.Lock()
|
|
||||||
out, ok := c.jobListener.jobs[job]
|
|
||||||
if ok {
|
|
||||||
out <- result
|
|
||||||
delete(c.jobListener.jobs, job)
|
|
||||||
}
|
|
||||||
c.jobListener.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) startJob(ch chan<- string, job string, args ...interface{}) (int, error) {
|
|
||||||
if ch != nil {
|
|
||||||
c.jobListener.Lock()
|
|
||||||
defer c.jobListener.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
var p dbus.ObjectPath
|
|
||||||
err := c.sysobj.Call(job, 0, args...).Store(&p)
|
|
||||||
if err != nil {
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if ch != nil {
|
|
||||||
c.jobListener.jobs[p] = ch
|
|
||||||
}
|
|
||||||
|
|
||||||
// ignore error since 0 is fine if conversion fails
|
|
||||||
jobID, _ := strconv.Atoi(path.Base(string(p)))
|
|
||||||
|
|
||||||
return jobID, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// StartUnit enqueues a start job and depending jobs, if any (unless otherwise
|
|
||||||
// specified by the mode string).
|
|
||||||
//
|
|
||||||
// Takes the unit to activate, plus a mode string. The mode needs to be one of
|
|
||||||
// replace, fail, isolate, ignore-dependencies, ignore-requirements. If
|
|
||||||
// "replace" the call will start the unit and its dependencies, possibly
|
|
||||||
// replacing already queued jobs that conflict with this. If "fail" the call
|
|
||||||
// will start the unit and its dependencies, but will fail if this would change
|
|
||||||
// an already queued job. If "isolate" the call will start the unit in question
|
|
||||||
// and terminate all units that aren't dependencies of it. If
|
|
||||||
// "ignore-dependencies" it will start a unit but ignore all its dependencies.
|
|
||||||
// If "ignore-requirements" it will start a unit but only ignore the
|
|
||||||
// requirement dependencies. It is not recommended to make use of the latter
|
|
||||||
// two options.
|
|
||||||
//
|
|
||||||
// If the provided channel is non-nil, a result string will be sent to it upon
|
|
||||||
// job completion: one of done, canceled, timeout, failed, dependency, skipped.
|
|
||||||
// done indicates successful execution of a job. canceled indicates that a job
|
|
||||||
// has been canceled before it finished execution. timeout indicates that the
|
|
||||||
// job timeout was reached. failed indicates that the job failed. dependency
|
|
||||||
// indicates that a job this job has been depending on failed and the job hence
|
|
||||||
// has been removed too. skipped indicates that a job was skipped because it
|
|
||||||
// didn't apply to the units current state.
|
|
||||||
//
|
|
||||||
// If no error occurs, the ID of the underlying systemd job will be returned. There
|
|
||||||
// does exist the possibility for no error to be returned, but for the returned job
|
|
||||||
// ID to be 0. In this case, the actual underlying ID is not 0 and this datapoint
|
|
||||||
// should not be considered authoritative.
|
|
||||||
//
|
|
||||||
// If an error does occur, it will be returned to the user alongside a job ID of 0.
|
|
||||||
func (c *Conn) StartUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.StartUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// StopUnit is similar to StartUnit but stops the specified unit rather
|
|
||||||
// than starting it.
|
|
||||||
func (c *Conn) StopUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.StopUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReloadUnit reloads a unit. Reloading is done only if the unit is already running and fails otherwise.
|
|
||||||
func (c *Conn) ReloadUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.ReloadUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// RestartUnit restarts a service. If a service is restarted that isn't
|
|
||||||
// running it will be started.
|
|
||||||
func (c *Conn) RestartUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.RestartUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// TryRestartUnit is like RestartUnit, except that a service that isn't running
|
|
||||||
// is not affected by the restart.
|
|
||||||
func (c *Conn) TryRestartUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.TryRestartUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReloadOrRestartUnit attempts a reload if the unit supports it and use a restart
|
|
||||||
// otherwise.
|
|
||||||
func (c *Conn) ReloadOrRestartUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.ReloadOrRestartUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReloadOrTryRestartUnit attempts a reload if the unit supports it and use a "Try"
|
|
||||||
// flavored restart otherwise.
|
|
||||||
func (c *Conn) ReloadOrTryRestartUnit(name string, mode string, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.ReloadOrTryRestartUnit", name, mode)
|
|
||||||
}
|
|
||||||
|
|
||||||
// StartTransientUnit() may be used to create and start a transient unit, which
|
|
||||||
// will be released as soon as it is not running or referenced anymore or the
|
|
||||||
// system is rebooted. name is the unit name including suffix, and must be
|
|
||||||
// unique. mode is the same as in StartUnit(), properties contains properties
|
|
||||||
// of the unit.
|
|
||||||
func (c *Conn) StartTransientUnit(name string, mode string, properties []Property, ch chan<- string) (int, error) {
|
|
||||||
return c.startJob(ch, "org.freedesktop.systemd1.Manager.StartTransientUnit", name, mode, properties, make([]PropertyCollection, 0))
|
|
||||||
}
|
|
||||||
|
|
||||||
// KillUnit takes the unit name and a UNIX signal number to send. All of the unit's
|
|
||||||
// processes are killed.
|
|
||||||
func (c *Conn) KillUnit(name string, signal int32) {
|
|
||||||
c.sysobj.Call("org.freedesktop.systemd1.Manager.KillUnit", 0, name, "all", signal).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ResetFailedUnit resets the "failed" state of a specific unit.
|
|
||||||
func (c *Conn) ResetFailedUnit(name string) error {
|
|
||||||
return c.sysobj.Call("org.freedesktop.systemd1.Manager.ResetFailedUnit", 0, name).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SystemState returns the systemd state. Equivalent to `systemctl is-system-running`.
|
|
||||||
func (c *Conn) SystemState() (*Property, error) {
|
|
||||||
var err error
|
|
||||||
var prop dbus.Variant
|
|
||||||
|
|
||||||
obj := c.sysconn.Object("org.freedesktop.systemd1", "/org/freedesktop/systemd1")
|
|
||||||
err = obj.Call("org.freedesktop.DBus.Properties.Get", 0, "org.freedesktop.systemd1.Manager", "SystemState").Store(&prop)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &Property{Name: "SystemState", Value: prop}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// getProperties takes the unit path and returns all of its dbus object properties, for the given dbus interface
|
|
||||||
func (c *Conn) getProperties(path dbus.ObjectPath, dbusInterface string) (map[string]interface{}, error) {
|
|
||||||
var err error
|
|
||||||
var props map[string]dbus.Variant
|
|
||||||
|
|
||||||
if !path.IsValid() {
|
|
||||||
return nil, fmt.Errorf("invalid unit name: %v", path)
|
|
||||||
}
|
|
||||||
|
|
||||||
obj := c.sysconn.Object("org.freedesktop.systemd1", path)
|
|
||||||
err = obj.Call("org.freedesktop.DBus.Properties.GetAll", 0, dbusInterface).Store(&props)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
out := make(map[string]interface{}, len(props))
|
|
||||||
for k, v := range props {
|
|
||||||
out[k] = v.Value()
|
|
||||||
}
|
|
||||||
|
|
||||||
return out, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetUnitProperties takes the (unescaped) unit name and returns all of its dbus object properties.
|
|
||||||
func (c *Conn) GetUnitProperties(unit string) (map[string]interface{}, error) {
|
|
||||||
path := unitPath(unit)
|
|
||||||
return c.getProperties(path, "org.freedesktop.systemd1.Unit")
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetUnitPathProperties takes the (escaped) unit path and returns all of its dbus object properties.
|
|
||||||
func (c *Conn) GetUnitPathProperties(path dbus.ObjectPath) (map[string]interface{}, error) {
|
|
||||||
return c.getProperties(path, "org.freedesktop.systemd1.Unit")
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetAllProperties takes the (unescaped) unit name and returns all of its dbus object properties.
|
|
||||||
func (c *Conn) GetAllProperties(unit string) (map[string]interface{}, error) {
|
|
||||||
path := unitPath(unit)
|
|
||||||
return c.getProperties(path, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) getProperty(unit string, dbusInterface string, propertyName string) (*Property, error) {
|
|
||||||
var err error
|
|
||||||
var prop dbus.Variant
|
|
||||||
|
|
||||||
path := unitPath(unit)
|
|
||||||
if !path.IsValid() {
|
|
||||||
return nil, errors.New("invalid unit name: " + unit)
|
|
||||||
}
|
|
||||||
|
|
||||||
obj := c.sysconn.Object("org.freedesktop.systemd1", path)
|
|
||||||
err = obj.Call("org.freedesktop.DBus.Properties.Get", 0, dbusInterface, propertyName).Store(&prop)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &Property{Name: propertyName, Value: prop}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) GetUnitProperty(unit string, propertyName string) (*Property, error) {
|
|
||||||
return c.getProperty(unit, "org.freedesktop.systemd1.Unit", propertyName)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetServiceProperty returns property for given service name and property name
|
|
||||||
func (c *Conn) GetServiceProperty(service string, propertyName string) (*Property, error) {
|
|
||||||
return c.getProperty(service, "org.freedesktop.systemd1.Service", propertyName)
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetUnitTypeProperties returns the extra properties for a unit, specific to the unit type.
|
|
||||||
// Valid values for unitType: Service, Socket, Target, Device, Mount, Automount, Snapshot, Timer, Swap, Path, Slice, Scope
|
|
||||||
// return "dbus.Error: Unknown interface" if the unitType is not the correct type of the unit
|
|
||||||
func (c *Conn) GetUnitTypeProperties(unit string, unitType string) (map[string]interface{}, error) {
|
|
||||||
path := unitPath(unit)
|
|
||||||
return c.getProperties(path, "org.freedesktop.systemd1."+unitType)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetUnitProperties() may be used to modify certain unit properties at runtime.
|
|
||||||
// Not all properties may be changed at runtime, but many resource management
|
|
||||||
// settings (primarily those in systemd.cgroup(5)) may. The changes are applied
|
|
||||||
// instantly, and stored on disk for future boots, unless runtime is true, in which
|
|
||||||
// case the settings only apply until the next reboot. name is the name of the unit
|
|
||||||
// to modify. properties are the settings to set, encoded as an array of property
|
|
||||||
// name and value pairs.
|
|
||||||
func (c *Conn) SetUnitProperties(name string, runtime bool, properties ...Property) error {
|
|
||||||
return c.sysobj.Call("org.freedesktop.systemd1.Manager.SetUnitProperties", 0, name, runtime, properties).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) GetUnitTypeProperty(unit string, unitType string, propertyName string) (*Property, error) {
|
|
||||||
return c.getProperty(unit, "org.freedesktop.systemd1."+unitType, propertyName)
|
|
||||||
}
|
|
||||||
|
|
||||||
type UnitStatus struct {
|
|
||||||
Name string // The primary unit name as string
|
|
||||||
Description string // The human readable description string
|
|
||||||
LoadState string // The load state (i.e. whether the unit file has been loaded successfully)
|
|
||||||
ActiveState string // The active state (i.e. whether the unit is currently started or not)
|
|
||||||
SubState string // The sub state (a more fine-grained version of the active state that is specific to the unit type, which the active state is not)
|
|
||||||
Followed string // A unit that is being followed in its state by this unit, if there is any, otherwise the empty string.
|
|
||||||
Path dbus.ObjectPath // The unit object path
|
|
||||||
JobId uint32 // If there is a job queued for the job unit the numeric job id, 0 otherwise
|
|
||||||
JobType string // The job type as string
|
|
||||||
JobPath dbus.ObjectPath // The job object path
|
|
||||||
}
|
|
||||||
|
|
||||||
type storeFunc func(retvalues ...interface{}) error
|
|
||||||
|
|
||||||
func (c *Conn) listUnitsInternal(f storeFunc) ([]UnitStatus, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := f(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
status := make([]UnitStatus, len(result))
|
|
||||||
statusInterface := make([]interface{}, len(status))
|
|
||||||
for i := range status {
|
|
||||||
statusInterface[i] = &status[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, statusInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return status, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnits returns an array with all currently loaded units. Note that
|
|
||||||
// units may be known by multiple names at the same time, and hence there might
|
|
||||||
// be more unit names loaded than actual units behind them.
|
|
||||||
// Also note that a unit is only loaded if it is active and/or enabled.
|
|
||||||
// Units that are both disabled and inactive will thus not be returned.
|
|
||||||
func (c *Conn) ListUnits() ([]UnitStatus, error) {
|
|
||||||
return c.listUnitsInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnits", 0).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnitsFiltered returns an array with units filtered by state.
|
|
||||||
// It takes a list of units' statuses to filter.
|
|
||||||
func (c *Conn) ListUnitsFiltered(states []string) ([]UnitStatus, error) {
|
|
||||||
return c.listUnitsInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnitsFiltered", 0, states).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnitsByPatterns returns an array with units.
|
|
||||||
// It takes a list of units' statuses and names to filter.
|
|
||||||
// Note that units may be known by multiple names at the same time,
|
|
||||||
// and hence there might be more unit names loaded than actual units behind them.
|
|
||||||
func (c *Conn) ListUnitsByPatterns(states []string, patterns []string) ([]UnitStatus, error) {
|
|
||||||
return c.listUnitsInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnitsByPatterns", 0, states, patterns).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnitsByNames returns an array with units. It takes a list of units'
|
|
||||||
// names and returns an UnitStatus array. Comparing to ListUnitsByPatterns
|
|
||||||
// method, this method returns statuses even for inactive or non-existing
|
|
||||||
// units. Input array should contain exact unit names, but not patterns.
|
|
||||||
// Note: Requires systemd v230 or higher
|
|
||||||
func (c *Conn) ListUnitsByNames(units []string) ([]UnitStatus, error) {
|
|
||||||
return c.listUnitsInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnitsByNames", 0, units).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
type UnitFile struct {
|
|
||||||
Path string
|
|
||||||
Type string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) listUnitFilesInternal(f storeFunc) ([]UnitFile, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := f(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
files := make([]UnitFile, len(result))
|
|
||||||
fileInterface := make([]interface{}, len(files))
|
|
||||||
for i := range files {
|
|
||||||
fileInterface[i] = &files[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, fileInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return files, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnitFiles returns an array of all available units on disk.
|
|
||||||
func (c *Conn) ListUnitFiles() ([]UnitFile, error) {
|
|
||||||
return c.listUnitFilesInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnitFiles", 0).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUnitFilesByPatterns returns an array of all available units on disk matched the patterns.
|
|
||||||
func (c *Conn) ListUnitFilesByPatterns(states []string, patterns []string) ([]UnitFile, error) {
|
|
||||||
return c.listUnitFilesInternal(c.sysobj.Call("org.freedesktop.systemd1.Manager.ListUnitFilesByPatterns", 0, states, patterns).Store)
|
|
||||||
}
|
|
||||||
|
|
||||||
type LinkUnitFileChange EnableUnitFileChange
|
|
||||||
|
|
||||||
// LinkUnitFiles() links unit files (that are located outside of the
|
|
||||||
// usual unit search paths) into the unit search path.
|
|
||||||
//
|
|
||||||
// It takes a list of absolute paths to unit files to link and two
|
|
||||||
// booleans. The first boolean controls whether the unit shall be
|
|
||||||
// enabled for runtime only (true, /run), or persistently (false,
|
|
||||||
// /etc).
|
|
||||||
// The second controls whether symlinks pointing to other units shall
|
|
||||||
// be replaced if necessary.
|
|
||||||
//
|
|
||||||
// This call returns a list of the changes made. The list consists of
|
|
||||||
// structures with three strings: the type of the change (one of symlink
|
|
||||||
// or unlink), the file name of the symlink and the destination of the
|
|
||||||
// symlink.
|
|
||||||
func (c *Conn) LinkUnitFiles(files []string, runtime bool, force bool) ([]LinkUnitFileChange, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := c.sysobj.Call("org.freedesktop.systemd1.Manager.LinkUnitFiles", 0, files, runtime, force).Store(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
changes := make([]LinkUnitFileChange, len(result))
|
|
||||||
changesInterface := make([]interface{}, len(changes))
|
|
||||||
for i := range changes {
|
|
||||||
changesInterface[i] = &changes[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, changesInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return changes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// EnableUnitFiles() may be used to enable one or more units in the system (by
|
|
||||||
// creating symlinks to them in /etc or /run).
|
|
||||||
//
|
|
||||||
// It takes a list of unit files to enable (either just file names or full
|
|
||||||
// absolute paths if the unit files are residing outside the usual unit
|
|
||||||
// search paths), and two booleans: the first controls whether the unit shall
|
|
||||||
// be enabled for runtime only (true, /run), or persistently (false, /etc).
|
|
||||||
// The second one controls whether symlinks pointing to other units shall
|
|
||||||
// be replaced if necessary.
|
|
||||||
//
|
|
||||||
// This call returns one boolean and an array with the changes made. The
|
|
||||||
// boolean signals whether the unit files contained any enablement
|
|
||||||
// information (i.e. an [Install]) section. The changes list consists of
|
|
||||||
// structures with three strings: the type of the change (one of symlink
|
|
||||||
// or unlink), the file name of the symlink and the destination of the
|
|
||||||
// symlink.
|
|
||||||
func (c *Conn) EnableUnitFiles(files []string, runtime bool, force bool) (bool, []EnableUnitFileChange, error) {
|
|
||||||
var carries_install_info bool
|
|
||||||
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := c.sysobj.Call("org.freedesktop.systemd1.Manager.EnableUnitFiles", 0, files, runtime, force).Store(&carries_install_info, &result)
|
|
||||||
if err != nil {
|
|
||||||
return false, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
changes := make([]EnableUnitFileChange, len(result))
|
|
||||||
changesInterface := make([]interface{}, len(changes))
|
|
||||||
for i := range changes {
|
|
||||||
changesInterface[i] = &changes[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, changesInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return false, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return carries_install_info, changes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type EnableUnitFileChange struct {
|
|
||||||
Type string // Type of the change (one of symlink or unlink)
|
|
||||||
Filename string // File name of the symlink
|
|
||||||
Destination string // Destination of the symlink
|
|
||||||
}
|
|
||||||
|
|
||||||
// DisableUnitFiles() may be used to disable one or more units in the system (by
|
|
||||||
// removing symlinks to them from /etc or /run).
|
|
||||||
//
|
|
||||||
// It takes a list of unit files to disable (either just file names or full
|
|
||||||
// absolute paths if the unit files are residing outside the usual unit
|
|
||||||
// search paths), and one boolean: whether the unit was enabled for runtime
|
|
||||||
// only (true, /run), or persistently (false, /etc).
|
|
||||||
//
|
|
||||||
// This call returns an array with the changes made. The changes list
|
|
||||||
// consists of structures with three strings: the type of the change (one of
|
|
||||||
// symlink or unlink), the file name of the symlink and the destination of the
|
|
||||||
// symlink.
|
|
||||||
func (c *Conn) DisableUnitFiles(files []string, runtime bool) ([]DisableUnitFileChange, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := c.sysobj.Call("org.freedesktop.systemd1.Manager.DisableUnitFiles", 0, files, runtime).Store(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
changes := make([]DisableUnitFileChange, len(result))
|
|
||||||
changesInterface := make([]interface{}, len(changes))
|
|
||||||
for i := range changes {
|
|
||||||
changesInterface[i] = &changes[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, changesInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return changes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type DisableUnitFileChange struct {
|
|
||||||
Type string // Type of the change (one of symlink or unlink)
|
|
||||||
Filename string // File name of the symlink
|
|
||||||
Destination string // Destination of the symlink
|
|
||||||
}
|
|
||||||
|
|
||||||
// MaskUnitFiles masks one or more units in the system
|
|
||||||
//
|
|
||||||
// It takes three arguments:
|
|
||||||
// * list of units to mask (either just file names or full
|
|
||||||
// absolute paths if the unit files are residing outside
|
|
||||||
// the usual unit search paths)
|
|
||||||
// * runtime to specify whether the unit was enabled for runtime
|
|
||||||
// only (true, /run/systemd/..), or persistently (false, /etc/systemd/..)
|
|
||||||
// * force flag
|
|
||||||
func (c *Conn) MaskUnitFiles(files []string, runtime bool, force bool) ([]MaskUnitFileChange, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := c.sysobj.Call("org.freedesktop.systemd1.Manager.MaskUnitFiles", 0, files, runtime, force).Store(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
changes := make([]MaskUnitFileChange, len(result))
|
|
||||||
changesInterface := make([]interface{}, len(changes))
|
|
||||||
for i := range changes {
|
|
||||||
changesInterface[i] = &changes[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, changesInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return changes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type MaskUnitFileChange struct {
|
|
||||||
Type string // Type of the change (one of symlink or unlink)
|
|
||||||
Filename string // File name of the symlink
|
|
||||||
Destination string // Destination of the symlink
|
|
||||||
}
|
|
||||||
|
|
||||||
// UnmaskUnitFiles unmasks one or more units in the system
|
|
||||||
//
|
|
||||||
// It takes two arguments:
|
|
||||||
// * list of unit files to mask (either just file names or full
|
|
||||||
// absolute paths if the unit files are residing outside
|
|
||||||
// the usual unit search paths)
|
|
||||||
// * runtime to specify whether the unit was enabled for runtime
|
|
||||||
// only (true, /run/systemd/..), or persistently (false, /etc/systemd/..)
|
|
||||||
func (c *Conn) UnmaskUnitFiles(files []string, runtime bool) ([]UnmaskUnitFileChange, error) {
|
|
||||||
result := make([][]interface{}, 0)
|
|
||||||
err := c.sysobj.Call("org.freedesktop.systemd1.Manager.UnmaskUnitFiles", 0, files, runtime).Store(&result)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
resultInterface := make([]interface{}, len(result))
|
|
||||||
for i := range result {
|
|
||||||
resultInterface[i] = result[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
changes := make([]UnmaskUnitFileChange, len(result))
|
|
||||||
changesInterface := make([]interface{}, len(changes))
|
|
||||||
for i := range changes {
|
|
||||||
changesInterface[i] = &changes[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dbus.Store(resultInterface, changesInterface...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return changes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type UnmaskUnitFileChange struct {
|
|
||||||
Type string // Type of the change (one of symlink or unlink)
|
|
||||||
Filename string // File name of the symlink
|
|
||||||
Destination string // Destination of the symlink
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reload instructs systemd to scan for and reload unit files. This is
|
|
||||||
// equivalent to a 'systemctl daemon-reload'.
|
|
||||||
func (c *Conn) Reload() error {
|
|
||||||
return c.sysobj.Call("org.freedesktop.systemd1.Manager.Reload", 0).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
func unitPath(name string) dbus.ObjectPath {
|
|
||||||
return dbus.ObjectPath("/org/freedesktop/systemd1/unit/" + PathBusEscape(name))
|
|
||||||
}
|
|
||||||
|
|
||||||
// unitName returns the unescaped base element of the supplied escaped path
|
|
||||||
func unitName(dpath dbus.ObjectPath) string {
|
|
||||||
return pathBusUnescape(path.Base(string(dpath)))
|
|
||||||
}
|
|
237
vendor/github.com/coreos/go-systemd/dbus/properties.go
generated
vendored
237
vendor/github.com/coreos/go-systemd/dbus/properties.go
generated
vendored
|
@ -1,237 +0,0 @@
|
||||||
// Copyright 2015 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/godbus/dbus"
|
|
||||||
)
|
|
||||||
|
|
||||||
// From the systemd docs:
|
|
||||||
//
|
|
||||||
// The properties array of StartTransientUnit() may take many of the settings
|
|
||||||
// that may also be configured in unit files. Not all parameters are currently
|
|
||||||
// accepted though, but we plan to cover more properties with future release.
|
|
||||||
// Currently you may set the Description, Slice and all dependency types of
|
|
||||||
// units, as well as RemainAfterExit, ExecStart for service units,
|
|
||||||
// TimeoutStopUSec and PIDs for scope units, and CPUAccounting, CPUShares,
|
|
||||||
// BlockIOAccounting, BlockIOWeight, BlockIOReadBandwidth,
|
|
||||||
// BlockIOWriteBandwidth, BlockIODeviceWeight, MemoryAccounting, MemoryLimit,
|
|
||||||
// DevicePolicy, DeviceAllow for services/scopes/slices. These fields map
|
|
||||||
// directly to their counterparts in unit files and as normal D-Bus object
|
|
||||||
// properties. The exception here is the PIDs field of scope units which is
|
|
||||||
// used for construction of the scope only and specifies the initial PIDs to
|
|
||||||
// add to the scope object.
|
|
||||||
|
|
||||||
type Property struct {
|
|
||||||
Name string
|
|
||||||
Value dbus.Variant
|
|
||||||
}
|
|
||||||
|
|
||||||
type PropertyCollection struct {
|
|
||||||
Name string
|
|
||||||
Properties []Property
|
|
||||||
}
|
|
||||||
|
|
||||||
type execStart struct {
|
|
||||||
Path string // the binary path to execute
|
|
||||||
Args []string // an array with all arguments to pass to the executed command, starting with argument 0
|
|
||||||
UncleanIsFailure bool // a boolean whether it should be considered a failure if the process exits uncleanly
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropExecStart sets the ExecStart service property. The first argument is a
|
|
||||||
// slice with the binary path to execute followed by the arguments to pass to
|
|
||||||
// the executed command. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.service.html#ExecStart=
|
|
||||||
func PropExecStart(command []string, uncleanIsFailure bool) Property {
|
|
||||||
execStarts := []execStart{
|
|
||||||
execStart{
|
|
||||||
Path: command[0],
|
|
||||||
Args: command,
|
|
||||||
UncleanIsFailure: uncleanIsFailure,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
return Property{
|
|
||||||
Name: "ExecStart",
|
|
||||||
Value: dbus.MakeVariant(execStarts),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRemainAfterExit sets the RemainAfterExit service property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.service.html#RemainAfterExit=
|
|
||||||
func PropRemainAfterExit(b bool) Property {
|
|
||||||
return Property{
|
|
||||||
Name: "RemainAfterExit",
|
|
||||||
Value: dbus.MakeVariant(b),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropType sets the Type service property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.service.html#Type=
|
|
||||||
func PropType(t string) Property {
|
|
||||||
return Property{
|
|
||||||
Name: "Type",
|
|
||||||
Value: dbus.MakeVariant(t),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropDescription sets the Description unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit#Description=
|
|
||||||
func PropDescription(desc string) Property {
|
|
||||||
return Property{
|
|
||||||
Name: "Description",
|
|
||||||
Value: dbus.MakeVariant(desc),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func propDependency(name string, units []string) Property {
|
|
||||||
return Property{
|
|
||||||
Name: name,
|
|
||||||
Value: dbus.MakeVariant(units),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequires sets the Requires unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Requires=
|
|
||||||
func PropRequires(units ...string) Property {
|
|
||||||
return propDependency("Requires", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequiresOverridable sets the RequiresOverridable unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#RequiresOverridable=
|
|
||||||
func PropRequiresOverridable(units ...string) Property {
|
|
||||||
return propDependency("RequiresOverridable", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequisite sets the Requisite unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Requisite=
|
|
||||||
func PropRequisite(units ...string) Property {
|
|
||||||
return propDependency("Requisite", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequisiteOverridable sets the RequisiteOverridable unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#RequisiteOverridable=
|
|
||||||
func PropRequisiteOverridable(units ...string) Property {
|
|
||||||
return propDependency("RequisiteOverridable", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropWants sets the Wants unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Wants=
|
|
||||||
func PropWants(units ...string) Property {
|
|
||||||
return propDependency("Wants", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropBindsTo sets the BindsTo unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#BindsTo=
|
|
||||||
func PropBindsTo(units ...string) Property {
|
|
||||||
return propDependency("BindsTo", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequiredBy sets the RequiredBy unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#RequiredBy=
|
|
||||||
func PropRequiredBy(units ...string) Property {
|
|
||||||
return propDependency("RequiredBy", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequiredByOverridable sets the RequiredByOverridable unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#RequiredByOverridable=
|
|
||||||
func PropRequiredByOverridable(units ...string) Property {
|
|
||||||
return propDependency("RequiredByOverridable", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropWantedBy sets the WantedBy unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#WantedBy=
|
|
||||||
func PropWantedBy(units ...string) Property {
|
|
||||||
return propDependency("WantedBy", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropBoundBy sets the BoundBy unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/main/systemd.unit.html#BoundBy=
|
|
||||||
func PropBoundBy(units ...string) Property {
|
|
||||||
return propDependency("BoundBy", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropConflicts sets the Conflicts unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Conflicts=
|
|
||||||
func PropConflicts(units ...string) Property {
|
|
||||||
return propDependency("Conflicts", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropConflictedBy sets the ConflictedBy unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#ConflictedBy=
|
|
||||||
func PropConflictedBy(units ...string) Property {
|
|
||||||
return propDependency("ConflictedBy", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropBefore sets the Before unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Before=
|
|
||||||
func PropBefore(units ...string) Property {
|
|
||||||
return propDependency("Before", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropAfter sets the After unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#After=
|
|
||||||
func PropAfter(units ...string) Property {
|
|
||||||
return propDependency("After", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropOnFailure sets the OnFailure unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#OnFailure=
|
|
||||||
func PropOnFailure(units ...string) Property {
|
|
||||||
return propDependency("OnFailure", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropTriggers sets the Triggers unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#Triggers=
|
|
||||||
func PropTriggers(units ...string) Property {
|
|
||||||
return propDependency("Triggers", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropTriggeredBy sets the TriggeredBy unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#TriggeredBy=
|
|
||||||
func PropTriggeredBy(units ...string) Property {
|
|
||||||
return propDependency("TriggeredBy", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropPropagatesReloadTo sets the PropagatesReloadTo unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#PropagatesReloadTo=
|
|
||||||
func PropPropagatesReloadTo(units ...string) Property {
|
|
||||||
return propDependency("PropagatesReloadTo", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropRequiresMountsFor sets the RequiresMountsFor unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.unit.html#RequiresMountsFor=
|
|
||||||
func PropRequiresMountsFor(units ...string) Property {
|
|
||||||
return propDependency("RequiresMountsFor", units)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropSlice sets the Slice unit property. See
|
|
||||||
// http://www.freedesktop.org/software/systemd/man/systemd.resource-control.html#Slice=
|
|
||||||
func PropSlice(slice string) Property {
|
|
||||||
return Property{
|
|
||||||
Name: "Slice",
|
|
||||||
Value: dbus.MakeVariant(slice),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropPids sets the PIDs field of scope units used in the initial construction
|
|
||||||
// of the scope only and specifies the initial PIDs to add to the scope object.
|
|
||||||
// See https://www.freedesktop.org/wiki/Software/systemd/ControlGroupInterface/#properties
|
|
||||||
func PropPids(pids ...uint32) Property {
|
|
||||||
return Property{
|
|
||||||
Name: "PIDs",
|
|
||||||
Value: dbus.MakeVariant(pids),
|
|
||||||
}
|
|
||||||
}
|
|
47
vendor/github.com/coreos/go-systemd/dbus/set.go
generated
vendored
47
vendor/github.com/coreos/go-systemd/dbus/set.go
generated
vendored
|
@ -1,47 +0,0 @@
|
||||||
// Copyright 2015 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
type set struct {
|
|
||||||
data map[string]bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *set) Add(value string) {
|
|
||||||
s.data[value] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *set) Remove(value string) {
|
|
||||||
delete(s.data, value)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *set) Contains(value string) (exists bool) {
|
|
||||||
_, exists = s.data[value]
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *set) Length() int {
|
|
||||||
return len(s.data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *set) Values() (values []string) {
|
|
||||||
for val := range s.data {
|
|
||||||
values = append(values, val)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func newSet() *set {
|
|
||||||
return &set{make(map[string]bool)}
|
|
||||||
}
|
|
333
vendor/github.com/coreos/go-systemd/dbus/subscription.go
generated
vendored
333
vendor/github.com/coreos/go-systemd/dbus/subscription.go
generated
vendored
|
@ -1,333 +0,0 @@
|
||||||
// Copyright 2015 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"log"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/godbus/dbus"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
cleanIgnoreInterval = int64(10 * time.Second)
|
|
||||||
ignoreInterval = int64(30 * time.Millisecond)
|
|
||||||
)
|
|
||||||
|
|
||||||
// Subscribe sets up this connection to subscribe to all systemd dbus events.
|
|
||||||
// This is required before calling SubscribeUnits. When the connection closes
|
|
||||||
// systemd will automatically stop sending signals so there is no need to
|
|
||||||
// explicitly call Unsubscribe().
|
|
||||||
func (c *Conn) Subscribe() error {
|
|
||||||
c.sigconn.BusObject().Call("org.freedesktop.DBus.AddMatch", 0,
|
|
||||||
"type='signal',interface='org.freedesktop.systemd1.Manager',member='UnitNew'")
|
|
||||||
c.sigconn.BusObject().Call("org.freedesktop.DBus.AddMatch", 0,
|
|
||||||
"type='signal',interface='org.freedesktop.DBus.Properties',member='PropertiesChanged'")
|
|
||||||
|
|
||||||
return c.sigobj.Call("org.freedesktop.systemd1.Manager.Subscribe", 0).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unsubscribe this connection from systemd dbus events.
|
|
||||||
func (c *Conn) Unsubscribe() error {
|
|
||||||
return c.sigobj.Call("org.freedesktop.systemd1.Manager.Unsubscribe", 0).Store()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) dispatch() {
|
|
||||||
ch := make(chan *dbus.Signal, signalBuffer)
|
|
||||||
|
|
||||||
c.sigconn.Signal(ch)
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
signal, ok := <-ch
|
|
||||||
if !ok {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if signal.Name == "org.freedesktop.systemd1.Manager.JobRemoved" {
|
|
||||||
c.jobComplete(signal)
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.subStateSubscriber.updateCh == nil &&
|
|
||||||
c.propertiesSubscriber.updateCh == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
var unitPath dbus.ObjectPath
|
|
||||||
switch signal.Name {
|
|
||||||
case "org.freedesktop.systemd1.Manager.JobRemoved":
|
|
||||||
unitName := signal.Body[2].(string)
|
|
||||||
c.sysobj.Call("org.freedesktop.systemd1.Manager.GetUnit", 0, unitName).Store(&unitPath)
|
|
||||||
case "org.freedesktop.systemd1.Manager.UnitNew":
|
|
||||||
unitPath = signal.Body[1].(dbus.ObjectPath)
|
|
||||||
case "org.freedesktop.DBus.Properties.PropertiesChanged":
|
|
||||||
if signal.Body[0].(string) == "org.freedesktop.systemd1.Unit" {
|
|
||||||
unitPath = signal.Path
|
|
||||||
|
|
||||||
if len(signal.Body) >= 2 {
|
|
||||||
if changed, ok := signal.Body[1].(map[string]dbus.Variant); ok {
|
|
||||||
c.sendPropertiesUpdate(unitPath, changed)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if unitPath == dbus.ObjectPath("") {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
c.sendSubStateUpdate(unitPath)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SubscribeUnits returns two unbuffered channels which will receive all changed units every
|
|
||||||
// interval. Deleted units are sent as nil.
|
|
||||||
func (c *Conn) SubscribeUnits(interval time.Duration) (<-chan map[string]*UnitStatus, <-chan error) {
|
|
||||||
return c.SubscribeUnitsCustom(interval, 0, func(u1, u2 *UnitStatus) bool { return *u1 != *u2 }, nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SubscribeUnitsCustom is like SubscribeUnits but lets you specify the buffer
|
|
||||||
// size of the channels, the comparison function for detecting changes and a filter
|
|
||||||
// function for cutting down on the noise that your channel receives.
|
|
||||||
func (c *Conn) SubscribeUnitsCustom(interval time.Duration, buffer int, isChanged func(*UnitStatus, *UnitStatus) bool, filterUnit func(string) bool) (<-chan map[string]*UnitStatus, <-chan error) {
|
|
||||||
old := make(map[string]*UnitStatus)
|
|
||||||
statusChan := make(chan map[string]*UnitStatus, buffer)
|
|
||||||
errChan := make(chan error, buffer)
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
timerChan := time.After(interval)
|
|
||||||
|
|
||||||
units, err := c.ListUnits()
|
|
||||||
if err == nil {
|
|
||||||
cur := make(map[string]*UnitStatus)
|
|
||||||
for i := range units {
|
|
||||||
if filterUnit != nil && filterUnit(units[i].Name) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
cur[units[i].Name] = &units[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
// add all new or changed units
|
|
||||||
changed := make(map[string]*UnitStatus)
|
|
||||||
for n, u := range cur {
|
|
||||||
if oldU, ok := old[n]; !ok || isChanged(oldU, u) {
|
|
||||||
changed[n] = u
|
|
||||||
}
|
|
||||||
delete(old, n)
|
|
||||||
}
|
|
||||||
|
|
||||||
// add all deleted units
|
|
||||||
for oldN := range old {
|
|
||||||
changed[oldN] = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
old = cur
|
|
||||||
|
|
||||||
if len(changed) != 0 {
|
|
||||||
statusChan <- changed
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
errChan <- err
|
|
||||||
}
|
|
||||||
|
|
||||||
<-timerChan
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
return statusChan, errChan
|
|
||||||
}
|
|
||||||
|
|
||||||
type SubStateUpdate struct {
|
|
||||||
UnitName string
|
|
||||||
SubState string
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetSubStateSubscriber writes to updateCh when any unit's substate changes.
|
|
||||||
// Although this writes to updateCh on every state change, the reported state
|
|
||||||
// may be more recent than the change that generated it (due to an unavoidable
|
|
||||||
// race in the systemd dbus interface). That is, this method provides a good
|
|
||||||
// way to keep a current view of all units' states, but is not guaranteed to
|
|
||||||
// show every state transition they go through. Furthermore, state changes
|
|
||||||
// will only be written to the channel with non-blocking writes. If updateCh
|
|
||||||
// is full, it attempts to write an error to errCh; if errCh is full, the error
|
|
||||||
// passes silently.
|
|
||||||
func (c *Conn) SetSubStateSubscriber(updateCh chan<- *SubStateUpdate, errCh chan<- error) {
|
|
||||||
if c == nil {
|
|
||||||
msg := "nil receiver"
|
|
||||||
select {
|
|
||||||
case errCh <- errors.New(msg):
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", msg)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
c.subStateSubscriber.Lock()
|
|
||||||
defer c.subStateSubscriber.Unlock()
|
|
||||||
c.subStateSubscriber.updateCh = updateCh
|
|
||||||
c.subStateSubscriber.errCh = errCh
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) sendSubStateUpdate(unitPath dbus.ObjectPath) {
|
|
||||||
c.subStateSubscriber.Lock()
|
|
||||||
defer c.subStateSubscriber.Unlock()
|
|
||||||
|
|
||||||
if c.subStateSubscriber.updateCh == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
isIgnored := c.shouldIgnore(unitPath)
|
|
||||||
defer c.cleanIgnore()
|
|
||||||
if isIgnored {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
info, err := c.GetUnitPathProperties(unitPath)
|
|
||||||
if err != nil {
|
|
||||||
select {
|
|
||||||
case c.subStateSubscriber.errCh <- err:
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", err)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer c.updateIgnore(unitPath, info)
|
|
||||||
|
|
||||||
name, ok := info["Id"].(string)
|
|
||||||
if !ok {
|
|
||||||
msg := "failed to cast info.Id"
|
|
||||||
select {
|
|
||||||
case c.subStateSubscriber.errCh <- errors.New(msg):
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", err)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
substate, ok := info["SubState"].(string)
|
|
||||||
if !ok {
|
|
||||||
msg := "failed to cast info.SubState"
|
|
||||||
select {
|
|
||||||
case c.subStateSubscriber.errCh <- errors.New(msg):
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", msg)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
update := &SubStateUpdate{name, substate}
|
|
||||||
select {
|
|
||||||
case c.subStateSubscriber.updateCh <- update:
|
|
||||||
default:
|
|
||||||
msg := "update channel is full"
|
|
||||||
select {
|
|
||||||
case c.subStateSubscriber.errCh <- errors.New(msg):
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", msg)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// The ignore functions work around a wart in the systemd dbus interface.
|
|
||||||
// Requesting the properties of an unloaded unit will cause systemd to send a
|
|
||||||
// pair of UnitNew/UnitRemoved signals. Because we need to get a unit's
|
|
||||||
// properties on UnitNew (as that's the only indication of a new unit coming up
|
|
||||||
// for the first time), we would enter an infinite loop if we did not attempt
|
|
||||||
// to detect and ignore these spurious signals. The signal themselves are
|
|
||||||
// indistinguishable from relevant ones, so we (somewhat hackishly) ignore an
|
|
||||||
// unloaded unit's signals for a short time after requesting its properties.
|
|
||||||
// This means that we will miss e.g. a transient unit being restarted
|
|
||||||
// *immediately* upon failure and also a transient unit being started
|
|
||||||
// immediately after requesting its status (with systemctl status, for example,
|
|
||||||
// because this causes a UnitNew signal to be sent which then causes us to fetch
|
|
||||||
// the properties).
|
|
||||||
|
|
||||||
func (c *Conn) shouldIgnore(path dbus.ObjectPath) bool {
|
|
||||||
t, ok := c.subStateSubscriber.ignore[path]
|
|
||||||
return ok && t >= time.Now().UnixNano()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Conn) updateIgnore(path dbus.ObjectPath, info map[string]interface{}) {
|
|
||||||
loadState, ok := info["LoadState"].(string)
|
|
||||||
if !ok {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// unit is unloaded - it will trigger bad systemd dbus behavior
|
|
||||||
if loadState == "not-found" {
|
|
||||||
c.subStateSubscriber.ignore[path] = time.Now().UnixNano() + ignoreInterval
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// without this, ignore would grow unboundedly over time
|
|
||||||
func (c *Conn) cleanIgnore() {
|
|
||||||
now := time.Now().UnixNano()
|
|
||||||
if c.subStateSubscriber.cleanIgnore < now {
|
|
||||||
c.subStateSubscriber.cleanIgnore = now + cleanIgnoreInterval
|
|
||||||
|
|
||||||
for p, t := range c.subStateSubscriber.ignore {
|
|
||||||
if t < now {
|
|
||||||
delete(c.subStateSubscriber.ignore, p)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// PropertiesUpdate holds a map of a unit's changed properties
|
|
||||||
type PropertiesUpdate struct {
|
|
||||||
UnitName string
|
|
||||||
Changed map[string]dbus.Variant
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetPropertiesSubscriber writes to updateCh when any unit's properties
|
|
||||||
// change. Every property change reported by systemd will be sent; that is, no
|
|
||||||
// transitions will be "missed" (as they might be with SetSubStateSubscriber).
|
|
||||||
// However, state changes will only be written to the channel with non-blocking
|
|
||||||
// writes. If updateCh is full, it attempts to write an error to errCh; if
|
|
||||||
// errCh is full, the error passes silently.
|
|
||||||
func (c *Conn) SetPropertiesSubscriber(updateCh chan<- *PropertiesUpdate, errCh chan<- error) {
|
|
||||||
c.propertiesSubscriber.Lock()
|
|
||||||
defer c.propertiesSubscriber.Unlock()
|
|
||||||
c.propertiesSubscriber.updateCh = updateCh
|
|
||||||
c.propertiesSubscriber.errCh = errCh
|
|
||||||
}
|
|
||||||
|
|
||||||
// we don't need to worry about shouldIgnore() here because
|
|
||||||
// sendPropertiesUpdate doesn't call GetProperties()
|
|
||||||
func (c *Conn) sendPropertiesUpdate(unitPath dbus.ObjectPath, changedProps map[string]dbus.Variant) {
|
|
||||||
c.propertiesSubscriber.Lock()
|
|
||||||
defer c.propertiesSubscriber.Unlock()
|
|
||||||
|
|
||||||
if c.propertiesSubscriber.updateCh == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
update := &PropertiesUpdate{unitName(unitPath), changedProps}
|
|
||||||
|
|
||||||
select {
|
|
||||||
case c.propertiesSubscriber.updateCh <- update:
|
|
||||||
default:
|
|
||||||
msg := "update channel is full"
|
|
||||||
select {
|
|
||||||
case c.propertiesSubscriber.errCh <- errors.New(msg):
|
|
||||||
default:
|
|
||||||
log.Printf("full error channel while reporting: %s\n", msg)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
57
vendor/github.com/coreos/go-systemd/dbus/subscription_set.go
generated
vendored
57
vendor/github.com/coreos/go-systemd/dbus/subscription_set.go
generated
vendored
|
@ -1,57 +0,0 @@
|
||||||
// Copyright 2015 CoreOS, Inc.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
// you may not use this file except in compliance with the License.
|
|
||||||
// You may obtain a copy of the License at
|
|
||||||
//
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
//
|
|
||||||
// Unless required by applicable law or agreed to in writing, software
|
|
||||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
// See the License for the specific language governing permissions and
|
|
||||||
// limitations under the License.
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SubscriptionSet returns a subscription set which is like conn.Subscribe but
|
|
||||||
// can filter to only return events for a set of units.
|
|
||||||
type SubscriptionSet struct {
|
|
||||||
*set
|
|
||||||
conn *Conn
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *SubscriptionSet) filter(unit string) bool {
|
|
||||||
return !s.Contains(unit)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Subscribe starts listening for dbus events for all of the units in the set.
|
|
||||||
// Returns channels identical to conn.SubscribeUnits.
|
|
||||||
func (s *SubscriptionSet) Subscribe() (<-chan map[string]*UnitStatus, <-chan error) {
|
|
||||||
// TODO: Make fully evented by using systemd 209 with properties changed values
|
|
||||||
return s.conn.SubscribeUnitsCustom(time.Second, 0,
|
|
||||||
mismatchUnitStatus,
|
|
||||||
func(unit string) bool { return s.filter(unit) },
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSubscriptionSet returns a new subscription set.
|
|
||||||
func (conn *Conn) NewSubscriptionSet() *SubscriptionSet {
|
|
||||||
return &SubscriptionSet{newSet(), conn}
|
|
||||||
}
|
|
||||||
|
|
||||||
// mismatchUnitStatus returns true if the provided UnitStatus objects
|
|
||||||
// are not equivalent. false is returned if the objects are equivalent.
|
|
||||||
// Only the Name, Description and state-related fields are used in
|
|
||||||
// the comparison.
|
|
||||||
func mismatchUnitStatus(u1, u2 *UnitStatus) bool {
|
|
||||||
return u1.Name != u2.Name ||
|
|
||||||
u1.Description != u2.Description ||
|
|
||||||
u1.LoadState != u2.LoadState ||
|
|
||||||
u1.ActiveState != u2.ActiveState ||
|
|
||||||
u1.SubState != u2.SubState
|
|
||||||
}
|
|
16
vendor/github.com/ema/qdisc/.travis.yml
generated
vendored
16
vendor/github.com/ema/qdisc/.travis.yml
generated
vendored
|
@ -1,16 +0,0 @@
|
||||||
language: go
|
|
||||||
go:
|
|
||||||
- 1.x
|
|
||||||
env:
|
|
||||||
- GO111MODULE=on
|
|
||||||
os:
|
|
||||||
- linux
|
|
||||||
sudo: required
|
|
||||||
before_install:
|
|
||||||
- curl -sfL https://install.goreleaser.com/github.com/golangci/golangci-lint.sh | sh -s -- -b $(go env GOPATH)/bin v1.17.1
|
|
||||||
- go get -d ./...
|
|
||||||
script:
|
|
||||||
- go build -tags=gofuzz ./...
|
|
||||||
- go vet ./...
|
|
||||||
- golangci-lint run ./...
|
|
||||||
- go test -v -race -tags=integration ./...
|
|
10
vendor/github.com/ema/qdisc/LICENSE.md
generated
vendored
10
vendor/github.com/ema/qdisc/LICENSE.md
generated
vendored
|
@ -1,10 +0,0 @@
|
||||||
MIT License
|
|
||||||
===========
|
|
||||||
|
|
||||||
Copyright (C) 2017 Emanuele Rocca
|
|
||||||
|
|
||||||
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.
|
|
11
vendor/github.com/ema/qdisc/Makefile
generated
vendored
11
vendor/github.com/ema/qdisc/Makefile
generated
vendored
|
@ -1,11 +0,0 @@
|
||||||
build:
|
|
||||||
go fmt
|
|
||||||
go build
|
|
||||||
go vet
|
|
||||||
staticcheck
|
|
||||||
#golint -set_exit_status
|
|
||||||
go test -v -race -tags=integration
|
|
||||||
|
|
||||||
cover:
|
|
||||||
go test -coverprofile=coverage.out
|
|
||||||
go tool cover -html=coverage.out
|
|
26
vendor/github.com/ema/qdisc/README.md
generated
vendored
26
vendor/github.com/ema/qdisc/README.md
generated
vendored
|
@ -1,26 +0,0 @@
|
||||||
qdisc [![Build Status](https://travis-ci.org/ema/qdisc.svg?branch=master)](https://travis-ci.org/ema/qdisc)
|
|
||||||
=====
|
|
||||||
|
|
||||||
Package `qdisc` allows to get queuing discipline information via netlink,
|
|
||||||
similarly to what `tc -s qdisc show` does.
|
|
||||||
|
|
||||||
Example usage
|
|
||||||
-------------
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"github.com/ema/qdisc"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
info, err := qdisc.Get()
|
|
||||||
|
|
||||||
if err == nil {
|
|
||||||
for _, msg := range info {
|
|
||||||
fmt.Printf("%+v\n", msg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
299
vendor/github.com/ema/qdisc/get.go
generated
vendored
299
vendor/github.com/ema/qdisc/get.go
generated
vendored
|
@ -1,299 +0,0 @@
|
||||||
package qdisc
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"math"
|
|
||||||
"net"
|
|
||||||
|
|
||||||
"github.com/mdlayher/netlink"
|
|
||||||
"github.com/mdlayher/netlink/nlenc"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
TCA_UNSPEC = iota
|
|
||||||
TCA_KIND
|
|
||||||
TCA_OPTIONS
|
|
||||||
TCA_STATS
|
|
||||||
TCA_XSTATS
|
|
||||||
TCA_RATE
|
|
||||||
TCA_FCNT
|
|
||||||
TCA_STATS2
|
|
||||||
TCA_STAB
|
|
||||||
// __TCA_MAX
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
TCA_STATS_UNSPEC = iota
|
|
||||||
TCA_STATS_BASIC
|
|
||||||
TCA_STATS_RATE_EST
|
|
||||||
TCA_STATS_QUEUE
|
|
||||||
TCA_STATS_APP
|
|
||||||
TCA_STATS_RATE_EST64
|
|
||||||
// __TCA_STATS_MAX
|
|
||||||
)
|
|
||||||
|
|
||||||
// See struct tc_stats in /usr/include/linux/pkt_sched.h
|
|
||||||
type TC_Stats struct {
|
|
||||||
Bytes uint64
|
|
||||||
Packets uint32
|
|
||||||
Drops uint32
|
|
||||||
Overlimits uint32
|
|
||||||
Bps uint32
|
|
||||||
Pps uint32
|
|
||||||
Qlen uint32
|
|
||||||
Backlog uint32
|
|
||||||
}
|
|
||||||
|
|
||||||
// See /usr/include/linux/gen_stats.h
|
|
||||||
type TC_Stats2 struct {
|
|
||||||
// struct gnet_stats_basic
|
|
||||||
Bytes uint64
|
|
||||||
Packets uint32
|
|
||||||
// struct gnet_stats_queue
|
|
||||||
Qlen uint32
|
|
||||||
Backlog uint32
|
|
||||||
Drops uint32
|
|
||||||
Requeues uint32
|
|
||||||
Overlimits uint32
|
|
||||||
}
|
|
||||||
|
|
||||||
// See struct tc_fq_qd_stats /usr/include/linux/pkt_sched.h
|
|
||||||
type TC_Fq_Qd_Stats struct {
|
|
||||||
GcFlows uint64
|
|
||||||
HighprioPackets uint64
|
|
||||||
TcpRetrans uint64
|
|
||||||
Throttled uint64
|
|
||||||
FlowsPlimit uint64
|
|
||||||
PktsTooLong uint64
|
|
||||||
AllocationErrors uint64
|
|
||||||
TimeNextDelayedFlow int64
|
|
||||||
Flows uint32
|
|
||||||
InactiveFlows uint32
|
|
||||||
ThrottledFlows uint32
|
|
||||||
UnthrottleLatencyNs uint32
|
|
||||||
}
|
|
||||||
|
|
||||||
type QdiscInfo struct {
|
|
||||||
IfaceName string
|
|
||||||
Parent uint32
|
|
||||||
Handle uint32
|
|
||||||
Kind string
|
|
||||||
Bytes uint64
|
|
||||||
Packets uint32
|
|
||||||
Drops uint32
|
|
||||||
Requeues uint32
|
|
||||||
Overlimits uint32
|
|
||||||
GcFlows uint64
|
|
||||||
Throttled uint64
|
|
||||||
FlowsPlimit uint64
|
|
||||||
Qlen uint32
|
|
||||||
Backlog uint32
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseTCAStats(attr netlink.Attribute) TC_Stats {
|
|
||||||
var stats TC_Stats
|
|
||||||
stats.Bytes = nlenc.Uint64(attr.Data[0:8])
|
|
||||||
stats.Packets = nlenc.Uint32(attr.Data[8:12])
|
|
||||||
stats.Drops = nlenc.Uint32(attr.Data[12:16])
|
|
||||||
stats.Overlimits = nlenc.Uint32(attr.Data[16:20])
|
|
||||||
stats.Bps = nlenc.Uint32(attr.Data[20:24])
|
|
||||||
stats.Pps = nlenc.Uint32(attr.Data[24:28])
|
|
||||||
stats.Qlen = nlenc.Uint32(attr.Data[28:32])
|
|
||||||
stats.Backlog = nlenc.Uint32(attr.Data[32:36])
|
|
||||||
return stats
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseTCAStats2(attr netlink.Attribute) TC_Stats2 {
|
|
||||||
var stats TC_Stats2
|
|
||||||
|
|
||||||
nested, _ := netlink.UnmarshalAttributes(attr.Data)
|
|
||||||
|
|
||||||
for _, a := range nested {
|
|
||||||
switch a.Type {
|
|
||||||
case TCA_STATS_BASIC:
|
|
||||||
stats.Bytes = nlenc.Uint64(a.Data[0:8])
|
|
||||||
stats.Packets = nlenc.Uint32(a.Data[8:12])
|
|
||||||
case TCA_STATS_QUEUE:
|
|
||||||
stats.Qlen = nlenc.Uint32(a.Data[0:4])
|
|
||||||
stats.Backlog = nlenc.Uint32(a.Data[4:8])
|
|
||||||
stats.Drops = nlenc.Uint32(a.Data[8:12])
|
|
||||||
stats.Requeues = nlenc.Uint32(a.Data[12:16])
|
|
||||||
stats.Overlimits = nlenc.Uint32(a.Data[16:20])
|
|
||||||
default:
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return stats
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseTC_Fq_Qd_Stats(attr netlink.Attribute) (TC_Fq_Qd_Stats, error) {
|
|
||||||
var stats TC_Fq_Qd_Stats
|
|
||||||
|
|
||||||
nested, err := netlink.UnmarshalAttributes(attr.Data)
|
|
||||||
if err != nil {
|
|
||||||
return stats, err
|
|
||||||
}
|
|
||||||
|
|
||||||
pts := []*uint64{
|
|
||||||
&stats.GcFlows,
|
|
||||||
&stats.HighprioPackets,
|
|
||||||
&stats.TcpRetrans,
|
|
||||||
&stats.Throttled,
|
|
||||||
&stats.FlowsPlimit,
|
|
||||||
&stats.PktsTooLong,
|
|
||||||
&stats.AllocationErrors,
|
|
||||||
}
|
|
||||||
for _, a := range nested {
|
|
||||||
switch a.Type {
|
|
||||||
case TCA_STATS_APP:
|
|
||||||
for i := 0; i < len(pts) && (i+1)*8 <= len(a.Data); i++ {
|
|
||||||
*pts[i] = nlenc.Uint64(a.Data[i*8 : (i+1)*8])
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return stats, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func getQdiscMsgs(c *netlink.Conn) ([]netlink.Message, error) {
|
|
||||||
req := netlink.Message{
|
|
||||||
Header: netlink.Header{
|
|
||||||
Flags: netlink.Request | netlink.Dump,
|
|
||||||
Type: 38, // RTM_GETQDISC
|
|
||||||
},
|
|
||||||
Data: make([]byte, 20),
|
|
||||||
}
|
|
||||||
|
|
||||||
// Perform a request, receive replies, and validate the replies
|
|
||||||
msgs, err := c.Execute(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to execute request: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return msgs, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// See https://tools.ietf.org/html/rfc3549#section-3.1.3
|
|
||||||
func parseMessage(msg netlink.Message) (QdiscInfo, error) {
|
|
||||||
var m QdiscInfo
|
|
||||||
var s TC_Stats
|
|
||||||
var s2 TC_Stats2
|
|
||||||
var s_fq TC_Fq_Qd_Stats
|
|
||||||
|
|
||||||
/*
|
|
||||||
struct tcmsg {
|
|
||||||
unsigned char tcm_family;
|
|
||||||
unsigned char tcm__pad1;
|
|
||||||
unsigned short tcm__pad2;
|
|
||||||
int tcm_ifindex;
|
|
||||||
__u32 tcm_handle;
|
|
||||||
__u32 tcm_parent;
|
|
||||||
__u32 tcm_info;
|
|
||||||
};
|
|
||||||
*/
|
|
||||||
|
|
||||||
if len(msg.Data) < 20 {
|
|
||||||
return m, fmt.Errorf("short message, len=%d", len(msg.Data))
|
|
||||||
}
|
|
||||||
|
|
||||||
ifaceIdx := nlenc.Uint32(msg.Data[4:8])
|
|
||||||
|
|
||||||
m.Handle = nlenc.Uint32(msg.Data[8:12])
|
|
||||||
m.Parent = nlenc.Uint32(msg.Data[12:16])
|
|
||||||
|
|
||||||
if m.Parent == math.MaxUint32 {
|
|
||||||
m.Parent = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// The first 20 bytes are taken by tcmsg
|
|
||||||
attrs, err := netlink.UnmarshalAttributes(msg.Data[20:])
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return m, fmt.Errorf("failed to unmarshal attributes: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, attr := range attrs {
|
|
||||||
switch attr.Type {
|
|
||||||
case TCA_KIND:
|
|
||||||
m.Kind = nlenc.String(attr.Data)
|
|
||||||
case TCA_STATS2:
|
|
||||||
s_fq, err = parseTC_Fq_Qd_Stats(attr)
|
|
||||||
if err != nil {
|
|
||||||
return m, err
|
|
||||||
}
|
|
||||||
if s_fq.GcFlows > 0 {
|
|
||||||
m.GcFlows = s_fq.GcFlows
|
|
||||||
}
|
|
||||||
if s_fq.Throttled > 0 {
|
|
||||||
m.Throttled = s_fq.Throttled
|
|
||||||
}
|
|
||||||
if s_fq.FlowsPlimit > 0 {
|
|
||||||
m.FlowsPlimit = s_fq.FlowsPlimit
|
|
||||||
}
|
|
||||||
|
|
||||||
s2 = parseTCAStats2(attr)
|
|
||||||
m.Bytes = s2.Bytes
|
|
||||||
m.Packets = s2.Packets
|
|
||||||
m.Drops = s2.Drops
|
|
||||||
// requeues only available in TCA_STATS2, not in TCA_STATS
|
|
||||||
m.Requeues = s2.Requeues
|
|
||||||
m.Overlimits = s2.Overlimits
|
|
||||||
m.Qlen = s2.Qlen
|
|
||||||
m.Backlog = s2.Backlog
|
|
||||||
case TCA_STATS:
|
|
||||||
// Legacy
|
|
||||||
s = parseTCAStats(attr)
|
|
||||||
m.Bytes = s.Bytes
|
|
||||||
m.Packets = s.Packets
|
|
||||||
m.Drops = s.Drops
|
|
||||||
m.Overlimits = s.Overlimits
|
|
||||||
m.Qlen = s.Qlen
|
|
||||||
m.Backlog = s.Backlog
|
|
||||||
default:
|
|
||||||
// TODO: TCA_OPTIONS and TCA_XSTATS
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
iface, err := net.InterfaceByIndex(int(ifaceIdx))
|
|
||||||
|
|
||||||
if err == nil {
|
|
||||||
m.IfaceName = iface.Name
|
|
||||||
}
|
|
||||||
|
|
||||||
return m, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func getAndParse(c *netlink.Conn) ([]QdiscInfo, error) {
|
|
||||||
var res []QdiscInfo
|
|
||||||
|
|
||||||
msgs, err := getQdiscMsgs(c)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, msg := range msgs {
|
|
||||||
m, err := parseMessage(msg)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
res = append(res, m)
|
|
||||||
}
|
|
||||||
|
|
||||||
return res, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func Get() ([]QdiscInfo, error) {
|
|
||||||
const familyRoute = 0
|
|
||||||
|
|
||||||
c, err := netlink.Dial(familyRoute, nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("failed to dial netlink: %v", err)
|
|
||||||
}
|
|
||||||
defer c.Close()
|
|
||||||
|
|
||||||
return getAndParse(c)
|
|
||||||
}
|
|
9
vendor/github.com/ema/qdisc/go.mod
generated
vendored
9
vendor/github.com/ema/qdisc/go.mod
generated
vendored
|
@ -1,9 +0,0 @@
|
||||||
module github.com/ema/qdisc
|
|
||||||
|
|
||||||
go 1.12
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/jsimonetti/rtnetlink v0.0.0-20190830100107-3784a6c7c552 // indirect
|
|
||||||
github.com/mdlayher/netlink v0.0.0-20190828143259-340058475d09
|
|
||||||
golang.org/x/sys v0.0.0-20190902133755-9109b7679e13 // indirect
|
|
||||||
)
|
|
20
vendor/github.com/ema/qdisc/go.sum
generated
vendored
20
vendor/github.com/ema/qdisc/go.sum
generated
vendored
|
@ -1,20 +0,0 @@
|
||||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
|
||||||
github.com/google/go-cmp v0.3.1 h1:Xye71clBPdm5HgqGwUkwhbynsUJZhDbS20FvLhQ2izg=
|
|
||||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
|
||||||
github.com/jsimonetti/rtnetlink v0.0.0-20190606172950-9527aa82566a/go.mod h1:Oz+70psSo5OFh8DBl0Zv2ACw7Esh6pPUphlvZG9x7uw=
|
|
||||||
github.com/jsimonetti/rtnetlink v0.0.0-20190830100107-3784a6c7c552 h1:Ve/e6edHdAHn+8/24Xco7IhQCv3u5Dab2qZNvR9e5/U=
|
|
||||||
github.com/jsimonetti/rtnetlink v0.0.0-20190830100107-3784a6c7c552/go.mod h1:Oz+70psSo5OFh8DBl0Zv2ACw7Esh6pPUphlvZG9x7uw=
|
|
||||||
github.com/mdlayher/netlink v0.0.0-20190409211403-11939a169225/go.mod h1:eQB3mZE4aiYnlUsyGGCOpPETfdQq4Jhsgf1fk3cwQaA=
|
|
||||||
github.com/mdlayher/netlink v0.0.0-20190828143259-340058475d09 h1:U2vuol6i4UF6MSpZJclH4HHiLRMoq1NAzxpIpCUJK/Y=
|
|
||||||
github.com/mdlayher/netlink v0.0.0-20190828143259-340058475d09/go.mod h1:KxeJAFOFLG6AjpyDkQ/iIhxygIUKD+vcwqcnu43w/+M=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190827160401-ba9fcec4b297 h1:k7pJ2yAPLPgbskkFdhRCsA77k2fySZ1zf2zCjvQCiIM=
|
|
||||||
golang.org/x/net v0.0.0-20190827160401-ba9fcec4b297/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190411185658-b44545bcd369/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190902133755-9109b7679e13 h1:tdsQdquKbTNMsSZLqnLELJGzCANp9oXhu6zFBW6ODx4=
|
|
||||||
golang.org/x/sys v0.0.0-20190902133755-9109b7679e13/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
22
vendor/github.com/go-kit/kit/LICENSE
generated
vendored
22
vendor/github.com/go-kit/kit/LICENSE
generated
vendored
|
@ -1,22 +0,0 @@
|
||||||
The MIT License (MIT)
|
|
||||||
|
|
||||||
Copyright (c) 2015 Peter Bourgon
|
|
||||||
|
|
||||||
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.
|
|
||||||
|
|
151
vendor/github.com/go-kit/kit/log/README.md
generated
vendored
151
vendor/github.com/go-kit/kit/log/README.md
generated
vendored
|
@ -1,151 +0,0 @@
|
||||||
# package log
|
|
||||||
|
|
||||||
`package log` provides a minimal interface for structured logging in services.
|
|
||||||
It may be wrapped to encode conventions, enforce type-safety, provide leveled
|
|
||||||
logging, and so on. It can be used for both typical application log events,
|
|
||||||
and log-structured data streams.
|
|
||||||
|
|
||||||
## Structured logging
|
|
||||||
|
|
||||||
Structured logging is, basically, conceding to the reality that logs are
|
|
||||||
_data_, and warrant some level of schematic rigor. Using a stricter,
|
|
||||||
key/value-oriented message format for our logs, containing contextual and
|
|
||||||
semantic information, makes it much easier to get insight into the
|
|
||||||
operational activity of the systems we build. Consequently, `package log` is
|
|
||||||
of the strong belief that "[the benefits of structured logging outweigh the
|
|
||||||
minimal effort involved](https://www.thoughtworks.com/radar/techniques/structured-logging)".
|
|
||||||
|
|
||||||
Migrating from unstructured to structured logging is probably a lot easier
|
|
||||||
than you'd expect.
|
|
||||||
|
|
||||||
```go
|
|
||||||
// Unstructured
|
|
||||||
log.Printf("HTTP server listening on %s", addr)
|
|
||||||
|
|
||||||
// Structured
|
|
||||||
logger.Log("transport", "HTTP", "addr", addr, "msg", "listening")
|
|
||||||
```
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
### Typical application logging
|
|
||||||
|
|
||||||
```go
|
|
||||||
w := log.NewSyncWriter(os.Stderr)
|
|
||||||
logger := log.NewLogfmtLogger(w)
|
|
||||||
logger.Log("question", "what is the meaning of life?", "answer", 42)
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// question="what is the meaning of life?" answer=42
|
|
||||||
```
|
|
||||||
|
|
||||||
### Contextual Loggers
|
|
||||||
|
|
||||||
```go
|
|
||||||
func main() {
|
|
||||||
var logger log.Logger
|
|
||||||
logger = log.NewLogfmtLogger(log.NewSyncWriter(os.Stderr))
|
|
||||||
logger = log.With(logger, "instance_id", 123)
|
|
||||||
|
|
||||||
logger.Log("msg", "starting")
|
|
||||||
NewWorker(log.With(logger, "component", "worker")).Run()
|
|
||||||
NewSlacker(log.With(logger, "component", "slacker")).Run()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// instance_id=123 msg=starting
|
|
||||||
// instance_id=123 component=worker msg=running
|
|
||||||
// instance_id=123 component=slacker msg=running
|
|
||||||
```
|
|
||||||
|
|
||||||
### Interact with stdlib logger
|
|
||||||
|
|
||||||
Redirect stdlib logger to Go kit logger.
|
|
||||||
|
|
||||||
```go
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
stdlog "log"
|
|
||||||
kitlog "github.com/go-kit/kit/log"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
logger := kitlog.NewJSONLogger(kitlog.NewSyncWriter(os.Stdout))
|
|
||||||
stdlog.SetOutput(kitlog.NewStdlibAdapter(logger))
|
|
||||||
stdlog.Print("I sure like pie")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// {"msg":"I sure like pie","ts":"2016/01/01 12:34:56"}
|
|
||||||
```
|
|
||||||
|
|
||||||
Or, if, for legacy reasons, you need to pipe all of your logging through the
|
|
||||||
stdlib log package, you can redirect Go kit logger to the stdlib logger.
|
|
||||||
|
|
||||||
```go
|
|
||||||
logger := kitlog.NewLogfmtLogger(kitlog.StdlibWriter{})
|
|
||||||
logger.Log("legacy", true, "msg", "at least it's something")
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// 2016/01/01 12:34:56 legacy=true msg="at least it's something"
|
|
||||||
```
|
|
||||||
|
|
||||||
### Timestamps and callers
|
|
||||||
|
|
||||||
```go
|
|
||||||
var logger log.Logger
|
|
||||||
logger = log.NewLogfmtLogger(log.NewSyncWriter(os.Stderr))
|
|
||||||
logger = log.With(logger, "ts", log.DefaultTimestampUTC, "caller", log.DefaultCaller)
|
|
||||||
|
|
||||||
logger.Log("msg", "hello")
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// ts=2016-01-01T12:34:56Z caller=main.go:15 msg=hello
|
|
||||||
```
|
|
||||||
|
|
||||||
## Levels
|
|
||||||
|
|
||||||
Log levels are supported via the [level package](https://godoc.org/github.com/go-kit/kit/log/level).
|
|
||||||
|
|
||||||
## Supported output formats
|
|
||||||
|
|
||||||
- [Logfmt](https://brandur.org/logfmt) ([see also](https://blog.codeship.com/logfmt-a-log-format-thats-easy-to-read-and-write))
|
|
||||||
- JSON
|
|
||||||
|
|
||||||
## Enhancements
|
|
||||||
|
|
||||||
`package log` is centered on the one-method Logger interface.
|
|
||||||
|
|
||||||
```go
|
|
||||||
type Logger interface {
|
|
||||||
Log(keyvals ...interface{}) error
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
This interface, and its supporting code like is the product of much iteration
|
|
||||||
and evaluation. For more details on the evolution of the Logger interface,
|
|
||||||
see [The Hunt for a Logger Interface](http://go-talks.appspot.com/github.com/ChrisHines/talks/structured-logging/structured-logging.slide#1),
|
|
||||||
a talk by [Chris Hines](https://github.com/ChrisHines).
|
|
||||||
Also, please see
|
|
||||||
[#63](https://github.com/go-kit/kit/issues/63),
|
|
||||||
[#76](https://github.com/go-kit/kit/pull/76),
|
|
||||||
[#131](https://github.com/go-kit/kit/issues/131),
|
|
||||||
[#157](https://github.com/go-kit/kit/pull/157),
|
|
||||||
[#164](https://github.com/go-kit/kit/issues/164), and
|
|
||||||
[#252](https://github.com/go-kit/kit/pull/252)
|
|
||||||
to review historical conversations about package log and the Logger interface.
|
|
||||||
|
|
||||||
Value-add packages and suggestions,
|
|
||||||
like improvements to [the leveled logger](https://godoc.org/github.com/go-kit/kit/log/level),
|
|
||||||
are of course welcome. Good proposals should
|
|
||||||
|
|
||||||
- Be composable with [contextual loggers](https://godoc.org/github.com/go-kit/kit/log#With),
|
|
||||||
- Not break the behavior of [log.Caller](https://godoc.org/github.com/go-kit/kit/log#Caller) in any wrapped contextual loggers, and
|
|
||||||
- Be friendly to packages that accept only an unadorned log.Logger.
|
|
||||||
|
|
||||||
## Benchmarks & comparisons
|
|
||||||
|
|
||||||
There are a few Go logging benchmarks and comparisons that include Go kit's package log.
|
|
||||||
|
|
||||||
- [imkira/go-loggers-bench](https://github.com/imkira/go-loggers-bench) includes kit/log
|
|
||||||
- [uber-common/zap](https://github.com/uber-common/zap), a zero-alloc logging library, includes a comparison with kit/log
|
|
116
vendor/github.com/go-kit/kit/log/doc.go
generated
vendored
116
vendor/github.com/go-kit/kit/log/doc.go
generated
vendored
|
@ -1,116 +0,0 @@
|
||||||
// Package log provides a structured logger.
|
|
||||||
//
|
|
||||||
// Structured logging produces logs easily consumed later by humans or
|
|
||||||
// machines. Humans might be interested in debugging errors, or tracing
|
|
||||||
// specific requests. Machines might be interested in counting interesting
|
|
||||||
// events, or aggregating information for off-line processing. In both cases,
|
|
||||||
// it is important that the log messages are structured and actionable.
|
|
||||||
// Package log is designed to encourage both of these best practices.
|
|
||||||
//
|
|
||||||
// Basic Usage
|
|
||||||
//
|
|
||||||
// The fundamental interface is Logger. Loggers create log events from
|
|
||||||
// key/value data. The Logger interface has a single method, Log, which
|
|
||||||
// accepts a sequence of alternating key/value pairs, which this package names
|
|
||||||
// keyvals.
|
|
||||||
//
|
|
||||||
// type Logger interface {
|
|
||||||
// Log(keyvals ...interface{}) error
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// Here is an example of a function using a Logger to create log events.
|
|
||||||
//
|
|
||||||
// func RunTask(task Task, logger log.Logger) string {
|
|
||||||
// logger.Log("taskID", task.ID, "event", "starting task")
|
|
||||||
// ...
|
|
||||||
// logger.Log("taskID", task.ID, "event", "task complete")
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// The keys in the above example are "taskID" and "event". The values are
|
|
||||||
// task.ID, "starting task", and "task complete". Every key is followed
|
|
||||||
// immediately by its value.
|
|
||||||
//
|
|
||||||
// Keys are usually plain strings. Values may be any type that has a sensible
|
|
||||||
// encoding in the chosen log format. With structured logging it is a good
|
|
||||||
// idea to log simple values without formatting them. This practice allows
|
|
||||||
// the chosen logger to encode values in the most appropriate way.
|
|
||||||
//
|
|
||||||
// Contextual Loggers
|
|
||||||
//
|
|
||||||
// A contextual logger stores keyvals that it includes in all log events.
|
|
||||||
// Building appropriate contextual loggers reduces repetition and aids
|
|
||||||
// consistency in the resulting log output. With and WithPrefix add context to
|
|
||||||
// a logger. We can use With to improve the RunTask example.
|
|
||||||
//
|
|
||||||
// func RunTask(task Task, logger log.Logger) string {
|
|
||||||
// logger = log.With(logger, "taskID", task.ID)
|
|
||||||
// logger.Log("event", "starting task")
|
|
||||||
// ...
|
|
||||||
// taskHelper(task.Cmd, logger)
|
|
||||||
// ...
|
|
||||||
// logger.Log("event", "task complete")
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// The improved version emits the same log events as the original for the
|
|
||||||
// first and last calls to Log. Passing the contextual logger to taskHelper
|
|
||||||
// enables each log event created by taskHelper to include the task.ID even
|
|
||||||
// though taskHelper does not have access to that value. Using contextual
|
|
||||||
// loggers this way simplifies producing log output that enables tracing the
|
|
||||||
// life cycle of individual tasks. (See the Contextual example for the full
|
|
||||||
// code of the above snippet.)
|
|
||||||
//
|
|
||||||
// Dynamic Contextual Values
|
|
||||||
//
|
|
||||||
// A Valuer function stored in a contextual logger generates a new value each
|
|
||||||
// time an event is logged. The Valuer example demonstrates how this feature
|
|
||||||
// works.
|
|
||||||
//
|
|
||||||
// Valuers provide the basis for consistently logging timestamps and source
|
|
||||||
// code location. The log package defines several valuers for that purpose.
|
|
||||||
// See Timestamp, DefaultTimestamp, DefaultTimestampUTC, Caller, and
|
|
||||||
// DefaultCaller. A common logger initialization sequence that ensures all log
|
|
||||||
// entries contain a timestamp and source location looks like this:
|
|
||||||
//
|
|
||||||
// logger := log.NewLogfmtLogger(log.NewSyncWriter(os.Stdout))
|
|
||||||
// logger = log.With(logger, "ts", log.DefaultTimestampUTC, "caller", log.DefaultCaller)
|
|
||||||
//
|
|
||||||
// Concurrent Safety
|
|
||||||
//
|
|
||||||
// Applications with multiple goroutines want each log event written to the
|
|
||||||
// same logger to remain separate from other log events. Package log provides
|
|
||||||
// two simple solutions for concurrent safe logging.
|
|
||||||
//
|
|
||||||
// NewSyncWriter wraps an io.Writer and serializes each call to its Write
|
|
||||||
// method. Using a SyncWriter has the benefit that the smallest practical
|
|
||||||
// portion of the logging logic is performed within a mutex, but it requires
|
|
||||||
// the formatting Logger to make only one call to Write per log event.
|
|
||||||
//
|
|
||||||
// NewSyncLogger wraps any Logger and serializes each call to its Log method.
|
|
||||||
// Using a SyncLogger has the benefit that it guarantees each log event is
|
|
||||||
// handled atomically within the wrapped logger, but it typically serializes
|
|
||||||
// both the formatting and output logic. Use a SyncLogger if the formatting
|
|
||||||
// logger may perform multiple writes per log event.
|
|
||||||
//
|
|
||||||
// Error Handling
|
|
||||||
//
|
|
||||||
// This package relies on the practice of wrapping or decorating loggers with
|
|
||||||
// other loggers to provide composable pieces of functionality. It also means
|
|
||||||
// that Logger.Log must return an error because some
|
|
||||||
// implementations—especially those that output log data to an io.Writer—may
|
|
||||||
// encounter errors that cannot be handled locally. This in turn means that
|
|
||||||
// Loggers that wrap other loggers should return errors from the wrapped
|
|
||||||
// logger up the stack.
|
|
||||||
//
|
|
||||||
// Fortunately, the decorator pattern also provides a way to avoid the
|
|
||||||
// necessity to check for errors every time an application calls Logger.Log.
|
|
||||||
// An application required to panic whenever its Logger encounters
|
|
||||||
// an error could initialize its logger as follows.
|
|
||||||
//
|
|
||||||
// fmtlogger := log.NewLogfmtLogger(log.NewSyncWriter(os.Stdout))
|
|
||||||
// logger := log.LoggerFunc(func(keyvals ...interface{}) error {
|
|
||||||
// if err := fmtlogger.Log(keyvals...); err != nil {
|
|
||||||
// panic(err)
|
|
||||||
// }
|
|
||||||
// return nil
|
|
||||||
// })
|
|
||||||
package log
|
|
91
vendor/github.com/go-kit/kit/log/json_logger.go
generated
vendored
91
vendor/github.com/go-kit/kit/log/json_logger.go
generated
vendored
|
@ -1,91 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding"
|
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"reflect"
|
|
||||||
)
|
|
||||||
|
|
||||||
type jsonLogger struct {
|
|
||||||
io.Writer
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewJSONLogger returns a Logger that encodes keyvals to the Writer as a
|
|
||||||
// single JSON object. Each log event produces no more than one call to
|
|
||||||
// w.Write. The passed Writer must be safe for concurrent use by multiple
|
|
||||||
// goroutines if the returned Logger will be used concurrently.
|
|
||||||
func NewJSONLogger(w io.Writer) Logger {
|
|
||||||
return &jsonLogger{w}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *jsonLogger) Log(keyvals ...interface{}) error {
|
|
||||||
n := (len(keyvals) + 1) / 2 // +1 to handle case when len is odd
|
|
||||||
m := make(map[string]interface{}, n)
|
|
||||||
for i := 0; i < len(keyvals); i += 2 {
|
|
||||||
k := keyvals[i]
|
|
||||||
var v interface{} = ErrMissingValue
|
|
||||||
if i+1 < len(keyvals) {
|
|
||||||
v = keyvals[i+1]
|
|
||||||
}
|
|
||||||
merge(m, k, v)
|
|
||||||
}
|
|
||||||
enc := json.NewEncoder(l.Writer)
|
|
||||||
enc.SetEscapeHTML(false)
|
|
||||||
return enc.Encode(m)
|
|
||||||
}
|
|
||||||
|
|
||||||
func merge(dst map[string]interface{}, k, v interface{}) {
|
|
||||||
var key string
|
|
||||||
switch x := k.(type) {
|
|
||||||
case string:
|
|
||||||
key = x
|
|
||||||
case fmt.Stringer:
|
|
||||||
key = safeString(x)
|
|
||||||
default:
|
|
||||||
key = fmt.Sprint(x)
|
|
||||||
}
|
|
||||||
|
|
||||||
// We want json.Marshaler and encoding.TextMarshaller to take priority over
|
|
||||||
// err.Error() and v.String(). But json.Marshall (called later) does that by
|
|
||||||
// default so we force a no-op if it's one of those 2 case.
|
|
||||||
switch x := v.(type) {
|
|
||||||
case json.Marshaler:
|
|
||||||
case encoding.TextMarshaler:
|
|
||||||
case error:
|
|
||||||
v = safeError(x)
|
|
||||||
case fmt.Stringer:
|
|
||||||
v = safeString(x)
|
|
||||||
}
|
|
||||||
|
|
||||||
dst[key] = v
|
|
||||||
}
|
|
||||||
|
|
||||||
func safeString(str fmt.Stringer) (s string) {
|
|
||||||
defer func() {
|
|
||||||
if panicVal := recover(); panicVal != nil {
|
|
||||||
if v := reflect.ValueOf(str); v.Kind() == reflect.Ptr && v.IsNil() {
|
|
||||||
s = "NULL"
|
|
||||||
} else {
|
|
||||||
panic(panicVal)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
s = str.String()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func safeError(err error) (s interface{}) {
|
|
||||||
defer func() {
|
|
||||||
if panicVal := recover(); panicVal != nil {
|
|
||||||
if v := reflect.ValueOf(err); v.Kind() == reflect.Ptr && v.IsNil() {
|
|
||||||
s = nil
|
|
||||||
} else {
|
|
||||||
panic(panicVal)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
s = err.Error()
|
|
||||||
return
|
|
||||||
}
|
|
22
vendor/github.com/go-kit/kit/log/level/doc.go
generated
vendored
22
vendor/github.com/go-kit/kit/log/level/doc.go
generated
vendored
|
@ -1,22 +0,0 @@
|
||||||
// Package level implements leveled logging on top of Go kit's log package. To
|
|
||||||
// use the level package, create a logger as per normal in your func main, and
|
|
||||||
// wrap it with level.NewFilter.
|
|
||||||
//
|
|
||||||
// var logger log.Logger
|
|
||||||
// logger = log.NewLogfmtLogger(os.Stderr)
|
|
||||||
// logger = level.NewFilter(logger, level.AllowInfo()) // <--
|
|
||||||
// logger = log.With(logger, "ts", log.DefaultTimestampUTC)
|
|
||||||
//
|
|
||||||
// Then, at the callsites, use one of the level.Debug, Info, Warn, or Error
|
|
||||||
// helper methods to emit leveled log events.
|
|
||||||
//
|
|
||||||
// logger.Log("foo", "bar") // as normal, no level
|
|
||||||
// level.Debug(logger).Log("request_id", reqID, "trace_data", trace.Get())
|
|
||||||
// if value > 100 {
|
|
||||||
// level.Error(logger).Log("value", value)
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// NewFilter allows precise control over what happens when a log event is
|
|
||||||
// emitted without a level key, or if a squelched level is used. Check the
|
|
||||||
// Option functions for details.
|
|
||||||
package level
|
|
205
vendor/github.com/go-kit/kit/log/level/level.go
generated
vendored
205
vendor/github.com/go-kit/kit/log/level/level.go
generated
vendored
|
@ -1,205 +0,0 @@
|
||||||
package level
|
|
||||||
|
|
||||||
import "github.com/go-kit/kit/log"
|
|
||||||
|
|
||||||
// Error returns a logger that includes a Key/ErrorValue pair.
|
|
||||||
func Error(logger log.Logger) log.Logger {
|
|
||||||
return log.WithPrefix(logger, Key(), ErrorValue())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Warn returns a logger that includes a Key/WarnValue pair.
|
|
||||||
func Warn(logger log.Logger) log.Logger {
|
|
||||||
return log.WithPrefix(logger, Key(), WarnValue())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Info returns a logger that includes a Key/InfoValue pair.
|
|
||||||
func Info(logger log.Logger) log.Logger {
|
|
||||||
return log.WithPrefix(logger, Key(), InfoValue())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Debug returns a logger that includes a Key/DebugValue pair.
|
|
||||||
func Debug(logger log.Logger) log.Logger {
|
|
||||||
return log.WithPrefix(logger, Key(), DebugValue())
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewFilter wraps next and implements level filtering. See the commentary on
|
|
||||||
// the Option functions for a detailed description of how to configure levels.
|
|
||||||
// If no options are provided, all leveled log events created with Debug,
|
|
||||||
// Info, Warn or Error helper methods are squelched and non-leveled log
|
|
||||||
// events are passed to next unmodified.
|
|
||||||
func NewFilter(next log.Logger, options ...Option) log.Logger {
|
|
||||||
l := &logger{
|
|
||||||
next: next,
|
|
||||||
}
|
|
||||||
for _, option := range options {
|
|
||||||
option(l)
|
|
||||||
}
|
|
||||||
return l
|
|
||||||
}
|
|
||||||
|
|
||||||
type logger struct {
|
|
||||||
next log.Logger
|
|
||||||
allowed level
|
|
||||||
squelchNoLevel bool
|
|
||||||
errNotAllowed error
|
|
||||||
errNoLevel error
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *logger) Log(keyvals ...interface{}) error {
|
|
||||||
var hasLevel, levelAllowed bool
|
|
||||||
for i := 1; i < len(keyvals); i += 2 {
|
|
||||||
if v, ok := keyvals[i].(*levelValue); ok {
|
|
||||||
hasLevel = true
|
|
||||||
levelAllowed = l.allowed&v.level != 0
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !hasLevel && l.squelchNoLevel {
|
|
||||||
return l.errNoLevel
|
|
||||||
}
|
|
||||||
if hasLevel && !levelAllowed {
|
|
||||||
return l.errNotAllowed
|
|
||||||
}
|
|
||||||
return l.next.Log(keyvals...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Option sets a parameter for the leveled logger.
|
|
||||||
type Option func(*logger)
|
|
||||||
|
|
||||||
// AllowAll is an alias for AllowDebug.
|
|
||||||
func AllowAll() Option {
|
|
||||||
return AllowDebug()
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllowDebug allows error, warn, info and debug level log events to pass.
|
|
||||||
func AllowDebug() Option {
|
|
||||||
return allowed(levelError | levelWarn | levelInfo | levelDebug)
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllowInfo allows error, warn and info level log events to pass.
|
|
||||||
func AllowInfo() Option {
|
|
||||||
return allowed(levelError | levelWarn | levelInfo)
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllowWarn allows error and warn level log events to pass.
|
|
||||||
func AllowWarn() Option {
|
|
||||||
return allowed(levelError | levelWarn)
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllowError allows only error level log events to pass.
|
|
||||||
func AllowError() Option {
|
|
||||||
return allowed(levelError)
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllowNone allows no leveled log events to pass.
|
|
||||||
func AllowNone() Option {
|
|
||||||
return allowed(0)
|
|
||||||
}
|
|
||||||
|
|
||||||
func allowed(allowed level) Option {
|
|
||||||
return func(l *logger) { l.allowed = allowed }
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrNotAllowed sets the error to return from Log when it squelches a log
|
|
||||||
// event disallowed by the configured Allow[Level] option. By default,
|
|
||||||
// ErrNotAllowed is nil; in this case the log event is squelched with no
|
|
||||||
// error.
|
|
||||||
func ErrNotAllowed(err error) Option {
|
|
||||||
return func(l *logger) { l.errNotAllowed = err }
|
|
||||||
}
|
|
||||||
|
|
||||||
// SquelchNoLevel instructs Log to squelch log events with no level, so that
|
|
||||||
// they don't proceed through to the wrapped logger. If SquelchNoLevel is set
|
|
||||||
// to true and a log event is squelched in this way, the error value
|
|
||||||
// configured with ErrNoLevel is returned to the caller.
|
|
||||||
func SquelchNoLevel(squelch bool) Option {
|
|
||||||
return func(l *logger) { l.squelchNoLevel = squelch }
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrNoLevel sets the error to return from Log when it squelches a log event
|
|
||||||
// with no level. By default, ErrNoLevel is nil; in this case the log event is
|
|
||||||
// squelched with no error.
|
|
||||||
func ErrNoLevel(err error) Option {
|
|
||||||
return func(l *logger) { l.errNoLevel = err }
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewInjector wraps next and returns a logger that adds a Key/level pair to
|
|
||||||
// the beginning of log events that don't already contain a level. In effect,
|
|
||||||
// this gives a default level to logs without a level.
|
|
||||||
func NewInjector(next log.Logger, level Value) log.Logger {
|
|
||||||
return &injector{
|
|
||||||
next: next,
|
|
||||||
level: level,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type injector struct {
|
|
||||||
next log.Logger
|
|
||||||
level interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *injector) Log(keyvals ...interface{}) error {
|
|
||||||
for i := 1; i < len(keyvals); i += 2 {
|
|
||||||
if _, ok := keyvals[i].(*levelValue); ok {
|
|
||||||
return l.next.Log(keyvals...)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
kvs := make([]interface{}, len(keyvals)+2)
|
|
||||||
kvs[0], kvs[1] = key, l.level
|
|
||||||
copy(kvs[2:], keyvals)
|
|
||||||
return l.next.Log(kvs...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Value is the interface that each of the canonical level values implement.
|
|
||||||
// It contains unexported methods that prevent types from other packages from
|
|
||||||
// implementing it and guaranteeing that NewFilter can distinguish the levels
|
|
||||||
// defined in this package from all other values.
|
|
||||||
type Value interface {
|
|
||||||
String() string
|
|
||||||
levelVal()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Key returns the unique key added to log events by the loggers in this
|
|
||||||
// package.
|
|
||||||
func Key() interface{} { return key }
|
|
||||||
|
|
||||||
// ErrorValue returns the unique value added to log events by Error.
|
|
||||||
func ErrorValue() Value { return errorValue }
|
|
||||||
|
|
||||||
// WarnValue returns the unique value added to log events by Warn.
|
|
||||||
func WarnValue() Value { return warnValue }
|
|
||||||
|
|
||||||
// InfoValue returns the unique value added to log events by Info.
|
|
||||||
func InfoValue() Value { return infoValue }
|
|
||||||
|
|
||||||
// DebugValue returns the unique value added to log events by Warn.
|
|
||||||
func DebugValue() Value { return debugValue }
|
|
||||||
|
|
||||||
var (
|
|
||||||
// key is of type interface{} so that it allocates once during package
|
|
||||||
// initialization and avoids allocating every time the value is added to a
|
|
||||||
// []interface{} later.
|
|
||||||
key interface{} = "level"
|
|
||||||
|
|
||||||
errorValue = &levelValue{level: levelError, name: "error"}
|
|
||||||
warnValue = &levelValue{level: levelWarn, name: "warn"}
|
|
||||||
infoValue = &levelValue{level: levelInfo, name: "info"}
|
|
||||||
debugValue = &levelValue{level: levelDebug, name: "debug"}
|
|
||||||
)
|
|
||||||
|
|
||||||
type level byte
|
|
||||||
|
|
||||||
const (
|
|
||||||
levelDebug level = 1 << iota
|
|
||||||
levelInfo
|
|
||||||
levelWarn
|
|
||||||
levelError
|
|
||||||
)
|
|
||||||
|
|
||||||
type levelValue struct {
|
|
||||||
name string
|
|
||||||
level
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *levelValue) String() string { return v.name }
|
|
||||||
func (v *levelValue) levelVal() {}
|
|
135
vendor/github.com/go-kit/kit/log/log.go
generated
vendored
135
vendor/github.com/go-kit/kit/log/log.go
generated
vendored
|
@ -1,135 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import "errors"
|
|
||||||
|
|
||||||
// Logger is the fundamental interface for all log operations. Log creates a
|
|
||||||
// log event from keyvals, a variadic sequence of alternating keys and values.
|
|
||||||
// Implementations must be safe for concurrent use by multiple goroutines. In
|
|
||||||
// particular, any implementation of Logger that appends to keyvals or
|
|
||||||
// modifies or retains any of its elements must make a copy first.
|
|
||||||
type Logger interface {
|
|
||||||
Log(keyvals ...interface{}) error
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrMissingValue is appended to keyvals slices with odd length to substitute
|
|
||||||
// the missing value.
|
|
||||||
var ErrMissingValue = errors.New("(MISSING)")
|
|
||||||
|
|
||||||
// With returns a new contextual logger with keyvals prepended to those passed
|
|
||||||
// to calls to Log. If logger is also a contextual logger created by With or
|
|
||||||
// WithPrefix, keyvals is appended to the existing context.
|
|
||||||
//
|
|
||||||
// The returned Logger replaces all value elements (odd indexes) containing a
|
|
||||||
// Valuer with their generated value for each call to its Log method.
|
|
||||||
func With(logger Logger, keyvals ...interface{}) Logger {
|
|
||||||
if len(keyvals) == 0 {
|
|
||||||
return logger
|
|
||||||
}
|
|
||||||
l := newContext(logger)
|
|
||||||
kvs := append(l.keyvals, keyvals...)
|
|
||||||
if len(kvs)%2 != 0 {
|
|
||||||
kvs = append(kvs, ErrMissingValue)
|
|
||||||
}
|
|
||||||
return &context{
|
|
||||||
logger: l.logger,
|
|
||||||
// Limiting the capacity of the stored keyvals ensures that a new
|
|
||||||
// backing array is created if the slice must grow in Log or With.
|
|
||||||
// Using the extra capacity without copying risks a data race that
|
|
||||||
// would violate the Logger interface contract.
|
|
||||||
keyvals: kvs[:len(kvs):len(kvs)],
|
|
||||||
hasValuer: l.hasValuer || containsValuer(keyvals),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithPrefix returns a new contextual logger with keyvals prepended to those
|
|
||||||
// passed to calls to Log. If logger is also a contextual logger created by
|
|
||||||
// With or WithPrefix, keyvals is prepended to the existing context.
|
|
||||||
//
|
|
||||||
// The returned Logger replaces all value elements (odd indexes) containing a
|
|
||||||
// Valuer with their generated value for each call to its Log method.
|
|
||||||
func WithPrefix(logger Logger, keyvals ...interface{}) Logger {
|
|
||||||
if len(keyvals) == 0 {
|
|
||||||
return logger
|
|
||||||
}
|
|
||||||
l := newContext(logger)
|
|
||||||
// Limiting the capacity of the stored keyvals ensures that a new
|
|
||||||
// backing array is created if the slice must grow in Log or With.
|
|
||||||
// Using the extra capacity without copying risks a data race that
|
|
||||||
// would violate the Logger interface contract.
|
|
||||||
n := len(l.keyvals) + len(keyvals)
|
|
||||||
if len(keyvals)%2 != 0 {
|
|
||||||
n++
|
|
||||||
}
|
|
||||||
kvs := make([]interface{}, 0, n)
|
|
||||||
kvs = append(kvs, keyvals...)
|
|
||||||
if len(kvs)%2 != 0 {
|
|
||||||
kvs = append(kvs, ErrMissingValue)
|
|
||||||
}
|
|
||||||
kvs = append(kvs, l.keyvals...)
|
|
||||||
return &context{
|
|
||||||
logger: l.logger,
|
|
||||||
keyvals: kvs,
|
|
||||||
hasValuer: l.hasValuer || containsValuer(keyvals),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// context is the Logger implementation returned by With and WithPrefix. It
|
|
||||||
// wraps a Logger and holds keyvals that it includes in all log events. Its
|
|
||||||
// Log method calls bindValues to generate values for each Valuer in the
|
|
||||||
// context keyvals.
|
|
||||||
//
|
|
||||||
// A context must always have the same number of stack frames between calls to
|
|
||||||
// its Log method and the eventual binding of Valuers to their value. This
|
|
||||||
// requirement comes from the functional requirement to allow a context to
|
|
||||||
// resolve application call site information for a Caller stored in the
|
|
||||||
// context. To do this we must be able to predict the number of logging
|
|
||||||
// functions on the stack when bindValues is called.
|
|
||||||
//
|
|
||||||
// Two implementation details provide the needed stack depth consistency.
|
|
||||||
//
|
|
||||||
// 1. newContext avoids introducing an additional layer when asked to
|
|
||||||
// wrap another context.
|
|
||||||
// 2. With and WithPrefix avoid introducing an additional layer by
|
|
||||||
// returning a newly constructed context with a merged keyvals rather
|
|
||||||
// than simply wrapping the existing context.
|
|
||||||
type context struct {
|
|
||||||
logger Logger
|
|
||||||
keyvals []interface{}
|
|
||||||
hasValuer bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func newContext(logger Logger) *context {
|
|
||||||
if c, ok := logger.(*context); ok {
|
|
||||||
return c
|
|
||||||
}
|
|
||||||
return &context{logger: logger}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log replaces all value elements (odd indexes) containing a Valuer in the
|
|
||||||
// stored context with their generated value, appends keyvals, and passes the
|
|
||||||
// result to the wrapped Logger.
|
|
||||||
func (l *context) Log(keyvals ...interface{}) error {
|
|
||||||
kvs := append(l.keyvals, keyvals...)
|
|
||||||
if len(kvs)%2 != 0 {
|
|
||||||
kvs = append(kvs, ErrMissingValue)
|
|
||||||
}
|
|
||||||
if l.hasValuer {
|
|
||||||
// If no keyvals were appended above then we must copy l.keyvals so
|
|
||||||
// that future log events will reevaluate the stored Valuers.
|
|
||||||
if len(keyvals) == 0 {
|
|
||||||
kvs = append([]interface{}{}, l.keyvals...)
|
|
||||||
}
|
|
||||||
bindValues(kvs[:len(l.keyvals)])
|
|
||||||
}
|
|
||||||
return l.logger.Log(kvs...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoggerFunc is an adapter to allow use of ordinary functions as Loggers. If
|
|
||||||
// f is a function with the appropriate signature, LoggerFunc(f) is a Logger
|
|
||||||
// object that calls f.
|
|
||||||
type LoggerFunc func(...interface{}) error
|
|
||||||
|
|
||||||
// Log implements Logger by calling f(keyvals...).
|
|
||||||
func (f LoggerFunc) Log(keyvals ...interface{}) error {
|
|
||||||
return f(keyvals...)
|
|
||||||
}
|
|
62
vendor/github.com/go-kit/kit/log/logfmt_logger.go
generated
vendored
62
vendor/github.com/go-kit/kit/log/logfmt_logger.go
generated
vendored
|
@ -1,62 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"io"
|
|
||||||
"sync"
|
|
||||||
|
|
||||||
"github.com/go-logfmt/logfmt"
|
|
||||||
)
|
|
||||||
|
|
||||||
type logfmtEncoder struct {
|
|
||||||
*logfmt.Encoder
|
|
||||||
buf bytes.Buffer
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l *logfmtEncoder) Reset() {
|
|
||||||
l.Encoder.Reset()
|
|
||||||
l.buf.Reset()
|
|
||||||
}
|
|
||||||
|
|
||||||
var logfmtEncoderPool = sync.Pool{
|
|
||||||
New: func() interface{} {
|
|
||||||
var enc logfmtEncoder
|
|
||||||
enc.Encoder = logfmt.NewEncoder(&enc.buf)
|
|
||||||
return &enc
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
type logfmtLogger struct {
|
|
||||||
w io.Writer
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewLogfmtLogger returns a logger that encodes keyvals to the Writer in
|
|
||||||
// logfmt format. Each log event produces no more than one call to w.Write.
|
|
||||||
// The passed Writer must be safe for concurrent use by multiple goroutines if
|
|
||||||
// the returned Logger will be used concurrently.
|
|
||||||
func NewLogfmtLogger(w io.Writer) Logger {
|
|
||||||
return &logfmtLogger{w}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (l logfmtLogger) Log(keyvals ...interface{}) error {
|
|
||||||
enc := logfmtEncoderPool.Get().(*logfmtEncoder)
|
|
||||||
enc.Reset()
|
|
||||||
defer logfmtEncoderPool.Put(enc)
|
|
||||||
|
|
||||||
if err := enc.EncodeKeyvals(keyvals...); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add newline to the end of the buffer
|
|
||||||
if err := enc.EndRecord(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// The Logger interface requires implementations to be safe for concurrent
|
|
||||||
// use by multiple goroutines. For this implementation that means making
|
|
||||||
// only one call to l.w.Write() for each call to Log.
|
|
||||||
if _, err := l.w.Write(enc.buf.Bytes()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
8
vendor/github.com/go-kit/kit/log/nop_logger.go
generated
vendored
8
vendor/github.com/go-kit/kit/log/nop_logger.go
generated
vendored
|
@ -1,8 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
type nopLogger struct{}
|
|
||||||
|
|
||||||
// NewNopLogger returns a logger that doesn't do anything.
|
|
||||||
func NewNopLogger() Logger { return nopLogger{} }
|
|
||||||
|
|
||||||
func (nopLogger) Log(...interface{}) error { return nil }
|
|
116
vendor/github.com/go-kit/kit/log/stdlib.go
generated
vendored
116
vendor/github.com/go-kit/kit/log/stdlib.go
generated
vendored
|
@ -1,116 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import (
|
|
||||||
"io"
|
|
||||||
"log"
|
|
||||||
"regexp"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
// StdlibWriter implements io.Writer by invoking the stdlib log.Print. It's
|
|
||||||
// designed to be passed to a Go kit logger as the writer, for cases where
|
|
||||||
// it's necessary to redirect all Go kit log output to the stdlib logger.
|
|
||||||
//
|
|
||||||
// If you have any choice in the matter, you shouldn't use this. Prefer to
|
|
||||||
// redirect the stdlib log to the Go kit logger via NewStdlibAdapter.
|
|
||||||
type StdlibWriter struct{}
|
|
||||||
|
|
||||||
// Write implements io.Writer.
|
|
||||||
func (w StdlibWriter) Write(p []byte) (int, error) {
|
|
||||||
log.Print(strings.TrimSpace(string(p)))
|
|
||||||
return len(p), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// StdlibAdapter wraps a Logger and allows it to be passed to the stdlib
|
|
||||||
// logger's SetOutput. It will extract date/timestamps, filenames, and
|
|
||||||
// messages, and place them under relevant keys.
|
|
||||||
type StdlibAdapter struct {
|
|
||||||
Logger
|
|
||||||
timestampKey string
|
|
||||||
fileKey string
|
|
||||||
messageKey string
|
|
||||||
}
|
|
||||||
|
|
||||||
// StdlibAdapterOption sets a parameter for the StdlibAdapter.
|
|
||||||
type StdlibAdapterOption func(*StdlibAdapter)
|
|
||||||
|
|
||||||
// TimestampKey sets the key for the timestamp field. By default, it's "ts".
|
|
||||||
func TimestampKey(key string) StdlibAdapterOption {
|
|
||||||
return func(a *StdlibAdapter) { a.timestampKey = key }
|
|
||||||
}
|
|
||||||
|
|
||||||
// FileKey sets the key for the file and line field. By default, it's "caller".
|
|
||||||
func FileKey(key string) StdlibAdapterOption {
|
|
||||||
return func(a *StdlibAdapter) { a.fileKey = key }
|
|
||||||
}
|
|
||||||
|
|
||||||
// MessageKey sets the key for the actual log message. By default, it's "msg".
|
|
||||||
func MessageKey(key string) StdlibAdapterOption {
|
|
||||||
return func(a *StdlibAdapter) { a.messageKey = key }
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewStdlibAdapter returns a new StdlibAdapter wrapper around the passed
|
|
||||||
// logger. It's designed to be passed to log.SetOutput.
|
|
||||||
func NewStdlibAdapter(logger Logger, options ...StdlibAdapterOption) io.Writer {
|
|
||||||
a := StdlibAdapter{
|
|
||||||
Logger: logger,
|
|
||||||
timestampKey: "ts",
|
|
||||||
fileKey: "caller",
|
|
||||||
messageKey: "msg",
|
|
||||||
}
|
|
||||||
for _, option := range options {
|
|
||||||
option(&a)
|
|
||||||
}
|
|
||||||
return a
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a StdlibAdapter) Write(p []byte) (int, error) {
|
|
||||||
result := subexps(p)
|
|
||||||
keyvals := []interface{}{}
|
|
||||||
var timestamp string
|
|
||||||
if date, ok := result["date"]; ok && date != "" {
|
|
||||||
timestamp = date
|
|
||||||
}
|
|
||||||
if time, ok := result["time"]; ok && time != "" {
|
|
||||||
if timestamp != "" {
|
|
||||||
timestamp += " "
|
|
||||||
}
|
|
||||||
timestamp += time
|
|
||||||
}
|
|
||||||
if timestamp != "" {
|
|
||||||
keyvals = append(keyvals, a.timestampKey, timestamp)
|
|
||||||
}
|
|
||||||
if file, ok := result["file"]; ok && file != "" {
|
|
||||||
keyvals = append(keyvals, a.fileKey, file)
|
|
||||||
}
|
|
||||||
if msg, ok := result["msg"]; ok {
|
|
||||||
keyvals = append(keyvals, a.messageKey, msg)
|
|
||||||
}
|
|
||||||
if err := a.Logger.Log(keyvals...); err != nil {
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
return len(p), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
logRegexpDate = `(?P<date>[0-9]{4}/[0-9]{2}/[0-9]{2})?[ ]?`
|
|
||||||
logRegexpTime = `(?P<time>[0-9]{2}:[0-9]{2}:[0-9]{2}(\.[0-9]+)?)?[ ]?`
|
|
||||||
logRegexpFile = `(?P<file>.+?:[0-9]+)?`
|
|
||||||
logRegexpMsg = `(: )?(?P<msg>.*)`
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
logRegexp = regexp.MustCompile(logRegexpDate + logRegexpTime + logRegexpFile + logRegexpMsg)
|
|
||||||
)
|
|
||||||
|
|
||||||
func subexps(line []byte) map[string]string {
|
|
||||||
m := logRegexp.FindSubmatch(line)
|
|
||||||
if len(m) < len(logRegexp.SubexpNames()) {
|
|
||||||
return map[string]string{}
|
|
||||||
}
|
|
||||||
result := map[string]string{}
|
|
||||||
for i, name := range logRegexp.SubexpNames() {
|
|
||||||
result[name] = string(m[i])
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
116
vendor/github.com/go-kit/kit/log/sync.go
generated
vendored
116
vendor/github.com/go-kit/kit/log/sync.go
generated
vendored
|
@ -1,116 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import (
|
|
||||||
"io"
|
|
||||||
"sync"
|
|
||||||
"sync/atomic"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SwapLogger wraps another logger that may be safely replaced while other
|
|
||||||
// goroutines use the SwapLogger concurrently. The zero value for a SwapLogger
|
|
||||||
// will discard all log events without error.
|
|
||||||
//
|
|
||||||
// SwapLogger serves well as a package global logger that can be changed by
|
|
||||||
// importers.
|
|
||||||
type SwapLogger struct {
|
|
||||||
logger atomic.Value
|
|
||||||
}
|
|
||||||
|
|
||||||
type loggerStruct struct {
|
|
||||||
Logger
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log implements the Logger interface by forwarding keyvals to the currently
|
|
||||||
// wrapped logger. It does not log anything if the wrapped logger is nil.
|
|
||||||
func (l *SwapLogger) Log(keyvals ...interface{}) error {
|
|
||||||
s, ok := l.logger.Load().(loggerStruct)
|
|
||||||
if !ok || s.Logger == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return s.Log(keyvals...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Swap replaces the currently wrapped logger with logger. Swap may be called
|
|
||||||
// concurrently with calls to Log from other goroutines.
|
|
||||||
func (l *SwapLogger) Swap(logger Logger) {
|
|
||||||
l.logger.Store(loggerStruct{logger})
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSyncWriter returns a new writer that is safe for concurrent use by
|
|
||||||
// multiple goroutines. Writes to the returned writer are passed on to w. If
|
|
||||||
// another write is already in progress, the calling goroutine blocks until
|
|
||||||
// the writer is available.
|
|
||||||
//
|
|
||||||
// If w implements the following interface, so does the returned writer.
|
|
||||||
//
|
|
||||||
// interface {
|
|
||||||
// Fd() uintptr
|
|
||||||
// }
|
|
||||||
func NewSyncWriter(w io.Writer) io.Writer {
|
|
||||||
switch w := w.(type) {
|
|
||||||
case fdWriter:
|
|
||||||
return &fdSyncWriter{fdWriter: w}
|
|
||||||
default:
|
|
||||||
return &syncWriter{Writer: w}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// syncWriter synchronizes concurrent writes to an io.Writer.
|
|
||||||
type syncWriter struct {
|
|
||||||
sync.Mutex
|
|
||||||
io.Writer
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write writes p to the underlying io.Writer. If another write is already in
|
|
||||||
// progress, the calling goroutine blocks until the syncWriter is available.
|
|
||||||
func (w *syncWriter) Write(p []byte) (n int, err error) {
|
|
||||||
w.Lock()
|
|
||||||
n, err = w.Writer.Write(p)
|
|
||||||
w.Unlock()
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// fdWriter is an io.Writer that also has an Fd method. The most common
|
|
||||||
// example of an fdWriter is an *os.File.
|
|
||||||
type fdWriter interface {
|
|
||||||
io.Writer
|
|
||||||
Fd() uintptr
|
|
||||||
}
|
|
||||||
|
|
||||||
// fdSyncWriter synchronizes concurrent writes to an fdWriter.
|
|
||||||
type fdSyncWriter struct {
|
|
||||||
sync.Mutex
|
|
||||||
fdWriter
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write writes p to the underlying io.Writer. If another write is already in
|
|
||||||
// progress, the calling goroutine blocks until the fdSyncWriter is available.
|
|
||||||
func (w *fdSyncWriter) Write(p []byte) (n int, err error) {
|
|
||||||
w.Lock()
|
|
||||||
n, err = w.fdWriter.Write(p)
|
|
||||||
w.Unlock()
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// syncLogger provides concurrent safe logging for another Logger.
|
|
||||||
type syncLogger struct {
|
|
||||||
mu sync.Mutex
|
|
||||||
logger Logger
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewSyncLogger returns a logger that synchronizes concurrent use of the
|
|
||||||
// wrapped logger. When multiple goroutines use the SyncLogger concurrently
|
|
||||||
// only one goroutine will be allowed to log to the wrapped logger at a time.
|
|
||||||
// The other goroutines will block until the logger is available.
|
|
||||||
func NewSyncLogger(logger Logger) Logger {
|
|
||||||
return &syncLogger{logger: logger}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Log logs keyvals to the underlying Logger. If another log is already in
|
|
||||||
// progress, the calling goroutine blocks until the syncLogger is available.
|
|
||||||
func (l *syncLogger) Log(keyvals ...interface{}) error {
|
|
||||||
l.mu.Lock()
|
|
||||||
err := l.logger.Log(keyvals...)
|
|
||||||
l.mu.Unlock()
|
|
||||||
return err
|
|
||||||
}
|
|
110
vendor/github.com/go-kit/kit/log/value.go
generated
vendored
110
vendor/github.com/go-kit/kit/log/value.go
generated
vendored
|
@ -1,110 +0,0 @@
|
||||||
package log
|
|
||||||
|
|
||||||
import (
|
|
||||||
"runtime"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// A Valuer generates a log value. When passed to With or WithPrefix in a
|
|
||||||
// value element (odd indexes), it represents a dynamic value which is re-
|
|
||||||
// evaluated with each log event.
|
|
||||||
type Valuer func() interface{}
|
|
||||||
|
|
||||||
// bindValues replaces all value elements (odd indexes) containing a Valuer
|
|
||||||
// with their generated value.
|
|
||||||
func bindValues(keyvals []interface{}) {
|
|
||||||
for i := 1; i < len(keyvals); i += 2 {
|
|
||||||
if v, ok := keyvals[i].(Valuer); ok {
|
|
||||||
keyvals[i] = v()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// containsValuer returns true if any of the value elements (odd indexes)
|
|
||||||
// contain a Valuer.
|
|
||||||
func containsValuer(keyvals []interface{}) bool {
|
|
||||||
for i := 1; i < len(keyvals); i += 2 {
|
|
||||||
if _, ok := keyvals[i].(Valuer); ok {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Timestamp returns a timestamp Valuer. It invokes the t function to get the
|
|
||||||
// time; unless you are doing something tricky, pass time.Now.
|
|
||||||
//
|
|
||||||
// Most users will want to use DefaultTimestamp or DefaultTimestampUTC, which
|
|
||||||
// are TimestampFormats that use the RFC3339Nano format.
|
|
||||||
func Timestamp(t func() time.Time) Valuer {
|
|
||||||
return func() interface{} { return t() }
|
|
||||||
}
|
|
||||||
|
|
||||||
// TimestampFormat returns a timestamp Valuer with a custom time format. It
|
|
||||||
// invokes the t function to get the time to format; unless you are doing
|
|
||||||
// something tricky, pass time.Now. The layout string is passed to
|
|
||||||
// Time.Format.
|
|
||||||
//
|
|
||||||
// Most users will want to use DefaultTimestamp or DefaultTimestampUTC, which
|
|
||||||
// are TimestampFormats that use the RFC3339Nano format.
|
|
||||||
func TimestampFormat(t func() time.Time, layout string) Valuer {
|
|
||||||
return func() interface{} {
|
|
||||||
return timeFormat{
|
|
||||||
time: t(),
|
|
||||||
layout: layout,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// A timeFormat represents an instant in time and a layout used when
|
|
||||||
// marshaling to a text format.
|
|
||||||
type timeFormat struct {
|
|
||||||
time time.Time
|
|
||||||
layout string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tf timeFormat) String() string {
|
|
||||||
return tf.time.Format(tf.layout)
|
|
||||||
}
|
|
||||||
|
|
||||||
// MarshalText implements encoding.TextMarshaller.
|
|
||||||
func (tf timeFormat) MarshalText() (text []byte, err error) {
|
|
||||||
// The following code adapted from the standard library time.Time.Format
|
|
||||||
// method. Using the same undocumented magic constant to extend the size
|
|
||||||
// of the buffer as seen there.
|
|
||||||
b := make([]byte, 0, len(tf.layout)+10)
|
|
||||||
b = tf.time.AppendFormat(b, tf.layout)
|
|
||||||
return b, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Caller returns a Valuer that returns a file and line from a specified depth
|
|
||||||
// in the callstack. Users will probably want to use DefaultCaller.
|
|
||||||
func Caller(depth int) Valuer {
|
|
||||||
return func() interface{} {
|
|
||||||
_, file, line, _ := runtime.Caller(depth)
|
|
||||||
idx := strings.LastIndexByte(file, '/')
|
|
||||||
// using idx+1 below handles both of following cases:
|
|
||||||
// idx == -1 because no "/" was found, or
|
|
||||||
// idx >= 0 and we want to start at the character after the found "/".
|
|
||||||
return file[idx+1:] + ":" + strconv.Itoa(line)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
// DefaultTimestamp is a Valuer that returns the current wallclock time,
|
|
||||||
// respecting time zones, when bound.
|
|
||||||
DefaultTimestamp = TimestampFormat(time.Now, time.RFC3339Nano)
|
|
||||||
|
|
||||||
// DefaultTimestampUTC is a Valuer that returns the current time in UTC
|
|
||||||
// when bound.
|
|
||||||
DefaultTimestampUTC = TimestampFormat(
|
|
||||||
func() time.Time { return time.Now().UTC() },
|
|
||||||
time.RFC3339Nano,
|
|
||||||
)
|
|
||||||
|
|
||||||
// DefaultCaller is a Valuer that returns the file and line where the Log
|
|
||||||
// method was invoked. It can only be used with log.With.
|
|
||||||
DefaultCaller = Caller(3)
|
|
||||||
)
|
|
1
vendor/github.com/go-logfmt/logfmt/.gitignore
generated
vendored
1
vendor/github.com/go-logfmt/logfmt/.gitignore
generated
vendored
|
@ -1 +0,0 @@
|
||||||
.vscode/
|
|
18
vendor/github.com/go-logfmt/logfmt/.travis.yml
generated
vendored
18
vendor/github.com/go-logfmt/logfmt/.travis.yml
generated
vendored
|
@ -1,18 +0,0 @@
|
||||||
language: go
|
|
||||||
sudo: false
|
|
||||||
go:
|
|
||||||
- "1.7.x"
|
|
||||||
- "1.8.x"
|
|
||||||
- "1.9.x"
|
|
||||||
- "1.10.x"
|
|
||||||
- "1.11.x"
|
|
||||||
- "1.12.x"
|
|
||||||
- "1.13.x"
|
|
||||||
- "tip"
|
|
||||||
|
|
||||||
before_install:
|
|
||||||
- go get github.com/mattn/goveralls
|
|
||||||
- go get golang.org/x/tools/cmd/cover
|
|
||||||
|
|
||||||
script:
|
|
||||||
- goveralls -service=travis-ci
|
|
48
vendor/github.com/go-logfmt/logfmt/CHANGELOG.md
generated
vendored
48
vendor/github.com/go-logfmt/logfmt/CHANGELOG.md
generated
vendored
|
@ -1,48 +0,0 @@
|
||||||
# Changelog
|
|
||||||
All notable changes to this project will be documented in this file.
|
|
||||||
|
|
||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
|
||||||
|
|
||||||
## [0.5.0] - 2020-01-03
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
- Remove the dependency on github.com/kr/logfmt by [@ChrisHines]
|
|
||||||
- Move fuzz code to github.com/go-logfmt/fuzzlogfmt by [@ChrisHines]
|
|
||||||
|
|
||||||
## [0.4.0] - 2018-11-21
|
|
||||||
|
|
||||||
### Added
|
|
||||||
- Go module support by [@ChrisHines]
|
|
||||||
- CHANGELOG by [@ChrisHines]
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
- Drop invalid runes from keys instead of returning ErrInvalidKey by [@ChrisHines]
|
|
||||||
- On panic while printing, attempt to print panic value by [@bboreham]
|
|
||||||
|
|
||||||
## [0.3.0] - 2016-11-15
|
|
||||||
### Added
|
|
||||||
- Pool buffers for quoted strings and byte slices by [@nussjustin]
|
|
||||||
### Fixed
|
|
||||||
- Fuzz fix, quote invalid UTF-8 values by [@judwhite]
|
|
||||||
|
|
||||||
## [0.2.0] - 2016-05-08
|
|
||||||
### Added
|
|
||||||
- Encoder.EncodeKeyvals by [@ChrisHines]
|
|
||||||
|
|
||||||
## [0.1.0] - 2016-03-28
|
|
||||||
### Added
|
|
||||||
- Encoder by [@ChrisHines]
|
|
||||||
- Decoder by [@ChrisHines]
|
|
||||||
- MarshalKeyvals by [@ChrisHines]
|
|
||||||
|
|
||||||
[0.5.0]: https://github.com/go-logfmt/logfmt/compare/v0.4.0...v0.5.0
|
|
||||||
[0.4.0]: https://github.com/go-logfmt/logfmt/compare/v0.3.0...v0.4.0
|
|
||||||
[0.3.0]: https://github.com/go-logfmt/logfmt/compare/v0.2.0...v0.3.0
|
|
||||||
[0.2.0]: https://github.com/go-logfmt/logfmt/compare/v0.1.0...v0.2.0
|
|
||||||
[0.1.0]: https://github.com/go-logfmt/logfmt/commits/v0.1.0
|
|
||||||
|
|
||||||
[@ChrisHines]: https://github.com/ChrisHines
|
|
||||||
[@bboreham]: https://github.com/bboreham
|
|
||||||
[@judwhite]: https://github.com/judwhite
|
|
||||||
[@nussjustin]: https://github.com/nussjustin
|
|
22
vendor/github.com/go-logfmt/logfmt/LICENSE
generated
vendored
22
vendor/github.com/go-logfmt/logfmt/LICENSE
generated
vendored
|
@ -1,22 +0,0 @@
|
||||||
The MIT License (MIT)
|
|
||||||
|
|
||||||
Copyright (c) 2015 go-logfmt
|
|
||||||
|
|
||||||
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.
|
|
||||||
|
|
33
vendor/github.com/go-logfmt/logfmt/README.md
generated
vendored
33
vendor/github.com/go-logfmt/logfmt/README.md
generated
vendored
|
@ -1,33 +0,0 @@
|
||||||
[![GoDoc](https://godoc.org/github.com/go-logfmt/logfmt?status.svg)](https://godoc.org/github.com/go-logfmt/logfmt)
|
|
||||||
[![Go Report Card](https://goreportcard.com/badge/go-logfmt/logfmt)](https://goreportcard.com/report/go-logfmt/logfmt)
|
|
||||||
[![TravisCI](https://travis-ci.org/go-logfmt/logfmt.svg?branch=master)](https://travis-ci.org/go-logfmt/logfmt)
|
|
||||||
[![Coverage Status](https://coveralls.io/repos/github/go-logfmt/logfmt/badge.svg?branch=master)](https://coveralls.io/github/go-logfmt/logfmt?branch=master)
|
|
||||||
|
|
||||||
# logfmt
|
|
||||||
|
|
||||||
Package logfmt implements utilities to marshal and unmarshal data in the [logfmt
|
|
||||||
format](https://brandur.org/logfmt). It provides an API similar to
|
|
||||||
[encoding/json](http://golang.org/pkg/encoding/json/) and
|
|
||||||
[encoding/xml](http://golang.org/pkg/encoding/xml/).
|
|
||||||
|
|
||||||
The logfmt format was first documented by Brandur Leach in [this
|
|
||||||
article](https://brandur.org/logfmt). The format has not been formally
|
|
||||||
standardized. The most authoritative public specification to date has been the
|
|
||||||
documentation of a Go Language [package](http://godoc.org/github.com/kr/logfmt)
|
|
||||||
written by Blake Mizerany and Keith Rarick.
|
|
||||||
|
|
||||||
## Goals
|
|
||||||
|
|
||||||
This project attempts to conform as closely as possible to the prior art, while
|
|
||||||
also removing ambiguity where necessary to provide well behaved encoder and
|
|
||||||
decoder implementations.
|
|
||||||
|
|
||||||
## Non-goals
|
|
||||||
|
|
||||||
This project does not attempt to formally standardize the logfmt format. In the
|
|
||||||
event that logfmt is standardized this project would take conforming to the
|
|
||||||
standard as a goal.
|
|
||||||
|
|
||||||
## Versioning
|
|
||||||
|
|
||||||
Package logfmt publishes releases via [semver](http://semver.org/) compatible Git tags prefixed with a single 'v'.
|
|
237
vendor/github.com/go-logfmt/logfmt/decode.go
generated
vendored
237
vendor/github.com/go-logfmt/logfmt/decode.go
generated
vendored
|
@ -1,237 +0,0 @@
|
||||||
package logfmt
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"unicode/utf8"
|
|
||||||
)
|
|
||||||
|
|
||||||
// A Decoder reads and decodes logfmt records from an input stream.
|
|
||||||
type Decoder struct {
|
|
||||||
pos int
|
|
||||||
key []byte
|
|
||||||
value []byte
|
|
||||||
lineNum int
|
|
||||||
s *bufio.Scanner
|
|
||||||
err error
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewDecoder returns a new decoder that reads from r.
|
|
||||||
//
|
|
||||||
// The decoder introduces its own buffering and may read data from r beyond
|
|
||||||
// the logfmt records requested.
|
|
||||||
func NewDecoder(r io.Reader) *Decoder {
|
|
||||||
dec := &Decoder{
|
|
||||||
s: bufio.NewScanner(r),
|
|
||||||
}
|
|
||||||
return dec
|
|
||||||
}
|
|
||||||
|
|
||||||
// ScanRecord advances the Decoder to the next record, which can then be
|
|
||||||
// parsed with the ScanKeyval method. It returns false when decoding stops,
|
|
||||||
// either by reaching the end of the input or an error. After ScanRecord
|
|
||||||
// returns false, the Err method will return any error that occurred during
|
|
||||||
// decoding, except that if it was io.EOF, Err will return nil.
|
|
||||||
func (dec *Decoder) ScanRecord() bool {
|
|
||||||
if dec.err != nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if !dec.s.Scan() {
|
|
||||||
dec.err = dec.s.Err()
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
dec.lineNum++
|
|
||||||
dec.pos = 0
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// ScanKeyval advances the Decoder to the next key/value pair of the current
|
|
||||||
// record, which can then be retrieved with the Key and Value methods. It
|
|
||||||
// returns false when decoding stops, either by reaching the end of the
|
|
||||||
// current record or an error.
|
|
||||||
func (dec *Decoder) ScanKeyval() bool {
|
|
||||||
dec.key, dec.value = nil, nil
|
|
||||||
if dec.err != nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
line := dec.s.Bytes()
|
|
||||||
|
|
||||||
// garbage
|
|
||||||
for p, c := range line[dec.pos:] {
|
|
||||||
if c > ' ' {
|
|
||||||
dec.pos += p
|
|
||||||
goto key
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dec.pos = len(line)
|
|
||||||
return false
|
|
||||||
|
|
||||||
key:
|
|
||||||
const invalidKeyError = "invalid key"
|
|
||||||
|
|
||||||
start, multibyte := dec.pos, false
|
|
||||||
for p, c := range line[dec.pos:] {
|
|
||||||
switch {
|
|
||||||
case c == '=':
|
|
||||||
dec.pos += p
|
|
||||||
if dec.pos > start {
|
|
||||||
dec.key = line[start:dec.pos]
|
|
||||||
if multibyte && bytes.ContainsRune(dec.key, utf8.RuneError) {
|
|
||||||
dec.syntaxError(invalidKeyError)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if dec.key == nil {
|
|
||||||
dec.unexpectedByte(c)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
goto equal
|
|
||||||
case c == '"':
|
|
||||||
dec.pos += p
|
|
||||||
dec.unexpectedByte(c)
|
|
||||||
return false
|
|
||||||
case c <= ' ':
|
|
||||||
dec.pos += p
|
|
||||||
if dec.pos > start {
|
|
||||||
dec.key = line[start:dec.pos]
|
|
||||||
if multibyte && bytes.ContainsRune(dec.key, utf8.RuneError) {
|
|
||||||
dec.syntaxError(invalidKeyError)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
case c >= utf8.RuneSelf:
|
|
||||||
multibyte = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dec.pos = len(line)
|
|
||||||
if dec.pos > start {
|
|
||||||
dec.key = line[start:dec.pos]
|
|
||||||
if multibyte && bytes.ContainsRune(dec.key, utf8.RuneError) {
|
|
||||||
dec.syntaxError(invalidKeyError)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
|
|
||||||
equal:
|
|
||||||
dec.pos++
|
|
||||||
if dec.pos >= len(line) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
switch c := line[dec.pos]; {
|
|
||||||
case c <= ' ':
|
|
||||||
return true
|
|
||||||
case c == '"':
|
|
||||||
goto qvalue
|
|
||||||
}
|
|
||||||
|
|
||||||
// value
|
|
||||||
start = dec.pos
|
|
||||||
for p, c := range line[dec.pos:] {
|
|
||||||
switch {
|
|
||||||
case c == '=' || c == '"':
|
|
||||||
dec.pos += p
|
|
||||||
dec.unexpectedByte(c)
|
|
||||||
return false
|
|
||||||
case c <= ' ':
|
|
||||||
dec.pos += p
|
|
||||||
if dec.pos > start {
|
|
||||||
dec.value = line[start:dec.pos]
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dec.pos = len(line)
|
|
||||||
if dec.pos > start {
|
|
||||||
dec.value = line[start:dec.pos]
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
|
|
||||||
qvalue:
|
|
||||||
const (
|
|
||||||
untermQuote = "unterminated quoted value"
|
|
||||||
invalidQuote = "invalid quoted value"
|
|
||||||
)
|
|
||||||
|
|
||||||
hasEsc, esc := false, false
|
|
||||||
start = dec.pos
|
|
||||||
for p, c := range line[dec.pos+1:] {
|
|
||||||
switch {
|
|
||||||
case esc:
|
|
||||||
esc = false
|
|
||||||
case c == '\\':
|
|
||||||
hasEsc, esc = true, true
|
|
||||||
case c == '"':
|
|
||||||
dec.pos += p + 2
|
|
||||||
if hasEsc {
|
|
||||||
v, ok := unquoteBytes(line[start:dec.pos])
|
|
||||||
if !ok {
|
|
||||||
dec.syntaxError(invalidQuote)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
dec.value = v
|
|
||||||
} else {
|
|
||||||
start++
|
|
||||||
end := dec.pos - 1
|
|
||||||
if end > start {
|
|
||||||
dec.value = line[start:end]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
dec.pos = len(line)
|
|
||||||
dec.syntaxError(untermQuote)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Key returns the most recent key found by a call to ScanKeyval. The returned
|
|
||||||
// slice may point to internal buffers and is only valid until the next call
|
|
||||||
// to ScanRecord. It does no allocation.
|
|
||||||
func (dec *Decoder) Key() []byte {
|
|
||||||
return dec.key
|
|
||||||
}
|
|
||||||
|
|
||||||
// Value returns the most recent value found by a call to ScanKeyval. The
|
|
||||||
// returned slice may point to internal buffers and is only valid until the
|
|
||||||
// next call to ScanRecord. It does no allocation when the value has no
|
|
||||||
// escape sequences.
|
|
||||||
func (dec *Decoder) Value() []byte {
|
|
||||||
return dec.value
|
|
||||||
}
|
|
||||||
|
|
||||||
// Err returns the first non-EOF error that was encountered by the Scanner.
|
|
||||||
func (dec *Decoder) Err() error {
|
|
||||||
return dec.err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dec *Decoder) syntaxError(msg string) {
|
|
||||||
dec.err = &SyntaxError{
|
|
||||||
Msg: msg,
|
|
||||||
Line: dec.lineNum,
|
|
||||||
Pos: dec.pos + 1,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dec *Decoder) unexpectedByte(c byte) {
|
|
||||||
dec.err = &SyntaxError{
|
|
||||||
Msg: fmt.Sprintf("unexpected %q", c),
|
|
||||||
Line: dec.lineNum,
|
|
||||||
Pos: dec.pos + 1,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// A SyntaxError represents a syntax error in the logfmt input stream.
|
|
||||||
type SyntaxError struct {
|
|
||||||
Msg string
|
|
||||||
Line int
|
|
||||||
Pos int
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *SyntaxError) Error() string {
|
|
||||||
return fmt.Sprintf("logfmt syntax error at pos %d on line %d: %s", e.Pos, e.Line, e.Msg)
|
|
||||||
}
|
|
6
vendor/github.com/go-logfmt/logfmt/doc.go
generated
vendored
6
vendor/github.com/go-logfmt/logfmt/doc.go
generated
vendored
|
@ -1,6 +0,0 @@
|
||||||
// Package logfmt implements utilities to marshal and unmarshal data in the
|
|
||||||
// logfmt format. The logfmt format records key/value pairs in a way that
|
|
||||||
// balances readability for humans and simplicity of computer parsing. It is
|
|
||||||
// most commonly used as a more human friendly alternative to JSON for
|
|
||||||
// structured logging.
|
|
||||||
package logfmt
|
|
322
vendor/github.com/go-logfmt/logfmt/encode.go
generated
vendored
322
vendor/github.com/go-logfmt/logfmt/encode.go
generated
vendored
|
@ -1,322 +0,0 @@
|
||||||
package logfmt
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"encoding"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"unicode/utf8"
|
|
||||||
)
|
|
||||||
|
|
||||||
// MarshalKeyvals returns the logfmt encoding of keyvals, a variadic sequence
|
|
||||||
// of alternating keys and values.
|
|
||||||
func MarshalKeyvals(keyvals ...interface{}) ([]byte, error) {
|
|
||||||
buf := &bytes.Buffer{}
|
|
||||||
if err := NewEncoder(buf).EncodeKeyvals(keyvals...); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return buf.Bytes(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// An Encoder writes logfmt data to an output stream.
|
|
||||||
type Encoder struct {
|
|
||||||
w io.Writer
|
|
||||||
scratch bytes.Buffer
|
|
||||||
needSep bool
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewEncoder returns a new encoder that writes to w.
|
|
||||||
func NewEncoder(w io.Writer) *Encoder {
|
|
||||||
return &Encoder{
|
|
||||||
w: w,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
space = []byte(" ")
|
|
||||||
equals = []byte("=")
|
|
||||||
newline = []byte("\n")
|
|
||||||
null = []byte("null")
|
|
||||||
)
|
|
||||||
|
|
||||||
// EncodeKeyval writes the logfmt encoding of key and value to the stream. A
|
|
||||||
// single space is written before the second and subsequent keys in a record.
|
|
||||||
// Nothing is written if a non-nil error is returned.
|
|
||||||
func (enc *Encoder) EncodeKeyval(key, value interface{}) error {
|
|
||||||
enc.scratch.Reset()
|
|
||||||
if enc.needSep {
|
|
||||||
if _, err := enc.scratch.Write(space); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if err := writeKey(&enc.scratch, key); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if _, err := enc.scratch.Write(equals); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := writeValue(&enc.scratch, value); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, err := enc.w.Write(enc.scratch.Bytes())
|
|
||||||
enc.needSep = true
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// EncodeKeyvals writes the logfmt encoding of keyvals to the stream. Keyvals
|
|
||||||
// is a variadic sequence of alternating keys and values. Keys of unsupported
|
|
||||||
// type are skipped along with their corresponding value. Values of
|
|
||||||
// unsupported type or that cause a MarshalerError are replaced by their error
|
|
||||||
// but do not cause EncodeKeyvals to return an error. If a non-nil error is
|
|
||||||
// returned some key/value pairs may not have be written.
|
|
||||||
func (enc *Encoder) EncodeKeyvals(keyvals ...interface{}) error {
|
|
||||||
if len(keyvals) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if len(keyvals)%2 == 1 {
|
|
||||||
keyvals = append(keyvals, nil)
|
|
||||||
}
|
|
||||||
for i := 0; i < len(keyvals); i += 2 {
|
|
||||||
k, v := keyvals[i], keyvals[i+1]
|
|
||||||
err := enc.EncodeKeyval(k, v)
|
|
||||||
if err == ErrUnsupportedKeyType {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if _, ok := err.(*MarshalerError); ok || err == ErrUnsupportedValueType {
|
|
||||||
v = err
|
|
||||||
err = enc.EncodeKeyval(k, v)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// MarshalerError represents an error encountered while marshaling a value.
|
|
||||||
type MarshalerError struct {
|
|
||||||
Type reflect.Type
|
|
||||||
Err error
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *MarshalerError) Error() string {
|
|
||||||
return "error marshaling value of type " + e.Type.String() + ": " + e.Err.Error()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrNilKey is returned by Marshal functions and Encoder methods if a key is
|
|
||||||
// a nil interface or pointer value.
|
|
||||||
var ErrNilKey = errors.New("nil key")
|
|
||||||
|
|
||||||
// ErrInvalidKey is returned by Marshal functions and Encoder methods if, after
|
|
||||||
// dropping invalid runes, a key is empty.
|
|
||||||
var ErrInvalidKey = errors.New("invalid key")
|
|
||||||
|
|
||||||
// ErrUnsupportedKeyType is returned by Encoder methods if a key has an
|
|
||||||
// unsupported type.
|
|
||||||
var ErrUnsupportedKeyType = errors.New("unsupported key type")
|
|
||||||
|
|
||||||
// ErrUnsupportedValueType is returned by Encoder methods if a value has an
|
|
||||||
// unsupported type.
|
|
||||||
var ErrUnsupportedValueType = errors.New("unsupported value type")
|
|
||||||
|
|
||||||
func writeKey(w io.Writer, key interface{}) error {
|
|
||||||
if key == nil {
|
|
||||||
return ErrNilKey
|
|
||||||
}
|
|
||||||
|
|
||||||
switch k := key.(type) {
|
|
||||||
case string:
|
|
||||||
return writeStringKey(w, k)
|
|
||||||
case []byte:
|
|
||||||
if k == nil {
|
|
||||||
return ErrNilKey
|
|
||||||
}
|
|
||||||
return writeBytesKey(w, k)
|
|
||||||
case encoding.TextMarshaler:
|
|
||||||
kb, err := safeMarshal(k)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if kb == nil {
|
|
||||||
return ErrNilKey
|
|
||||||
}
|
|
||||||
return writeBytesKey(w, kb)
|
|
||||||
case fmt.Stringer:
|
|
||||||
ks, ok := safeString(k)
|
|
||||||
if !ok {
|
|
||||||
return ErrNilKey
|
|
||||||
}
|
|
||||||
return writeStringKey(w, ks)
|
|
||||||
default:
|
|
||||||
rkey := reflect.ValueOf(key)
|
|
||||||
switch rkey.Kind() {
|
|
||||||
case reflect.Array, reflect.Chan, reflect.Func, reflect.Map, reflect.Slice, reflect.Struct:
|
|
||||||
return ErrUnsupportedKeyType
|
|
||||||
case reflect.Ptr:
|
|
||||||
if rkey.IsNil() {
|
|
||||||
return ErrNilKey
|
|
||||||
}
|
|
||||||
return writeKey(w, rkey.Elem().Interface())
|
|
||||||
}
|
|
||||||
return writeStringKey(w, fmt.Sprint(k))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// keyRuneFilter returns r for all valid key runes, and -1 for all invalid key
|
|
||||||
// runes. When used as the mapping function for strings.Map and bytes.Map
|
|
||||||
// functions it causes them to remove invalid key runes from strings or byte
|
|
||||||
// slices respectively.
|
|
||||||
func keyRuneFilter(r rune) rune {
|
|
||||||
if r <= ' ' || r == '=' || r == '"' || r == utf8.RuneError {
|
|
||||||
return -1
|
|
||||||
}
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeStringKey(w io.Writer, key string) error {
|
|
||||||
k := strings.Map(keyRuneFilter, key)
|
|
||||||
if k == "" {
|
|
||||||
return ErrInvalidKey
|
|
||||||
}
|
|
||||||
_, err := io.WriteString(w, k)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeBytesKey(w io.Writer, key []byte) error {
|
|
||||||
k := bytes.Map(keyRuneFilter, key)
|
|
||||||
if len(k) == 0 {
|
|
||||||
return ErrInvalidKey
|
|
||||||
}
|
|
||||||
_, err := w.Write(k)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeValue(w io.Writer, value interface{}) error {
|
|
||||||
switch v := value.(type) {
|
|
||||||
case nil:
|
|
||||||
return writeBytesValue(w, null)
|
|
||||||
case string:
|
|
||||||
return writeStringValue(w, v, true)
|
|
||||||
case []byte:
|
|
||||||
return writeBytesValue(w, v)
|
|
||||||
case encoding.TextMarshaler:
|
|
||||||
vb, err := safeMarshal(v)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if vb == nil {
|
|
||||||
vb = null
|
|
||||||
}
|
|
||||||
return writeBytesValue(w, vb)
|
|
||||||
case error:
|
|
||||||
se, ok := safeError(v)
|
|
||||||
return writeStringValue(w, se, ok)
|
|
||||||
case fmt.Stringer:
|
|
||||||
ss, ok := safeString(v)
|
|
||||||
return writeStringValue(w, ss, ok)
|
|
||||||
default:
|
|
||||||
rvalue := reflect.ValueOf(value)
|
|
||||||
switch rvalue.Kind() {
|
|
||||||
case reflect.Array, reflect.Chan, reflect.Func, reflect.Map, reflect.Slice, reflect.Struct:
|
|
||||||
return ErrUnsupportedValueType
|
|
||||||
case reflect.Ptr:
|
|
||||||
if rvalue.IsNil() {
|
|
||||||
return writeBytesValue(w, null)
|
|
||||||
}
|
|
||||||
return writeValue(w, rvalue.Elem().Interface())
|
|
||||||
}
|
|
||||||
return writeStringValue(w, fmt.Sprint(v), true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func needsQuotedValueRune(r rune) bool {
|
|
||||||
return r <= ' ' || r == '=' || r == '"' || r == utf8.RuneError
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeStringValue(w io.Writer, value string, ok bool) error {
|
|
||||||
var err error
|
|
||||||
if ok && value == "null" {
|
|
||||||
_, err = io.WriteString(w, `"null"`)
|
|
||||||
} else if strings.IndexFunc(value, needsQuotedValueRune) != -1 {
|
|
||||||
_, err = writeQuotedString(w, value)
|
|
||||||
} else {
|
|
||||||
_, err = io.WriteString(w, value)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeBytesValue(w io.Writer, value []byte) error {
|
|
||||||
var err error
|
|
||||||
if bytes.IndexFunc(value, needsQuotedValueRune) != -1 {
|
|
||||||
_, err = writeQuotedBytes(w, value)
|
|
||||||
} else {
|
|
||||||
_, err = w.Write(value)
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// EndRecord writes a newline character to the stream and resets the encoder
|
|
||||||
// to the beginning of a new record.
|
|
||||||
func (enc *Encoder) EndRecord() error {
|
|
||||||
_, err := enc.w.Write(newline)
|
|
||||||
if err == nil {
|
|
||||||
enc.needSep = false
|
|
||||||
}
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reset resets the encoder to the beginning of a new record.
|
|
||||||
func (enc *Encoder) Reset() {
|
|
||||||
enc.needSep = false
|
|
||||||
}
|
|
||||||
|
|
||||||
func safeError(err error) (s string, ok bool) {
|
|
||||||
defer func() {
|
|
||||||
if panicVal := recover(); panicVal != nil {
|
|
||||||
if v := reflect.ValueOf(err); v.Kind() == reflect.Ptr && v.IsNil() {
|
|
||||||
s, ok = "null", false
|
|
||||||
} else {
|
|
||||||
s, ok = fmt.Sprintf("PANIC:%v", panicVal), false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
s, ok = err.Error(), true
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func safeString(str fmt.Stringer) (s string, ok bool) {
|
|
||||||
defer func() {
|
|
||||||
if panicVal := recover(); panicVal != nil {
|
|
||||||
if v := reflect.ValueOf(str); v.Kind() == reflect.Ptr && v.IsNil() {
|
|
||||||
s, ok = "null", false
|
|
||||||
} else {
|
|
||||||
s, ok = fmt.Sprintf("PANIC:%v", panicVal), true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
s, ok = str.String(), true
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func safeMarshal(tm encoding.TextMarshaler) (b []byte, err error) {
|
|
||||||
defer func() {
|
|
||||||
if panicVal := recover(); panicVal != nil {
|
|
||||||
if v := reflect.ValueOf(tm); v.Kind() == reflect.Ptr && v.IsNil() {
|
|
||||||
b, err = nil, nil
|
|
||||||
} else {
|
|
||||||
b, err = nil, fmt.Errorf("panic when marshalling: %s", panicVal)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
b, err = tm.MarshalText()
|
|
||||||
if err != nil {
|
|
||||||
return nil, &MarshalerError{
|
|
||||||
Type: reflect.TypeOf(tm),
|
|
||||||
Err: err,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
3
vendor/github.com/go-logfmt/logfmt/go.mod
generated
vendored
3
vendor/github.com/go-logfmt/logfmt/go.mod
generated
vendored
|
@ -1,3 +0,0 @@
|
||||||
module github.com/go-logfmt/logfmt
|
|
||||||
|
|
||||||
go 1.13
|
|
277
vendor/github.com/go-logfmt/logfmt/jsonstring.go
generated
vendored
277
vendor/github.com/go-logfmt/logfmt/jsonstring.go
generated
vendored
|
@ -1,277 +0,0 @@
|
||||||
package logfmt
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"io"
|
|
||||||
"strconv"
|
|
||||||
"sync"
|
|
||||||
"unicode"
|
|
||||||
"unicode/utf16"
|
|
||||||
"unicode/utf8"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Taken from Go's encoding/json and modified for use here.
|
|
||||||
|
|
||||||
// Copyright 2010 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
var hex = "0123456789abcdef"
|
|
||||||
|
|
||||||
var bufferPool = sync.Pool{
|
|
||||||
New: func() interface{} {
|
|
||||||
return &bytes.Buffer{}
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
func getBuffer() *bytes.Buffer {
|
|
||||||
return bufferPool.Get().(*bytes.Buffer)
|
|
||||||
}
|
|
||||||
|
|
||||||
func poolBuffer(buf *bytes.Buffer) {
|
|
||||||
buf.Reset()
|
|
||||||
bufferPool.Put(buf)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NOTE: keep in sync with writeQuotedBytes below.
|
|
||||||
func writeQuotedString(w io.Writer, s string) (int, error) {
|
|
||||||
buf := getBuffer()
|
|
||||||
buf.WriteByte('"')
|
|
||||||
start := 0
|
|
||||||
for i := 0; i < len(s); {
|
|
||||||
if b := s[i]; b < utf8.RuneSelf {
|
|
||||||
if 0x20 <= b && b != '\\' && b != '"' {
|
|
||||||
i++
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if start < i {
|
|
||||||
buf.WriteString(s[start:i])
|
|
||||||
}
|
|
||||||
switch b {
|
|
||||||
case '\\', '"':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte(b)
|
|
||||||
case '\n':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('n')
|
|
||||||
case '\r':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('r')
|
|
||||||
case '\t':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('t')
|
|
||||||
default:
|
|
||||||
// This encodes bytes < 0x20 except for \n, \r, and \t.
|
|
||||||
buf.WriteString(`\u00`)
|
|
||||||
buf.WriteByte(hex[b>>4])
|
|
||||||
buf.WriteByte(hex[b&0xF])
|
|
||||||
}
|
|
||||||
i++
|
|
||||||
start = i
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
c, size := utf8.DecodeRuneInString(s[i:])
|
|
||||||
if c == utf8.RuneError {
|
|
||||||
if start < i {
|
|
||||||
buf.WriteString(s[start:i])
|
|
||||||
}
|
|
||||||
buf.WriteString(`\ufffd`)
|
|
||||||
i += size
|
|
||||||
start = i
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
i += size
|
|
||||||
}
|
|
||||||
if start < len(s) {
|
|
||||||
buf.WriteString(s[start:])
|
|
||||||
}
|
|
||||||
buf.WriteByte('"')
|
|
||||||
n, err := w.Write(buf.Bytes())
|
|
||||||
poolBuffer(buf)
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// NOTE: keep in sync with writeQuoteString above.
|
|
||||||
func writeQuotedBytes(w io.Writer, s []byte) (int, error) {
|
|
||||||
buf := getBuffer()
|
|
||||||
buf.WriteByte('"')
|
|
||||||
start := 0
|
|
||||||
for i := 0; i < len(s); {
|
|
||||||
if b := s[i]; b < utf8.RuneSelf {
|
|
||||||
if 0x20 <= b && b != '\\' && b != '"' {
|
|
||||||
i++
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if start < i {
|
|
||||||
buf.Write(s[start:i])
|
|
||||||
}
|
|
||||||
switch b {
|
|
||||||
case '\\', '"':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte(b)
|
|
||||||
case '\n':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('n')
|
|
||||||
case '\r':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('r')
|
|
||||||
case '\t':
|
|
||||||
buf.WriteByte('\\')
|
|
||||||
buf.WriteByte('t')
|
|
||||||
default:
|
|
||||||
// This encodes bytes < 0x20 except for \n, \r, and \t.
|
|
||||||
buf.WriteString(`\u00`)
|
|
||||||
buf.WriteByte(hex[b>>4])
|
|
||||||
buf.WriteByte(hex[b&0xF])
|
|
||||||
}
|
|
||||||
i++
|
|
||||||
start = i
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
c, size := utf8.DecodeRune(s[i:])
|
|
||||||
if c == utf8.RuneError {
|
|
||||||
if start < i {
|
|
||||||
buf.Write(s[start:i])
|
|
||||||
}
|
|
||||||
buf.WriteString(`\ufffd`)
|
|
||||||
i += size
|
|
||||||
start = i
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
i += size
|
|
||||||
}
|
|
||||||
if start < len(s) {
|
|
||||||
buf.Write(s[start:])
|
|
||||||
}
|
|
||||||
buf.WriteByte('"')
|
|
||||||
n, err := w.Write(buf.Bytes())
|
|
||||||
poolBuffer(buf)
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// getu4 decodes \uXXXX from the beginning of s, returning the hex value,
|
|
||||||
// or it returns -1.
|
|
||||||
func getu4(s []byte) rune {
|
|
||||||
if len(s) < 6 || s[0] != '\\' || s[1] != 'u' {
|
|
||||||
return -1
|
|
||||||
}
|
|
||||||
r, err := strconv.ParseUint(string(s[2:6]), 16, 64)
|
|
||||||
if err != nil {
|
|
||||||
return -1
|
|
||||||
}
|
|
||||||
return rune(r)
|
|
||||||
}
|
|
||||||
|
|
||||||
func unquoteBytes(s []byte) (t []byte, ok bool) {
|
|
||||||
if len(s) < 2 || s[0] != '"' || s[len(s)-1] != '"' {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s = s[1 : len(s)-1]
|
|
||||||
|
|
||||||
// Check for unusual characters. If there are none,
|
|
||||||
// then no unquoting is needed, so return a slice of the
|
|
||||||
// original bytes.
|
|
||||||
r := 0
|
|
||||||
for r < len(s) {
|
|
||||||
c := s[r]
|
|
||||||
if c == '\\' || c == '"' || c < ' ' {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if c < utf8.RuneSelf {
|
|
||||||
r++
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
rr, size := utf8.DecodeRune(s[r:])
|
|
||||||
if rr == utf8.RuneError {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
r += size
|
|
||||||
}
|
|
||||||
if r == len(s) {
|
|
||||||
return s, true
|
|
||||||
}
|
|
||||||
|
|
||||||
b := make([]byte, len(s)+2*utf8.UTFMax)
|
|
||||||
w := copy(b, s[0:r])
|
|
||||||
for r < len(s) {
|
|
||||||
// Out of room? Can only happen if s is full of
|
|
||||||
// malformed UTF-8 and we're replacing each
|
|
||||||
// byte with RuneError.
|
|
||||||
if w >= len(b)-2*utf8.UTFMax {
|
|
||||||
nb := make([]byte, (len(b)+utf8.UTFMax)*2)
|
|
||||||
copy(nb, b[0:w])
|
|
||||||
b = nb
|
|
||||||
}
|
|
||||||
switch c := s[r]; {
|
|
||||||
case c == '\\':
|
|
||||||
r++
|
|
||||||
if r >= len(s) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
switch s[r] {
|
|
||||||
default:
|
|
||||||
return
|
|
||||||
case '"', '\\', '/', '\'':
|
|
||||||
b[w] = s[r]
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 'b':
|
|
||||||
b[w] = '\b'
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 'f':
|
|
||||||
b[w] = '\f'
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 'n':
|
|
||||||
b[w] = '\n'
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 'r':
|
|
||||||
b[w] = '\r'
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 't':
|
|
||||||
b[w] = '\t'
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
case 'u':
|
|
||||||
r--
|
|
||||||
rr := getu4(s[r:])
|
|
||||||
if rr < 0 {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
r += 6
|
|
||||||
if utf16.IsSurrogate(rr) {
|
|
||||||
rr1 := getu4(s[r:])
|
|
||||||
if dec := utf16.DecodeRune(rr, rr1); dec != unicode.ReplacementChar {
|
|
||||||
// A valid pair; consume.
|
|
||||||
r += 6
|
|
||||||
w += utf8.EncodeRune(b[w:], dec)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
// Invalid surrogate; fall back to replacement rune.
|
|
||||||
rr = unicode.ReplacementChar
|
|
||||||
}
|
|
||||||
w += utf8.EncodeRune(b[w:], rr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Quote, control characters are invalid.
|
|
||||||
case c == '"', c < ' ':
|
|
||||||
return
|
|
||||||
|
|
||||||
// ASCII
|
|
||||||
case c < utf8.RuneSelf:
|
|
||||||
b[w] = c
|
|
||||||
r++
|
|
||||||
w++
|
|
||||||
|
|
||||||
// Coerce to well-formed UTF-8.
|
|
||||||
default:
|
|
||||||
rr, size := utf8.DecodeRune(s[r:])
|
|
||||||
r += size
|
|
||||||
w += utf8.EncodeRune(b[w:], rr)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return b[0:w], true
|
|
||||||
}
|
|
46
vendor/github.com/godbus/dbus/.travis.yml
generated
vendored
46
vendor/github.com/godbus/dbus/.travis.yml
generated
vendored
|
@ -1,46 +0,0 @@
|
||||||
dist: precise
|
|
||||||
language: go
|
|
||||||
go_import_path: github.com/godbus/dbus
|
|
||||||
sudo: true
|
|
||||||
|
|
||||||
go:
|
|
||||||
- 1.7.3
|
|
||||||
- 1.8.7
|
|
||||||
- 1.9.5
|
|
||||||
- 1.10.1
|
|
||||||
- tip
|
|
||||||
|
|
||||||
env:
|
|
||||||
global:
|
|
||||||
matrix:
|
|
||||||
- TARGET=amd64
|
|
||||||
- TARGET=arm64
|
|
||||||
- TARGET=arm
|
|
||||||
- TARGET=386
|
|
||||||
- TARGET=ppc64le
|
|
||||||
|
|
||||||
matrix:
|
|
||||||
fast_finish: true
|
|
||||||
allow_failures:
|
|
||||||
- go: tip
|
|
||||||
exclude:
|
|
||||||
- go: tip
|
|
||||||
env: TARGET=arm
|
|
||||||
- go: tip
|
|
||||||
env: TARGET=arm64
|
|
||||||
- go: tip
|
|
||||||
env: TARGET=386
|
|
||||||
- go: tip
|
|
||||||
env: TARGET=ppc64le
|
|
||||||
|
|
||||||
addons:
|
|
||||||
apt:
|
|
||||||
packages:
|
|
||||||
- dbus
|
|
||||||
- dbus-x11
|
|
||||||
|
|
||||||
before_install:
|
|
||||||
|
|
||||||
script:
|
|
||||||
- go test -v -race ./... # Run all the tests with the race detector enabled
|
|
||||||
- go vet ./... # go vet is the official Go static analyzer
|
|
50
vendor/github.com/godbus/dbus/CONTRIBUTING.md
generated
vendored
50
vendor/github.com/godbus/dbus/CONTRIBUTING.md
generated
vendored
|
@ -1,50 +0,0 @@
|
||||||
# How to Contribute
|
|
||||||
|
|
||||||
## Getting Started
|
|
||||||
|
|
||||||
- Fork the repository on GitHub
|
|
||||||
- Read the [README](README.markdown) for build and test instructions
|
|
||||||
- Play with the project, submit bugs, submit patches!
|
|
||||||
|
|
||||||
## Contribution Flow
|
|
||||||
|
|
||||||
This is a rough outline of what a contributor's workflow looks like:
|
|
||||||
|
|
||||||
- Create a topic branch from where you want to base your work (usually master).
|
|
||||||
- Make commits of logical units.
|
|
||||||
- Make sure your commit messages are in the proper format (see below).
|
|
||||||
- Push your changes to a topic branch in your fork of the repository.
|
|
||||||
- Make sure the tests pass, and add any new tests as appropriate.
|
|
||||||
- Submit a pull request to the original repository.
|
|
||||||
|
|
||||||
Thanks for your contributions!
|
|
||||||
|
|
||||||
### Format of the Commit Message
|
|
||||||
|
|
||||||
We follow a rough convention for commit messages that is designed to answer two
|
|
||||||
questions: what changed and why. The subject line should feature the what and
|
|
||||||
the body of the commit should describe the why.
|
|
||||||
|
|
||||||
```
|
|
||||||
scripts: add the test-cluster command
|
|
||||||
|
|
||||||
this uses tmux to setup a test cluster that you can easily kill and
|
|
||||||
start for debugging.
|
|
||||||
|
|
||||||
Fixes #38
|
|
||||||
```
|
|
||||||
|
|
||||||
The format can be described more formally as follows:
|
|
||||||
|
|
||||||
```
|
|
||||||
<subsystem>: <what changed>
|
|
||||||
<BLANK LINE>
|
|
||||||
<why this change was made>
|
|
||||||
<BLANK LINE>
|
|
||||||
<footer>
|
|
||||||
```
|
|
||||||
|
|
||||||
The first line is the subject and should be no longer than 70 characters, the
|
|
||||||
second line is always blank, and other lines should be wrapped at 80 characters.
|
|
||||||
This allows the message to be easier to read on GitHub as well as in various
|
|
||||||
git tools.
|
|
25
vendor/github.com/godbus/dbus/LICENSE
generated
vendored
25
vendor/github.com/godbus/dbus/LICENSE
generated
vendored
|
@ -1,25 +0,0 @@
|
||||||
Copyright (c) 2013, Georg Reinke (<guelfey at gmail dot com>), Google
|
|
||||||
All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
|
||||||
modification, are permitted provided that the following conditions
|
|
||||||
are met:
|
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
|
||||||
this list of conditions and the following disclaimer.
|
|
||||||
|
|
||||||
2. 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.
|
|
||||||
|
|
||||||
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
|
|
||||||
HOLDER 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.
|
|
3
vendor/github.com/godbus/dbus/MAINTAINERS
generated
vendored
3
vendor/github.com/godbus/dbus/MAINTAINERS
generated
vendored
|
@ -1,3 +0,0 @@
|
||||||
Brandon Philips <brandon@ifup.org> (@philips)
|
|
||||||
Brian Waldon <brian@waldon.cc> (@bcwaldon)
|
|
||||||
John Southworth <jsouthwo@brocade.com> (@jsouthworth)
|
|
44
vendor/github.com/godbus/dbus/README.markdown
generated
vendored
44
vendor/github.com/godbus/dbus/README.markdown
generated
vendored
|
@ -1,44 +0,0 @@
|
||||||
[![Build Status](https://travis-ci.org/godbus/dbus.svg?branch=master)](https://travis-ci.org/godbus/dbus)
|
|
||||||
|
|
||||||
dbus
|
|
||||||
----
|
|
||||||
|
|
||||||
dbus is a simple library that implements native Go client bindings for the
|
|
||||||
D-Bus message bus system.
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
* Complete native implementation of the D-Bus message protocol
|
|
||||||
* Go-like API (channels for signals / asynchronous method calls, Goroutine-safe connections)
|
|
||||||
* Subpackages that help with the introspection / property interfaces
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
|
|
||||||
This packages requires Go 1.7. If you installed it and set up your GOPATH, just run:
|
|
||||||
|
|
||||||
```
|
|
||||||
go get github.com/godbus/dbus
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want to use the subpackages, you can install them the same way.
|
|
||||||
|
|
||||||
### Usage
|
|
||||||
|
|
||||||
The complete package documentation and some simple examples are available at
|
|
||||||
[godoc.org](http://godoc.org/github.com/godbus/dbus). Also, the
|
|
||||||
[_examples](https://github.com/godbus/dbus/tree/master/_examples) directory
|
|
||||||
gives a short overview over the basic usage.
|
|
||||||
|
|
||||||
#### Projects using godbus
|
|
||||||
- [notify](https://github.com/esiqveland/notify) provides desktop notifications over dbus into a library.
|
|
||||||
- [go-bluetooth](https://github.com/muka/go-bluetooth) provides a bluetooth client over bluez dbus API.
|
|
||||||
|
|
||||||
Please note that the API is considered unstable for now and may change without
|
|
||||||
further notice.
|
|
||||||
|
|
||||||
### License
|
|
||||||
|
|
||||||
go.dbus is available under the Simplified BSD License; see LICENSE for the full
|
|
||||||
text.
|
|
||||||
|
|
||||||
Nearly all of the credit for this library goes to github.com/guelfey/go.dbus.
|
|
252
vendor/github.com/godbus/dbus/auth.go
generated
vendored
252
vendor/github.com/godbus/dbus/auth.go
generated
vendored
|
@ -1,252 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"bytes"
|
|
||||||
"errors"
|
|
||||||
"io"
|
|
||||||
"os"
|
|
||||||
"strconv"
|
|
||||||
)
|
|
||||||
|
|
||||||
// AuthStatus represents the Status of an authentication mechanism.
|
|
||||||
type AuthStatus byte
|
|
||||||
|
|
||||||
const (
|
|
||||||
// AuthOk signals that authentication is finished; the next command
|
|
||||||
// from the server should be an OK.
|
|
||||||
AuthOk AuthStatus = iota
|
|
||||||
|
|
||||||
// AuthContinue signals that additional data is needed; the next command
|
|
||||||
// from the server should be a DATA.
|
|
||||||
AuthContinue
|
|
||||||
|
|
||||||
// AuthError signals an error; the server sent invalid data or some
|
|
||||||
// other unexpected thing happened and the current authentication
|
|
||||||
// process should be aborted.
|
|
||||||
AuthError
|
|
||||||
)
|
|
||||||
|
|
||||||
type authState byte
|
|
||||||
|
|
||||||
const (
|
|
||||||
waitingForData authState = iota
|
|
||||||
waitingForOk
|
|
||||||
waitingForReject
|
|
||||||
)
|
|
||||||
|
|
||||||
// Auth defines the behaviour of an authentication mechanism.
|
|
||||||
type Auth interface {
|
|
||||||
// Return the name of the mechnism, the argument to the first AUTH command
|
|
||||||
// and the next status.
|
|
||||||
FirstData() (name, resp []byte, status AuthStatus)
|
|
||||||
|
|
||||||
// Process the given DATA command, and return the argument to the DATA
|
|
||||||
// command and the next status. If len(resp) == 0, no DATA command is sent.
|
|
||||||
HandleData(data []byte) (resp []byte, status AuthStatus)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Auth authenticates the connection, trying the given list of authentication
|
|
||||||
// mechanisms (in that order). If nil is passed, the EXTERNAL and
|
|
||||||
// DBUS_COOKIE_SHA1 mechanisms are tried for the current user. For private
|
|
||||||
// connections, this method must be called before sending any messages to the
|
|
||||||
// bus. Auth must not be called on shared connections.
|
|
||||||
func (conn *Conn) Auth(methods []Auth) error {
|
|
||||||
if methods == nil {
|
|
||||||
uid := strconv.Itoa(os.Getuid())
|
|
||||||
methods = []Auth{AuthExternal(uid), AuthCookieSha1(uid, getHomeDir())}
|
|
||||||
}
|
|
||||||
in := bufio.NewReader(conn.transport)
|
|
||||||
err := conn.transport.SendNullByte()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = authWriteLine(conn.transport, []byte("AUTH"))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
s, err := authReadLine(in)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(s) < 2 || !bytes.Equal(s[0], []byte("REJECTED")) {
|
|
||||||
return errors.New("dbus: authentication protocol error")
|
|
||||||
}
|
|
||||||
s = s[1:]
|
|
||||||
for _, v := range s {
|
|
||||||
for _, m := range methods {
|
|
||||||
if name, data, status := m.FirstData(); bytes.Equal(v, name) {
|
|
||||||
var ok bool
|
|
||||||
err = authWriteLine(conn.transport, []byte("AUTH"), []byte(v), data)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
switch status {
|
|
||||||
case AuthOk:
|
|
||||||
err, ok = conn.tryAuth(m, waitingForOk, in)
|
|
||||||
case AuthContinue:
|
|
||||||
err, ok = conn.tryAuth(m, waitingForData, in)
|
|
||||||
default:
|
|
||||||
panic("dbus: invalid authentication status")
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if ok {
|
|
||||||
if conn.transport.SupportsUnixFDs() {
|
|
||||||
err = authWriteLine(conn, []byte("NEGOTIATE_UNIX_FD"))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
line, err := authReadLine(in)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
switch {
|
|
||||||
case bytes.Equal(line[0], []byte("AGREE_UNIX_FD")):
|
|
||||||
conn.EnableUnixFDs()
|
|
||||||
conn.unixFD = true
|
|
||||||
case bytes.Equal(line[0], []byte("ERROR")):
|
|
||||||
default:
|
|
||||||
return errors.New("dbus: authentication protocol error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
err = authWriteLine(conn.transport, []byte("BEGIN"))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
go conn.inWorker()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return errors.New("dbus: authentication failed")
|
|
||||||
}
|
|
||||||
|
|
||||||
// tryAuth tries to authenticate with m as the mechanism, using state as the
|
|
||||||
// initial authState and in for reading input. It returns (nil, true) on
|
|
||||||
// success, (nil, false) on a REJECTED and (someErr, false) if some other
|
|
||||||
// error occured.
|
|
||||||
func (conn *Conn) tryAuth(m Auth, state authState, in *bufio.Reader) (error, bool) {
|
|
||||||
for {
|
|
||||||
s, err := authReadLine(in)
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
switch {
|
|
||||||
case state == waitingForData && string(s[0]) == "DATA":
|
|
||||||
if len(s) != 2 {
|
|
||||||
err = authWriteLine(conn.transport, []byte("ERROR"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
data, status := m.HandleData(s[1])
|
|
||||||
switch status {
|
|
||||||
case AuthOk, AuthContinue:
|
|
||||||
if len(data) != 0 {
|
|
||||||
err = authWriteLine(conn.transport, []byte("DATA"), data)
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if status == AuthOk {
|
|
||||||
state = waitingForOk
|
|
||||||
}
|
|
||||||
case AuthError:
|
|
||||||
err = authWriteLine(conn.transport, []byte("ERROR"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case state == waitingForData && string(s[0]) == "REJECTED":
|
|
||||||
return nil, false
|
|
||||||
case state == waitingForData && string(s[0]) == "ERROR":
|
|
||||||
err = authWriteLine(conn.transport, []byte("CANCEL"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
state = waitingForReject
|
|
||||||
case state == waitingForData && string(s[0]) == "OK":
|
|
||||||
if len(s) != 2 {
|
|
||||||
err = authWriteLine(conn.transport, []byte("CANCEL"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
state = waitingForReject
|
|
||||||
}
|
|
||||||
conn.uuid = string(s[1])
|
|
||||||
return nil, true
|
|
||||||
case state == waitingForData:
|
|
||||||
err = authWriteLine(conn.transport, []byte("ERROR"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
case state == waitingForOk && string(s[0]) == "OK":
|
|
||||||
if len(s) != 2 {
|
|
||||||
err = authWriteLine(conn.transport, []byte("CANCEL"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
state = waitingForReject
|
|
||||||
}
|
|
||||||
conn.uuid = string(s[1])
|
|
||||||
return nil, true
|
|
||||||
case state == waitingForOk && string(s[0]) == "REJECTED":
|
|
||||||
return nil, false
|
|
||||||
case state == waitingForOk && (string(s[0]) == "DATA" ||
|
|
||||||
string(s[0]) == "ERROR"):
|
|
||||||
|
|
||||||
err = authWriteLine(conn.transport, []byte("CANCEL"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
state = waitingForReject
|
|
||||||
case state == waitingForOk:
|
|
||||||
err = authWriteLine(conn.transport, []byte("ERROR"))
|
|
||||||
if err != nil {
|
|
||||||
return err, false
|
|
||||||
}
|
|
||||||
case state == waitingForReject && string(s[0]) == "REJECTED":
|
|
||||||
return nil, false
|
|
||||||
case state == waitingForReject:
|
|
||||||
return errors.New("dbus: authentication protocol error"), false
|
|
||||||
default:
|
|
||||||
panic("dbus: invalid auth state")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// authReadLine reads a line and separates it into its fields.
|
|
||||||
func authReadLine(in *bufio.Reader) ([][]byte, error) {
|
|
||||||
data, err := in.ReadBytes('\n')
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
data = bytes.TrimSuffix(data, []byte("\r\n"))
|
|
||||||
return bytes.Split(data, []byte{' '}), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// authWriteLine writes the given line in the authentication protocol format
|
|
||||||
// (elements of data separated by a " " and terminated by "\r\n").
|
|
||||||
func authWriteLine(out io.Writer, data ...[]byte) error {
|
|
||||||
buf := make([]byte, 0)
|
|
||||||
for i, v := range data {
|
|
||||||
buf = append(buf, v...)
|
|
||||||
if i != len(data)-1 {
|
|
||||||
buf = append(buf, ' ')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
buf = append(buf, '\r')
|
|
||||||
buf = append(buf, '\n')
|
|
||||||
n, err := out.Write(buf)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if n != len(buf) {
|
|
||||||
return io.ErrUnexpectedEOF
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
16
vendor/github.com/godbus/dbus/auth_anonymous.go
generated
vendored
16
vendor/github.com/godbus/dbus/auth_anonymous.go
generated
vendored
|
@ -1,16 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
// AuthAnonymous returns an Auth that uses the ANONYMOUS mechanism.
|
|
||||||
func AuthAnonymous() Auth {
|
|
||||||
return &authAnonymous{}
|
|
||||||
}
|
|
||||||
|
|
||||||
type authAnonymous struct{}
|
|
||||||
|
|
||||||
func (a *authAnonymous) FirstData() (name, resp []byte, status AuthStatus) {
|
|
||||||
return []byte("ANONYMOUS"), nil, AuthOk
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a *authAnonymous) HandleData(data []byte) (resp []byte, status AuthStatus) {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
26
vendor/github.com/godbus/dbus/auth_external.go
generated
vendored
26
vendor/github.com/godbus/dbus/auth_external.go
generated
vendored
|
@ -1,26 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/hex"
|
|
||||||
)
|
|
||||||
|
|
||||||
// AuthExternal returns an Auth that authenticates as the given user with the
|
|
||||||
// EXTERNAL mechanism.
|
|
||||||
func AuthExternal(user string) Auth {
|
|
||||||
return authExternal{user}
|
|
||||||
}
|
|
||||||
|
|
||||||
// AuthExternal implements the EXTERNAL authentication mechanism.
|
|
||||||
type authExternal struct {
|
|
||||||
user string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a authExternal) FirstData() ([]byte, []byte, AuthStatus) {
|
|
||||||
b := make([]byte, 2*len(a.user))
|
|
||||||
hex.Encode(b, []byte(a.user))
|
|
||||||
return []byte("EXTERNAL"), b, AuthOk
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a authExternal) HandleData(b []byte) ([]byte, AuthStatus) {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
102
vendor/github.com/godbus/dbus/auth_sha1.go
generated
vendored
102
vendor/github.com/godbus/dbus/auth_sha1.go
generated
vendored
|
@ -1,102 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"bytes"
|
|
||||||
"crypto/rand"
|
|
||||||
"crypto/sha1"
|
|
||||||
"encoding/hex"
|
|
||||||
"os"
|
|
||||||
)
|
|
||||||
|
|
||||||
// AuthCookieSha1 returns an Auth that authenticates as the given user with the
|
|
||||||
// DBUS_COOKIE_SHA1 mechanism. The home parameter should specify the home
|
|
||||||
// directory of the user.
|
|
||||||
func AuthCookieSha1(user, home string) Auth {
|
|
||||||
return authCookieSha1{user, home}
|
|
||||||
}
|
|
||||||
|
|
||||||
type authCookieSha1 struct {
|
|
||||||
user, home string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a authCookieSha1) FirstData() ([]byte, []byte, AuthStatus) {
|
|
||||||
b := make([]byte, 2*len(a.user))
|
|
||||||
hex.Encode(b, []byte(a.user))
|
|
||||||
return []byte("DBUS_COOKIE_SHA1"), b, AuthContinue
|
|
||||||
}
|
|
||||||
|
|
||||||
func (a authCookieSha1) HandleData(data []byte) ([]byte, AuthStatus) {
|
|
||||||
challenge := make([]byte, len(data)/2)
|
|
||||||
_, err := hex.Decode(challenge, data)
|
|
||||||
if err != nil {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
||||||
b := bytes.Split(challenge, []byte{' '})
|
|
||||||
if len(b) != 3 {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
||||||
context := b[0]
|
|
||||||
id := b[1]
|
|
||||||
svchallenge := b[2]
|
|
||||||
cookie := a.getCookie(context, id)
|
|
||||||
if cookie == nil {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
||||||
clchallenge := a.generateChallenge()
|
|
||||||
if clchallenge == nil {
|
|
||||||
return nil, AuthError
|
|
||||||
}
|
|
||||||
hash := sha1.New()
|
|
||||||
hash.Write(bytes.Join([][]byte{svchallenge, clchallenge, cookie}, []byte{':'}))
|
|
||||||
hexhash := make([]byte, 2*hash.Size())
|
|
||||||
hex.Encode(hexhash, hash.Sum(nil))
|
|
||||||
data = append(clchallenge, ' ')
|
|
||||||
data = append(data, hexhash...)
|
|
||||||
resp := make([]byte, 2*len(data))
|
|
||||||
hex.Encode(resp, data)
|
|
||||||
return resp, AuthOk
|
|
||||||
}
|
|
||||||
|
|
||||||
// getCookie searches for the cookie identified by id in context and returns
|
|
||||||
// the cookie content or nil. (Since HandleData can't return a specific error,
|
|
||||||
// but only whether an error occured, this function also doesn't bother to
|
|
||||||
// return an error.)
|
|
||||||
func (a authCookieSha1) getCookie(context, id []byte) []byte {
|
|
||||||
file, err := os.Open(a.home + "/.dbus-keyrings/" + string(context))
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
rd := bufio.NewReader(file)
|
|
||||||
for {
|
|
||||||
line, err := rd.ReadBytes('\n')
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
line = line[:len(line)-1]
|
|
||||||
b := bytes.Split(line, []byte{' '})
|
|
||||||
if len(b) != 3 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if bytes.Equal(b[0], id) {
|
|
||||||
return b[2]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// generateChallenge returns a random, hex-encoded challenge, or nil on error
|
|
||||||
// (see above).
|
|
||||||
func (a authCookieSha1) generateChallenge() []byte {
|
|
||||||
b := make([]byte, 16)
|
|
||||||
n, err := rand.Read(b)
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if n != 16 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
enc := make([]byte, 32)
|
|
||||||
hex.Encode(enc, b)
|
|
||||||
return enc
|
|
||||||
}
|
|
60
vendor/github.com/godbus/dbus/call.go
generated
vendored
60
vendor/github.com/godbus/dbus/call.go
generated
vendored
|
@ -1,60 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
var errSignature = errors.New("dbus: mismatched signature")
|
|
||||||
|
|
||||||
// Call represents a pending or completed method call.
|
|
||||||
type Call struct {
|
|
||||||
Destination string
|
|
||||||
Path ObjectPath
|
|
||||||
Method string
|
|
||||||
Args []interface{}
|
|
||||||
|
|
||||||
// Strobes when the call is complete.
|
|
||||||
Done chan *Call
|
|
||||||
|
|
||||||
// After completion, the error status. If this is non-nil, it may be an
|
|
||||||
// error message from the peer (with Error as its type) or some other error.
|
|
||||||
Err error
|
|
||||||
|
|
||||||
// Holds the response once the call is done.
|
|
||||||
Body []interface{}
|
|
||||||
|
|
||||||
// tracks context and canceler
|
|
||||||
ctx context.Context
|
|
||||||
ctxCanceler context.CancelFunc
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Call) Context() context.Context {
|
|
||||||
if c.ctx == nil {
|
|
||||||
return context.Background()
|
|
||||||
}
|
|
||||||
|
|
||||||
return c.ctx
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Call) ContextCancel() {
|
|
||||||
if c.ctxCanceler != nil {
|
|
||||||
c.ctxCanceler()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store stores the body of the reply into the provided pointers. It returns
|
|
||||||
// an error if the signatures of the body and retvalues don't match, or if
|
|
||||||
// the error status is not nil.
|
|
||||||
func (c *Call) Store(retvalues ...interface{}) error {
|
|
||||||
if c.Err != nil {
|
|
||||||
return c.Err
|
|
||||||
}
|
|
||||||
|
|
||||||
return Store(c.Body, retvalues...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Call) done() {
|
|
||||||
c.Done <- c
|
|
||||||
c.ContextCancel()
|
|
||||||
}
|
|
847
vendor/github.com/godbus/dbus/conn.go
generated
vendored
847
vendor/github.com/godbus/dbus/conn.go
generated
vendored
|
@ -1,847 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
"io"
|
|
||||||
"os"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
systemBus *Conn
|
|
||||||
systemBusLck sync.Mutex
|
|
||||||
sessionBus *Conn
|
|
||||||
sessionBusLck sync.Mutex
|
|
||||||
)
|
|
||||||
|
|
||||||
// ErrClosed is the error returned by calls on a closed connection.
|
|
||||||
var ErrClosed = errors.New("dbus: connection closed by user")
|
|
||||||
|
|
||||||
// Conn represents a connection to a message bus (usually, the system or
|
|
||||||
// session bus).
|
|
||||||
//
|
|
||||||
// Connections are either shared or private. Shared connections
|
|
||||||
// are shared between calls to the functions that return them. As a result,
|
|
||||||
// the methods Close, Auth and Hello must not be called on them.
|
|
||||||
//
|
|
||||||
// Multiple goroutines may invoke methods on a connection simultaneously.
|
|
||||||
type Conn struct {
|
|
||||||
transport
|
|
||||||
|
|
||||||
busObj BusObject
|
|
||||||
unixFD bool
|
|
||||||
uuid string
|
|
||||||
|
|
||||||
handler Handler
|
|
||||||
signalHandler SignalHandler
|
|
||||||
serialGen SerialGenerator
|
|
||||||
|
|
||||||
names *nameTracker
|
|
||||||
calls *callTracker
|
|
||||||
outHandler *outputHandler
|
|
||||||
|
|
||||||
eavesdropped chan<- *Message
|
|
||||||
eavesdroppedLck sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
// SessionBus returns a shared connection to the session bus, connecting to it
|
|
||||||
// if not already done.
|
|
||||||
func SessionBus() (conn *Conn, err error) {
|
|
||||||
sessionBusLck.Lock()
|
|
||||||
defer sessionBusLck.Unlock()
|
|
||||||
if sessionBus != nil {
|
|
||||||
return sessionBus, nil
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
if conn != nil {
|
|
||||||
sessionBus = conn
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
conn, err = SessionBusPrivate()
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err = conn.Auth(nil); err != nil {
|
|
||||||
conn.Close()
|
|
||||||
conn = nil
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err = conn.Hello(); err != nil {
|
|
||||||
conn.Close()
|
|
||||||
conn = nil
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func getSessionBusAddress() (string, error) {
|
|
||||||
if address := os.Getenv("DBUS_SESSION_BUS_ADDRESS"); address != "" && address != "autolaunch:" {
|
|
||||||
return address, nil
|
|
||||||
|
|
||||||
} else if address := tryDiscoverDbusSessionBusAddress(); address != "" {
|
|
||||||
os.Setenv("DBUS_SESSION_BUS_ADDRESS", address)
|
|
||||||
return address, nil
|
|
||||||
}
|
|
||||||
return getSessionBusPlatformAddress()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SessionBusPrivate returns a new private connection to the session bus.
|
|
||||||
func SessionBusPrivate(opts ...ConnOption) (*Conn, error) {
|
|
||||||
address, err := getSessionBusAddress()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return Dial(address, opts...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SessionBusPrivate returns a new private connection to the session bus.
|
|
||||||
//
|
|
||||||
// Deprecated: use SessionBusPrivate with options instead.
|
|
||||||
func SessionBusPrivateHandler(handler Handler, signalHandler SignalHandler) (*Conn, error) {
|
|
||||||
return SessionBusPrivate(WithHandler(handler), WithSignalHandler(signalHandler))
|
|
||||||
}
|
|
||||||
|
|
||||||
// SystemBus returns a shared connection to the system bus, connecting to it if
|
|
||||||
// not already done.
|
|
||||||
func SystemBus() (conn *Conn, err error) {
|
|
||||||
systemBusLck.Lock()
|
|
||||||
defer systemBusLck.Unlock()
|
|
||||||
if systemBus != nil {
|
|
||||||
return systemBus, nil
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
if conn != nil {
|
|
||||||
systemBus = conn
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
conn, err = SystemBusPrivate()
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err = conn.Auth(nil); err != nil {
|
|
||||||
conn.Close()
|
|
||||||
conn = nil
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if err = conn.Hello(); err != nil {
|
|
||||||
conn.Close()
|
|
||||||
conn = nil
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// SystemBusPrivate returns a new private connection to the system bus.
|
|
||||||
// Note: this connection is not ready to use. One must perform Auth and Hello
|
|
||||||
// on the connection before it is useable.
|
|
||||||
func SystemBusPrivate(opts ...ConnOption) (*Conn, error) {
|
|
||||||
return Dial(getSystemBusPlatformAddress(), opts...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SystemBusPrivateHandler returns a new private connection to the system bus, using the provided handlers.
|
|
||||||
//
|
|
||||||
// Deprecated: use SystemBusPrivate with options instead.
|
|
||||||
func SystemBusPrivateHandler(handler Handler, signalHandler SignalHandler) (*Conn, error) {
|
|
||||||
return SystemBusPrivate(WithHandler(handler), WithSignalHandler(signalHandler))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dial establishes a new private connection to the message bus specified by address.
|
|
||||||
func Dial(address string, opts ...ConnOption) (*Conn, error) {
|
|
||||||
tr, err := getTransport(address)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return newConn(tr, opts...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// DialHandler establishes a new private connection to the message bus specified by address, using the supplied handlers.
|
|
||||||
//
|
|
||||||
// Deprecated: use Dial with options instead.
|
|
||||||
func DialHandler(address string, handler Handler, signalHandler SignalHandler) (*Conn, error) {
|
|
||||||
return Dial(address, WithSignalHandler(signalHandler))
|
|
||||||
}
|
|
||||||
|
|
||||||
// ConnOption is a connection option.
|
|
||||||
type ConnOption func(conn *Conn) error
|
|
||||||
|
|
||||||
// WithHandler overrides the default handler.
|
|
||||||
func WithHandler(handler Handler) ConnOption {
|
|
||||||
return func(conn *Conn) error {
|
|
||||||
conn.handler = handler
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithSignalHandler overrides the default signal handler.
|
|
||||||
func WithSignalHandler(handler SignalHandler) ConnOption {
|
|
||||||
return func(conn *Conn) error {
|
|
||||||
conn.signalHandler = handler
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithSerialGenerator overrides the default signals generator.
|
|
||||||
func WithSerialGenerator(gen SerialGenerator) ConnOption {
|
|
||||||
return func(conn *Conn) error {
|
|
||||||
conn.serialGen = gen
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewConn creates a new private *Conn from an already established connection.
|
|
||||||
func NewConn(conn io.ReadWriteCloser, opts ...ConnOption) (*Conn, error) {
|
|
||||||
return newConn(genericTransport{conn}, opts...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewConnHandler creates a new private *Conn from an already established connection, using the supplied handlers.
|
|
||||||
//
|
|
||||||
// Deprecated: use NewConn with options instead.
|
|
||||||
func NewConnHandler(conn io.ReadWriteCloser, handler Handler, signalHandler SignalHandler) (*Conn, error) {
|
|
||||||
return NewConn(genericTransport{conn}, WithHandler(handler), WithSignalHandler(signalHandler))
|
|
||||||
}
|
|
||||||
|
|
||||||
// newConn creates a new *Conn from a transport.
|
|
||||||
func newConn(tr transport, opts ...ConnOption) (*Conn, error) {
|
|
||||||
conn := new(Conn)
|
|
||||||
conn.transport = tr
|
|
||||||
for _, opt := range opts {
|
|
||||||
if err := opt(conn); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
conn.calls = newCallTracker()
|
|
||||||
if conn.handler == nil {
|
|
||||||
conn.handler = NewDefaultHandler()
|
|
||||||
}
|
|
||||||
if conn.signalHandler == nil {
|
|
||||||
conn.signalHandler = NewDefaultSignalHandler()
|
|
||||||
}
|
|
||||||
if conn.serialGen == nil {
|
|
||||||
conn.serialGen = newSerialGenerator()
|
|
||||||
}
|
|
||||||
conn.outHandler = &outputHandler{conn: conn}
|
|
||||||
conn.names = newNameTracker()
|
|
||||||
conn.busObj = conn.Object("org.freedesktop.DBus", "/org/freedesktop/DBus")
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// BusObject returns the object owned by the bus daemon which handles
|
|
||||||
// administrative requests.
|
|
||||||
func (conn *Conn) BusObject() BusObject {
|
|
||||||
return conn.busObj
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close closes the connection. Any blocked operations will return with errors
|
|
||||||
// and the channels passed to Eavesdrop and Signal are closed. This method must
|
|
||||||
// not be called on shared connections.
|
|
||||||
func (conn *Conn) Close() error {
|
|
||||||
conn.outHandler.close()
|
|
||||||
if term, ok := conn.signalHandler.(Terminator); ok {
|
|
||||||
term.Terminate()
|
|
||||||
}
|
|
||||||
|
|
||||||
if term, ok := conn.handler.(Terminator); ok {
|
|
||||||
term.Terminate()
|
|
||||||
}
|
|
||||||
|
|
||||||
conn.eavesdroppedLck.Lock()
|
|
||||||
if conn.eavesdropped != nil {
|
|
||||||
close(conn.eavesdropped)
|
|
||||||
}
|
|
||||||
conn.eavesdroppedLck.Unlock()
|
|
||||||
|
|
||||||
return conn.transport.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Eavesdrop causes conn to send all incoming messages to the given channel
|
|
||||||
// without further processing. Method replies, errors and signals will not be
|
|
||||||
// sent to the appropiate channels and method calls will not be handled. If nil
|
|
||||||
// is passed, the normal behaviour is restored.
|
|
||||||
//
|
|
||||||
// The caller has to make sure that ch is sufficiently buffered;
|
|
||||||
// if a message arrives when a write to ch is not possible, the message is
|
|
||||||
// discarded.
|
|
||||||
func (conn *Conn) Eavesdrop(ch chan<- *Message) {
|
|
||||||
conn.eavesdroppedLck.Lock()
|
|
||||||
conn.eavesdropped = ch
|
|
||||||
conn.eavesdroppedLck.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
// getSerial returns an unused serial.
|
|
||||||
func (conn *Conn) getSerial() uint32 {
|
|
||||||
return conn.serialGen.GetSerial()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hello sends the initial org.freedesktop.DBus.Hello call. This method must be
|
|
||||||
// called after authentication, but before sending any other messages to the
|
|
||||||
// bus. Hello must not be called for shared connections.
|
|
||||||
func (conn *Conn) Hello() error {
|
|
||||||
var s string
|
|
||||||
err := conn.busObj.Call("org.freedesktop.DBus.Hello", 0).Store(&s)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
conn.names.acquireUniqueConnectionName(s)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// inWorker runs in an own goroutine, reading incoming messages from the
|
|
||||||
// transport and dispatching them appropiately.
|
|
||||||
func (conn *Conn) inWorker() {
|
|
||||||
for {
|
|
||||||
msg, err := conn.ReadMessage()
|
|
||||||
if err != nil {
|
|
||||||
if _, ok := err.(InvalidMessageError); !ok {
|
|
||||||
// Some read error occured (usually EOF); we can't really do
|
|
||||||
// anything but to shut down all stuff and returns errors to all
|
|
||||||
// pending replies.
|
|
||||||
conn.Close()
|
|
||||||
conn.calls.finalizeAllWithError(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// invalid messages are ignored
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
conn.eavesdroppedLck.Lock()
|
|
||||||
if conn.eavesdropped != nil {
|
|
||||||
select {
|
|
||||||
case conn.eavesdropped <- msg:
|
|
||||||
default:
|
|
||||||
}
|
|
||||||
conn.eavesdroppedLck.Unlock()
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
conn.eavesdroppedLck.Unlock()
|
|
||||||
dest, _ := msg.Headers[FieldDestination].value.(string)
|
|
||||||
found := dest == "" ||
|
|
||||||
!conn.names.uniqueNameIsKnown() ||
|
|
||||||
conn.names.isKnownName(dest)
|
|
||||||
if !found {
|
|
||||||
// Eavesdropped a message, but no channel for it is registered.
|
|
||||||
// Ignore it.
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
switch msg.Type {
|
|
||||||
case TypeError:
|
|
||||||
conn.serialGen.RetireSerial(conn.calls.handleDBusError(msg))
|
|
||||||
case TypeMethodReply:
|
|
||||||
conn.serialGen.RetireSerial(conn.calls.handleReply(msg))
|
|
||||||
case TypeSignal:
|
|
||||||
conn.handleSignal(msg)
|
|
||||||
case TypeMethodCall:
|
|
||||||
go conn.handleCall(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (conn *Conn) handleSignal(msg *Message) {
|
|
||||||
iface := msg.Headers[FieldInterface].value.(string)
|
|
||||||
member := msg.Headers[FieldMember].value.(string)
|
|
||||||
// as per http://dbus.freedesktop.org/doc/dbus-specification.html ,
|
|
||||||
// sender is optional for signals.
|
|
||||||
sender, _ := msg.Headers[FieldSender].value.(string)
|
|
||||||
if iface == "org.freedesktop.DBus" && sender == "org.freedesktop.DBus" {
|
|
||||||
if member == "NameLost" {
|
|
||||||
// If we lost the name on the bus, remove it from our
|
|
||||||
// tracking list.
|
|
||||||
name, ok := msg.Body[0].(string)
|
|
||||||
if !ok {
|
|
||||||
panic("Unable to read the lost name")
|
|
||||||
}
|
|
||||||
conn.names.loseName(name)
|
|
||||||
} else if member == "NameAcquired" {
|
|
||||||
// If we acquired the name on the bus, add it to our
|
|
||||||
// tracking list.
|
|
||||||
name, ok := msg.Body[0].(string)
|
|
||||||
if !ok {
|
|
||||||
panic("Unable to read the acquired name")
|
|
||||||
}
|
|
||||||
conn.names.acquireName(name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
signal := &Signal{
|
|
||||||
Sender: sender,
|
|
||||||
Path: msg.Headers[FieldPath].value.(ObjectPath),
|
|
||||||
Name: iface + "." + member,
|
|
||||||
Body: msg.Body,
|
|
||||||
}
|
|
||||||
conn.signalHandler.DeliverSignal(iface, member, signal)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Names returns the list of all names that are currently owned by this
|
|
||||||
// connection. The slice is always at least one element long, the first element
|
|
||||||
// being the unique name of the connection.
|
|
||||||
func (conn *Conn) Names() []string {
|
|
||||||
return conn.names.listKnownNames()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Object returns the object identified by the given destination name and path.
|
|
||||||
func (conn *Conn) Object(dest string, path ObjectPath) BusObject {
|
|
||||||
return &Object{conn, dest, path}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (conn *Conn) sendMessage(msg *Message) {
|
|
||||||
conn.sendMessageAndIfClosed(msg, func() {})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (conn *Conn) sendMessageAndIfClosed(msg *Message, ifClosed func()) {
|
|
||||||
err := conn.outHandler.sendAndIfClosed(msg, ifClosed)
|
|
||||||
conn.calls.handleSendError(msg, err)
|
|
||||||
if err != nil {
|
|
||||||
conn.serialGen.RetireSerial(msg.serial)
|
|
||||||
} else if msg.Type != TypeMethodCall {
|
|
||||||
conn.serialGen.RetireSerial(msg.serial)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Send sends the given message to the message bus. You usually don't need to
|
|
||||||
// use this; use the higher-level equivalents (Call / Go, Emit and Export)
|
|
||||||
// instead. If msg is a method call and NoReplyExpected is not set, a non-nil
|
|
||||||
// call is returned and the same value is sent to ch (which must be buffered)
|
|
||||||
// once the call is complete. Otherwise, ch is ignored and a Call structure is
|
|
||||||
// returned of which only the Err member is valid.
|
|
||||||
func (conn *Conn) Send(msg *Message, ch chan *Call) *Call {
|
|
||||||
return conn.send(context.Background(), msg, ch)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SendWithContext acts like Send but takes a context
|
|
||||||
func (conn *Conn) SendWithContext(ctx context.Context, msg *Message, ch chan *Call) *Call {
|
|
||||||
return conn.send(ctx, msg, ch)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (conn *Conn) send(ctx context.Context, msg *Message, ch chan *Call) *Call {
|
|
||||||
if ctx == nil {
|
|
||||||
panic("nil context")
|
|
||||||
}
|
|
||||||
|
|
||||||
var call *Call
|
|
||||||
ctx, canceler := context.WithCancel(ctx)
|
|
||||||
msg.serial = conn.getSerial()
|
|
||||||
if msg.Type == TypeMethodCall && msg.Flags&FlagNoReplyExpected == 0 {
|
|
||||||
if ch == nil {
|
|
||||||
ch = make(chan *Call, 5)
|
|
||||||
} else if cap(ch) == 0 {
|
|
||||||
panic("dbus: unbuffered channel passed to (*Conn).Send")
|
|
||||||
}
|
|
||||||
call = new(Call)
|
|
||||||
call.Destination, _ = msg.Headers[FieldDestination].value.(string)
|
|
||||||
call.Path, _ = msg.Headers[FieldPath].value.(ObjectPath)
|
|
||||||
iface, _ := msg.Headers[FieldInterface].value.(string)
|
|
||||||
member, _ := msg.Headers[FieldMember].value.(string)
|
|
||||||
call.Method = iface + "." + member
|
|
||||||
call.Args = msg.Body
|
|
||||||
call.Done = ch
|
|
||||||
call.ctx = ctx
|
|
||||||
call.ctxCanceler = canceler
|
|
||||||
conn.calls.track(msg.serial, call)
|
|
||||||
go func() {
|
|
||||||
<-ctx.Done()
|
|
||||||
conn.calls.handleSendError(msg, ctx.Err())
|
|
||||||
}()
|
|
||||||
conn.sendMessageAndIfClosed(msg, func() {
|
|
||||||
conn.calls.handleSendError(msg, ErrClosed)
|
|
||||||
canceler()
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
canceler()
|
|
||||||
call = &Call{Err: nil}
|
|
||||||
conn.sendMessageAndIfClosed(msg, func() {
|
|
||||||
call = &Call{Err: ErrClosed}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return call
|
|
||||||
}
|
|
||||||
|
|
||||||
// sendError creates an error message corresponding to the parameters and sends
|
|
||||||
// it to conn.out.
|
|
||||||
func (conn *Conn) sendError(err error, dest string, serial uint32) {
|
|
||||||
var e *Error
|
|
||||||
switch em := err.(type) {
|
|
||||||
case Error:
|
|
||||||
e = &em
|
|
||||||
case *Error:
|
|
||||||
e = em
|
|
||||||
case DBusError:
|
|
||||||
name, body := em.DBusError()
|
|
||||||
e = NewError(name, body)
|
|
||||||
default:
|
|
||||||
e = MakeFailedError(err)
|
|
||||||
}
|
|
||||||
msg := new(Message)
|
|
||||||
msg.Type = TypeError
|
|
||||||
msg.serial = conn.getSerial()
|
|
||||||
msg.Headers = make(map[HeaderField]Variant)
|
|
||||||
if dest != "" {
|
|
||||||
msg.Headers[FieldDestination] = MakeVariant(dest)
|
|
||||||
}
|
|
||||||
msg.Headers[FieldErrorName] = MakeVariant(e.Name)
|
|
||||||
msg.Headers[FieldReplySerial] = MakeVariant(serial)
|
|
||||||
msg.Body = e.Body
|
|
||||||
if len(e.Body) > 0 {
|
|
||||||
msg.Headers[FieldSignature] = MakeVariant(SignatureOf(e.Body...))
|
|
||||||
}
|
|
||||||
conn.sendMessage(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
// sendReply creates a method reply message corresponding to the parameters and
|
|
||||||
// sends it to conn.out.
|
|
||||||
func (conn *Conn) sendReply(dest string, serial uint32, values ...interface{}) {
|
|
||||||
msg := new(Message)
|
|
||||||
msg.Type = TypeMethodReply
|
|
||||||
msg.serial = conn.getSerial()
|
|
||||||
msg.Headers = make(map[HeaderField]Variant)
|
|
||||||
if dest != "" {
|
|
||||||
msg.Headers[FieldDestination] = MakeVariant(dest)
|
|
||||||
}
|
|
||||||
msg.Headers[FieldReplySerial] = MakeVariant(serial)
|
|
||||||
msg.Body = values
|
|
||||||
if len(values) > 0 {
|
|
||||||
msg.Headers[FieldSignature] = MakeVariant(SignatureOf(values...))
|
|
||||||
}
|
|
||||||
conn.sendMessage(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (conn *Conn) defaultSignalAction(fn func(h *defaultSignalHandler, ch chan<- *Signal), ch chan<- *Signal) {
|
|
||||||
if !isDefaultSignalHandler(conn.signalHandler) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
handler := conn.signalHandler.(*defaultSignalHandler)
|
|
||||||
fn(handler, ch)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Signal registers the given channel to be passed all received signal messages.
|
|
||||||
// The caller has to make sure that ch is sufficiently buffered; if a message
|
|
||||||
// arrives when a write to c is not possible, it is discarded.
|
|
||||||
//
|
|
||||||
// Multiple of these channels can be registered at the same time.
|
|
||||||
//
|
|
||||||
// These channels are "overwritten" by Eavesdrop; i.e., if there currently is a
|
|
||||||
// channel for eavesdropped messages, this channel receives all signals, and
|
|
||||||
// none of the channels passed to Signal will receive any signals.
|
|
||||||
func (conn *Conn) Signal(ch chan<- *Signal) {
|
|
||||||
conn.defaultSignalAction((*defaultSignalHandler).addSignal, ch)
|
|
||||||
}
|
|
||||||
|
|
||||||
// RemoveSignal removes the given channel from the list of the registered channels.
|
|
||||||
func (conn *Conn) RemoveSignal(ch chan<- *Signal) {
|
|
||||||
conn.defaultSignalAction((*defaultSignalHandler).removeSignal, ch)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SupportsUnixFDs returns whether the underlying transport supports passing of
|
|
||||||
// unix file descriptors. If this is false, method calls containing unix file
|
|
||||||
// descriptors will return an error and emitted signals containing them will
|
|
||||||
// not be sent.
|
|
||||||
func (conn *Conn) SupportsUnixFDs() bool {
|
|
||||||
return conn.unixFD
|
|
||||||
}
|
|
||||||
|
|
||||||
// Error represents a D-Bus message of type Error.
|
|
||||||
type Error struct {
|
|
||||||
Name string
|
|
||||||
Body []interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewError(name string, body []interface{}) *Error {
|
|
||||||
return &Error{name, body}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e Error) Error() string {
|
|
||||||
if len(e.Body) >= 1 {
|
|
||||||
s, ok := e.Body[0].(string)
|
|
||||||
if ok {
|
|
||||||
return s
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return e.Name
|
|
||||||
}
|
|
||||||
|
|
||||||
// Signal represents a D-Bus message of type Signal. The name member is given in
|
|
||||||
// "interface.member" notation, e.g. org.freedesktop.D-Bus.NameLost.
|
|
||||||
type Signal struct {
|
|
||||||
Sender string
|
|
||||||
Path ObjectPath
|
|
||||||
Name string
|
|
||||||
Body []interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// transport is a D-Bus transport.
|
|
||||||
type transport interface {
|
|
||||||
// Read and Write raw data (for example, for the authentication protocol).
|
|
||||||
io.ReadWriteCloser
|
|
||||||
|
|
||||||
// Send the initial null byte used for the EXTERNAL mechanism.
|
|
||||||
SendNullByte() error
|
|
||||||
|
|
||||||
// Returns whether this transport supports passing Unix FDs.
|
|
||||||
SupportsUnixFDs() bool
|
|
||||||
|
|
||||||
// Signal the transport that Unix FD passing is enabled for this connection.
|
|
||||||
EnableUnixFDs()
|
|
||||||
|
|
||||||
// Read / send a message, handling things like Unix FDs.
|
|
||||||
ReadMessage() (*Message, error)
|
|
||||||
SendMessage(*Message) error
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
transports = make(map[string]func(string) (transport, error))
|
|
||||||
)
|
|
||||||
|
|
||||||
func getTransport(address string) (transport, error) {
|
|
||||||
var err error
|
|
||||||
var t transport
|
|
||||||
|
|
||||||
addresses := strings.Split(address, ";")
|
|
||||||
for _, v := range addresses {
|
|
||||||
i := strings.IndexRune(v, ':')
|
|
||||||
if i == -1 {
|
|
||||||
err = errors.New("dbus: invalid bus address (no transport)")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
f := transports[v[:i]]
|
|
||||||
if f == nil {
|
|
||||||
err = errors.New("dbus: invalid bus address (invalid or unsupported transport)")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
t, err = f(v[i+1:])
|
|
||||||
if err == nil {
|
|
||||||
return t, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// dereferenceAll returns a slice that, assuming that vs is a slice of pointers
|
|
||||||
// of arbitrary types, containes the values that are obtained from dereferencing
|
|
||||||
// all elements in vs.
|
|
||||||
func dereferenceAll(vs []interface{}) []interface{} {
|
|
||||||
for i := range vs {
|
|
||||||
v := reflect.ValueOf(vs[i])
|
|
||||||
v = v.Elem()
|
|
||||||
vs[i] = v.Interface()
|
|
||||||
}
|
|
||||||
return vs
|
|
||||||
}
|
|
||||||
|
|
||||||
// getKey gets a key from a the list of keys. Returns "" on error / not found...
|
|
||||||
func getKey(s, key string) string {
|
|
||||||
for _, keyEqualsValue := range strings.Split(s, ",") {
|
|
||||||
keyValue := strings.SplitN(keyEqualsValue, "=", 2)
|
|
||||||
if len(keyValue) == 2 && keyValue[0] == key {
|
|
||||||
return keyValue[1]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
type outputHandler struct {
|
|
||||||
conn *Conn
|
|
||||||
sendLck sync.Mutex
|
|
||||||
closed struct {
|
|
||||||
isClosed bool
|
|
||||||
lck sync.RWMutex
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *outputHandler) sendAndIfClosed(msg *Message, ifClosed func()) error {
|
|
||||||
h.closed.lck.RLock()
|
|
||||||
defer h.closed.lck.RUnlock()
|
|
||||||
if h.closed.isClosed {
|
|
||||||
ifClosed()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
h.sendLck.Lock()
|
|
||||||
defer h.sendLck.Unlock()
|
|
||||||
return h.conn.SendMessage(msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *outputHandler) close() {
|
|
||||||
h.closed.lck.Lock()
|
|
||||||
defer h.closed.lck.Unlock()
|
|
||||||
h.closed.isClosed = true
|
|
||||||
}
|
|
||||||
|
|
||||||
type serialGenerator struct {
|
|
||||||
lck sync.Mutex
|
|
||||||
nextSerial uint32
|
|
||||||
serialUsed map[uint32]bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func newSerialGenerator() *serialGenerator {
|
|
||||||
return &serialGenerator{
|
|
||||||
serialUsed: map[uint32]bool{0: true},
|
|
||||||
nextSerial: 1,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gen *serialGenerator) GetSerial() uint32 {
|
|
||||||
gen.lck.Lock()
|
|
||||||
defer gen.lck.Unlock()
|
|
||||||
n := gen.nextSerial
|
|
||||||
for gen.serialUsed[n] {
|
|
||||||
n++
|
|
||||||
}
|
|
||||||
gen.serialUsed[n] = true
|
|
||||||
gen.nextSerial = n + 1
|
|
||||||
return n
|
|
||||||
}
|
|
||||||
|
|
||||||
func (gen *serialGenerator) RetireSerial(serial uint32) {
|
|
||||||
gen.lck.Lock()
|
|
||||||
defer gen.lck.Unlock()
|
|
||||||
delete(gen.serialUsed, serial)
|
|
||||||
}
|
|
||||||
|
|
||||||
type nameTracker struct {
|
|
||||||
lck sync.RWMutex
|
|
||||||
unique string
|
|
||||||
names map[string]struct{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func newNameTracker() *nameTracker {
|
|
||||||
return &nameTracker{names: map[string]struct{}{}}
|
|
||||||
}
|
|
||||||
func (tracker *nameTracker) acquireUniqueConnectionName(name string) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
defer tracker.lck.Unlock()
|
|
||||||
tracker.unique = name
|
|
||||||
}
|
|
||||||
func (tracker *nameTracker) acquireName(name string) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
defer tracker.lck.Unlock()
|
|
||||||
tracker.names[name] = struct{}{}
|
|
||||||
}
|
|
||||||
func (tracker *nameTracker) loseName(name string) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
defer tracker.lck.Unlock()
|
|
||||||
delete(tracker.names, name)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *nameTracker) uniqueNameIsKnown() bool {
|
|
||||||
tracker.lck.RLock()
|
|
||||||
defer tracker.lck.RUnlock()
|
|
||||||
return tracker.unique != ""
|
|
||||||
}
|
|
||||||
func (tracker *nameTracker) isKnownName(name string) bool {
|
|
||||||
tracker.lck.RLock()
|
|
||||||
defer tracker.lck.RUnlock()
|
|
||||||
_, ok := tracker.names[name]
|
|
||||||
return ok || name == tracker.unique
|
|
||||||
}
|
|
||||||
func (tracker *nameTracker) listKnownNames() []string {
|
|
||||||
tracker.lck.RLock()
|
|
||||||
defer tracker.lck.RUnlock()
|
|
||||||
out := make([]string, 0, len(tracker.names)+1)
|
|
||||||
out = append(out, tracker.unique)
|
|
||||||
for k := range tracker.names {
|
|
||||||
out = append(out, k)
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
type callTracker struct {
|
|
||||||
calls map[uint32]*Call
|
|
||||||
lck sync.RWMutex
|
|
||||||
}
|
|
||||||
|
|
||||||
func newCallTracker() *callTracker {
|
|
||||||
return &callTracker{calls: map[uint32]*Call{}}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) track(sn uint32, call *Call) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
tracker.calls[sn] = call
|
|
||||||
tracker.lck.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) handleReply(msg *Message) uint32 {
|
|
||||||
serial := msg.Headers[FieldReplySerial].value.(uint32)
|
|
||||||
tracker.lck.RLock()
|
|
||||||
_, ok := tracker.calls[serial]
|
|
||||||
tracker.lck.RUnlock()
|
|
||||||
if ok {
|
|
||||||
tracker.finalizeWithBody(serial, msg.Body)
|
|
||||||
}
|
|
||||||
return serial
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) handleDBusError(msg *Message) uint32 {
|
|
||||||
serial := msg.Headers[FieldReplySerial].value.(uint32)
|
|
||||||
tracker.lck.RLock()
|
|
||||||
_, ok := tracker.calls[serial]
|
|
||||||
tracker.lck.RUnlock()
|
|
||||||
if ok {
|
|
||||||
name, _ := msg.Headers[FieldErrorName].value.(string)
|
|
||||||
tracker.finalizeWithError(serial, Error{name, msg.Body})
|
|
||||||
}
|
|
||||||
return serial
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) handleSendError(msg *Message, err error) {
|
|
||||||
if err == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
tracker.lck.RLock()
|
|
||||||
_, ok := tracker.calls[msg.serial]
|
|
||||||
tracker.lck.RUnlock()
|
|
||||||
if ok {
|
|
||||||
tracker.finalizeWithError(msg.serial, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// finalize was the only func that did not strobe Done
|
|
||||||
func (tracker *callTracker) finalize(sn uint32) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
defer tracker.lck.Unlock()
|
|
||||||
c, ok := tracker.calls[sn]
|
|
||||||
if ok {
|
|
||||||
delete(tracker.calls, sn)
|
|
||||||
c.ContextCancel()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) finalizeWithBody(sn uint32, body []interface{}) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
c, ok := tracker.calls[sn]
|
|
||||||
if ok {
|
|
||||||
delete(tracker.calls, sn)
|
|
||||||
}
|
|
||||||
tracker.lck.Unlock()
|
|
||||||
if ok {
|
|
||||||
c.Body = body
|
|
||||||
c.done()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) finalizeWithError(sn uint32, err error) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
c, ok := tracker.calls[sn]
|
|
||||||
if ok {
|
|
||||||
delete(tracker.calls, sn)
|
|
||||||
}
|
|
||||||
tracker.lck.Unlock()
|
|
||||||
if ok {
|
|
||||||
c.Err = err
|
|
||||||
c.done()
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (tracker *callTracker) finalizeAllWithError(err error) {
|
|
||||||
tracker.lck.Lock()
|
|
||||||
closedCalls := make([]*Call, 0, len(tracker.calls))
|
|
||||||
for sn := range tracker.calls {
|
|
||||||
closedCalls = append(closedCalls, tracker.calls[sn])
|
|
||||||
}
|
|
||||||
tracker.calls = map[uint32]*Call{}
|
|
||||||
tracker.lck.Unlock()
|
|
||||||
for _, call := range closedCalls {
|
|
||||||
call.Err = err
|
|
||||||
call.done()
|
|
||||||
}
|
|
||||||
}
|
|
37
vendor/github.com/godbus/dbus/conn_darwin.go
generated
vendored
37
vendor/github.com/godbus/dbus/conn_darwin.go
generated
vendored
|
@ -1,37 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
)
|
|
||||||
|
|
||||||
const defaultSystemBusAddress = "unix:path=/opt/local/var/run/dbus/system_bus_socket"
|
|
||||||
|
|
||||||
func getSessionBusPlatformAddress() (string, error) {
|
|
||||||
cmd := exec.Command("launchctl", "getenv", "DBUS_LAUNCHD_SESSION_BUS_SOCKET")
|
|
||||||
b, err := cmd.CombinedOutput()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(b) == 0 {
|
|
||||||
return "", errors.New("dbus: couldn't determine address of session bus")
|
|
||||||
}
|
|
||||||
|
|
||||||
return "unix:path=" + string(b[:len(b)-1]), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func getSystemBusPlatformAddress() string {
|
|
||||||
address := os.Getenv("DBUS_LAUNCHD_SESSION_BUS_SOCKET")
|
|
||||||
if address != "" {
|
|
||||||
return fmt.Sprintf("unix:path=%s", address)
|
|
||||||
}
|
|
||||||
return defaultSystemBusAddress
|
|
||||||
}
|
|
||||||
|
|
||||||
func tryDiscoverDbusSessionBusAddress() string {
|
|
||||||
return ""
|
|
||||||
}
|
|
91
vendor/github.com/godbus/dbus/conn_other.go
generated
vendored
91
vendor/github.com/godbus/dbus/conn_other.go
generated
vendored
|
@ -1,91 +0,0 @@
|
||||||
// +build !darwin
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
"os/user"
|
|
||||||
"path"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
func getSessionBusPlatformAddress() (string, error) {
|
|
||||||
cmd := exec.Command("dbus-launch")
|
|
||||||
b, err := cmd.CombinedOutput()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
i := bytes.IndexByte(b, '=')
|
|
||||||
j := bytes.IndexByte(b, '\n')
|
|
||||||
|
|
||||||
if i == -1 || j == -1 {
|
|
||||||
return "", errors.New("dbus: couldn't determine address of session bus")
|
|
||||||
}
|
|
||||||
|
|
||||||
env, addr := string(b[0:i]), string(b[i+1:j])
|
|
||||||
os.Setenv(env, addr)
|
|
||||||
|
|
||||||
return addr, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// tryDiscoverDbusSessionBusAddress tries to discover an existing dbus session
|
|
||||||
// and return the value of its DBUS_SESSION_BUS_ADDRESS.
|
|
||||||
// It tries different techniques employed by different operating systems,
|
|
||||||
// returning the first valid address it finds, or an empty string.
|
|
||||||
//
|
|
||||||
// * /run/user/<uid>/bus if this exists, it *is* the bus socket. present on
|
|
||||||
// Ubuntu 18.04
|
|
||||||
// * /run/user/<uid>/dbus-session: if this exists, it can be parsed for the bus
|
|
||||||
// address. present on Ubuntu 16.04
|
|
||||||
//
|
|
||||||
// See https://dbus.freedesktop.org/doc/dbus-launch.1.html
|
|
||||||
func tryDiscoverDbusSessionBusAddress() string {
|
|
||||||
if runtimeDirectory, err := getRuntimeDirectory(); err == nil {
|
|
||||||
|
|
||||||
if runUserBusFile := path.Join(runtimeDirectory, "bus"); fileExists(runUserBusFile) {
|
|
||||||
// if /run/user/<uid>/bus exists, that file itself
|
|
||||||
// *is* the unix socket, so return its path
|
|
||||||
return fmt.Sprintf("unix:path=%s", runUserBusFile)
|
|
||||||
}
|
|
||||||
if runUserSessionDbusFile := path.Join(runtimeDirectory, "dbus-session"); fileExists(runUserSessionDbusFile) {
|
|
||||||
// if /run/user/<uid>/dbus-session exists, it's a
|
|
||||||
// text file // containing the address of the socket, e.g.:
|
|
||||||
// DBUS_SESSION_BUS_ADDRESS=unix:abstract=/tmp/dbus-E1c73yNqrG
|
|
||||||
|
|
||||||
if f, err := ioutil.ReadFile(runUserSessionDbusFile); err == nil {
|
|
||||||
fileContent := string(f)
|
|
||||||
|
|
||||||
prefix := "DBUS_SESSION_BUS_ADDRESS="
|
|
||||||
|
|
||||||
if strings.HasPrefix(fileContent, prefix) {
|
|
||||||
address := strings.TrimRight(strings.TrimPrefix(fileContent, prefix), "\n\r")
|
|
||||||
return address
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
func getRuntimeDirectory() (string, error) {
|
|
||||||
if currentUser, err := user.Current(); err != nil {
|
|
||||||
return "", err
|
|
||||||
} else {
|
|
||||||
return fmt.Sprintf("/run/user/%s", currentUser.Uid), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func fileExists(filename string) bool {
|
|
||||||
if _, err := os.Stat(filename); !os.IsNotExist(err) {
|
|
||||||
return true
|
|
||||||
} else {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
18
vendor/github.com/godbus/dbus/conn_unix.go
generated
vendored
18
vendor/github.com/godbus/dbus/conn_unix.go
generated
vendored
|
@ -1,18 +0,0 @@
|
||||||
//+build !windows,!solaris,!darwin
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"fmt"
|
|
||||||
)
|
|
||||||
|
|
||||||
const defaultSystemBusAddress = "unix:path=/var/run/dbus/system_bus_socket"
|
|
||||||
|
|
||||||
func getSystemBusPlatformAddress() string {
|
|
||||||
address := os.Getenv("DBUS_SYSTEM_BUS_ADDRESS")
|
|
||||||
if address != "" {
|
|
||||||
return fmt.Sprintf("unix:path=%s", address)
|
|
||||||
}
|
|
||||||
return defaultSystemBusAddress
|
|
||||||
}
|
|
15
vendor/github.com/godbus/dbus/conn_windows.go
generated
vendored
15
vendor/github.com/godbus/dbus/conn_windows.go
generated
vendored
|
@ -1,15 +0,0 @@
|
||||||
//+build windows
|
|
||||||
|
|
||||||
package dbus
|
|
||||||
|
|
||||||
import "os"
|
|
||||||
|
|
||||||
const defaultSystemBusAddress = "tcp:host=127.0.0.1,port=12434"
|
|
||||||
|
|
||||||
func getSystemBusPlatformAddress() string {
|
|
||||||
address := os.Getenv("DBUS_SYSTEM_BUS_ADDRESS")
|
|
||||||
if address != "" {
|
|
||||||
return address
|
|
||||||
}
|
|
||||||
return defaultSystemBusAddress
|
|
||||||
}
|
|
427
vendor/github.com/godbus/dbus/dbus.go
generated
vendored
427
vendor/github.com/godbus/dbus/dbus.go
generated
vendored
|
@ -1,427 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
byteType = reflect.TypeOf(byte(0))
|
|
||||||
boolType = reflect.TypeOf(false)
|
|
||||||
uint8Type = reflect.TypeOf(uint8(0))
|
|
||||||
int16Type = reflect.TypeOf(int16(0))
|
|
||||||
uint16Type = reflect.TypeOf(uint16(0))
|
|
||||||
intType = reflect.TypeOf(int(0))
|
|
||||||
uintType = reflect.TypeOf(uint(0))
|
|
||||||
int32Type = reflect.TypeOf(int32(0))
|
|
||||||
uint32Type = reflect.TypeOf(uint32(0))
|
|
||||||
int64Type = reflect.TypeOf(int64(0))
|
|
||||||
uint64Type = reflect.TypeOf(uint64(0))
|
|
||||||
float64Type = reflect.TypeOf(float64(0))
|
|
||||||
stringType = reflect.TypeOf("")
|
|
||||||
signatureType = reflect.TypeOf(Signature{""})
|
|
||||||
objectPathType = reflect.TypeOf(ObjectPath(""))
|
|
||||||
variantType = reflect.TypeOf(Variant{Signature{""}, nil})
|
|
||||||
interfacesType = reflect.TypeOf([]interface{}{})
|
|
||||||
interfaceType = reflect.TypeOf((*interface{})(nil)).Elem()
|
|
||||||
unixFDType = reflect.TypeOf(UnixFD(0))
|
|
||||||
unixFDIndexType = reflect.TypeOf(UnixFDIndex(0))
|
|
||||||
)
|
|
||||||
|
|
||||||
// An InvalidTypeError signals that a value which cannot be represented in the
|
|
||||||
// D-Bus wire format was passed to a function.
|
|
||||||
type InvalidTypeError struct {
|
|
||||||
Type reflect.Type
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e InvalidTypeError) Error() string {
|
|
||||||
return "dbus: invalid type " + e.Type.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store copies the values contained in src to dest, which must be a slice of
|
|
||||||
// pointers. It converts slices of interfaces from src to corresponding structs
|
|
||||||
// in dest. An error is returned if the lengths of src and dest or the types of
|
|
||||||
// their elements don't match.
|
|
||||||
func Store(src []interface{}, dest ...interface{}) error {
|
|
||||||
if len(src) != len(dest) {
|
|
||||||
return errors.New("dbus.Store: length mismatch")
|
|
||||||
}
|
|
||||||
|
|
||||||
for i := range src {
|
|
||||||
if err := storeInterfaces(src[i], dest[i]); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeInterfaces(src, dest interface{}) error {
|
|
||||||
return store(reflect.ValueOf(dest), reflect.ValueOf(src))
|
|
||||||
}
|
|
||||||
|
|
||||||
func store(dest, src reflect.Value) error {
|
|
||||||
if dest.Kind() == reflect.Ptr {
|
|
||||||
return store(dest.Elem(), src)
|
|
||||||
}
|
|
||||||
switch src.Kind() {
|
|
||||||
case reflect.Slice:
|
|
||||||
return storeSlice(dest, src)
|
|
||||||
case reflect.Map:
|
|
||||||
return storeMap(dest, src)
|
|
||||||
default:
|
|
||||||
return storeBase(dest, src)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeBase(dest, src reflect.Value) error {
|
|
||||||
return setDest(dest, src)
|
|
||||||
}
|
|
||||||
|
|
||||||
func setDest(dest, src reflect.Value) error {
|
|
||||||
if !isVariant(src.Type()) && isVariant(dest.Type()) {
|
|
||||||
//special conversion for dbus.Variant
|
|
||||||
dest.Set(reflect.ValueOf(MakeVariant(src.Interface())))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if isVariant(src.Type()) && !isVariant(dest.Type()) {
|
|
||||||
src = getVariantValue(src)
|
|
||||||
}
|
|
||||||
if !src.Type().ConvertibleTo(dest.Type()) {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: cannot convert %s to %s",
|
|
||||||
src.Type(), dest.Type())
|
|
||||||
}
|
|
||||||
dest.Set(src.Convert(dest.Type()))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func kindsAreCompatible(dest, src reflect.Type) bool {
|
|
||||||
switch {
|
|
||||||
case isVariant(dest):
|
|
||||||
return true
|
|
||||||
case dest.Kind() == reflect.Interface:
|
|
||||||
return true
|
|
||||||
default:
|
|
||||||
return dest.Kind() == src.Kind()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func isConvertibleTo(dest, src reflect.Type) bool {
|
|
||||||
switch {
|
|
||||||
case isVariant(dest):
|
|
||||||
return true
|
|
||||||
case dest.Kind() == reflect.Interface:
|
|
||||||
return true
|
|
||||||
case dest.Kind() == reflect.Slice:
|
|
||||||
return src.Kind() == reflect.Slice &&
|
|
||||||
isConvertibleTo(dest.Elem(), src.Elem())
|
|
||||||
case dest.Kind() == reflect.Struct:
|
|
||||||
return src == interfacesType
|
|
||||||
default:
|
|
||||||
return src.ConvertibleTo(dest)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeMap(dest, src reflect.Value) error {
|
|
||||||
switch {
|
|
||||||
case !kindsAreCompatible(dest.Type(), src.Type()):
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"map: cannot store a value of %s into %s",
|
|
||||||
src.Type(), dest.Type())
|
|
||||||
case isVariant(dest.Type()):
|
|
||||||
return storeMapIntoVariant(dest, src)
|
|
||||||
case dest.Kind() == reflect.Interface:
|
|
||||||
return storeMapIntoInterface(dest, src)
|
|
||||||
case isConvertibleTo(dest.Type().Key(), src.Type().Key()) &&
|
|
||||||
isConvertibleTo(dest.Type().Elem(), src.Type().Elem()):
|
|
||||||
return storeMapIntoMap(dest, src)
|
|
||||||
default:
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"map: cannot convert a value of %s into %s",
|
|
||||||
src.Type(), dest.Type())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeMapIntoVariant(dest, src reflect.Value) error {
|
|
||||||
dv := reflect.MakeMap(src.Type())
|
|
||||||
err := store(dv, src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return storeBase(dest, dv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeMapIntoInterface(dest, src reflect.Value) error {
|
|
||||||
var dv reflect.Value
|
|
||||||
if isVariant(src.Type().Elem()) {
|
|
||||||
//Convert variants to interface{} recursively when converting
|
|
||||||
//to interface{}
|
|
||||||
dv = reflect.MakeMap(
|
|
||||||
reflect.MapOf(src.Type().Key(), interfaceType))
|
|
||||||
} else {
|
|
||||||
dv = reflect.MakeMap(src.Type())
|
|
||||||
}
|
|
||||||
err := store(dv, src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return storeBase(dest, dv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeMapIntoMap(dest, src reflect.Value) error {
|
|
||||||
if dest.IsNil() {
|
|
||||||
dest.Set(reflect.MakeMap(dest.Type()))
|
|
||||||
}
|
|
||||||
keys := src.MapKeys()
|
|
||||||
for _, key := range keys {
|
|
||||||
dkey := key.Convert(dest.Type().Key())
|
|
||||||
dval := reflect.New(dest.Type().Elem()).Elem()
|
|
||||||
err := store(dval, getVariantValue(src.MapIndex(key)))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dest.SetMapIndex(dkey, dval)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeSlice(dest, src reflect.Value) error {
|
|
||||||
switch {
|
|
||||||
case src.Type() == interfacesType && dest.Kind() == reflect.Struct:
|
|
||||||
//The decoder always decodes structs as slices of interface{}
|
|
||||||
return storeStruct(dest, src)
|
|
||||||
case !kindsAreCompatible(dest.Type(), src.Type()):
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"slice: cannot store a value of %s into %s",
|
|
||||||
src.Type(), dest.Type())
|
|
||||||
case isVariant(dest.Type()):
|
|
||||||
return storeSliceIntoVariant(dest, src)
|
|
||||||
case dest.Kind() == reflect.Interface:
|
|
||||||
return storeSliceIntoInterface(dest, src)
|
|
||||||
case isConvertibleTo(dest.Type().Elem(), src.Type().Elem()):
|
|
||||||
return storeSliceIntoSlice(dest, src)
|
|
||||||
default:
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"slice: cannot convert a value of %s into %s",
|
|
||||||
src.Type(), dest.Type())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeStruct(dest, src reflect.Value) error {
|
|
||||||
if isVariant(dest.Type()) {
|
|
||||||
return storeBase(dest, src)
|
|
||||||
}
|
|
||||||
dval := make([]interface{}, 0, dest.NumField())
|
|
||||||
dtype := dest.Type()
|
|
||||||
for i := 0; i < dest.NumField(); i++ {
|
|
||||||
field := dest.Field(i)
|
|
||||||
ftype := dtype.Field(i)
|
|
||||||
if ftype.PkgPath != "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if ftype.Tag.Get("dbus") == "-" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
dval = append(dval, field.Addr().Interface())
|
|
||||||
}
|
|
||||||
if src.Len() != len(dval) {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"destination struct does not have "+
|
|
||||||
"enough fields need: %d have: %d",
|
|
||||||
src.Len(), len(dval))
|
|
||||||
}
|
|
||||||
return Store(src.Interface().([]interface{}), dval...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeSliceIntoVariant(dest, src reflect.Value) error {
|
|
||||||
dv := reflect.MakeSlice(src.Type(), src.Len(), src.Cap())
|
|
||||||
err := store(dv, src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return storeBase(dest, dv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeSliceIntoInterface(dest, src reflect.Value) error {
|
|
||||||
var dv reflect.Value
|
|
||||||
if isVariant(src.Type().Elem()) {
|
|
||||||
//Convert variants to interface{} recursively when converting
|
|
||||||
//to interface{}
|
|
||||||
dv = reflect.MakeSlice(reflect.SliceOf(interfaceType),
|
|
||||||
src.Len(), src.Cap())
|
|
||||||
} else {
|
|
||||||
dv = reflect.MakeSlice(src.Type(), src.Len(), src.Cap())
|
|
||||||
}
|
|
||||||
err := store(dv, src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return storeBase(dest, dv)
|
|
||||||
}
|
|
||||||
|
|
||||||
func storeSliceIntoSlice(dest, src reflect.Value) error {
|
|
||||||
if dest.IsNil() || dest.Len() < src.Len() {
|
|
||||||
dest.Set(reflect.MakeSlice(dest.Type(), src.Len(), src.Cap()))
|
|
||||||
}
|
|
||||||
if dest.Len() != src.Len() {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"dbus.Store: type mismatch: "+
|
|
||||||
"slices are different lengths "+
|
|
||||||
"need: %d have: %d",
|
|
||||||
src.Len(), dest.Len())
|
|
||||||
}
|
|
||||||
for i := 0; i < src.Len(); i++ {
|
|
||||||
err := store(dest.Index(i), getVariantValue(src.Index(i)))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func getVariantValue(in reflect.Value) reflect.Value {
|
|
||||||
if isVariant(in.Type()) {
|
|
||||||
return reflect.ValueOf(in.Interface().(Variant).Value())
|
|
||||||
}
|
|
||||||
return in
|
|
||||||
}
|
|
||||||
|
|
||||||
func isVariant(t reflect.Type) bool {
|
|
||||||
return t == variantType
|
|
||||||
}
|
|
||||||
|
|
||||||
// An ObjectPath is an object path as defined by the D-Bus spec.
|
|
||||||
type ObjectPath string
|
|
||||||
|
|
||||||
// IsValid returns whether the object path is valid.
|
|
||||||
func (o ObjectPath) IsValid() bool {
|
|
||||||
s := string(o)
|
|
||||||
if len(s) == 0 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if s[0] != '/' {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if s[len(s)-1] == '/' && len(s) != 1 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
// probably not used, but technically possible
|
|
||||||
if s == "/" {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
split := strings.Split(s[1:], "/")
|
|
||||||
for _, v := range split {
|
|
||||||
if len(v) == 0 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, c := range v {
|
|
||||||
if !isMemberChar(c) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// A UnixFD is a Unix file descriptor sent over the wire. See the package-level
|
|
||||||
// documentation for more information about Unix file descriptor passsing.
|
|
||||||
type UnixFD int32
|
|
||||||
|
|
||||||
// A UnixFDIndex is the representation of a Unix file descriptor in a message.
|
|
||||||
type UnixFDIndex uint32
|
|
||||||
|
|
||||||
// alignment returns the alignment of values of type t.
|
|
||||||
func alignment(t reflect.Type) int {
|
|
||||||
switch t {
|
|
||||||
case variantType:
|
|
||||||
return 1
|
|
||||||
case objectPathType:
|
|
||||||
return 4
|
|
||||||
case signatureType:
|
|
||||||
return 1
|
|
||||||
case interfacesType:
|
|
||||||
return 4
|
|
||||||
}
|
|
||||||
switch t.Kind() {
|
|
||||||
case reflect.Uint8:
|
|
||||||
return 1
|
|
||||||
case reflect.Uint16, reflect.Int16:
|
|
||||||
return 2
|
|
||||||
case reflect.Uint, reflect.Int, reflect.Uint32, reflect.Int32, reflect.String, reflect.Array, reflect.Slice, reflect.Map:
|
|
||||||
return 4
|
|
||||||
case reflect.Uint64, reflect.Int64, reflect.Float64, reflect.Struct:
|
|
||||||
return 8
|
|
||||||
case reflect.Ptr:
|
|
||||||
return alignment(t.Elem())
|
|
||||||
}
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
// isKeyType returns whether t is a valid type for a D-Bus dict.
|
|
||||||
func isKeyType(t reflect.Type) bool {
|
|
||||||
switch t.Kind() {
|
|
||||||
case reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64,
|
|
||||||
reflect.Int16, reflect.Int32, reflect.Int64, reflect.Float64,
|
|
||||||
reflect.String, reflect.Uint, reflect.Int:
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// isValidInterface returns whether s is a valid name for an interface.
|
|
||||||
func isValidInterface(s string) bool {
|
|
||||||
if len(s) == 0 || len(s) > 255 || s[0] == '.' {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
elem := strings.Split(s, ".")
|
|
||||||
if len(elem) < 2 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, v := range elem {
|
|
||||||
if len(v) == 0 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if v[0] >= '0' && v[0] <= '9' {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, c := range v {
|
|
||||||
if !isMemberChar(c) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// isValidMember returns whether s is a valid name for a member.
|
|
||||||
func isValidMember(s string) bool {
|
|
||||||
if len(s) == 0 || len(s) > 255 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
i := strings.Index(s, ".")
|
|
||||||
if i != -1 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if s[0] >= '0' && s[0] <= '9' {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, c := range s {
|
|
||||||
if !isMemberChar(c) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func isMemberChar(c rune) bool {
|
|
||||||
return (c >= '0' && c <= '9') || (c >= 'A' && c <= 'Z') ||
|
|
||||||
(c >= 'a' && c <= 'z') || c == '_'
|
|
||||||
}
|
|
235
vendor/github.com/godbus/dbus/decoder.go
generated
vendored
235
vendor/github.com/godbus/dbus/decoder.go
generated
vendored
|
@ -1,235 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/binary"
|
|
||||||
"io"
|
|
||||||
"reflect"
|
|
||||||
)
|
|
||||||
|
|
||||||
type decoder struct {
|
|
||||||
in io.Reader
|
|
||||||
order binary.ByteOrder
|
|
||||||
pos int
|
|
||||||
}
|
|
||||||
|
|
||||||
// newDecoder returns a new decoder that reads values from in. The input is
|
|
||||||
// expected to be in the given byte order.
|
|
||||||
func newDecoder(in io.Reader, order binary.ByteOrder) *decoder {
|
|
||||||
dec := new(decoder)
|
|
||||||
dec.in = in
|
|
||||||
dec.order = order
|
|
||||||
return dec
|
|
||||||
}
|
|
||||||
|
|
||||||
// align aligns the input to the given boundary and panics on error.
|
|
||||||
func (dec *decoder) align(n int) {
|
|
||||||
if dec.pos%n != 0 {
|
|
||||||
newpos := (dec.pos + n - 1) & ^(n - 1)
|
|
||||||
empty := make([]byte, newpos-dec.pos)
|
|
||||||
if _, err := io.ReadFull(dec.in, empty); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
dec.pos = newpos
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calls binary.Read(dec.in, dec.order, v) and panics on read errors.
|
|
||||||
func (dec *decoder) binread(v interface{}) {
|
|
||||||
if err := binary.Read(dec.in, dec.order, v); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dec *decoder) Decode(sig Signature) (vs []interface{}, err error) {
|
|
||||||
defer func() {
|
|
||||||
var ok bool
|
|
||||||
v := recover()
|
|
||||||
if err, ok = v.(error); ok {
|
|
||||||
if err == io.EOF || err == io.ErrUnexpectedEOF {
|
|
||||||
err = FormatError("unexpected EOF")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
vs = make([]interface{}, 0)
|
|
||||||
s := sig.str
|
|
||||||
for s != "" {
|
|
||||||
err, rem := validSingle(s, 0)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
v := dec.decode(s[:len(s)-len(rem)], 0)
|
|
||||||
vs = append(vs, v)
|
|
||||||
s = rem
|
|
||||||
}
|
|
||||||
return vs, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (dec *decoder) decode(s string, depth int) interface{} {
|
|
||||||
dec.align(alignment(typeFor(s)))
|
|
||||||
switch s[0] {
|
|
||||||
case 'y':
|
|
||||||
var b [1]byte
|
|
||||||
if _, err := dec.in.Read(b[:]); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
dec.pos++
|
|
||||||
return b[0]
|
|
||||||
case 'b':
|
|
||||||
i := dec.decode("u", depth).(uint32)
|
|
||||||
switch {
|
|
||||||
case i == 0:
|
|
||||||
return false
|
|
||||||
case i == 1:
|
|
||||||
return true
|
|
||||||
default:
|
|
||||||
panic(FormatError("invalid value for boolean"))
|
|
||||||
}
|
|
||||||
case 'n':
|
|
||||||
var i int16
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 2
|
|
||||||
return i
|
|
||||||
case 'i':
|
|
||||||
var i int32
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 4
|
|
||||||
return i
|
|
||||||
case 'x':
|
|
||||||
var i int64
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 8
|
|
||||||
return i
|
|
||||||
case 'q':
|
|
||||||
var i uint16
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 2
|
|
||||||
return i
|
|
||||||
case 'u':
|
|
||||||
var i uint32
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 4
|
|
||||||
return i
|
|
||||||
case 't':
|
|
||||||
var i uint64
|
|
||||||
dec.binread(&i)
|
|
||||||
dec.pos += 8
|
|
||||||
return i
|
|
||||||
case 'd':
|
|
||||||
var f float64
|
|
||||||
dec.binread(&f)
|
|
||||||
dec.pos += 8
|
|
||||||
return f
|
|
||||||
case 's':
|
|
||||||
length := dec.decode("u", depth).(uint32)
|
|
||||||
b := make([]byte, int(length)+1)
|
|
||||||
if _, err := io.ReadFull(dec.in, b); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
dec.pos += int(length) + 1
|
|
||||||
return string(b[:len(b)-1])
|
|
||||||
case 'o':
|
|
||||||
return ObjectPath(dec.decode("s", depth).(string))
|
|
||||||
case 'g':
|
|
||||||
length := dec.decode("y", depth).(byte)
|
|
||||||
b := make([]byte, int(length)+1)
|
|
||||||
if _, err := io.ReadFull(dec.in, b); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
dec.pos += int(length) + 1
|
|
||||||
sig, err := ParseSignature(string(b[:len(b)-1]))
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
return sig
|
|
||||||
case 'v':
|
|
||||||
if depth >= 64 {
|
|
||||||
panic(FormatError("input exceeds container depth limit"))
|
|
||||||
}
|
|
||||||
var variant Variant
|
|
||||||
sig := dec.decode("g", depth).(Signature)
|
|
||||||
if len(sig.str) == 0 {
|
|
||||||
panic(FormatError("variant signature is empty"))
|
|
||||||
}
|
|
||||||
err, rem := validSingle(sig.str, 0)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
if rem != "" {
|
|
||||||
panic(FormatError("variant signature has multiple types"))
|
|
||||||
}
|
|
||||||
variant.sig = sig
|
|
||||||
variant.value = dec.decode(sig.str, depth+1)
|
|
||||||
return variant
|
|
||||||
case 'h':
|
|
||||||
return UnixFDIndex(dec.decode("u", depth).(uint32))
|
|
||||||
case 'a':
|
|
||||||
if len(s) > 1 && s[1] == '{' {
|
|
||||||
ksig := s[2:3]
|
|
||||||
vsig := s[3 : len(s)-1]
|
|
||||||
v := reflect.MakeMap(reflect.MapOf(typeFor(ksig), typeFor(vsig)))
|
|
||||||
if depth >= 63 {
|
|
||||||
panic(FormatError("input exceeds container depth limit"))
|
|
||||||
}
|
|
||||||
length := dec.decode("u", depth).(uint32)
|
|
||||||
// Even for empty maps, the correct padding must be included
|
|
||||||
dec.align(8)
|
|
||||||
spos := dec.pos
|
|
||||||
for dec.pos < spos+int(length) {
|
|
||||||
dec.align(8)
|
|
||||||
if !isKeyType(v.Type().Key()) {
|
|
||||||
panic(InvalidTypeError{v.Type()})
|
|
||||||
}
|
|
||||||
kv := dec.decode(ksig, depth+2)
|
|
||||||
vv := dec.decode(vsig, depth+2)
|
|
||||||
v.SetMapIndex(reflect.ValueOf(kv), reflect.ValueOf(vv))
|
|
||||||
}
|
|
||||||
return v.Interface()
|
|
||||||
}
|
|
||||||
if depth >= 64 {
|
|
||||||
panic(FormatError("input exceeds container depth limit"))
|
|
||||||
}
|
|
||||||
length := dec.decode("u", depth).(uint32)
|
|
||||||
v := reflect.MakeSlice(reflect.SliceOf(typeFor(s[1:])), 0, 0)
|
|
||||||
// Even for empty arrays, the correct padding must be included
|
|
||||||
align := alignment(typeFor(s[1:]))
|
|
||||||
if len(s) > 1 && s[1] == '(' {
|
|
||||||
//Special case for arrays of structs
|
|
||||||
//structs decode as a slice of interface{} values
|
|
||||||
//but the dbus alignment does not match this
|
|
||||||
align = 8
|
|
||||||
}
|
|
||||||
dec.align(align)
|
|
||||||
spos := dec.pos
|
|
||||||
for dec.pos < spos+int(length) {
|
|
||||||
ev := dec.decode(s[1:], depth+1)
|
|
||||||
v = reflect.Append(v, reflect.ValueOf(ev))
|
|
||||||
}
|
|
||||||
return v.Interface()
|
|
||||||
case '(':
|
|
||||||
if depth >= 64 {
|
|
||||||
panic(FormatError("input exceeds container depth limit"))
|
|
||||||
}
|
|
||||||
dec.align(8)
|
|
||||||
v := make([]interface{}, 0)
|
|
||||||
s = s[1 : len(s)-1]
|
|
||||||
for s != "" {
|
|
||||||
err, rem := validSingle(s, 0)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
ev := dec.decode(s[:len(s)-len(rem)], depth+1)
|
|
||||||
v = append(v, ev)
|
|
||||||
s = rem
|
|
||||||
}
|
|
||||||
return v
|
|
||||||
default:
|
|
||||||
panic(SignatureError{Sig: s})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// A FormatError is an error in the wire format.
|
|
||||||
type FormatError string
|
|
||||||
|
|
||||||
func (e FormatError) Error() string {
|
|
||||||
return "dbus: wire format error: " + string(e)
|
|
||||||
}
|
|
321
vendor/github.com/godbus/dbus/default_handler.go
generated
vendored
321
vendor/github.com/godbus/dbus/default_handler.go
generated
vendored
|
@ -1,321 +0,0 @@
|
||||||
package dbus
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"reflect"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
)
|
|
||||||
|
|
||||||
func newIntrospectIntf(h *defaultHandler) *exportedIntf {
|
|
||||||
methods := make(map[string]Method)
|
|
||||||
methods["Introspect"] = exportedMethod{
|
|
||||||
reflect.ValueOf(func(msg Message) (string, *Error) {
|
|
||||||
path := msg.Headers[FieldPath].value.(ObjectPath)
|
|
||||||
return h.introspectPath(path), nil
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
return newExportedIntf(methods, true)
|
|
||||||
}
|
|
||||||
|
|
||||||
//NewDefaultHandler returns an instance of the default
|
|
||||||
//call handler. This is useful if you want to implement only
|
|
||||||
//one of the two handlers but not both.
|
|
||||||
//
|
|
||||||
// Deprecated: this is the default value, don't use it, it will be unexported.
|
|
||||||
func NewDefaultHandler() *defaultHandler {
|
|
||||||
h := &defaultHandler{
|
|
||||||
objects: make(map[ObjectPath]*exportedObj),
|
|
||||||
defaultIntf: make(map[string]*exportedIntf),
|
|
||||||
}
|
|
||||||
h.defaultIntf["org.freedesktop.DBus.Introspectable"] = newIntrospectIntf(h)
|
|
||||||
return h
|
|
||||||
}
|
|
||||||
|
|
||||||
type defaultHandler struct {
|
|
||||||
sync.RWMutex
|
|
||||||
objects map[ObjectPath]*exportedObj
|
|
||||||
defaultIntf map[string]*exportedIntf
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *defaultHandler) PathExists(path ObjectPath) bool {
|
|
||||||
_, ok := h.objects[path]
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *defaultHandler) introspectPath(path ObjectPath) string {
|
|
||||||
subpath := make(map[string]struct{})
|
|
||||||
var xml bytes.Buffer
|
|
||||||
xml.WriteString("<node>")
|
|
||||||
for obj, _ := range h.objects {
|
|
||||||
p := string(path)
|
|
||||||
if p != "/" {
|
|
||||||
p += "/"
|
|
||||||
}
|
|
||||||
if strings.HasPrefix(string(obj), p) {
|
|
||||||
node_name := strings.Split(string(obj[len(p):]), "/")[0]
|
|
||||||
subpath[node_name] = struct{}{}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for s, _ := range subpath {
|
|
||||||
xml.WriteString("\n\t<node name=\"" + s + "\"/>")
|
|
||||||
}
|
|
||||||
xml.WriteString("\n</node>")
|
|
||||||
return xml.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *defaultHandler) LookupObject(path ObjectPath) (ServerObject, bool) {
|
|
||||||
h.RLock()
|
|
||||||
defer h.RUnlock()
|
|
||||||
object, ok := h.objects[path]
|
|
||||||
if ok {
|
|
||||||
return object, ok
|
|
||||||
}
|
|
||||||
|
|
||||||
// If an object wasn't found for this exact path,
|
|
||||||
// look for a matching subtree registration
|
|
||||||
subtreeObject := newExportedObject()
|
|
||||||
path = path[:strings.LastIndex(string(path), "/")]
|
|
||||||
for len(path) > 0 {
|
|
||||||
object, ok = h.objects[path]
|
|
||||||
if ok {
|
|
||||||
for name, iface := range object.interfaces {
|
|
||||||
// Only include this handler if it registered for the subtree
|
|
||||||
if iface.isFallbackInterface() {
|
|
||||||
subtreeObject.interfaces[name] = iface
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
path = path[:strings.LastIndex(string(path), "/")]
|
|
||||||
}
|
|
||||||
|
|
||||||
for name, intf := range h.defaultIntf {
|
|
||||||
if _, exists := subtreeObject.interfaces[name]; exists {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
subtreeObject.interfaces[name] = intf
|
|
||||||
}
|
|
||||||
|
|
||||||
return subtreeObject, true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *defaultHandler) AddObject(path ObjectPath, object *exportedObj) {
|
|
||||||
h.Lock()
|
|
||||||
h.objects[path] = object
|
|
||||||
h.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *defaultHandler) DeleteObject(path ObjectPath) {
|
|
||||||
h.Lock()
|
|
||||||
delete(h.objects, path)
|
|
||||||
h.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
type exportedMethod struct {
|
|
||||||
reflect.Value
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m exportedMethod) Call(args ...interface{}) ([]interface{}, error) {
|
|
||||||
t := m.Type()
|
|
||||||
|
|
||||||
params := make([]reflect.Value, len(args))
|
|
||||||
for i := 0; i < len(args); i++ {
|
|
||||||
params[i] = reflect.ValueOf(args[i]).Elem()
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := m.Value.Call(params)
|
|
||||||
|
|
||||||
err := ret[t.NumOut()-1].Interface().(*Error)
|
|
||||||
ret = ret[:t.NumOut()-1]
|
|
||||||
out := make([]interface{}, len(ret))
|
|
||||||
for i, val := range ret {
|
|
||||||
out[i] = val.Interface()
|
|
||||||
}
|
|
||||||
if err == nil {
|
|
||||||
//concrete type to interface nil is a special case
|
|
||||||
return out, nil
|
|
||||||
}
|
|
||||||
return out, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m exportedMethod) NumArguments() int {
|
|
||||||
return m.Value.Type().NumIn()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m exportedMethod) ArgumentValue(i int) interface{} {
|
|
||||||
return reflect.Zero(m.Type().In(i)).Interface()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m exportedMethod) NumReturns() int {
|
|
||||||
return m.Value.Type().NumOut()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m exportedMethod) ReturnValue(i int) interface{} {
|
|
||||||
return reflect.Zero(m.Type().Out(i)).Interface()
|
|
||||||
}
|
|
||||||
|
|
||||||
func newExportedObject() *exportedObj {
|
|
||||||
return &exportedObj{
|
|
||||||
interfaces: make(map[string]*exportedIntf),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type exportedObj struct {
|
|
||||||
mu sync.RWMutex
|
|
||||||
interfaces map[string]*exportedIntf
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedObj) LookupInterface(name string) (Interface, bool) {
|
|
||||||
if name == "" {
|
|
||||||
return obj, true
|
|
||||||
}
|
|
||||||
obj.mu.RLock()
|
|
||||||
defer obj.mu.RUnlock()
|
|
||||||
intf, exists := obj.interfaces[name]
|
|
||||||
return intf, exists
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedObj) AddInterface(name string, iface *exportedIntf) {
|
|
||||||
obj.mu.Lock()
|
|
||||||
defer obj.mu.Unlock()
|
|
||||||
obj.interfaces[name] = iface
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedObj) DeleteInterface(name string) {
|
|
||||||
obj.mu.Lock()
|
|
||||||
defer obj.mu.Unlock()
|
|
||||||
delete(obj.interfaces, name)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedObj) LookupMethod(name string) (Method, bool) {
|
|
||||||
obj.mu.RLock()
|
|
||||||
defer obj.mu.RUnlock()
|
|
||||||
for _, intf := range obj.interfaces {
|
|
||||||
method, exists := intf.LookupMethod(name)
|
|
||||||
if exists {
|
|
||||||
return method, exists
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedObj) isFallbackInterface() bool {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func newExportedIntf(methods map[string]Method, includeSubtree bool) *exportedIntf {
|
|
||||||
return &exportedIntf{
|
|
||||||
methods: methods,
|
|
||||||
includeSubtree: includeSubtree,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type exportedIntf struct {
|
|
||||||
methods map[string]Method
|
|
||||||
|
|
||||||
// Whether or not this export is for the entire subtree
|
|
||||||
includeSubtree bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedIntf) LookupMethod(name string) (Method, bool) {
|
|
||||||
out, exists := obj.methods[name]
|
|
||||||
return out, exists
|
|
||||||
}
|
|
||||||
|
|
||||||
func (obj *exportedIntf) isFallbackInterface() bool {
|
|
||||||
return obj.includeSubtree
|
|
||||||
}
|
|
||||||
|
|
||||||
//NewDefaultSignalHandler returns an instance of the default
|
|
||||||
//signal handler. This is useful if you want to implement only
|
|
||||||
//one of the two handlers but not both.
|
|
||||||
//
|
|
||||||
// Deprecated: this is the default value, don't use it, it will be unexported.
|
|
||||||
func NewDefaultSignalHandler() *defaultSignalHandler {
|
|
||||||
return &defaultSignalHandler{
|
|
||||||
closeChan: make(chan struct{}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func isDefaultSignalHandler(handler SignalHandler) bool {
|
|
||||||
_, ok := handler.(*defaultSignalHandler)
|
|
||||||
return ok
|
|
||||||
}
|
|
||||||
|
|
||||||
type defaultSignalHandler struct {
|
|
||||||
sync.RWMutex
|
|
||||||
closed bool
|
|
||||||
signals []chan<- *Signal
|
|
||||||
closeChan chan struct{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sh *defaultSignalHandler) DeliverSignal(intf, name string, signal *Signal) {
|
|
||||||
sh.RLock()
|
|
||||||
defer sh.RUnlock()
|
|
||||||
if sh.closed {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
for _, ch := range sh.signals {
|
|
||||||
select {
|
|
||||||
case ch <- signal:
|
|
||||||
case <-sh.closeChan:
|
|
||||||
return
|
|
||||||
default:
|
|
||||||
go func(ch chan<- *Signal) {
|
|
||||||
select {
|
|
||||||
case ch <- signal:
|
|
||||||
case <-sh.closeChan:
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}(ch)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sh *defaultSignalHandler) Init() error {
|
|
||||||
sh.Lock()
|
|
||||||
sh.signals = make([]chan<- *Signal, 0)
|
|
||||||
sh.closeChan = make(chan struct{})
|
|
||||||
sh.Unlock()
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sh *defaultSignalHandler) Terminate() {
|
|
||||||
sh.Lock()
|
|
||||||
if !sh.closed {
|
|
||||||
close(sh.closeChan)
|
|
||||||
}
|
|
||||||
sh.closed = true
|
|
||||||
for _, ch := range sh.signals {
|
|
||||||
close(ch)
|
|
||||||
}
|
|
||||||
sh.signals = nil
|
|
||||||
sh.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sh *defaultSignalHandler) addSignal(ch chan<- *Signal) {
|
|
||||||
sh.Lock()
|
|
||||||
defer sh.Unlock()
|
|
||||||
if sh.closed {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
sh.signals = append(sh.signals, ch)
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sh *defaultSignalHandler) removeSignal(ch chan<- *Signal) {
|
|
||||||
sh.Lock()
|
|
||||||
defer sh.Unlock()
|
|
||||||
if sh.closed {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
for i := len(sh.signals) - 1; i >= 0; i-- {
|
|
||||||
if ch == sh.signals[i] {
|
|
||||||
copy(sh.signals[i:], sh.signals[i+1:])
|
|
||||||
sh.signals[len(sh.signals)-1] = nil
|
|
||||||
sh.signals = sh.signals[:len(sh.signals)-1]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
69
vendor/github.com/godbus/dbus/doc.go
generated
vendored
69
vendor/github.com/godbus/dbus/doc.go
generated
vendored
|
@ -1,69 +0,0 @@
|
||||||
/*
|
|
||||||
Package dbus implements bindings to the D-Bus message bus system.
|
|
||||||
|
|
||||||
To use the message bus API, you first need to connect to a bus (usually the
|
|
||||||
session or system bus). The acquired connection then can be used to call methods
|
|
||||||
on remote objects and emit or receive signals. Using the Export method, you can
|
|
||||||
arrange D-Bus methods calls to be directly translated to method calls on a Go
|
|
||||||
value.
|
|
||||||
|
|
||||||
Conversion Rules
|
|
||||||
|
|
||||||
For outgoing messages, Go types are automatically converted to the
|
|
||||||
corresponding D-Bus types. The following types are directly encoded as their
|
|
||||||
respective D-Bus equivalents:
|
|
||||||
|
|
||||||
Go type | D-Bus type
|
|
||||||
------------+-----------
|
|
||||||
byte | BYTE
|
|
||||||
bool | BOOLEAN
|
|
||||||
int16 | INT16
|
|
||||||
uint16 | UINT16
|
|
||||||
int | INT32
|
|
||||||
uint | UINT32
|
|
||||||
int32 | INT32
|
|
||||||
uint32 | UINT32
|
|
||||||
int64 | INT64
|
|
||||||
uint64 | UINT64
|
|
||||||
float64 | DOUBLE
|
|
||||||
string | STRING
|
|
||||||
ObjectPath | OBJECT_PATH
|
|
||||||
Signature | SIGNATURE
|
|
||||||
Variant | VARIANT
|
|
||||||
interface{} | VARIANT
|
|
||||||
UnixFDIndex | UNIX_FD
|
|
||||||
|
|
||||||
Slices and arrays encode as ARRAYs of their element type.
|
|
||||||
|
|
||||||
Maps encode as DICTs, provided that their key type can be used as a key for
|
|
||||||
a DICT.
|
|
||||||
|
|
||||||
Structs other than Variant and Signature encode as a STRUCT containing their
|
|
||||||
exported fields. Fields whose tags contain `dbus:"-"` and unexported fields will
|
|
||||||
be skipped.
|
|
||||||
|
|
||||||
Pointers encode as the value they're pointed to.
|
|
||||||
|
|
||||||
Types convertible to one of the base types above will be mapped as the
|
|
||||||
base type.
|
|
||||||
|
|
||||||
Trying to encode any other type or a slice, map or struct containing an
|
|
||||||
unsupported type will result in an InvalidTypeError.
|
|
||||||
|
|
||||||
For incoming messages, the inverse of these rules are used, with the exception
|
|
||||||
of STRUCTs. Incoming STRUCTS are represented as a slice of empty interfaces
|
|
||||||
containing the struct fields in the correct order. The Store function can be
|
|
||||||
used to convert such values to Go structs.
|
|
||||||
|
|
||||||
Unix FD passing
|
|
||||||
|
|
||||||
Handling Unix file descriptors deserves special mention. To use them, you should
|
|
||||||
first check that they are supported on a connection by calling SupportsUnixFDs.
|
|
||||||
If it returns true, all method of Connection will translate messages containing
|
|
||||||
UnixFD's to messages that are accompanied by the given file descriptors with the
|
|
||||||
UnixFD values being substituted by the correct indices. Similarily, the indices
|
|
||||||
of incoming messages are automatically resolved. It shouldn't be necessary to use
|
|
||||||
UnixFDIndex.
|
|
||||||
|
|
||||||
*/
|
|
||||||
package dbus
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue