line-client

LINE messaging integration via Chrome extension gateway. Send/read LINE messages, manage contacts, groups, profile, and reactions. Authenticate with QR code login. Provides HMAC-signed API access through the Chrome extension gateway (line-chrome-gw.line-apps.com).

install
source · Clone the upstream repo
git clone https://github.com/2manslkh/line-api
Claude Code · Install into ~/.claude/skills/
git clone --depth=1 https://github.com/2manslkh/line-api ~/.claude/skills/2manslkh-line-api-line-client
manifest: SKILL.md
source content

LINE Client Skill

Full LINE messaging client via the Chrome extension gateway JSON API.

Repo & Files

  • Repo:
    /data/workspace/line-client
    (github.com/2manslkh/line-api)
  • Main client:
    src/chrome_client.py
    LineChromeClient
  • QR login:
    src/auth/qr_login.py
    QRLogin
  • HMAC signer:
    src/hmac/signer.js
    (Node.js, auto-starts on port 18944)
  • Token storage:
    ~/.line-client/tokens.json
  • Certificate cache:
    ~/.line-client/sqr_cert
  • WASM files:
    lstm.wasm
    +
    lstmSandbox.js
    (required, in repo root)

Quick Start

import json
from pathlib import Path
from src.chrome_client import LineChromeClient

tokens = json.loads((Path.home() / ".line-client" / "tokens.json").read_text())
client = LineChromeClient(auth_token=tokens["auth_token"])

# Check if chat has E2EE before sending
if client.is_e2ee_chat("C..."):
    print("⚠️  E2EE chat — send_message won't work, see E2EE section below")
else:
    client.send_message("C...", "Hello!")

# Send a message (non-E2EE chats only)
client.send_message("U...", "Hello!")

# Get profile
profile = client.get_profile()

Tokens expire in ~7 days. If expired (

APIError(10051)
), re-run QR login.

⚠️ E2EE Limitation

Groups with E2EE v2 enabled cannot receive messages via

send_message()
.

  • Error: code 82 "types mismatch" when sending to E2EE chats
  • Messages in E2EE chats appear with
    chunks
    field (encrypted) instead of plaintext
    text
  • Use
    client.is_e2ee_chat(chat_id)
    to check before sending
  • E2EE encryption module exists in
    src/e2ee/crypto.py
    but is NOT wired into
    send_message
    yet
  • Reading messages from E2EE chats works (returns encrypted chunks)
  • Sending requires encrypting with each member's Curve25519 public key — not yet implemented

Workaround: For E2EE groups, use the LINE Official Account API or have the user send manually.

QR Login (Authentication)

QR login requires user interaction: scan QR on phone + enter PIN.

from src.hmac import HmacSigner
from src.auth.qr_login import QRLogin
import qrcode

signer = HmacSigner(mode="server")
login = QRLogin(signer)
result = login.run(
    on_qr=lambda url: send_qr_image_to_user(qrcode.make(url)),
    on_pin=lambda pin: send_pin_to_user_IMMEDIATELY(pin),  # TIME SENSITIVE!
    on_status=lambda msg: print(msg),
)
# result.auth_token, result.mid, result.refresh_token

Critical: The PIN must reach the user within ~60 seconds. Send it the instant

on_pin
fires.

QR Login State Machine

  1. createSession
    → session ID
  2. createQrCode
    → callback URL (append
    ?secret={curve25519_pubkey}&e2eeVersion=1
    )
  3. checkQrCodeVerified
    — poll until scan (uses
    X-Line-Session-ID
    , no
    origin
    header)
  4. verifyCertificate
    — MUST be called even if it fails (required state transition!)
  5. createPinCode
    → 6-digit PIN (skip if cert verified in step 4)
  6. checkPinCodeVerified
    — poll until user enters PIN
  7. qrCodeLoginV2
    → JWT token + certificate + refresh token

Server-Side Login Script

python scripts/qr_login_server.py /tmp/qr.png

Emits JSON events on stdout:

{"event": "qr", "path": "...", "url": "..."}
,
{"event": "pin", "pin": "123456"}
,
{"event": "done", "mid": "U..."}
.

Standalone Login (Recommended for Agents)

Best for automation — writes QR/PIN/status to files for fast pickup:

