Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/babylonhealth/stevenson
Stevenson is a Vapor framework designed to build integrations between Slack apps, GitHub, JIRA and CI services (CircleCI).
https://github.com/babylonhealth/stevenson
bot circle-ci fastlane ios server-side-swift slack vapor
Last synced: about 2 hours ago
JSON representation
Stevenson is a Vapor framework designed to build integrations between Slack apps, GitHub, JIRA and CI services (CircleCI).
- Host: GitHub
- URL: https://github.com/babylonhealth/stevenson
- Owner: babylonhealth
- License: mit
- Created: 2019-02-28T14:58:38.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2021-07-15T12:49:11.000Z (over 3 years ago)
- Last Synced: 2023-03-02T11:51:05.083Z (over 1 year ago)
- Topics: bot, circle-ci, fastlane, ios, server-side-swift, slack, vapor
- Language: Swift
- Homepage:
- Size: 421 KB
- Stars: 58
- Watchers: 16
- Forks: 2
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Codeowners: .github/CODEOWNERS
Awesome Lists containing this project
README
# 🤖 Stevenson Bot
`Stevenson` is a Vapor framework designed to build integrations between Slack apps, GitHub, JIRA and CI services (CircleCI).
This project also contains implementation of the Slack app used by Babylon iOS team (if you want to know more about how our team works, check our [playbook](https://github.com/babylonhealth/ios-playbook))## 🚀 Usage
To use `Stevenson` in your app add it as a dependency to your `Package.swift`:
```swift
dependencies: [
.package(url: "https://github.com/babylonhealth/Stevenson.git", .branch("master")),
]
```and then import it to your project:
```swift
import Stevenson
```### Supported services
`Stevenson` comes with implementation of Slack [slash commands](https://api.slack.com/slash-commands), GitHub, JIRA and CircleCI APIs. At the moment it does not implement complete set of these APIs but only provides bare minimum required for the functionality of the app.
To create these services, use corresponding type constructors providing required values. It's advised but not required to store these values in the environment variables:
```swift
let slack = SlackService(
token: Environment.get("SLACK_TOKEN")
)let ci = CircleCIService(
token: Environment.get("CIRCLECI_TOKEN")
)let jira = JiraService(
baseURL: Environment.get("JIRA_BASE_URL").flatMap(URL.init(string:)),
username: Environment.get("JIRA_USERNAME"),
password: Environment.get("JIRA_TOKEN")
)let github = GitHubService(
username: Environment.get("GITHUB_USERNAME"),
token: Environment.get("GITHUB_TOKEN")
)
```### Creating a Slack command
To create a Slack [slash command](https://api.slack.com/slash-commands) start with registering it in your Slack app following Slack documentation.
_Note: instead of using your own Slack app you may use Slack Slash Commands app. In this case you will register your slash commands in this app but the process will be pretty much the same._
Then use the `SlackCommand` type to implement it:
```swift
let myAmazingCommand = SlackCommand(
// This name should be the same name that you used to register a command in your Slack app
name: "myAmazingCommand",
// This message will be sent back to Slack when you call your command with `/myAmazingCommand help`
help: "Some command usage instructions",
// This closure is where the command is actually implemented
run: { metadata, request in
/**
Parse `metadata` here, do something useful,
i.e. invoke a CI job, and send a response back to Slack
*/
}
)
```
Then register a route in your app with a path that matches the command name (usually Slack sends commands as `POST` requests):
```swift
router.post(myAmazingCommand.name) { request -> Future in
try slack.handle(command: myAmazingCommand, on: request)
}
```For more details check the commands implemented in the app.
## 💻 Development
To develop locally on this repo:
* [Install Vapor locally](http://docs.vapor.codes/3.0/install/macos/)
* Run `vapor xcode` to create the Xcode project
* Open the Xcode project and work in it
* You'll need to define some environment variables in your scheme (Edit Scheme > "Run" Action > "Arguments" tab) if you want to try to run the [Babylon Stevenson app](BabylonCommands.md) locally. Typically declare [the same env variables](BabylonCommands.md#environment-variables) that are defined in the hosting environment (like Heroku or AWS on the server side).
* Hit Cmd-R to run the Vapor server locally. It will listen at `http://localhost:8080`
* Try it out by sending fake Slack payloads mimicking a Slack slash commandFor example to simulate `/fastlane somelane someargs`, use this (adapt the `&text=` value and the `/fastlane` endpoint to your needs)
```bash
curl --request POST \
--url http://localhost:8080/fastlane \
--header 'content-type: application/x-www-form-urlencoded' \
--data 'token=__SLACK_TOKEN__&channel_name=__SOME_SLACK_CHANNEL__&text=somelane%20someargs'
```## 🕹 Adding a new Slack command to the app
If you need to create a new Slack command:
1. Go to the Slack Commands config page for your team's Slack app: `https://api.slack.com/apps//slash-commands`
2. Click on "Create New Command"
* Fill in the slash command (e.g. `/foo`)
* Enter the URL this command will trigger – e.g. if your instance will be hosted on Heroku it will look like `https://.herokuapp.com/`, replacing `` with the name of your Heroku app instance (e.g. `stevenson-bot`) and `` by the command name (e.g. `foo`)
* Fill in the short description and the hint for the command
* Hit "Save"
3. Open the project in Xcode and add a new handler for the Slack command:
* Implement your command as a static function/constant in a `extension SlackCommand` (see `MainCommand.swift` or `SlackCommand+Fastlane` as examples):
```swift
extension SlackCommand {
static func (/* optional params if needed */) {
SlackCommand(
name: "",
help: "...",
allowedChannels: [...],
run: { ... }
)
}
}
```
* Open `configure.swift` and add that newly-created command to the list of handled commands```swift
routes(router: router, slack: slack, commands: [
.fastlane(ci),
...,
.()
])
```## 🚢 Deployment on Heroku
There are just example instructions if you plan to host the app on [Heroku](https://dashboard.heroku.com/apps).
_You can skip step 1 and 2 if you have already set it up locally._
1. Install the Heroku CLI.
```bash
brew install heroku/brew/heroku
heroku login
```
2. Navigate to the local repo, and configure the Heroku remote using the CLI.
```bash
heroku git:remote -a
```
3. Push the `master` branch to deploy.
```bash
git checkout master
git push heroku master
```Alternatively, you can deploy a specific branch manually by going to the deploy page on Heroku dashboard and using the "Manual Deploy" section at the very bottom.
Once the app is deployed, if you need to debug things, you can see the logs using `heroku logs -a `.
To set the aforementioned environment variables with the real values on Heroku:
* Go to your Heroku dashboard
* Navigate to Settings
* set the environment variables like `SLACK_TOKEN` etc## 📖 Documentation
* Learn more about the Stevenson's instance we use at Babylon and the commands that we implemented using this framework to support our processes [here](BabylonCommands.md)
* For instructions on how to use Vapor, visit the Vapor web framework's [documentation](http://docs.vapor.codes).## ✨ Take a look at our other OSS projects!
* [Bento](https://github.com/babylonhealth/Bento): Swift library for building component-based interfaces on top of UITableView and UICollectionView 🍱
* [DrawerKit](https://github.com/babylonhealth/DrawerKit): DrawerKit lets an UIViewController modally present another UIViewController in a manner similar to the way Apple's Maps app works.
* [ReactiveFeedback](https://github.com/babylonhealth/ReactiveFeedback): Unidirectional reactive architecture
* 🚧 [Wall-E](https://github.com/babylonhealth/Wall-E): A bot that monitors and manages your pull requests by ensuring they are merged when they're ready and don't stack up in your repository 🤓