DEV Community

Cover image for Connecting a Biometric Fingerprint Device to a Rails Web App Using Python β€” A Complete Walkthrough
Lakshay Tyagi
Lakshay Tyagi

Posted on • Originally published at imlakshay08-complete-ruby-on-rails.hashnode.dev

Connecting a Biometric Fingerprint Device to a Rails Web App Using Python β€” A Complete Walkthrough

πŸ‘‹ Introduction

When I was building a gym management system for a real gym in New Delhi, one of the most interesting challenges was connecting a physical biometric fingerprint device to my cloud-hosted Ruby on Rails app.

The gym wanted to:

  • βœ… Track member attendance via fingerprint scanning
  • βœ… Automatically deny access to members with expired subscriptions
  • βœ… Prevent duplicate check-ins
  • βœ… See attendance on the admin dashboard in real-time

The catch? The biometric device only speaks to the local network. My Rails app is hosted on Render (cloud). They can't talk to each other directly.

The solution? A Python bridge script running on the gym's laptop that reads fingerprint punches from the device and forwards them to the Rails API via HTTP.

This post covers the entire pipeline β€” Python bridge β†’ Rails API β†’ Database β€” with real code from production.


πŸ—οΈ Architecture: The Full Pipeline

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  ZK Fingerprint      β”‚
β”‚  Biometric Device    β”‚
β”‚  (192.168.1.201)     β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
           β”‚ pyzk SDK (TCP)
           β–Ό
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  Python Bridge Script    β”‚
β”‚  (bridge.py)             β”‚
β”‚  Runs on gym laptop      β”‚
β”‚  Polls every 20 seconds  β”‚
β”‚  Deduplicates locally    β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
           β”‚ HTTP POST (JSON)
           β–Ό
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  Ruby on Rails API (Render cloud)    β”‚
β”‚  POST /api/biometric_attendances     β”‚
β”‚                                      β”‚
β”‚  1. Find biometric mapping           β”‚
β”‚  2. Check for duplicate punches      β”‚
β”‚  3. Validate subscription            β”‚
β”‚  4. Store attendance                 β”‚
β”‚  5. Return ALLOWED / DENIED          β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
           β”‚
           β–Ό
β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  MySQL Database (CleverCloud)        β”‚
β”‚                                      β”‚
β”‚  trn_member_biometric_mappings       β”‚
β”‚  trn_member_attendances              β”‚
β”‚  trn_member_subscriptions            β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
Enter fullscreen mode Exit fullscreen mode

Three components, two languages, one seamless flow.


🐍 Part 1: The Python Bridge (Gym Laptop Side)

The Problem

The biometric device (a ZK-based fingerprint scanner) is on the gym's local network at 192.168.1.201. It stores fingerprints and punch records internally. It has no concept of "calling a web API."

My Rails app is hosted on Render β€” a cloud server that the device can't reach directly.

Solution: A Python script that acts as the middleman β€” reads from the device using the pyzk SDK, and forwards punches to Rails via HTTP.

Configuration

# biometric_bridge/config.py

DEVICE_IP = "192.168.1.201"
DEVICE_PORT = 4370
DEVICE_TIMEOUT = 5

RAILS_API_URL = "https://spine-fitness.com/api/biometric_attendances"
COMP_CODE = "SF"

POLL_INTERVAL_SECONDS = 20
Enter fullscreen mode Exit fullscreen mode

Key decisions:

  • Port 4370 β€” Standard ZK biometric device communication port
  • 20-second polling β€” Balances real-time feel vs. not overwhelming the device
  • Company code "SF" β€” Supports multi-tenant architecture (future-proof for multiple gyms)

The Bridge Script

# biometric_bridge/bridge.py

from zk import ZK
import requests
import time
from datetime import datetime
from config import *

def send_to_rails(payload):
    try:
        response = requests.post(
            RAILS_API_URL,
            json=payload,
            timeout=5
        )
        print(f"Sent: {payload} | Response: {response.status_code}")
    except Exception as e:
        print("Rails API error:", e)

def main():
    zk = ZK(
        DEVICE_IP,
        port=DEVICE_PORT,
        timeout=DEVICE_TIMEOUT,
        password=0,
        force_udp=False,
        ommit_ping=False
    )

    print("Connecting to biometric device...")

    try:
        conn = zk.connect()
        conn.disable_device()

        print("Connected to device")
        print("Fetching attendance logs...")

        last_sent = set()

        while True:
            attendances = conn.get_attendance()

            for att in attendances:
                key = f"{att.user_id}-{att.timestamp}"

                # Prevent duplicate sending
                if key in last_sent:
                    continue

                payload = {
                    "compcode": COMP_CODE,
                    "user_id": att.user_id,
                    "timestamp": att.timestamp.strftime("%Y-%m-%d %H:%M:%S"),
                    "device_sn": conn.serial_number
                }

                send_to_rails(payload)
                last_sent.add(key)

            time.sleep(POLL_INTERVAL_SECONDS)

    except Exception as e:
        print("Device connection error:", e)

    finally:
        try:
            conn.enable_device()
            conn.disconnect()
        except:
            pass

if __name__ == "__main__":
    main()
