133 lines
3.7 KiB
Go
133 lines
3.7 KiB
Go
package query
|
|
|
|
import (
|
|
"io"
|
|
"net/http"
|
|
|
|
"github.com/influxdata/flux"
|
|
"github.com/influxdata/flux/csv"
|
|
)
|
|
|
|
const (
|
|
NoContentDialectType = "no-content"
|
|
NoContentWErrDialectType = "no-content-with-error"
|
|
)
|
|
|
|
// AddDialectMappings adds the mappings for the no-content dialects.
|
|
func AddDialectMappings(mappings flux.DialectMappings) error {
|
|
if err := mappings.Add(NoContentDialectType, func() flux.Dialect {
|
|
return NewNoContentDialect()
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
return mappings.Add(NoContentWErrDialectType, func() flux.Dialect {
|
|
return NewNoContentWithErrorDialect()
|
|
})
|
|
}
|
|
|
|
// NoContentDialect is a dialect that provides an Encoder that discards query results.
|
|
// When invoking `dialect.Encoder().Encode(writer, results)`, `results` get consumed,
|
|
// while the `writer` is left intact.
|
|
// It is an HTTPDialect that sets the response status code to 204 NoContent.
|
|
type NoContentDialect struct{}
|
|
|
|
func NewNoContentDialect() *NoContentDialect {
|
|
return &NoContentDialect{}
|
|
}
|
|
|
|
func (d *NoContentDialect) Encoder() flux.MultiResultEncoder {
|
|
return &NoContentEncoder{}
|
|
}
|
|
|
|
func (d *NoContentDialect) DialectType() flux.DialectType {
|
|
return NoContentDialectType
|
|
}
|
|
|
|
func (d *NoContentDialect) SetHeaders(w http.ResponseWriter) {
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
|
|
type NoContentEncoder struct{}
|
|
|
|
func (e *NoContentEncoder) Encode(w io.Writer, results flux.ResultIterator) (int64, error) {
|
|
defer results.Release()
|
|
// Consume and discard results.
|
|
for results.More() {
|
|
if err := results.Next().Tables().Do(func(tbl flux.Table) error {
|
|
return tbl.Do(func(cr flux.ColReader) error {
|
|
return nil
|
|
})
|
|
}); err != nil {
|
|
return 0, err
|
|
}
|
|
}
|
|
// Do not write anything.
|
|
return 0, nil
|
|
}
|
|
|
|
// NoContentWithErrorDialect is a dialect that provides an Encoder that discards query results,
|
|
// but it encodes runtime errors from the Flux query in CSV format.
|
|
// To discover if there was any runtime error in the query, one should check the response size.
|
|
// If it is equal to zero, then no error was present.
|
|
// Otherwise one can decode the response body to get the error. For example:
|
|
// ```
|
|
// _, err = csv.NewResultDecoder(csv.ResultDecoderConfig{}).Decode(bytes.NewReader(res))
|
|
// if err != nil {
|
|
// // we got some runtime error
|
|
// }
|
|
// ```
|
|
type NoContentWithErrorDialect struct {
|
|
csv.ResultEncoderConfig
|
|
}
|
|
|
|
func NewNoContentWithErrorDialect() *NoContentWithErrorDialect {
|
|
return &NoContentWithErrorDialect{
|
|
ResultEncoderConfig: csv.DefaultEncoderConfig(),
|
|
}
|
|
}
|
|
|
|
func (d *NoContentWithErrorDialect) Encoder() flux.MultiResultEncoder {
|
|
return &NoContentWithErrorEncoder{
|
|
errorEncoder: csv.NewResultEncoder(d.ResultEncoderConfig),
|
|
}
|
|
}
|
|
|
|
func (d *NoContentWithErrorDialect) DialectType() flux.DialectType {
|
|
return NoContentWErrDialectType
|
|
}
|
|
|
|
func (d *NoContentWithErrorDialect) SetHeaders(w http.ResponseWriter) {
|
|
w.Header().Set("Content-Type", "text/csv; charset=utf-8")
|
|
w.Header().Set("Transfer-Encoding", "chunked")
|
|
}
|
|
|
|
type NoContentWithErrorEncoder struct {
|
|
errorEncoder *csv.ResultEncoder
|
|
}
|
|
|
|
func (e *NoContentWithErrorEncoder) Encode(w io.Writer, results flux.ResultIterator) (int64, error) {
|
|
// Make sure we release results.
|
|
// Remember, it is safe to call `Release` multiple times.
|
|
defer results.Release()
|
|
// Consume and discard results, but keep an eye on errors.
|
|
for results.More() {
|
|
if err := results.Next().Tables().Do(func(tbl flux.Table) error {
|
|
return tbl.Do(func(cr flux.ColReader) error {
|
|
return nil
|
|
})
|
|
}); err != nil {
|
|
// If there is an error, then encode it in the response.
|
|
if encErr := e.errorEncoder.EncodeError(w, err); encErr != nil {
|
|
return 0, encErr
|
|
}
|
|
}
|
|
}
|
|
// Now Release in order to populate the error, if present.
|
|
results.Release()
|
|
err := results.Err()
|
|
if err != nil {
|
|
return 0, e.errorEncoder.EncodeError(w, err)
|
|
}
|
|
return 0, nil
|
|
}
|