Skip to content
This repository has been archived by the owner on Dec 23, 2023. It is now read-only.

Commit

Permalink
Merge pull request #39 from b3n4kh/feature/goreleaser
Browse files Browse the repository at this point in the history
Feature/goreleaser
  • Loading branch information
muety authored Jan 24, 2023
2 parents df44f7f + e74ffdd commit dc15c5b
Show file tree
Hide file tree
Showing 6 changed files with 182 additions and 12 deletions.
44 changes: 44 additions & 0 deletions .github/workflows/build.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
name: build

on:
push:
tags:
- '**'

permissions:
contents: write
packages: write

jobs:
goreleaser:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v3
with:
fetch-depth: 0
- name: Fetch all tags
run: git fetch --force --tags
- name: Set up Go
uses: actions/setup-go@v3
with:
go-version: 1.19
cache: true
- name: get ghcr owner repository
run: |
echo "GHCR_OWNER=${GITHUB_REPOSITORY_OWNER,,}" >>${GITHUB_ENV}
- name: Login to GitHub Container Registry
uses: docker/login-action@v2
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Run GoReleaser
uses: goreleaser/goreleaser-action@v4
with:
distribution: goreleaser
version: latest
args: release --rm-dist
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
GHCR_OWNER: ${{ env.GHCR_OWNER }}
6 changes: 5 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,8 @@
config*.yml
!config.default.yml
mailwhale
*.db
node_modules
package-lock.json
package.json
dist
*.db
47 changes: 47 additions & 0 deletions .goreleaser.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
---
builds:
- id: mailwhale-builds
binary: mailwhale
dir: .
before:
hooks:
- sudo npm install -g yarn
- yarn --cwd webui
- yarn --cwd webui build
- cp config.default.yml config.yml
archives:
- id: mailwhale-archives
builds:
- mailwhale-builds
files:
- LICENSE
- README.md
- webui/public/*
- config.yml
- templates/*
- version.txt
replacements:
darwin: Darwin
linux: Linux
386: i386
amd64: x86_64
name_template: "{{ .Binary }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}"
dockers:
- ids:
- mailwhale-builds
goos: linux
goarch: amd64
dockerfile: Dockerfile-goreleaser
extra_files:
- "version.txt"
- "config.yml"
- "LICENSE"
- "README.md"
- "templates/"
- "webui/"

image_templates:
- "ghcr.io/{{ .Env.GHCR_OWNER }}/mailwhale:latest"
- "ghcr.io/{{ .Env.GHCR_OWNER }}/mailwhale:{{ .Tag }}"
release:
mode: replace
4 changes: 2 additions & 2 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# Build Stage

FROM golang:1.18 AS api-build-env
FROM golang:1.19 AS api-build-env

WORKDIR /src
ADD . .
Expand All @@ -11,7 +11,7 @@ RUN cp /src/mailwhale . && \
cp /src/version.txt . && \
cp -r /src/templates .

FROM node:14 AS ui-build-env
FROM node:18 AS ui-build-env

WORKDIR /src
ADD webui .
Expand Down
31 changes: 31 additions & 0 deletions Dockerfile-goreleaser
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
# Build Stage

FROM golang:1.19 AS api-build-env

WORKDIR /app

FROM node:18 AS ui-build-env

WORKDIR /src
ADD webui .
RUN yarn && \
yarn build

# Run Stage

# When running the application using `docker run`, you can pass environment variables
# to override config values using `-e` syntax.

FROM alpine
WORKDIR /app

COPY mailwhale README.md version.txt LICENSE config.yml templates/ ./
COPY --from=ui-build-env /src/public ./webui/public

ENV MW_ENV=prod
ENV MW_WEB_LISTEN_V4=0.0.0.0:3000
ENV MW_STORE_PATH=/data/data.json.db

VOLUME /data

ENTRYPOINT ./mailwhale
62 changes: 53 additions & 9 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,9 @@ As a developer, chances are high that at some point you need to teach your appli
Essentially, there are two options. Either you use a **professional mail sending service** or
you **include an SMTP client library** to your software and **plug your own mail server**.

Think of MailWhale like [Mailgun](https://mailgun.com), [SendGrid](https://sendgrid.com) or [SMTPeter](https://smtpeter.com), but open source and self-hosted. Or like [Postal](https://github.com/postalhq/postal) or [Cuttlefish](https://cuttlefish.io/), but less bloated and without running its own, internal SMTP server.
Think of MailWhale like [Mailgun](https://mailgun.com), [SendGrid](https://sendgrid.com) or [SMTPeter](https://smtpeter.com), but open source and self-hosted. Or like [Postal](https://github.com/postalhq/postal) or [Cuttlefish](https://cuttlefish.io/), but less bloated and without running its own, internal SMTP server.

However, if you want the best of both worlds – that is, send mails via simple HTTP calls and with no extra complexity,
However, if you want the best of both worlds – that is, send mails via simple HTTP calls and with no extra complexity,
but still use your own infrastructure – you may want to go with ✉️🐳.

You get a simple **REST API**, which you can call to send out e-mail. You can plug your self-hosted SMTP server, as well as Google Mail or **literally any other e-mail provider**.
Expand All @@ -29,12 +29,15 @@ Stay tuned, there is a lot more to come.
![](assets/screenshot01.png)

## 🚧 Project State

The project is in a very early stage and breaking changes are likely to happen. We'd recommend to not yet use this in production or at least expect non-trivial effort required to upgrade to a new version.

For a more stable and robust alternative to MailWhale, check out [postalsys/emailengine](https://github.com/postalsys/emailengine).

## 📦 Installation

### Compile from source

```bash
# 1. Clone repo
$ git clone https://github.com/muety/mailwhale.git
Expand All @@ -54,7 +57,37 @@ $ GO111MODULE=on go build
$ ./mailwhale
```

### With Docker
### From GitHub Release

```bash
# 1. Download latest release
curl -s https://api.github.com/repos/muety/mailwhale/releases/latest | jq -r ".assets[] | select(.name|match(\"Linux_$(arch).tar.gz\")) | .browser_download_url" | wget -qi -

# 2. Extract
mkdir mailwhale
tar xf mailwhale_*.tar.gz -C mailwhale
cd mailwhale

# 3.[Optional] Adapt config to your needs, i.e. set your SMTP server and credentials, etc.
# vi config.yml

# 4. Run it
./mailwhale
```

### With Docker Image

```bash
$ docker run -d \
-p 127.0.0.1:3000:3000 \
-v "$(pwd)/config.yml":/app/config.yml:ro \
-v mailwhale_data:/data \
--name mailwhale \
docker pull ghcr.io/muety/mailwhale
```

### Build custom Docker Image

```bash
# 1. Clone repo
$ git clone https://github.com/muety/mailwhale.git
Expand All @@ -78,18 +111,22 @@ $ docker run -d \
mailwhale
```

**Note:** An official Docker image is about to come. Also, there will be no need to mount your config file into the container, as everything will be configurable using environment variables eventually.
**Note:** An official Docker image is about to come. Also, there will be no need to mount your config file into the container, as everything will be configurable using environment variables eventually.

## ⌨️ Usage
First of all, you can get most tasks done through the web UI, available at http://localhost:3000.

First of all, you can get most tasks done through the web UI, available at <http://localhost:3000>.

### 1. Define a user
To get started with MailWhale, you need to create a **user** first. To do so, register a new user API or web UI. `security.allow_signup` needs to be set to `true`.

To get started with MailWhale, you need to create a **user** first. To do so, register a new user API or web UI. `security.allow_signup` needs to be set to `true`.

### 2. Create an API client
It is good practice to not authenticate against the API as a user directly. Instead, create an **API client** with limited privileges, that could easily be revoked in the future. A client is identified by a **client ID** and a **client secret** (or token), very similar to what you might already be familiar with from AWS APIs. Usually, such a client corresponds to an individual client application of yours, which wants to access MailWhale's API.

It is good practice to not authenticate against the API as a user directly. Instead, create an **API client** with limited privileges, that could easily be revoked in the future. A client is identified by a **client ID** and a **client secret** (or token), very similar to what you might already be familiar with from AWS APIs. Usually, such a client corresponds to an individual client application of yours, which wants to access MailWhale's API.

#### Request

```bash
$ curl -XPOST \
-u 'admin@local.host:admin' \
Expand All @@ -103,7 +140,8 @@ $ curl -XPOST \
```

#### Response
```

```json
{
"id": "SVNORFBUWGhxWGZSUUl0eA==",
"description": "My juicy web app",
Expand All @@ -129,6 +167,7 @@ $ echo "Authorization: Basic $(echo '<client_id>:<client_secret>' | base64)"
### 3. Send E-Mails

#### Plain text or HTML

```bash
$ curl -XPOST \
-u '<client_id>:<client_secret>' \
Expand All @@ -144,7 +183,9 @@ $ curl -XPOST \
You can also a `text` field instead, to send a plain text message.

#### Using a template

In case you have created a template using the web UI, you can reference it in a new mail like so:

```bash
$ curl -XPOST \
-u '<client_id>:<client_secret>' \
Expand All @@ -161,6 +202,7 @@ $ curl -XPOST \
```

## 🔧 Configuration Options

You can specify configuration options either via a config file (`config.yml`) or via environment variables. Here is an overview of all options.

| YAML Key | Environment Variable | Default | Description |
Expand All @@ -184,14 +226,16 @@ You can specify configuration options either via a config file (`config.yml`) or
| `security.block_list` | `MW_SECURITY_BLOCK_LIST` | `[]` | List of [regexes](https://regex101.com/) used to block certain recipient addresses |

### Sender verification & SPF Check

By default, mails are sent using a randomly generated address in the `From` header, which belongs to the domain configured via `mail.domain` (i.e. `user+abcdefgh@wakapi.dev`). Optionally, custom sender addresses can be configured on a per-API-client basis. However, it is recommended to properly configure [SPF](https://en.wikipedia.org/wiki/Sender_Policy_Framework) on that custom domain and instruct MailWhale to verify that configuration.

**As a user**, you need to configure your domain, which you want to use as part of your senders address (e.g. `example.org` for sending mails from `User Server <noreply@example.org>`), to publish an SPF record that delegates to the domain under which MailWhale is running (e.g. mailwhale.dev).

```
example.org. IN TXT v=spf1 include:mailwhale.dev
```

**As a server operator** of a MailWhale instance, you need to enable `mail.verify_senders` and set your `mail.domain` and `web.public_url`. For that domain, you need to configure an SPF record that allows your SMTP relay provider's (e.g. Mailbox.org, GMail, SendGrid, etc.) mail servers to be senders. Refer to your provider's documentation, e.g. [this](https://kb.mailbox.org/display/MBOKBEN/How+to+integrate+external+e-mail+accounts).
**As a server operator** of a MailWhale instance, you need to enable `mail.verify_senders` and set your `mail.domain` and `web.public_url`. For that domain, you need to configure an SPF record that allows your SMTP relay provider's (e.g. Mailbox.org, GMail, SendGrid, etc.) mail servers to be senders. Refer to your provider's documentation, e.g. [this](https://kb.mailbox.org/display/MBOKBEN/How+to+integrate+external+e-mail+accounts).

## 🚀 Features (planned)

Expand Down

0 comments on commit dc15c5b

Please sign in to comment.