# 1. Start HMAC signer
cd /path/to/line-client && node src/hmac/signer.js serve &

# 2. Run login (background)
python3 scripts/qr_login_standalone.py &

# 3. Watch for QR (send to user via messaging)
# File: /data/workspace/line_qr.png

# 4. Watch for PIN (relay to user IMMEDIATELY — 60s window!)
# File: /data/workspace/line_pin.txt

# 5. Check completion
# File: /data/workspace/line_done.txt → "OK:<mid>" or "FAILED"

Agent orchestration pattern:

# Start login in background
nohup python3 scripts/qr_login_standalone.py > /tmp/line_login.log 2>&1 &

# Poll for QR ready
while [ ! -f /data/workspace/line_status.txt ] || ! grep -q QR_READY /data/workspace/line_status.txt; do
  sleep 0.5
done
# → Send /data/workspace/line_qr.png to user

# Poll for PIN (fast — check every 0.2s)
while [ ! -f /data/workspace/line_pin.txt ]; do
  sleep 0.2
done
PIN=$(cat /data/workspace/line_pin.txt)
# → Send PIN to user IMMEDIATELY

# Poll for completion
while [ ! -f /data/workspace/line_done.txt ]; do
  sleep 1
done
# → Check result

Critical learnings:

  • Always clear
    ~/.line-client/sqr_cert
    before login to force PIN prompt
  • PIN window is ~60 seconds — relay speed is everything
  • The standalone script clears cert automatically
  • If running from a subagent, the subagent tool-call latency can eat the PIN window
  • Best approach: poll PIN file from main thread with minimal sleep interval

Dependencies

# Debian/Ubuntu (no pip needed)
apt-get install -y python3-requests python3-qrcode python3-pil python3-nacl python3-cryptography python3-httpx python3-rsa python3-pycryptodome

# Also need Crypto alias if pycryptodome installs as Cryptodome
ln -sf /usr/lib/python3/dist-packages/Cryptodome /usr/lib/python3/dist-packages/Crypto

All API Methods

Contacts & Friends

MethodArgsDescription
get_profile()
Get your own profile (displayName, mid, statusMessage, etc.)
get_contact(mid)
mid: strGet a single contact's profile
get_contacts(mids)
mids: list[str]Get multiple contacts
get_all_contact_ids()
List all friend MIDs
find_contact_by_userid(userid)
userid: strSearch by LINE ID
find_and_add_contact_by_mid(mid)
mid: strAdd friend by MID
find_contacts_by_phone(phones)
phones: list[str]Search by phone numbers
add_friend_by_mid(mid)
mid: strAdd friend (RelationService)
get_blocked_contact_ids()
List blocked MIDs
get_blocked_recommendation_ids()
List blocked recommendations
block_contact(mid)
mid: strBlock a contact
unblock_contact(mid)
mid: strUnblock a contact
block_recommendation(mid)
mid: strBlock a friend suggestion
update_contact_setting(mid, flag, value)
mid, flag: int, value: strUpdate contact setting (e.g. mute)
get_favorite_mids()
List favorited contact MIDs
get_recommendation_ids()
List friend suggestions

Messages

MethodArgsDescription
send_message(to, text, ...)
to: str, text: str, reply_to: str (opt)Send a text message. Supports replies via
reply_to=message_id
unsend_message(message_id)
message_id: strUnsend/delete a sent message
get_recent_messages(chat_id, count=50)
chat_id: strGet latest messages in a chat
get_previous_messages(chat_id, end_seq, count=50)
chat_id, end_seq: intPaginated history (older messages)
get_messages_by_ids(message_ids)
message_ids: list[str]Fetch specific messages
get_message_boxes(count=50)
Get chat list with last message (inbox view)
get_message_boxes_by_ids(chat_ids)
chat_ids: list[str]Get specific chats with last message
get_message_read_range(chat_ids)
chat_ids: list[str]Get read receipt info
send_chat_checked(chat_id, last_message_id)
chat_id, last_message_id: strMark messages as read
send_chat_removed(chat_id, last_message_id)
chat_id, last_message_id: strRemove chat from inbox
send_postback(to, postback_data)
to, postback_data: strSend postback (bot interactions)

Chats & Groups

