비트베이크

Implementing SMS Authentication in T3 Stack (Next.js + tRPC) in 5 Minutes for Solo Devs (No Paperwork)

2026-05-30T01:02:20.475Z

Hooded figure with digital code for a face, representing digital security and development.

What is the most frustrating part of building a side project or an MVP? For many developers, it’s integrating SMS authentication. Most legacy SMS APIs require tedious paperwork, such as business registration and telecom certificates. For solo developers or startups trying to rapidly test an idea, this is a massive barrier to entry.

In this guide, we will learn how to implement SMS verification in just 5 minutes with zero paperwork using EasyAuth and the highly popular T3 Stack (Next.js + tRPC).

Why Choose EasyAuth?

  • Zero Paperwork: Sign up and use the API instantly without submitting any business documents.
  • Automatic Caller ID: No need to go through telecom verification to register a sender number.
  • Simple Architecture: Just two endpoints complete the flow: POST /send and POST /verify.
  • Cost-Effective: Only 15-25 KRW per message (compared to the standard 30-50 KRW), with 10 free credits provided upon signup.

Let’s dive into integrating EasyAuth into the T3 Stack.

1. Setting up the tRPC Router (Backend)

In the T3 Stack, we don't need to create standard API routes. Instead, we use tRPC mutations. First, create a new router file at src/server/api/routers/sms.ts.

import { z } from "zod";
import { createTRPCRouter, publicProcedure } from "~/server/api/trpc";

// EasyAuth API Settings
const EASYAUTH_API_URL = "https://api.easyauth.co.kr";
const API_KEY = process.env.EASYAUTH_API_KEY!;

export const smsRouter = createTRPCRouter({
  // 1. Send OTP
  sendOtp: publicProcedure
    .input(z.object({ phoneNumber: z.string().min(10) }))
    .mutation(async ({ input }) => {
      const response = await fetch(`${EASYAUTH_API_URL}/send`, {
        method: "POST",
        headers: {
          "Content-Type": "application/json",
          "Authorization": `Bearer ${API_KEY}`,
        },
        body: JSON.stringify({ to: input.phoneNumber }),
      });

      if (!response.ok) throw new Error("Failed to send SMS.");
      return { success: true, message: "OTP sent successfully." };
    }),

  // 2. Verify OTP
  verifyOtp: publicProcedure
    .input(z.object({ 
      phoneNumber: z.string(),
      code: z.string().length(6) 
    }))
    .mutation(async ({ input }) => {
      const response = await fetch(`${EASYAUTH_API_URL}/verify`, {
        method: "POST",
        headers: {
          "Content-Type": "application/json",
          "Authorization": `Bearer ${API_KEY}`,
        },
        body: JSON.stringify({ 
          to: input.phoneNumber,
          code: input.code 
        }),
      });

      if (!response.ok) throw new Error("Invalid OTP code.");
      return { success: true, message: "Verification successful." };
    }),
});

Make sure to merge this router into your root API router in src/server/api/root.ts.

2. Frontend Integration

Now, let's create a Client Component in Next.js to handle the UI using tRPC hooks. We'll use standard React state for simplicity.

import { useState } from "react";
import { api } from "~/utils/api";

export default function SmsAuth() {
  const [phoneNumber, setPhoneNumber] = useState("");
  const [code, setCode] = useState("");
  const [isSent, setIsSent] = useState(false);

  const sendOtpMutation = api.sms.sendOtp.useMutation({
    onSuccess: () => setIsSent(true),
    onError: (err) => alert(err.message),
  });

  const verifyOtpMutation = api.sms.verifyOtp.useMutation({
    onSuccess: () => alert("Verification Complete!"),
    onError: (err) => alert(err.message),
  });

  const handleSend = () => sendOtpMutation.mutate({ phoneNumber });
  const handleVerify = () => verifyOtpMutation.mutate({ phoneNumber, code });

  return (
    <div>
      <h2>Phone Verification</h2>
      
      <div>
         setPhoneNumber(e.target.value)}
          disabled={isSent}
          className="flex-1 border p-2 rounded"
        /&gt;
        
          {isSent ? "Resend" : "Send OTP"}
        
      </div>

      {isSent &amp;&amp; (
        <div>
           setCode(e.target.value)}
            className="flex-1 border p-2 rounded"
          /&gt;
          
            Verify
          
        </div>
      )}
    </div>
  );
}

Conclusion

Without any complex paperwork or telecom reviews, we've successfully implemented SMS authentication using just two API endpoints (/send and /verify) combined with the power of the T3 Stack. EasyAuth handles all the complex backend logic out of the box, like storing OTP codes and checking expirations.

If you are building a toy project, a freelance gig, or a startup MVP, give EasyAuth a try. You can start testing instantly with 10 free credits provided upon signup, and scale your application effortlessly with an incredibly low cost of 15-25 KRW per message!

비트베이크에서 광고를 시작해보세요

광고 문의하기

다른 글 보기

2026-06-16T05:01:55.625Z

2026 다이소 여름 신상/인기템! 시원한 여름 꿀템 총정리

2026년 다이소 여름 신상부터 인기 쿨링템, 장마철 필수품, 홈캉스 아이템까지! 가성비 넘치는 다이소 여름 꿀템으로 시원하고 쾌적한 여름을 준비하는 완벽 가이드.

2026-06-16T05:01:31.367Z

지속 가능한 국내 워케이션: 2026년 숨은 보석 여행지

2026년 국내 워케이션 트렌드는 지속가능한 여행과 만납니다. 디지털 디톡스, 친환경 숙소, 로컬 체험을 통해 몸과 마음을 치유하고 지역 경제 활성화에 기여하는 숨은 명소 3곳을 소개합니다. 지금 바로 나만의 지속 가능한 워케이션을 계획해보세요!

2026-06-16T05:01:30.087Z

2026년 최신 의학 트렌드: AI와 정밀의료로 여는 초개인화 건강관리

2026년, AI와 정밀의료가 이끄는 초개인화 건강관리 시대가 열렸습니다. 딥러닝 기반 진단, 유전체 맞춤 치료, 웨어러블 및 디지털 치료제가 일상 속 건강을 혁신합니다. 미래 의학의 도전 과제와 현명한 건강 관리법을 알아보세요.

2026-06-16T05:01:16.613Z

2026 가을/겨울 출산준비물: 신생아 육아템 필수템 총정리

2026년 가을/겨울 출산을 앞둔 예비맘들을 위한 완벽 가이드! 최신 트렌드를 반영한 신생아 육아템 필수템부터 대형 육아용품 비교, 스마트한 케어 및 수유 용품, 쌀쌀한 날씨 대비 아기옷, 그리고 알뜰 구매 팁까지 모든 출산준비물을 총정리했습니다.

서비스

피드자주 묻는 질문고객센터

문의

비트베이크

레임스튜디오 | 사업자 등록번호 : 542-40-01042

경기도 남양주시 와부읍 수례로 116번길 16, 4층 402-제이270호

트위터인스타그램네이버 블로그