Skip to content

Commit

Permalink
beater/api/profile: fix pprof Content-Type (#3144) (#3145)
Browse files Browse the repository at this point in the history
If messageType is specified, check that it is the
expected value. Otherwise we assume pprof, and
optimistically attempt to decode.
  • Loading branch information
axw authored Jan 14, 2020
1 parent 5774a73 commit 279b81f
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 22 deletions.
41 changes: 28 additions & 13 deletions beater/api/profile/handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,8 +20,8 @@ package profile
import (
"fmt"
"io"
"mime"
"net/http"
"strings"

pprof_profile "github.com/google/pprof/profile"
"github.com/pkg/errors"
Expand All @@ -46,10 +46,12 @@ var (
)

const (
// TODO(axw) include messageType in pprofContentType; needs fix in agent
pprofContentType = "application/x-protobuf"
metadataContentType = "application/json"
requestContentType = "multipart/form-data"
pprofMediaType = "application/x-protobuf"
metadataMediaType = "application/json"
requestMediaType = "multipart/form-data"

// value for the "messagetype" param
pprofMessageType = "perftools.profiles.Profile"

metadataContentLengthLimit = 10 * 1024
profileContentLengthLimit = 10 * 1024 * 1024
Expand All @@ -68,7 +70,7 @@ func Handler(
err: errors.New("only POST requests are supported"),
}
}
if err := validateContentType(c.Request.Header, requestContentType); err != nil {
if _, err := validateContentType(c.Request.Header, requestMediaType); err != nil {
return nil, requestError{
id: request.IDResponseErrorsValidate,
err: err,
Expand Down Expand Up @@ -113,7 +115,7 @@ func Handler(

switch part.FormName() {
case "metadata":
if err := validateContentType(http.Header(part.Header), metadataContentType); err != nil {
if _, err := validateContentType(http.Header(part.Header), metadataMediaType); err != nil {
return nil, requestError{
id: request.IDResponseErrorsValidate,
err: errors.Wrap(err, "invalid metadata"),
Expand Down Expand Up @@ -152,12 +154,22 @@ func Handler(
tctx.Metadata = *metadata

case "profile":
if err := validateContentType(http.Header(part.Header), pprofContentType); err != nil {
params, err := validateContentType(http.Header(part.Header), pprofMediaType)
if err != nil {
return nil, requestError{
id: request.IDResponseErrorsValidate,
err: errors.Wrap(err, "invalid profile"),
}
}
if v, ok := params["messagetype"]; ok && v != pprofMessageType {
// If messagetype is specified, require that it matches.
// Otherwise we assume that it's pprof, and we'll error
// out below if it doesn't decode.
return nil, requestError{
id: request.IDResponseErrorsValidate,
err: errors.Wrapf(err, "expected messagetype %q, got %q", pprofMessageType, v),
}
}
r := &decoder.LimitedReader{R: part, N: totalLimitRemaining}
profile, err := pprof_profile.Parse(r)
if err != nil {
Expand Down Expand Up @@ -218,12 +230,15 @@ func Handler(
}
}

func validateContentType(header http.Header, contentType string) error {
got := header.Get(headers.ContentType)
if !strings.Contains(got, contentType) {
return fmt.Errorf("invalid content type %q, expected %q", got, contentType)
func validateContentType(header http.Header, expectedMediatype string) (params map[string]string, err error) {
mediatype, params, err := mime.ParseMediaType(header.Get(headers.ContentType))
if err != nil {
return nil, err
}
if mediatype != expectedMediatype {
return nil, fmt.Errorf("invalid content type %q, expected %q", mediatype, expectedMediatype)
}
return nil
return params, nil
}

type result struct {
Expand Down
36 changes: 27 additions & 9 deletions beater/api/profile/handler_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,8 @@ import (
"github.com/elastic/apm-server/publish"
)

const pprofContentType = `application/x-protobuf; messageType="perftools.profiles.Profile"`

func TestHandler(t *testing.T) {
var rateLimit, err = ratelimit.NewStore(1, 0, 0)
require.NoError(t, err)
Expand Down Expand Up @@ -127,7 +129,12 @@ func TestHandler(t *testing.T) {
id: request.IDResponseValidAccepted,
parts: []part{
heapProfilePart(),
heapProfilePart(),
part{
name: "profile",
// No messageType param specified, so pprof is assumed.
contentType: "application/x-protobuf",
body: heapProfileBody(),
},
part{
name: "metadata",
contentType: "application/json",
Expand All @@ -147,17 +154,28 @@ func TestHandler(t *testing.T) {
"ProfileInvalidContentType": {
id: request.IDResponseErrorsValidate,
parts: []part{{
name: "metadata",
name: "profile",
contentType: "text/plain",
body: strings.NewReader(""),
}},
body: prettyJSON(map[string]interface{}{"accepted": 0}),
},
"ProfileInvalidMessageType": {
id: request.IDResponseErrorsValidate,
parts: []part{{
name: "profile",
// Improperly formatted "messageType" param
// in Content-Type from APM Agent Go v1.6.0.
contentType: "application/x-protobuf; messageType=”perftools.profiles.Profile",
body: strings.NewReader(""),
}},
body: prettyJSON(map[string]interface{}{"accepted": 0}),
},
"ProfileInvalid": {
id: request.IDResponseErrorsDecode,
parts: []part{{
name: "profile",
contentType: "application/x-protobuf",
contentType: pprofContentType,
body: strings.NewReader("foo"),
}},
body: prettyJSON(map[string]interface{}{"accepted": 0}),
Expand All @@ -168,7 +186,7 @@ func TestHandler(t *testing.T) {
heapProfilePart(),
part{
name: "profile",
contentType: "application/x-protobuf",
contentType: pprofContentType,
body: strings.NewReader(strings.Repeat("*", 10*1024*1024)),
},
},
Expand Down Expand Up @@ -261,15 +279,15 @@ func emptyDec(_ *http.Request) (map[string]interface{}, error) {
}

func heapProfilePart() part {
return part{name: "profile", contentType: pprofContentType, body: heapProfileBody()}
}

func heapProfileBody() io.Reader {
var buf bytes.Buffer
if err := pprof.WriteHeapProfile(&buf); err != nil {
panic(err)
}
return part{
name: "profile",
contentType: "application/x-protobuf",
body: &buf,
}
return &buf
}

type part struct {
Expand Down

0 comments on commit 279b81f

Please sign in to comment.