MethodArgsDescription
get_chats(chat_ids, with_members=True, with_invitees=True)
chat_ids: list[str]Get chat/group details
get_all_chat_mids()
List all chat MIDs (groups + invites)
create_chat(name, target_mids)
name: str, target_mids: list[str]Create a new group chat
accept_chat_invitation(chat_id)
chat_id: strAccept group invite
reject_chat_invitation(chat_id)
chat_id: strReject group invite
invite_into_chat(chat_id, mids)
chat_id: str, mids: list[str]Invite users to group
cancel_chat_invitation(chat_id, mids)
chat_id: str, mids: list[str]Cancel pending invites
delete_other_from_chat(chat_id, mids)
chat_id: str, mids: list[str]Kick members from group
leave_chat(chat_id)
chat_id: strLeave a group chat
update_chat(chat_id, updates)
chat_id: str, updates: dictUpdate group name/settings
set_chat_hidden_status(chat_id, hidden)
chat_id: str, hidden: boolArchive/unarchive a chat
get_rooms(room_ids)
room_ids: list[str]Get legacy room info
invite_into_room(room_id, mids)
room_id: str, mids: list[str]Invite to legacy room
leave_room(room_id)
room_id: strLeave legacy room

Reactions

MethodArgsDescription
react(message_id, reaction_type)
message_id: str, type: intReact to a message. Types: 2=like, 3=love, 4=laugh, 5=surprised, 6=sad, 7=angry
cancel_reaction(message_id)
message_id: strRemove your reaction

Profile & Settings

MethodArgsDescription
update_profile_attributes(attr, value, meta={})
attr: int, value: strUpdate profile. Attrs: 2=DISPLAY_NAME, 16=STATUS_MESSAGE, 4=PICTURE_STATUS
update_status_message(message)
message: strShortcut: update status message
update_display_name(name)
name: strShortcut: update display name
get_settings()
Get all account settings
get_settings_attributes(attr_bitset)
attr_bitset: intGet specific settings
update_settings_attributes(attr_bitset, settings)
attr_bitset: int, settings: dictUpdate settings

Polling & Events

MethodArgsDescription
get_last_op_revision()
Get latest operation revision number
fetch_ops(count=50)
Fetch pending operations (may long-poll)
poll()
Generator yielding operations as they arrive
on_message(handler)
handler: Callable(msg, client)Start polling thread, calls handler on new messages. Op types: 26=SEND_MESSAGE, 27=RECEIVE_MESSAGE
stop()
Stop the polling thread

Other Services

MethodArgsDescription
get_server_time()
Get LINE server timestamp
get_configurations()
Get server configurations
get_rsa_key_info()
Get RSA key for auth
issue_channel_token(channel_id)
channel_id: strIssue channel token (LINE Login/LIFF)
get_buddy_detail(mid)
mid: strGet official account info
report_abuse(mid, category=0, reason="")
mid: strReport a user
add_friend_by_mid(mid)
mid: strAdd friend (RelationService)
logout()
Logout and invalidate token

MID Format

LINE identifies entities by MID:

  • U...
    or
    u...
    → User (toType=0)
  • C...
    or
    c...
    → Group chat (toType=2)
  • R...
    or
    r...
    → Room (toType=1)

The client auto-detects

toType
from the MID prefix when sending messages.

HMAC Signing

All API calls require

X-Hmac
header. The WASM signer handles this automatically:

  • Derives key from version "3.7.1" + access token via proprietary KDF (in lstm.wasm)
  • Signs
    path + body
    → base64 →
    X-Hmac
  • Server mode: ~13ms/sign (Node.js HTTP server on port 18944, auto-started)
  • Subprocess mode: ~2s/sign (fallback)

Error Handling

from src.chrome_client import APIError

try:
    client.send_message(mid, "test")
except APIError as e:
    print(e.code, e.api_message)
    # 10051 = session expired / invalid
    # 10052 = HTTP error from backend
    # 10102 = invalid arguments

Architecture

User's Phone (LINE app)
    ↕ (scan QR / enter PIN)
LINE Servers (line-chrome-gw.line-apps.com)
    ↕ (JSON REST + X-Hmac signing)
LineChromeClient (this repo)
    ↕ (WASM HMAC via Node.js signer)
lstm.wasm + lstmSandbox.js

The Chrome Gateway translates JSON ↔ Thrift internally. We never deal with Thrift binary — everything is clean JSON.