Enter fullscreen mode Exit fullscreen mode

How It Works Step by Step

1. Connect to the device

zk = ZK(DEVICE_IP, port=DEVICE_PORT, timeout=DEVICE_TIMEOUT, ...)
conn = zk.connect()
conn.disable_device()  # Prevents new operations while reading
Enter fullscreen mode Exit fullscreen mode

The pyzk library connects to the ZK device via TCP on port 4370. We temporarily disable the device during reads to prevent data corruption.

2. Poll every 20 seconds

while True:
    attendances = conn.get_attendance()
    # ... process ...
    time.sleep(POLL_INTERVAL_SECONDS)
Enter fullscreen mode Exit fullscreen mode

The script runs in an infinite loop, fetching all attendance records from the device. The device stores punches internally, so we get the full history each time.

3. Deduplicate locally

last_sent = set()

key = f"{att.user_id}-{att.timestamp}"
if key in last_sent:
    continue
Enter fullscreen mode Exit fullscreen mode

Since get_attendance() returns all historical records, we use an in-memory set to track what's already been sent. Only new punches get forwarded. This is deduplication layer 1 β€” the Rails API has its own deduplication as layer 2.

4. Forward to Rails API

payload = {
    "compcode": COMP_CODE,
    "user_id": att.user_id,
    "timestamp": att.timestamp.strftime("%Y-%m-%d %H:%M:%S"),
    "device_sn": conn.serial_number
}
send_to_rails(payload)
Enter fullscreen mode Exit fullscreen mode

Each punch becomes a clean JSON payload with all the context Rails needs.

Auto-Start on Windows

The gym staff shouldn't need to "start the bridge" manually. A .bat file handles this:

:: biometric_bridge/start_biometric.bat
cd C:\biometric_bridge
python bridge.py
Enter fullscreen mode Exit fullscreen mode

This can be placed in the Windows Startup folder so the bridge starts automatically when the gym opens and the laptop powers on.

Dependencies

# biometric_bridge/requirements.txt
pyzk
requests
Enter fullscreen mode Exit fullscreen mode

Just two dependencies β€” pyzk for ZK device communication and requests for HTTP calls. Minimal and reliable.


πŸ’Ž Part 2: The Rails API (Cloud Server Side)

The Route

# config/routes.rb
Rails.application.routes.draw do
  namespace :api do
    resources :biometric_attendances, only: [:create]
  end
end
Enter fullscreen mode Exit fullscreen mode

This gives us: POST /api/biometric_attendances

The Controller

When the Python bridge sends a punch, the Rails API processes it through a 4-step pipeline:

# app/controllers/api/biometric_attendances_controller.rb

class Api::BiometricAttendancesController < ApplicationController
  skip_before_action :verify_authenticity_token

  def create
    compcode       = params[:compcode].to_s
    device_user_id = params[:user_id].to_i
    device_sn      = params[:device_sn].to_s
    punch_time     = Time.zone.parse(params[:timestamp]) rescue Time.current

    # ── STEP 1: Find biometric mapping ──
    mapping = TrnMemberBiometricMapping.active.find_by(
      mbm_compcode:       compcode,
      mbm_device_user_id: device_user_id,
      mbm_device_sn:      device_sn
    )

    unless mapping
      render json: {
        status: false,
        message: "Biometric user not mapped"
      }, status: 404
      return
    end

    member = mapping.member

    # ── STEP 2: Ignore duplicate punches (same member, same minute) ──
    if duplicate_punch?(member.id, punch_time)
      render json: { status: true, message: "Duplicate ignored" }
      return
    end

    # ── STEP 3: Validate subscription ──
    subscription = latest_subscription(member.id, compcode)

    if subscription && subscription.ms_end_date >= Date.today
      att_status = "ALLOWED"
      reason     = nil
    else
      att_status = "DENIED"
      reason     = "Subscription expired"
    end

    # ── STEP 4: Store attendance ──
    TrnMemberAttendance.create!(
      att_compcode:       compcode,
      att_member_id:      member.id,
      att_device_user_id: device_user_id,
      att_device_sn:      device_sn,
      att_punch_time:     punch_time,
      att_punch_date:     punch_time.to_date,
      att_status:         att_status,
      att_reason:         reason
    )

    render json: { status: true, access: att_status }
  end

  private

  def duplicate_punch?(member_id, time)
    TrnMemberAttendance.where(
      att_member_id: member_id,
      att_punch_time: time.beginning_of_minute..time.end_of_minute
    ).exists?
  end

  def latest_subscription(member_id, compcode)
    TrnMemberSubscription
      .where(ms_compcode: compcode, ms_member_id: member_id)
      .order(ms_end_date: :desc)
      .first
  end
end
Enter fullscreen mode Exit fullscreen mode

πŸ›‘οΈ Two Layers of Deduplication

This is important β€” deduplication happens at both levels:

Layer Where How Why
Layer 1 Python bridge In-memory set of {user_id}-{timestamp} keys Prevents resending the same device record on every poll cycle
Layer 2 Rails API Database query for same member + same minute Catches duplicates if the bridge restarts (set resets), or if the device sends duplicate records
Member scans finger
       β”‚
       β–Ό
