Cordial Packages
Cordial packages include a number of Go language interfaces to Geneos.
commands
The commands package provides access to the Geneos Gateway REST Command API.
config
The config package provides a wrapper around viper
geneos
The geneos package provides a data model for Geneos XML configurations, both Gateway and Netprobe. It is only partially complete at this stage and is being extended as demand requires.
XML-RPC API
These packages wrap the original SOAP XML-RPC API interface:
The code is still in development and the API will evolve with releases. Feedback, via Issues, and Pull requests are welcome but without any guarantees if I'll have time to do them.
The documentation for the underlying API is here: XML-RPC API
While direct mappings from golang to the API are available in the xmlrpc package most users will want to look at the higher-level samplers and streams packages that try to implement an easier to use high-level interface.
Examples of use
The example package directory contains a number of simple implementations of common plugin types that show how to use the different types of data update methods.
The example/generic directory is described in further detail below. It uses this method to deliver updates:
func (s Samplers) UpdateTableFromSlice(rowdata interface{}) error
The other two methods both take maps as follows:
func (s *Samplers) UpdateTableFromMap(data interface{}) error
func (s *Samplers) UpdateTableFromMapDelta(newdata, olddata interface{}, interval time.Duration) error
The UpdateTableFromMapDelta()
also takes an time.Duration
interval that allows scaling of the difference between the two datasets.
Create a basic plugin
First, import the necessary packages
package generic
import (
"github.com/itrs-group/cordial/pkg/plugins"
"github.com/itrs-group/cordial/pkg/sampler"
)
Next, create two structs, one to hold the per-sample data and another to hold the Sampler amd any other local data that is needed for the lifetime of the sampler:
type GenericData struct {
RowName string
Column1 string
Column2 string
}
type GenericSampler struct {
samplers.Samplers
localdata string
}
Now create the required methods. There are three and they must meet this interface (from the samplers package):
type SamplerInstance interface {
New(plugins.Connection, string, string) *SamplerInstance
InitSampler(*SamplerInstance) (err error)
DoSample(*SamplerInstance) (err error)
}
First a New()
method that your main package will call to create an instance of the plugin - aka. the sampler - and do some housekeeping:
func New(s plugins.Connection, name string, group string) (*GenericSampler, error) {
c := new(GenericSampler)
c.Plugins = c
return c, c.New(s, name, group)
}
As an aside, this New() method has to work this way because it's how I have found to make the underlying type of Plugins take on that of the specific plugin package. Along with some internals this is how the plugin exposes it's own methods in the above interface correctly, without infinite recursion.
The next method is InitSampler()
which is called once upon start-up of the sampler instance. The first part of this example locates a parameter in the Geneos configurationa and assigns is to the local data struct.
func (g *GenericSampler) InitSampler() error {
example, err := g.Parameter("EXAMPLE")
if err != nil {
return nil
}
g.localdata = example
It is worth noting at this point that the InitSampler()
being called only once means that if there is any change in the Geneos configuration there is no way for the running program to notice. The XML-RPC API is stateless (we'll ignore the heartbeat functions for now) and these plugins may not notice a Netprobe or related restart. So, the Parameter()
call above is only an example and should probably be refreshed using a timer, but not every sample most likely.
The second part is required to initialise the helper methods which we'll used see below:
columns, columnnames, sortcol, err := g.ColumnInfo(GenericData{})
g.SetColumns(columns)
g.SetColumnNames(columnnames)
g.SetSortColumn(sortcol)
return g.Headline("example", g.localdata)
}
The final mandatory method is DoSample()
which is called to update the data:
func (p *GenericSampler) DoSample() error {
var rowdata = []GenericData{
{"row4", "data1", "data2"},
{"row2", "data1", "data2"},
{"row3", "data1", "data2"},
{"row1", "data1", "data2"},
}
return p.UpdateTableFromSlice(rowdata)
}
The call to UpdateTableFromSlice()
uses the column data initialised earlier to ensure the dataview is rendered correctly.
More features
You can use tags to control the rendering of the data, like this example of a CPU plugin for Windows:
// +build windows
package cpu
import (
"log"
"time"
"github.com/StackExchange/wmi"
"github.com/itrs-group/cordial/pkg/samplers"
)
// Win32_PerfRawData_PerfOS_Processor must be exported along with all it's
// fields so that methods in plugins package can output the results
type Win32_PerfRawData_PerfOS_Processor struct {
Name string `column:"cpuName"`
PercentUserTime uint64 `column:"% User Time,format=%.2f %%"`
PercentPrivilegedTime uint64 `column:"% Priv Time,format=%.2f %%"`
PercentIdleTime uint64 `column:"% Idle Time,format=%.2f %%"`
PercentProcessorTime uint64 `column:"% Proc Time,format=%.2f %%"`
PercentInterruptTime uint64 `column:"% Intr Time,format=%.2f %%"`
PercentDPCTime uint64 `column:"% DPC Time,format=%.2f %%"`
Timestamp_PerfTime uint64 `column:"OMIT"`
Frequency_PerfTime uint64 `column:"OMIT"`
}
// one entry for each CPU row in /proc/stats
type cpustat struct {
cpus map[string]Win32_PerfRawData_PerfOS_Processor
lastsample float64
frequency float64
}
The tag is column and the comma seperated tag values currently supported are:
name
- any value without an "=" is treated as a display name for the column created from this field. The special name "OMIT" means that the fields should not create a column, but the data will still be avilable for calculations etc. Any normal ASCII characters are permitted except a comma. No validation is done and the string is passed to the Netprobe as-is.
format=FORMAT
- FORMAT is a Printf
style format string used to render the value of the cell in the most appropriate way for the data
sort=[+|-][num]
- the sort tag defines which field - and only one field can be selected - should be used to sort the resulting rows published via the Map rendering methods. The valid values are an option leading + or - representing ascending or descending order and the option suffix "num" to indicate a numeric sort. "sort=" means to sort ascending in lexographical order, which is the same as "sort=+"
The sort tag only applies to those dataviews populated from maps like this call below:
func (p *CPUSampler) DoSample() (err error) {
...
err = p.UpdateTableFromMapDelta(stat.cpus, laststats.cpus, time.Duration(interval)*10*time.Millisecond)
The UpdateTableFromSlice()
shown in the generic example assumes that the slice has been passed in the order required. Maps on the other hand have no defined order and the package allows you to define the natural sort order. This can of course be overridden by the user of the Geneos Active Console.
Initialise and start-up
To use your plugin in a program, use it like this:
package main
import (
...
"github.com/itrs-group/cordial/pkg/plugins"
"github.com/itrs-group/cordial/pkg/streams"
"example/generic" // this will depend on how you name it
)
Do normal start-up configuration, process command line args etc. and then initialise the Sampler
connection like this:
func main() {
...
// connect to netprobe
url := fmt.Sprintf("http://%s:%v/xmlrpc", hostname, port)
s, err := plugins.Sampler(url, entityname, samplername)
if err != nil {
log.Fatal(err)
}
Once you have your sampler connection call the New()
method with dataview and group names. The group can be an empty string. Set the interval as a Go time.Duration
value. The default, if a zero is passed, is one second. One second is also the minimum interval.
Finally Start()
the sampler by passing a sync.WaitGroup
that you can later Wait()
on so the program doesn't exit while the sampler runs.
g, err := generic.New(s, "example", "SYSTEM")
defer g.Close()
g.SetInterval(interval)
g.Start(&wg)
wg.Wait()
}
Streams
Basic support for streams are included. Streams must be predefined in the Geneos configuration and sending messages to a non-existent stream name results in an error.
import (
"github.com/itrs-group/cordial/pkg/streams"
)
func main() {
...
streamsampler := "streams"
sp, err := streams.Sampler(fmt.Sprintf("http://%s:%v/xmlrpc", hostname, port), entityname, streamsampler)
if err != nil {
log.Fatal(err)
}
err := sp.WriteMessage("teststream", time.Now().String()+" this is a test")
if err != nil {
log.Fatal(err)
break
}
}
For convenience the streams package also acts as an io.Writer and io.StringWriter and so will respond to normal Go Write()
and WriteString()
calls. You must however call SetStreamName()
before trying to write messages this way. There is no validation of data content or length.
So, instead of the above you can also do:
sp.SetStreamName("teststream")
_, err := sp.WriteString(time.Now().String() + " this is a test")
if err != nil {
log.Fatal(err)
break
}
You can change the stream name as often as you want, but it will be easier to create multiple streams if you need to.
Note that the sampler name is always different to the normal dataview destination as the plugin on the Geneos side must be an api-streams one. Also there is no Close()
method. At the moment there is no direct support for heartbeats.
Secure Connections
The XML-RPC API packages support secure connections through the normal Go http.Client but it is quite common for individual Netprobe instances to run with self-signed certificates so there is a method to allow unverified certificates and this must be called immediately after getting the new sampler or stream like this:
u := &url.URL{Scheme: "https", Host: fmt.Sprintf("%s:%d", hostname, port), Path: "/xmlrpc"}
p, err := plugins.Sampler(u, entityname, samplername)
if err != nil {
log.Fatal(err)
}
p.InsecureSkipVerify()
Once allowed there is no way to turn this off until you create a new object. The setting is per sampler or stream so it can be on and off separately in the case where your program may send data to multiple Netprobes.