Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/spectralops/keyscope
Keyscope is a key and secret workflow (validation, invalidation, etc.) tool built in Rust
https://github.com/spectralops/keyscope
developer-tools devops pentest rust security testing vault
Last synced: about 2 hours ago
JSON representation
Keyscope is a key and secret workflow (validation, invalidation, etc.) tool built in Rust
- Host: GitHub
- URL: https://github.com/spectralops/keyscope
- Owner: SpectralOps
- License: apache-2.0
- Created: 2021-10-01T12:01:49.000Z (over 3 years ago)
- Default Branch: master
- Last Pushed: 2024-05-19T09:01:13.000Z (8 months ago)
- Last Synced: 2025-01-19T19:05:59.649Z (7 days ago)
- Topics: developer-tools, devops, pentest, rust, security, testing, vault
- Language: Rust
- Homepage:
- Size: 433 KB
- Stars: 387
- Watchers: 17
- Forks: 117
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE.txt
Awesome Lists containing this project
README
:white_check_mark: Automate your key and secret validation workflows
:cowboy_hat_face: Over 30 different providers
:robot: Export to JSON, audit via CSV
# :key: Keyscope
Keyscope is a key and secret workflow (validation, invalidation, etc.) tool built in Rust, powered by [`service_policy_kit`](https://github.com/spectralops/service-policy-kit).
Current workflows supported:
- Validation
# 🦀 Why Rust?
- With Rust, _"If it compiles, it works."_ and also, it compiles to many platforms.
- Rust is _fast_, has no _VM_, or unnecessary cruft (typically 5-8mb binaries with LOTS of code and libraries).
- Multi purpose, safe, and generalistic - makes for healthy and expressive [mission critical code](https://www.youtube.com/watch?v=ylOpCXI2EMM). Adding code or abstraction doesn't increase bloat, doesn't hurt performance, doesn't increase chance for bugs in a radical way (less edge cases).
- Amazing package manager: `Cargo`. Productive installing and running of tasks and examples.
- Rust is getting headlines in the security community as the go-to language for security tools. Equally interesting is offensive security + Rust [here](https://github.com/trickster0/OffensiveRust) and [here](https://github.com/skerkour/black-hat-rust).# :rocket: Quick Start
Grab a release from [releases](https://github.com/spectralops/keyscope/releases), or install via Homebrew:
```
brew tap spectralops/tap && brew install keyscope
```## Using keyscope
You can try out validating a key for a provider, say, Github (assuming the key is in the `GITHUB_TOKEN` environment variable):
```
$ keyscope validate github $GITHUB_TOKEN
```You can see which other providers are supported by running:
```
$ keyscope validate --list.
:
.twilio:validation
keyscope validate twilio -p twilio_1 -p twilio_2twitter:validation
keyscope validate twitter -p twitter_1zendesk:validation
keyscope validate zendesk -p zendesk_1 -p zendesk_2Total 44 providers available.
$
```And what parameters are required for a certain provider by running (say, `stripe`):
```
$ keyscope requirements stripeprovider stripe requires:
- param: p1
desc: stripe key
$
```Finally the general structure of the `validate` command is:
```
$ keyscope validate PROVIDER -p PARAM1 -p PARAM2 .. -p PARAM_N
```# :white_check_mark: Validation: key should be active
You can validate a specific provider like so:
```
$ keyscope validate twilio -p $TWILIO_KEY
```With the general pattern of:
```
$ keyscope validate PROVIDER -p PARAM1 -p PARAM2 ...
```The number of keys/params would change based on authentication type:
- `Bearer` - usually just a single key (token)
- `Basic Auth` - usually 2 keys: user, password
- `OAuth` - usually 2 keys: client_id, client_secret
- And others.Each provider in Keyscope will tell you what it requires using `requirements`:
```
$ keyscope requirements twilio
```You'll get a report:
```
$ keyscope --verbose validate stripe -p $STRIPE_KEY✔ stripe:validation: ok 766ms
Ran 1 interactions with 1 checks in 766ms
Success: 1
Failure: 0
Error: 0
Skipped: 0
```And an executable exit code that reflects success/failure.
You can use the `--verbose` flag to see API responses:
```
$ keyscope --verbose validate stripe -p $STRIPE_KEY✗ stripe:validation: failed 413ms
status_code: 200 != 401 UnauthorizedRan 1 interactions with 1 checks in 413ms
Success: 0
Failure: 1
Error: 0
Skipped: 0
```In this case the exit code is `1` (failure).
# :x: Validation: key should be inactive
When you are validating keys that are supposed to be inactive, you can use the `flip` flag. In this mode, a failed API access is a good thing, and the exit code will reflect that.
```
$ keyscope --flip validate stripe -p $STRIPE_KEY✔ stripe:validation: ok 766ms
Ran 1 interactions with 1 checks in 766ms
```In this case, the key is active - _which is bad for us_. Using `--flip`, the exit code will be `1` (failure).
# :runner: Setting up a validation job
## Audit active keys
You can set up a CI job (or other form of scheduled job you like) to perform an audit, by reading all parameters from a dedicated CSV file like so:
```
$ keyscope validate --csv-in report.csv
```The format of the CSV that you need to prepare should include a header line and look like this:
```
provider,key1,key2,key3
twilio,tw-key1,,,
```You can specify as many key columns as you like, as long as you provide an _empty_ value for providers which don't have that many keys, and all rows contain the same amount of cells.
## Audit inactive keys
If you have a dump of keys from your vault that are stale have expiry and should have been rotated, you want to test that they are all stale:
```
$ keyscope --flip validate --csv-in my-key-audit.csv
```# :link: Supported providers
We're always adding [new providers](src/defs.yaml), keep a link to this list or watch this repo to get updated.
We use our [`service_policy_kit`](https://github.com/spectralops/service-policy-kit) library to specify interactions with services and their policies, if you find a service [not in this list](src/defs.yaml) feel free to open an issue or contribute back.
**tester**
Tester: valid keyvalidation
`tester_1` - hookbin ID (https://hookb.in)
`tester_2` - fake key to put as a query param```
keyscope validate tester -p TESTER_1 -p TESTER_2
```**infura**
infura API keyvalidation
`infura_1` - infura API key
```
keyscope validate infura -p INFURA_1
```**covalenthq**
Covalent: valid keyvalidation
`covalenthq_1` - covalent token
```
keyscope validate covalenthq -p COVALENTHQ_1
```**asana**
Asana: valid tokenvalidation
`asana_1` - asana token
```
keyscope validate asana -p ASANA_1
```**bitly**
Bit.ly: valid access tokenvalidation
`bitly_1` - bit.ly token
```
keyscope validate bitly -p BITLY_1
```**ipstack**
ipstack access keyvalidation
`ipstack_1` - ipstack access key
```
keyscope validate ipstack -p IPSTACK_1
```**localytics**
Localytics: valid API credentialsvalidation
`localytics_1` - localytics user
`localytics_2` - localytics key```
keyscope validate localytics -p LOCALYTICS_1 -p LOCALYTICS_2
```**algolia**
Algolia: valid API credentialsvalidation
`algolia_1` - algolia application ID
`algolia_2` - algolia index
`algolia_3` - algolia API key```
keyscope validate algolia -p ALGOLIA_1 -p ALGOLIA_2 -p ALGOLIA_3
```**branchio**
branch.io: valid API credentialsvalidation
`branchio_1` - branch.io key
`branchio_2` - branch.io secret```
keyscope validate branchio -p BRANCHIO_1 -p BRANCHIO_2
```**browserstack**
browserstack: valid API credentialsvalidation
`browserstack_1` - browserstack key
`browserstack_2` - browserstack secret```
keyscope validate browserstack -p BROWSERSTACK_1 -p BROWSERSTACK_2
```**buildkite**
Buildkite: valid tokenvalidation
`buildkite_1` - buildkite token
```
keyscope validate buildkite -p BUILDKITE_1
```**datadog**
datadog: valid API credentialsvalidation
`datadog_1` - datadog API key
```
keyscope validate datadog -p DATADOG_1
```**github**
github: valid API credentialsvalidation
`github_1` - github token
```
keyscope validate github -p GITHUB_1
```**github-ent**
Github Enterprise: valid API tokenvalidation
`github-ent_1` - github enterprise instance (without http)
`github-ent_2` - github token```
keyscope validate github-ent -p GITHUB-ENT_1 -p GITHUB-ENT_2
```**dropbox**
dropbox: valid API credentialsvalidation
`dropbox_1` - dropbox token
```
keyscope validate dropbox -p DROPBOX_1
```**gitlab**
gitlab: valid API credentialsvalidation
`gitlab_1` - gitlab token
```
keyscope validate gitlab -p GITLAB_1
```**heroku**
heroku: valid API credentialsvalidation
`heroku_1` - heroku token
```
keyscope validate heroku -p HEROKU_1
```**mailchimp**
mailchimp: valid API credentialsvalidation
`mailchimp_1` - mailchimp datacenter ID
`mailchimp_2` - mailchimp key```
keyscope validate mailchimp -p MAILCHIMP_1 -p MAILCHIMP_2
```**mailgun**
mailgun: valid API credentialsvalidation
`mailgun_1` - mailgun key
```
keyscope validate mailgun -p MAILGUN_1
```**pagerduty**
pagerduty: valid API credentialsvalidation
`pagerduty_1` - pagerduty token
```
keyscope validate pagerduty -p PAGERDUTY_1
```**circleci**
circleci: valid API credentialsvalidation
`circleci_1` - circleci key
```
keyscope validate circleci -p CIRCLECI_1
```**facebook-access-token**
facebook: valid API tokenvalidation
`facebook-access-token_1` - facebook token
```
keyscope validate facebook-access-token -p FACEBOOK-ACCESS-TOKEN_1
```**salesforce**
salesforce: valid API credentialsvalidation
`salesforce_1` - salesforce instance name
`salesforce_2` - salesforce token```
keyscope validate salesforce -p SALESFORCE_1 -p SALESFORCE_2
```**jumpcloud**
jumpcloud: valid API credentialsvalidation
`jumpcloud_1` - jumpcloud key
```
keyscope validate jumpcloud -p JUMPCLOUD_1
```**saucelabs-us**
saucelabs-us: valid API credentialsvalidation
`saucelabs-us_1` - saucelabs user
`saucelabs-us_2` - saucelabs key```
keyscope validate saucelabs-us -p SAUCELABS-US_1 -p SAUCELABS-US_2
```**saucelabs-eu**
saucelabs-eu: valid API credentialsvalidation
`saucelabs-eu_1` - saucelabs user
`saucelabs-eu_2` - saucelabs key```
keyscope validate saucelabs-eu -p SAUCELABS-EU_1 -p SAUCELABS-EU_2
```**sendgrid**
sendgrid: valid API credentialsvalidation
`sendgrid_1` - sendgrid key
```
keyscope validate sendgrid -p SENDGRID_1
```**slack**
slack: valid API credentialsvalidation
`slack_1` - slack key
```
keyscope validate slack -p SLACK_1
```**slack-webhook**
slack-webook: valid API credentialsvalidation
`slack-webhook_1` - slack webhook
```
keyscope validate slack-webhook -p SLACK-WEBHOOK_1
```**stripe**
stripe: valid API credentialsvalidation
`stripe_1` - stripe key
```
keyscope validate stripe -p STRIPE_1
```**travisci**
travisci: valid API credentialsvalidation
`travisci_1` - travisci domain, choose 'org' or 'com'
`travisci_2` - travisci key```
keyscope validate travisci -p TRAVISCI_1 -p TRAVISCI_2
```**twilio**
twilio: valid API credentialsvalidation
`twilio_1` - twilio account sid
`twilio_2` - twilio token```
keyscope validate twilio -p TWILIO_1 -p TWILIO_2
```**twitter**
twitter: valid API credentialsvalidation
`twitter_1` - twitter API token
```
keyscope validate twitter -p TWITTER_1
```**zendesk**
zendesk: valid API credentialsvalidation
`zendesk_1` - zendesk domain
`zendesk_2` - zendesk key```
keyscope validate zendesk -p ZENDESK_1 -p ZENDESK_2
```**firebase**
firebase: valid API credentialsvalidation
`firebase_1` - firebase API key
`firebase_2` - firebase ID token```
keyscope validate firebase -p FIREBASE_1 -p FIREBASE_2
```**aws**
aws: valid API credentialsvalidation
`aws_1` - AWS ID
`aws_2` - AWS secret```
keyscope validate aws -p AWS_1 -p AWS_2
```**elastic-apm-secret**
Elastic APM: secret key validationvalidation
`elastic-apm-secret_1` - Elastic APM host address and port, including 'http/s' part
`elastic-apm-secret_2` - Elastic APM secret```
keyscope validate elastic-apm-secret -p ELASTIC-APM-SECRET_1 -p ELASTIC-APM-SECRET_2
```**artifactory**
Artifactory: token validationvalidation
`artifactory_1` - Artifactory host (including http(s) part)
`artifactory_2` - Artifactory token```
keyscope validate artifactory -p ARTIFACTORY_1 -p ARTIFACTORY_2
```**ibm-cos**
IBM: cloud object storage key validation (HMAC)validation
`ibm-cos_1` - IBM HMAC ID
`ibm-cos_2` - IBM HMAC secret```
keyscope validate ibm-cos -p IBM-COS_1 -p IBM-COS_2
```**ibm-iam**
IBM: cloud key validation (IAM)validation
`ibm-iam_1` - IBM cloud key
```
keyscope validate ibm-iam -p IBM-IAM_1
```**ibm-cloudant**
IBM: cloudant key validation (legacy)validation
`ibm-cloudant_1` - IBM cloudant hostname
`ibm-cloudant_2` - IBM cloudant user
`ibm-cloudant_3` - IBM cloudant key```
keyscope validate ibm-cloudant -p IBM-CLOUDANT_1 -p IBM-CLOUDANT_2 -p IBM-CLOUDANT_3
```**softlayer**
Softlayer: validate credentialsvalidation
`softlayer_1` - Softlayer hostname
`softlayer_2` - Softlayer token```
keyscope validate softlayer -p SOFTLAYER_1 -p SOFTLAYER_2
```**square**
Square: valid tokenvalidation
`square_1` - Square token
```
keyscope validate square -p SQUARE_1
```**telegram-bot**
telegram-bot: valid bot tokenvalidation
`telegram-bot_1` - bot key
```
keyscope validate telegram-bot -p TELEGRAM-BOT_1
```**bingmaps**
Bing Maps API: valid access tokenvalidation
`bingmaps_1` - Bing Maps token
```
keyscope validate bingmaps -p BINGMAPS_1
```**buttercms**
ButterCMS: valid bot tokenvalidation
`buttercms_1` - ButterCMS API key
```
keyscope validate buttercms -p BUTTERCMS_1
```**wakatime**
wakatime: valid api tokenvalidation
`wakatime_1` - WakeTime API key
```
keyscope validate wakatime -p WAKATIME_1
```**calendly**
calendly: valid API credentialsvalidation
`calendly_1` - calendly API key
```
keyscope validate calendly -p CALENDLY_1
```**shodan**
shodan: valid api tokenvalidation
`shodan_1` - Shodan API key
```
keyscope validate shodan -p SHODAN_1
```**opsgenie**
opsgenie: valid api tokenvalidation
`opsgenie_1` - opsgenie API key
```
keyscope validate opsgenie -p OPSGENIE_1
```**pendo**
pendo: valid api tokenvalidation
`pendo_1` - pendo API key
```
keyscope validate pendo -p PENDO_1
```**hubspot**
hubspot: valid api tokenvalidation
`hubspot_1` - hubspot API key
```
keyscope validate hubspot -p HUBSPOT_1
```**lokalise**
lokalise: valid api tokenvalidation
`lokalise_1` - lokalise token
```
keyscope validate lokalise -p LOKALISE_1
```# :cake: Adding your own providers
You can specify a custom definitions file (here [is an example](custom-defs.example.yaml)):
```
$ keyscope -f your-definitions.yaml validate --list
```Which is suitable for adding your own internal services, key issuing policies, and infrastructure.
You can also use custom definitions to test out new providers that you'd like to contribute back to keyscope :smile:
## Basics of a definition
All definitions represent an interaction. A request being made, and a policy that's being checked against it.
```yaml
providers:
hookbin:
validation:
#
# the request part
#
request:
params:
- name: hookbin_1
desc: hookbin ID (https://hookb.in)
- name: hookbin_2
desc: fake key to put as a query param
id: "postbin:validation"
desc: "Postbin: valid key"
# variable interpolation is good for all fields
uri: "https://hookb.in/{{hookbin_1}}?q={{hookbin_2}}"
method: post
# you can also use headers, body, form, basic_auth and more (see defs.yaml)#
# the policy part: all values are actually regular expressions and are matched against service response
#
response:
status_code: "200"
body: ok
```When in doubt, you can check keyscope's own [defs.yaml](src/defs.yaml) for real examples of how to use this infrastructure.
## Tutorial: adding Dropbox
To validate if a dropbox API key works, we first need to learn about the canonical way to authenticate against that API.
First stop, API docs:
- Dropbox has an [API Explorer](https://dropbox.github.io/dropbox-api-v2-explorer) which is super useful
Next stop, we want to find an API call that is a representative for:
- Has to be authenticated
- Has to indicate that when accessed successfully with our candidate key, the key has some authoritative value. Which means, that if exposed, contains significant risk.For this example, getting our current account sounds like something that only when we identify who we are - we're able to do.
We'll select [get_current_account](https://www.dropbox.com/developers/documentation/http/documentation#users-get_current_account).
Let's start forming our interaction. First the needed skeleton: containing the name of the provider (`dropbox`), its ID and description below, as well as parameters required and their name and description:
```yaml
dropbox:
validation:
request:
id: "dropbox:validation"
desc: "dropbox: valid API credentials"
params:
- name: dropbox_1
desc: dropbox token
```We keep the name of the parameter with a special convention that helps when feeding keyscope automatically:
```
PROVIDER_N
Where 'N' starts in 1 e.g.:
dropbox_1
dropbox_2
aws_1
...
```Then, details about actually making an HTTP call, as required by Dropbox (Bearer token authentication).
```yaml
uri: https://api.dropboxapi.com/2/users/get_current_account
method: post
headers:
Authorization:
- Bearer {{dropbox_1}}
```Note that per standard, all HTTP header fields are actually _arrays_. It's OK to always make an array of size _one_ if you only have one value (most common case).
We also see _variable interpolation_ here. Where `{{dropbox_1}}` will get replaced by keyscope in time before making the actual call.
Finally, we want to make sure we answer the question:
- What does it mean to have a successful call?
In our case, the Dropbox API call returns `HTTP OK` on success, which means a `200` status code.
And the final, complete result is this:
```yaml
dropbox:
validation:
request:
id: "dropbox:validation"
desc: "dropbox: valid API credentials"
params:
- name: dropbox_1
desc: dropbox token
uri: https://api.dropboxapi.com/2/users/get_current_account
method: post
headers:
Authorization:
- Bearer {{dropbox_1}}
response:
status_code: "200"
```Meanwhile, you can drop this provider in your own `providers.yaml` file and run keyscope:
```
$ keyscope -f providers.yaml validate dropbox -p MY_KEY
```Now you can keep this in your private `providers.yaml` file or contribute it back to keyscope if you think other people might enjoy using it - we're happy to accept pull requests.
# Thanks
To all [Contributors](https://github.com/spectralops/keyscope/graphs/contributors) - you make this happen, thanks!
# Copyright
Copyright (c) 2021 [@jondot](http://twitter.com/jondot). See [LICENSE](LICENSE.txt) for further details.