Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/go-fed/httpsig
Golang implementation of the HTTP Signatures RFC draft, with SSH support!
https://github.com/go-fed/httpsig
golang golang-library http http-signature http-signatures httpsig signatures signing ssh
Last synced: 2 days ago
JSON representation
Golang implementation of the HTTP Signatures RFC draft, with SSH support!
- Host: GitHub
- URL: https://github.com/go-fed/httpsig
- Owner: go-fed
- License: bsd-3-clause
- Created: 2018-05-15T20:49:36.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2024-01-19T14:56:22.000Z (12 months ago)
- Last Synced: 2024-10-29T18:38:41.443Z (2 months ago)
- Topics: golang, golang-library, http, http-signature, http-signatures, httpsig, signatures, signing, ssh
- Language: Go
- Homepage:
- Size: 73.2 KB
- Stars: 76
- Watchers: 3
- Forks: 29
- Open Issues: 9
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# httpsig
> HTTP Signatures made simple
[![Build Status][Build-Status-Image]][Build-Status-Url] [![Go Reference][Go-Reference-Image]][Go-Reference-Url]
[![Go Report Card][Go-Report-Card-Image]][Go-Report-Card-Url] [![License][License-Image]][License-Url]
[![Chat][Chat-Image]][Chat-Url] [![OpenCollective][OpenCollective-Image]][OpenCollective-Url]`go get github.com/go-fed/httpsig`
Implementation of [HTTP Signatures](https://tools.ietf.org/html/draft-cavage-http-signatures).
Supports many different combinations of MAC, HMAC signing of hash, or RSA
signing of hash schemes. Its goals are:* Have a very simple interface for signing and validating
* Support a variety of signing algorithms and combinations
* Support setting either headers (`Authorization` or `Signature`)
* Remaining flexible with headers included in the signing string
* Support both HTTP requests and responses
* Explicitly not support known-cryptographically weak algorithms
* Support automatic signing and validating Digest headers## How to use
`import "github.com/go-fed/httpsig"`
### Signing
Signing a request or response requires creating a new `Signer` and using it:
```
func sign(privateKey crypto.PrivateKey, pubKeyId string, r *http.Request) error {
prefs := []httpsig.Algorithm{httpsig.RSA_SHA512, httpsig.RSA_SHA256}
digestAlgorithm := DigestSha256
// The "Date" and "Digest" headers must already be set on r, as well as r.URL.
headersToSign := []string{httpsig.RequestTarget, "date", "digest"}
signer, chosenAlgo, err := httpsig.NewSigner(prefs, digestAlgorithm, headersToSign, httpsig.Signature)
if err != nil {
return err
}
// To sign the digest, we need to give the signer a copy of the body...
// ...but it is optional, no digest will be signed if given "nil"
body := ...
// If r were a http.ResponseWriter, call SignResponse instead.
return signer.SignRequest(privateKey, pubKeyId, r, body)
}
````Signer`s are not safe for concurrent use by goroutines, so be sure to guard
access:```
type server struct {
signer httpsig.Signer
mu *sync.Mutex
}func (s *server) handlerFunc(w http.ResponseWriter, r *http.Request) {
privateKey := ...
pubKeyId := ...
// Set headers and such on w
s.mu.Lock()
defer s.mu.Unlock()
// To sign the digest, we need to give the signer a copy of the response body...
// ...but it is optional, no digest will be signed if given "nil"
body := ...
err := s.signer.SignResponse(privateKey, pubKeyId, w, body)
if err != nil {
...
}
...
}
```The `pubKeyId` will be used at verification time.
### Verifying
Verifying requires an application to use the `pubKeyId` to both retrieve the key
needed for verification as well as determine the algorithm to use. Use a
`Verifier`:```
func verify(r *http.Request) error {
verifier, err := httpsig.NewVerifier(r)
if err != nil {
return err
}
pubKeyId := verifier.KeyId()
var algo httpsig.Algorithm = ...
var pubKey crypto.PublicKey = ...
// The verifier will verify the Digest in addition to the HTTP signature
return verifier.Verify(pubKey, algo)
}
````Verifier`s are not safe for concurrent use by goroutines, but since they are
constructed on a per-request or per-response basis it should not be a common
restriction.[Build-Status-Image]: https://travis-ci.org/go-fed/httpsig.svg?branch=master
[Build-Status-Url]: https://travis-ci.org/go-fed/httpsig
[Go-Reference-Image]: https://pkg.go.dev/badge/github.com/go-fed/httpsig
[Go-Reference-Url]: https://pkg.go.dev/github.com/go-fed/httpsig
[Go-Report-Card-Image]: https://goreportcard.com/badge/github.com/go-fed/httpsig
[Go-Report-Card-Url]: https://goreportcard.com/report/github.com/go-fed/httpsig
[License-Image]: https://img.shields.io/github/license/go-fed/httpsig?color=blue
[License-Url]: https://opensource.org/licenses/BSD-3-Clause
[Chat-Image]: https://img.shields.io/matrix/go-fed:feneas.org?server_fqdn=matrix.org
[Chat-Url]: https://matrix.to/#/!BLOSvIyKTDLIVjRKSc:feneas.org?via=feneas.org&via=matrix.org
[OpenCollective-Image]: https://img.shields.io/opencollective/backers/go-fed-activitypub-labs
[OpenCollective-Url]: https://opencollective.com/go-fed-activitypub-labs