π 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 β
ββββββββββββββββββββββββββββββββββββββββ
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
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()
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
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)
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
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)
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
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
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
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
π‘οΈ 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 β
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"}
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
π§© 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
Use a bridge pattern when hardware can't talk to cloud directly. A simple Python script solved the hardwareβcloud gap.
Deduplicate at every layer. Don't trust any single layer to handle it perfectly.
Store denied attempts. They're not failures β they're business intelligence.
Keep the bridge minimal. Two dependencies (
pyzk+requests), one config file, one script. Less can go wrong.Auto-start everything. The gym staff shouldn't need to know Python exists. A
.batfile in Startup and it just works.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)