From e69ec8c166be3ddd9cd611d7ef6ae52c6a1b2379 Mon Sep 17 00:00:00 2001 From: Valere JEANTET Date: Sun, 18 Mar 2018 13:26:13 +0100 Subject: [PATCH] processors: webhook - add error handlers --- core/webhook/webhook.go | 8 +++--- docs/data/processors/webfan.json | 12 ++++----- processors/webfan/docdoc.go | 12 ++++----- processors/webfan/readme.md | 28 ++++++++++----------- processors/webfan/webfan.go | 42 +++++++++++++++++--------------- processors/webhook.go | 2 +- 6 files changed, 54 insertions(+), 50 deletions(-) diff --git a/core/webhook/webhook.go b/core/webhook/webhook.go index 369dc9fa..b9fd8adb 100644 --- a/core/webhook/webhook.go +++ b/core/webhook/webhook.go @@ -54,14 +54,14 @@ func (w *webHook) buildURL(hookName string) string { return strings.ToLower("/h/" + slug.Make(w.pipelineLabel) + "/" + slug.Make(hookName)) } -func (w *webHook) buildNamedURL(hookName string) string { +func (w *webHook) buildShortURL(hookName string) string { return strings.ToLower("/_/" + hookName) } // Add a new route to a given http.HandlerFunc -func (w *webHook) AddNamed(hookName string, hf http.HandlerFunc) { +func (w *webHook) AddShort(hookName string, hf http.HandlerFunc) { - hUrl := w.buildNamedURL(hookName) + hUrl := w.buildShortURL(hookName) w.Hooks = append(w.Hooks, hookName) webHookMap.Store(hUrl, &Hook{ @@ -96,7 +96,7 @@ func (w *webHook) Delete(hookName string) { webHookMap.Delete(hUrl) Log.Debugf("WebHook unregisted [%s]", hUrl) - hUrl = w.buildNamedURL(hookName) + hUrl = w.buildShortURL(hookName) webHookMap.Delete(hUrl) Log.Debugf("WebHook unregisted [%s]", hUrl) } diff --git a/docs/data/processors/webfan.json b/docs/data/processors/webfan.json index 04545a3a..db1119be 100644 --- a/docs/data/processors/webfan.json +++ b/docs/data/processors/webfan.json @@ -18,9 +18,9 @@ }, { "Alias": "", - "DefaultValue": "\"json\"", - "Doc": "The codec used for input data. Input codecs are a convenient method for decoding\nyour data before it enters the input, without needing a separate filter in your bitfan pipeline\n\nDefault decode http request as json, response is json encoded.\nSet multiple codec with role to customize", - "ExampleLS": "", + "DefaultValue": null, + "Doc": "The codec used for posted data. Input codecs are a convenient method for decoding\nyour data before it enters the pipeline, without needing a separate filter in your bitfan pipeline\n\nDefault decode http request as plain text, response is json encoded.\nSet multiple codec with role to customize", + "ExampleLS": "codec =\u003e plain { role=\u003e\"encoder\"} codec =\u003e json { role=\u003e\"decoder\"}", "Name": "Codec", "PossibleValues": null, "Required": false, @@ -37,11 +37,11 @@ "Type": "string" }, { - "Alias": "conf", + "Alias": "pipeline", "DefaultValue": null, "Doc": "Path to pipeline's configuration to execute on request\nThis configuration should contains only a filter section an a output like ```output{pass{}}```", "ExampleLS": "", - "Name": "Conf", + "Name": "Pipeline", "PossibleValues": null, "Required": true, "Type": "string" @@ -49,7 +49,7 @@ { "Alias": "headers", "DefaultValue": null, - "Doc": "Headers to send back into each outgoing response", + "Doc": "Headers to send back into outgoing response", "ExampleLS": "{\"X-Processor\" =\u003e \"bitfan\"}", "Name": "Headers", "PossibleValues": null, diff --git a/processors/webfan/docdoc.go b/processors/webfan/docdoc.go index d9b416cb..45b49eda 100644 --- a/processors/webfan/docdoc.go +++ b/processors/webfan/docdoc.go @@ -25,12 +25,12 @@ func (p *processor) Doc() *doc.Processor { &doc.ProcessorOption{ Name: "Codec", Alias: "", - Doc: "The codec used for input data. Input codecs are a convenient method for decoding\nyour data before it enters the input, without needing a separate filter in your bitfan pipeline\n\nDefault decode http request as json, response is json encoded.\nSet multiple codec with role to customize", + Doc: "The codec used for posted data. Input codecs are a convenient method for decoding\nyour data before it enters the pipeline, without needing a separate filter in your bitfan pipeline\n\nDefault decode http request as plain text, response is json encoded.\nSet multiple codec with role to customize", Required: false, Type: "codec", - DefaultValue: "\"json\"", + DefaultValue: nil, PossibleValues: []string{}, - ExampleLS: "", + ExampleLS: "codec => plain { role=>\"encoder\"} codec => json { role=>\"decoder\"}", }, &doc.ProcessorOption{ Name: "Uri", @@ -43,8 +43,8 @@ func (p *processor) Doc() *doc.Processor { ExampleLS: "", }, &doc.ProcessorOption{ - Name: "Conf", - Alias: "conf", + Name: "Pipeline", + Alias: "pipeline", Doc: "Path to pipeline's configuration to execute on request\nThis configuration should contains only a filter section an a output like ```output{pass{}}```", Required: true, Type: "string", @@ -55,7 +55,7 @@ func (p *processor) Doc() *doc.Processor { &doc.ProcessorOption{ Name: "Headers", Alias: "headers", - Doc: "Headers to send back into each outgoing response", + Doc: "Headers to send back into outgoing response", Required: false, Type: "hash", DefaultValue: nil, diff --git a/processors/webfan/readme.md b/processors/webfan/readme.md index a5735c2f..34224ff7 100644 --- a/processors/webfan/readme.md +++ b/processors/webfan/readme.md @@ -4,24 +4,24 @@ ## Synopsys -| SETTING | TYPE | REQUIRED | DEFAULT VALUE | -|---------|--------|----------|---------------| -| Codec | codec | false | "json" | -| uri | string | true | "" | -| conf | string | true | "" | -| headers | hash | false | {} | +| SETTING | TYPE | REQUIRED | DEFAULT VALUE | +|----------|--------|----------|---------------| +| Codec | codec | false | ? | +| uri | string | true | "" | +| pipeline | string | true | "" | +| headers | hash | false | {} | ## Details ### Codec * Value type is codec -* Default value is `"json"` +* Default value is `?` -The codec used for input data. Input codecs are a convenient method for decoding -your data before it enters the input, without needing a separate filter in your bitfan pipeline +The codec used for posted data. Input codecs are a convenient method for decoding +your data before it enters the pipeline, without needing a separate filter in your bitfan pipeline -Default decode http request as json, response is json encoded. +Default decode http request as plain text, response is json encoded. Set multiple codec with role to customize ### uri @@ -31,7 +31,7 @@ Set multiple codec with role to customize URI path /_/path -### conf +### pipeline * This is a required setting. * Value type is string * Default value is `""` @@ -43,7 +43,7 @@ This configuration should contains only a filter section an a output like ```out * Value type is hash * Default value is `{}` -Headers to send back into each outgoing response +Headers to send back into outgoing response @@ -51,9 +51,9 @@ Headers to send back into each outgoing response ``` webfan{ - codec => "json" + codec => plain { role=>"encoder"} codec => json { role=>"decoder"} uri => "" - conf => "" + pipeline => "" {"X-Processor" => "bitfan"} } ``` diff --git a/processors/webfan/webfan.go b/processors/webfan/webfan.go index 3e2b1095..62171256 100644 --- a/processors/webfan/webfan.go +++ b/processors/webfan/webfan.go @@ -5,7 +5,6 @@ import ( "io" "net/http" "net/http/httputil" - "os" "sync" uuid "github.com/nu7hatch/gouuid" @@ -22,12 +21,12 @@ func New() processors.Processor { type options struct { processors.CommonOptions `mapstructure:",squash"` - // The codec used for input data. Input codecs are a convenient method for decoding - // your data before it enters the input, without needing a separate filter in your bitfan pipeline + // The codec used for posted data. Input codecs are a convenient method for decoding + // your data before it enters the pipeline, without needing a separate filter in your bitfan pipeline // - // Default decode http request as json, response is json encoded. + // Default decode http request as plain text, response is json encoded. // Set multiple codec with role to customize - // @Default "json" + // @ExampleLS codec => plain { role=>"encoder"} codec => json { role=>"decoder"} // @Type codec Codec codecs.CodecCollection @@ -36,9 +35,9 @@ type options struct { // Path to pipeline's configuration to execute on request // This configuration should contains only a filter section an a output like ```output{pass{}}``` - Conf string `mapstructure:"conf" validate:"required"` + Pipeline string `mapstructure:"pipeline" validate:"required"` - // Headers to send back into each outgoing response + // Headers to send back into outgoing response // @ExampleLS {"X-Processor" => "bitfan"} Headers map[string]string `mapstructure:"headers"` } @@ -47,10 +46,9 @@ type options struct { type processor struct { processors.Base - opt *options - wg *sync.WaitGroup - host string - ep *entrypoint.Entrypoint + opt *options + wg *sync.WaitGroup + ep *entrypoint.Entrypoint } func (p *processor) Configure(ctx processors.ProcessorContext, conf map[string]interface{}) error { @@ -61,10 +59,6 @@ func (p *processor) Configure(ctx processors.ProcessorContext, conf map[string]i return err } - if p.host, err = os.Hostname(); err != nil { - p.Logger.Warnf("can not get hostname : %v", err) - } - if p.opt.Codec.Enc == nil { p.opt.Codec.Enc = codecs.New("json", nil, ctx.Log(), ctx.ConfigWorkingLocation()) } @@ -77,12 +71,12 @@ func (p *processor) Configure(ctx processors.ProcessorContext, conf map[string]i } func (p *processor) Start(e processors.IPacket) error { p.wg = &sync.WaitGroup{} - p.WebHook.AddNamed(p.opt.Uri, p.HttpHandler) + p.WebHook.AddShort(p.opt.Uri, p.HttpHandler) var err error - p.ep, err = entrypoint.New(p.opt.Conf, p.ConfigWorkingLocation, entrypoint.CONTENT_REF) + p.ep, err = entrypoint.New(p.opt.Pipeline, p.ConfigWorkingLocation, entrypoint.CONTENT_REF) if err != nil { - p.Logger.Errorf("Error with entrypoint %s", p.opt.Conf) + p.Logger.Errorf("Error with entrypoint %s", p.opt.Pipeline) } return err @@ -98,6 +92,9 @@ func (p *processor) HttpHandler(w http.ResponseWriter, r *http.Request) { ppl, err := p.ep.Pipeline() if err != nil { p.Logger.Errorf("%s", err) + w.WriteHeader(http.StatusInternalServerError) + w.Header().Set("Content-Type", "text/plain; charset=utf-8") + w.Write([]byte(err.Error())) return } // pp.Println("ppl-->", ppl) @@ -105,6 +102,14 @@ func (p *processor) HttpHandler(w http.ResponseWriter, r *http.Request) { // Find Last Agent lastAgent := orderedAgentConfList[len(orderedAgentConfList)-1] + if lastAgent.Label != "pass" { + p.Logger.Errorf("Add an `output{pass{}}` into pipeline configuration") + w.WriteHeader(http.StatusInternalServerError) + w.Header().Set("Content-Type", "text/plain; charset=utf-8") + w.Write([]byte("Add an `output{pass{}}` into pipeline configuration")) + return + } + // pp.Println("lastAgent-->", lastAgent) // Connect its Recipient to this Receive // Find First agent @@ -159,7 +164,6 @@ func (p *processor) HttpHandler(w http.ResponseWriter, r *http.Request) { if dec, err = p.opt.Codec.NewDecoder(r.Body); err != nil { p.Logger.Errorln("decoder error : ", err.Error()) close(back) - close(back) return } headersBytes, _ := httputil.DumpRequest(r, false) diff --git a/processors/webhook.go b/processors/webhook.go index 15a1de31..9c3889b8 100644 --- a/processors/webhook.go +++ b/processors/webhook.go @@ -4,6 +4,6 @@ import "net/http" type WebHook interface { Add(string, http.HandlerFunc) - AddNamed(string, http.HandlerFunc) + AddShort(string, http.HandlerFunc) Unregister() }