Python bridge: "Already in last_sent?" ──YES──▢ Skip
       β”‚ NO
       β–Ό
Rails API: "Punch in same minute?" ──YES──▢ Return "Duplicate ignored"
       β”‚ NO
       β–Ό
Store attendance βœ…
Enter fullscreen mode Exit fullscreen mode

This double-layer approach means the system is resilient to:

  • Bridge restarts (set clears β†’ Layer 2 catches it)
  • Device quirks (some ZK devices record multiple entries per scan)
  • Network retries (if the bridge retries a failed request)

πŸ—„οΈ Database Design

The Bridge Table: trn_member_biometric_mappings

Column Type Purpose
mbm_compcode string Company code (multi-tenant)
mbm_device_user_id integer User ID on the biometric device
mbm_device_sn string Device serial number
mbm_member_id integer FK β†’ mst_members_lists.id
mbm_status string ACTIVE / INACTIVE

Why is this needed? The biometric device assigns its own user IDs (1, 2, 3...). These don't match your database. This table says "Device user #42 on device SN-ABC = Member Rahul Sharma (ID: 156)".

The Attendance Table: trn_member_attendances

Column Type Purpose
att_member_id integer FK β†’ member
att_device_user_id integer Device user ID
att_device_sn string Device serial
att_punch_time datetime Exact punch time
att_punch_date date For easy date-based queries
att_status string ALLOWED or DENIED
att_reason string Why denied (if applicable)

Key design decision: Even DENIED attempts are stored. This lets the gym owner see which expired members are still trying to come β€” useful for renewal follow-ups.


πŸ§ͺ Testing the Full Pipeline

Test with cURL (bypassing the Python bridge)

# Active member
curl -X POST https://spine-fitness.com/api/biometric_attendances \
  -H "Content-Type: application/json" \
  -d '{
    "compcode": "SF",
    "user_id": 42,
    "device_sn": "CRT5200-SN001",
    "timestamp": "2026-03-12 07:30:00"
  }'
# β†’ {"status":true,"access":"ALLOWED"}

# Expired member
# β†’ {"status":true,"access":"DENIED"}

# Unknown biometric ID
# β†’ {"status":false,"message":"Biometric user not mapped"}
Enter fullscreen mode Exit fullscreen mode

Test the Python bridge locally

cd biometric_bridge
pip install -r requirements.txt
python bridge.py
# β†’ Connecting to biometric device...
# β†’ Connected to device
# β†’ Fetching attendance logs...
# β†’ Sent: {...} | Response: 200
Enter fullscreen mode Exit fullscreen mode

🧩 Edge Cases I Solved

Edge Case Solution
Device returns ALL historical records every poll Python-side last_sent set filters to only new records
Member scans finger 3 times rapidly Rails-side 1-minute deduplication window
Bridge script crashes / laptop restarts .bat file in Startup folder auto-restarts; Rails Layer 2 catches re-sent duplicates
Gym has multiple devices device_sn is part of the mapping β€” same user_id on different devices = different members
Timestamp timezone mismatch Time.zone.parse with rescue Time.current fallback
Member renews subscription mid-day ms_end_date >= Date.today check means renewal takes effect immediately
Member leaves permanently Set mbm_status to INACTIVE β€” .active scope blocks without deleting data
Network/API timeout Python requests.post(timeout=5) with try/except β€” failed sends are logged, not fatal

πŸ“Š How This Powers the Dashboard

All attendance data flows to the admin dashboard in real-time:

  • 🟒 Active members β€” subscription valid, attendance tracked
  • 🟑 Expiring soon β€” auto-triggers WhatsApp reminders
  • πŸ”΄ Expired β€” DENIED attendance logged, visible to gym owner

The gym owner sees a member scan their finger, and within seconds the dashboard reflects it β€” all without touching a single register.


πŸ’‘ Key Takeaways

  1. Use a bridge pattern when hardware can't talk to cloud directly. A simple Python script solved the hardware↔cloud gap.

  2. Deduplicate at every layer. Don't trust any single layer to handle it perfectly.

  3. Store denied attempts. They're not failures β€” they're business intelligence.

  4. Keep the bridge minimal. Two dependencies (pyzk + requests), one config file, one script. Less can go wrong.

  5. Auto-start everything. The gym staff shouldn't need to know Python exists. A .bat file in Startup and it just works.

  6. Multi-language is fine. Python is better at hardware communication (pyzk), Rails is better at web apps. Use the right tool for each layer.


🏁 Conclusion

This feature taught me that production software often lives at the intersection of hardware and software. The biometric device, the Python bridge, the Rails API, and the MySQL database β€” four different technologies working together to create a seamless experience: member scans finger β†’ attendance appears on dashboard.

The most satisfying moment? Watching the gym owner check the dashboard and see real-time attendance without touching a single notebook.


πŸ”— Live App: spine-fitness.com
πŸ’» Full Source Code: GitHub
πŸ“‚ Python Bridge Code: biometric_bridge/

Found this useful? Drop a ❀️! Got questions about biometric integration or the Python bridge? Let's chat in the comments!

Top comments (0)