How I Set Up a Self-Hosted Poll Tool for Cevi Wetzikon

For Cevi Wetzikon, I was looking for an open-source tool to create polls and meeting date finders, ideally something I could host myself under a custom domain. After testing various solutions, I discovered Framadate – and it ticked all the boxes:

The official installation guide describes a manual setup on a Linux server. That works, but I wanted something more maintainable. So I decided to go with Docker, using Portainer for easy management – including automatic image updates via webhook.

Dockerfile

To containerize Framadate, I created the following Dockerfile from scratch:

FROM php:7.4-apache

RUN apt-get update && \
    apt-get install -y git libicu-dev libxml2-dev zip unzip zlib1g-dev g++ mariadb-client jq pandoc && \
    rm -rf /var/lib/apt/lists/*

RUN docker-php-ext-install pdo pdo_mysql intl && \
    a2enmod rewrite

# Framadate Folder
RUN mkdir -p /var/www/framadate
RUN git clone https://framagit.org/framasoft/framadate/framadate.git /var/www/framadate

# Apache Configuration
COPY ./framadate/apache.conf /etc/apache2/sites-available/000-default.conf
COPY ./framadate/framadate-admin-htaccess.txt /var/www/framadate/admin/.htaccess
COPY ./framadate/framadate-htaccess.txt /var/www/framadate/.htaccess

WORKDIR /var/www/framadate

# Composer
RUN php -r "readfile('https://getcomposer.org/installer');" | php -- --install-dir=/usr/local/bin --filename=composer && \
    composer install

EXPOSE 80
CMD ["apache2-foreground"]

Customizing Framadate

Since I wasn’t a big fan of some of the default Framadate UI and logic, I decided to customize parts of the application. I created my own versions of certain files and overwrote them directly via Dockerfile. This gives me full control and helps me keep track of what I’ve changed.

Examples:

COPY ./framadate/php-create_classic_poll.php /var/www/framadate/create_classic_poll.php
COPY ./framadate/./docs /var/www/framadate/docs
COPY ./framadate/script-mdtodocspage.sh /var/www/framadate/docs/script-mdtodocspage.sh
RUN chmod +x /var/www/framadate/docs/script-mdtodocspage.sh
RUN /var/www/framadate/docs/script-mdtodocspage.sh

This way, my Framadate instance runs the way I want it – and I understand what’s happening under the hood.

Deployment Workflow

I set up a GitHub Action to automatically build and push the Docker image to the GitHub Container Registry whenever I push changes. This image is then deployed to my server via Tailscale VPN and a Portainer Webhook – fully automated.

GithubAction.png

Docker Compose in Portainer

The following docker-compose.yml is used in Portainer on my Docker Ubuntu server. The Framadate image is re-pulled automatically whenever a new version is pushed:

version: '3'
services:
  framadate:
    image: ghcr.io/surmatik/framadate-ceviwetzikon:latest
    ports:
      - "8081:80"
    depends_on:
      - mysql
    volumes:
      - /opt/docker/framadate-ceviwetzikon/framadate-config.php:/var/www/framadate/app/inc/config.php
  mysql:
    image: mysql:5.7
    environment:
      - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD}
      - MYSQL_DATABASE=${MYSQL_DATABASE}
      - MYSQL_USER=${MYSQL_USER}
      - MYSQL_PASSWORD=${MYSQL_PASSWORD}
    volumes:
      - mysql_data:/var/lib/mysql
    command: --bind-address=0.0.0.0

volumes:
  mysql_data:

Environment variables:

Hosting and Access

The setup runs on my Docker-based Ubuntu server, fully managed with Portainer and secured through Tailscale. Everything stays up to date via webhook – and it just works.

DockerContainerNetzwerk.png

You can check it out live at:
pool.cevi-wetzikon.ch