commit
61c9db4a05
6 changed files with 2307 additions and 0 deletions
@ -0,0 +1,15 @@ |
|||||
|
# Copyright 2020 Keyporttech 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. |
||||
|
|
||||
|
FROM scratch |
||||
|
ADD "$BUILT_ARTIFACT" / |
||||
|
EXPOSE 8080 |
||||
|
ENTRYPOINT /giteainterceptor |
||||
@ -0,0 +1,201 @@ |
|||||
|
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: |
||||
|
|
||||
|
(a) You must give any other recipients of the Work or |
||||
|
Derivative Works a copy of this License; and |
||||
|
|
||||
|
(b) You must cause any modified files to carry prominent notices |
||||
|
stating that You changed the files; and |
||||
|
|
||||
|
(c) 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 |
||||
|
|
||||
|
(d) 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. |
||||
@ -0,0 +1,12 @@ |
|||||
|
# gitea-interceptor for Tekton |
||||
|
|
||||
|
This a custom gitea interceptor for [tekton triggers](https://github.com/tektoncd/triggers). |
||||
|
|
||||
|
The primary function is use the configured webhook secret key to validate payload encryption checksum. |
||||
|
|
||||
|
This code borrows from: |
||||
|
- https://github.com/keyporttech/gitea-tekton-interceptor |
||||
|
- https://github.com/tektoncd/triggers/blob/v0.17.1/cmd/interceptors/main.go |
||||
|
- https://github.com/tektoncd/triggers/blob/v0.17.1/pkg/interceptors/server/server.go |
||||
|
- https://github.com/tektoncd/triggers/blob/v0.17.1/pkg/interceptors/github/github.go |
||||
|
|
||||
@ -0,0 +1,11 @@ |
|||||
|
module itix.fr/giteainterceptor |
||||
|
|
||||
|
go 1.16 |
||||
|
|
||||
|
require ( |
||||
|
github.com/tektoncd/triggers v0.17.1 |
||||
|
go.uber.org/zap v1.19.0 |
||||
|
k8s.io/client-go v0.21.4 |
||||
|
knative.dev/pkg v0.0.0-20210827184538-2bd91f75571c |
||||
|
google.golang.org/grpc v1.40.0 |
||||
|
) |
||||
File diff suppressed because it is too large
@ -0,0 +1,295 @@ |
|||||
|
/* |
||||
|
Copyright 2020 The Tekton Authors |
||||
|
|
||||
|
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 main |
||||
|
|
||||
|
import ( |
||||
|
"bytes" |
||||
|
"context" |
||||
|
"errors" |
||||
|
"fmt" |
||||
|
"io" |
||||
|
"encoding/json" |
||||
|
"log" |
||||
|
"net" |
||||
|
"net/http" |
||||
|
"time" |
||||
|
"crypto/hmac" |
||||
|
"crypto/sha256" |
||||
|
"encoding/hex" |
||||
|
"hash" |
||||
|
|
||||
|
"google.golang.org/grpc/codes" |
||||
|
"go.uber.org/zap" |
||||
|
"k8s.io/client-go/rest" |
||||
|
secretInformer "knative.dev/pkg/client/injection/kube/informers/core/v1/secret" |
||||
|
"knative.dev/pkg/injection" |
||||
|
"knative.dev/pkg/logging" |
||||
|
"knative.dev/pkg/signals" |
||||
|
triggersv1 "github.com/tektoncd/triggers/pkg/apis/triggers/v1beta1" |
||||
|
corev1lister "k8s.io/client-go/listers/core/v1" |
||||
|
"github.com/tektoncd/triggers/pkg/interceptors" |
||||
|
) |
||||
|
|
||||
|
const ( |
||||
|
// Port is the port that the port that interceptor service listens on
|
||||
|
Port = 8080 |
||||
|
readTimeout = 5 * time.Second |
||||
|
writeTimeout = 20 * time.Second |
||||
|
idleTimeout = 60 * time.Second |
||||
|
) |
||||
|
|
||||
|
func main() { |
||||
|
// set up signals so we handle the first shutdown signal gracefully
|
||||
|
ctx := signals.NewContext() |
||||
|
|
||||
|
clusterConfig, err := rest.InClusterConfig() |
||||
|
if err != nil { |
||||
|
log.Fatalf("Failed to build config: %v", err) |
||||
|
} |
||||
|
|
||||
|
ctx, startInformer := injection.EnableInjectionOrDie(ctx, clusterConfig) |
||||
|
|
||||
|
zap, err := zap.NewProduction() |
||||
|
if err != nil { |
||||
|
log.Fatalf("failed to initialize logger: %s", err) |
||||
|
} |
||||
|
logger := zap.Sugar() |
||||
|
ctx = logging.WithLogger(ctx, logger) |
||||
|
defer func() { |
||||
|
if err := logger.Sync(); err != nil { |
||||
|
log.Fatalf("failed to sync the logger: %s", err) |
||||
|
} |
||||
|
}() |
||||
|
|
||||
|
secretLister := secretInformer.Get(ctx).Lister() |
||||
|
service := NewGiteaInterceptor(secretLister, logger) |
||||
|
startInformer() |
||||
|
|
||||
|
mux := http.NewServeMux() |
||||
|
mux.Handle("/", service) |
||||
|
mux.HandleFunc("/ready", readinessHandler) |
||||
|
|
||||
|
srv := &http.Server{ |
||||
|
Addr: fmt.Sprintf(":%d", Port), |
||||
|
BaseContext: func(listener net.Listener) context.Context { |
||||
|
return ctx |
||||
|
}, |
||||
|
ReadTimeout: readTimeout, |
||||
|
WriteTimeout: writeTimeout, |
||||
|
IdleTimeout: idleTimeout, |
||||
|
Handler: mux, |
||||
|
} |
||||
|
|
||||
|
logger.Infof("Listen and serve on port %d", Port) |
||||
|
if err := srv.ListenAndServe(); err != nil { |
||||
|
logger.Fatalf("failed to start interceptors service: %v", err) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
func readinessHandler(w http.ResponseWriter, r *http.Request) { |
||||
|
w.WriteHeader(http.StatusOK) |
||||
|
} |
||||
|
|
||||
|
func (gi *GiteaInterceptor) ServeHTTP(w http.ResponseWriter, r *http.Request) { |
||||
|
b, err := gi.executeInterceptor(r) |
||||
|
if err != nil { |
||||
|
switch e := err.(type) { |
||||
|
case Error: |
||||
|
gi.Logger.Infof("HTTP %d - %s", e.Status(), e) |
||||
|
http.Error(w, e.Error(), e.Status()) |
||||
|
default: |
||||
|
gi.Logger.Errorf("Non Status Error: %s", err) |
||||
|
http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError) |
||||
|
} |
||||
|
} |
||||
|
w.Header().Add("Content-Type", "application/json") |
||||
|
if _, err := w.Write(b); err != nil { |
||||
|
gi.Logger.Errorf("failed to write response: %s", err) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// Error represents a handler error. It provides methods for a HTTP status
|
||||
|
// code and embeds the built-in error interface.
|
||||
|
type Error interface { |
||||
|
error |
||||
|
Status() int |
||||
|
} |
||||
|
|
||||
|
// HTTPError represents an error with an associated HTTP status code.
|
||||
|
type HTTPError struct { |
||||
|
Code int |
||||
|
Err error |
||||
|
} |
||||
|
|
||||
|
// Allows HTTPError to satisfy the error interface.
|
||||
|
func (se HTTPError) Error() string { |
||||
|
return se.Err.Error() |
||||
|
} |
||||
|
|
||||
|
// Returns our HTTP status code.
|
||||
|
func (se HTTPError) Status() int { |
||||
|
return se.Code |
||||
|
} |
||||
|
|
||||
|
func badRequest(err error) HTTPError { |
||||
|
return HTTPError{Code: http.StatusBadRequest, Err: err} |
||||
|
} |
||||
|
|
||||
|
func internal(err error) HTTPError { |
||||
|
return HTTPError{Code: http.StatusInternalServerError, Err: err} |
||||
|
} |
||||
|
|
||||
|
func (gi *GiteaInterceptor) executeInterceptor(r *http.Request) ([]byte, error) { |
||||
|
// Create a context
|
||||
|
ctx, cancel := context.WithTimeout(r.Context(), 3*time.Second) |
||||
|
defer cancel() |
||||
|
|
||||
|
var body bytes.Buffer |
||||
|
defer r.Body.Close() |
||||
|
if _, err := io.Copy(&body, r.Body); err != nil { |
||||
|
return nil, internal(fmt.Errorf("failed to read body: %w", err)) |
||||
|
} |
||||
|
var ireq triggersv1.InterceptorRequest |
||||
|
if err := json.Unmarshal(body.Bytes(), &ireq); err != nil { |
||||
|
return nil, badRequest(fmt.Errorf("failed to parse body as InterceptorRequest: %w", err)) |
||||
|
} |
||||
|
gi.Logger.Debugf("Interceptor Request is: %+v", ireq) |
||||
|
iresp := gi.Process(ctx, &ireq) |
||||
|
gi.Logger.Infof("Interceptor response is: %+v", iresp) |
||||
|
respBytes, err := json.Marshal(iresp) |
||||
|
if err != nil { |
||||
|
return nil, internal(err) |
||||
|
} |
||||
|
return respBytes, nil |
||||
|
} |
||||
|
|
||||
|
// ErrInvalidContentType is returned when the content-type is not a JSON body.
|
||||
|
var ErrInvalidContentType = errors.New("form parameter encoding not supported, please change the hook to send JSON payloads") |
||||
|
|
||||
|
type GiteaInterceptor struct { |
||||
|
SecretLister corev1lister.SecretLister |
||||
|
Logger *zap.SugaredLogger |
||||
|
} |
||||
|
|
||||
|
func NewGiteaInterceptor(s corev1lister.SecretLister, l *zap.SugaredLogger) *GiteaInterceptor { |
||||
|
return &GiteaInterceptor{ |
||||
|
SecretLister: s, |
||||
|
Logger: l, |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
func (w *GiteaInterceptor) Process(ctx context.Context, r *triggersv1.InterceptorRequest) *triggersv1.InterceptorResponse { |
||||
|
headers := interceptors.Canonical(r.Header) |
||||
|
if v := headers.Get("Content-Type"); v == "application/x-www-form-urlencoded" { |
||||
|
return interceptors.Fail(codes.InvalidArgument, ErrInvalidContentType.Error()) |
||||
|
} |
||||
|
|
||||
|
p := triggersv1.GitHubInterceptor{} |
||||
|
if err := interceptors.UnmarshalParams(r.InterceptorParams, &p); err != nil { |
||||
|
return interceptors.Failf(codes.InvalidArgument, "failed to parse interceptor params: %v", err) |
||||
|
} |
||||
|
|
||||
|
// Check if the event type is in the allow-list
|
||||
|
if p.EventTypes != nil { |
||||
|
actualEvent := headers.Get("X-Gitea-Event") |
||||
|
isAllowed := false |
||||
|
for _, allowedEvent := range p.EventTypes { |
||||
|
if actualEvent == allowedEvent { |
||||
|
isAllowed = true |
||||
|
break |
||||
|
} |
||||
|
} |
||||
|
if !isAllowed { |
||||
|
return interceptors.Failf(codes.FailedPrecondition, "event type %s is not allowed", actualEvent) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// Next validate secrets
|
||||
|
if p.SecretRef != nil { |
||||
|
// Check the secret to see if it is empty
|
||||
|
if p.SecretRef.SecretKey == "" { |
||||
|
return interceptors.Fail(codes.FailedPrecondition, "gitea interceptor secretRef.secretKey is empty") |
||||
|
} |
||||
|
header := headers.Get("X-Gitea-Signature") |
||||
|
if header == "" { |
||||
|
return interceptors.Fail(codes.FailedPrecondition, "no X-Gitea-Signature header set") |
||||
|
} |
||||
|
|
||||
|
ns, _ := triggersv1.ParseTriggerID(r.Context.TriggerID) |
||||
|
secret, err := w.SecretLister.Secrets(ns).Get(p.SecretRef.SecretName) |
||||
|
if err != nil { |
||||
|
return interceptors.Failf(codes.FailedPrecondition, "error getting secret: %v", err) |
||||
|
} |
||||
|
secretToken := secret.Data[p.SecretRef.SecretKey] |
||||
|
|
||||
|
if err := validateSignature(header, []byte(r.Body), secretToken); err != nil { |
||||
|
return interceptors.Fail(codes.FailedPrecondition, err.Error()) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return &triggersv1.InterceptorResponse{ |
||||
|
Continue: true, |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// genMAC generates the HMAC signature for a message provided the secret key
|
||||
|
// and hashFunc.
|
||||
|
func genMAC(message, key []byte, hashFunc func() hash.Hash) []byte { |
||||
|
mac := hmac.New(hashFunc, key) |
||||
|
mac.Write(message) |
||||
|
return mac.Sum(nil) |
||||
|
} |
||||
|
|
||||
|
// checkMAC reports whether messageMAC is a valid HMAC tag for message.
|
||||
|
func checkMAC(message, messageMAC, key []byte, hashFunc func() hash.Hash) bool { |
||||
|
expectedMAC := genMAC(message, key, hashFunc) |
||||
|
return hmac.Equal(messageMAC, expectedMAC) |
||||
|
} |
||||
|
|
||||
|
// messageMAC returns the hex-decoded HMAC tag from the signature and its
|
||||
|
// corresponding hash function.
|
||||
|
func messageMAC(signature string) ([]byte, func() hash.Hash, error) { |
||||
|
if signature == "" { |
||||
|
return nil, nil, errors.New("missing signature") |
||||
|
} |
||||
|
|
||||
|
var hashFunc func() hash.Hash |
||||
|
|
||||
|
hashFunc = sha256.New |
||||
|
|
||||
|
buf, err := hex.DecodeString(signature) |
||||
|
if err != nil { |
||||
|
return nil, nil, fmt.Errorf("error decoding signature %q: %v", signature, err) |
||||
|
} |
||||
|
return buf, hashFunc, nil |
||||
|
} |
||||
|
|
||||
|
// validateSignature validates the signature for the given payload.
|
||||
|
// signature is the gitea hash signature delivered in the X-Gitea-Signature header.
|
||||
|
// payload is the JSON payload sent by gitea Webhooks.
|
||||
|
// secretToken is the gitea Webhook secret token.
|
||||
|
//
|
||||
|
func validateSignature(signature string, payload, secretToken []byte) error { |
||||
|
messageMAC, hashFunc, err := messageMAC(signature) |
||||
|
if err != nil { |
||||
|
return err |
||||
|
} |
||||
|
if !checkMAC(payload, messageMAC, secretToken, hashFunc) { |
||||
|
return errors.New("payload signature check failed") |
||||
|
} |
||||
|
return nil |
||||
|
} |
||||
Loading…
Reference in new issue