Unified OTP flow - works for both login and signup

This commit is contained in:
John Pinkyfloyd 2026-02-15 17:13:09 -08:00
parent 4ae3b8b3d8
commit c198b68ee0
2 changed files with 48 additions and 54 deletions

View file

@ -5,15 +5,16 @@
<cfscript>
/**
* Send OTP to phone number for signup
* Unified OTP: Send OTP to any phone number (login or signup)
*
* POST: { "phone": "5551234567" }
*
* Returns: { OK: true, UUID: "..." } or { OK: false, ERROR: "..." }
* Returns: { OK: true, UUID: "..." }
*
* If phone already has verified account, returns error.
* If phone has unverified account, resends OTP.
* Otherwise creates new user record with OTP.
* Works for any phone number:
* - Existing complete account: updates OTP, user will be logged in after verify
* - Existing incomplete account: updates OTP, user continues to registration after verify
* - New phone: creates user record, user continues to registration after verify
*/
function apiAbort(required struct payload) {
@ -54,29 +55,11 @@ try {
apiAbort({ "OK": false, "ERROR": "invalid_phone", "MESSAGE": "Please enter a valid 10-digit phone number" });
}
// Check if phone already has a COMPLETE account (verified AND has profile info)
// An account is only "complete" if they have a first name (meaning they finished signup)
// Check if phone already has ANY account
qExisting = queryTimed("
SELECT ID, UUID, FirstName
SELECT ID, UUID, FirstName, IsContactVerified, IsActive
FROM Users
WHERE ContactNumber = :phone
AND IsContactVerified > 0
AND FirstName IS NOT NULL
AND LENGTH(TRIM(FirstName)) > 0
LIMIT 1
", { phone: { value: phone, cfsqltype: "cf_sql_varchar" } }, { datasource: "payfrit" });
if (qExisting.recordCount > 0) {
apiAbort({ "OK": false, "ERROR": "phone_exists", "MESSAGE": "This phone number already has an account. Please login instead." });
}
// Check for incomplete account with this phone (verified but no profile, OR unverified)
// These accounts can be reused for signup
qIncomplete = queryTimed("
SELECT ID, UUID
FROM Users
WHERE ContactNumber = :phone
AND (IsContactVerified = 0 OR FirstName IS NULL OR LENGTH(TRIM(FirstName)) = 0)
LIMIT 1
", { phone: { value: phone, cfsqltype: "cf_sql_varchar" } }, { datasource: "payfrit" });
@ -87,21 +70,24 @@ try {
}
userUUID = "";
if (qIncomplete.recordCount > 0) {
// Update existing incomplete record with new OTP and reset for re-registration
userUUID = qIncomplete.UUID;
if (qExisting.recordCount > 0) {
// Existing account - just update OTP
userUUID = qExisting.UUID;
if (!len(trim(userUUID))) {
userUUID = replace(createUUID(), "-", "", "all");
}
queryTimed("
UPDATE Users
SET MobileVerifyCode = :otp,
IsContactVerified = 0,
IsActive = 0
UUID = :uuid
WHERE ID = :userId
", {
otp: { value: otp, cfsqltype: "cf_sql_varchar" },
userId: { value: qIncomplete.ID, cfsqltype: "cf_sql_integer" }
uuid: { value: userUUID, cfsqltype: "cf_sql_varchar" },
userId: { value: qExisting.ID, cfsqltype: "cf_sql_integer" }
}, { datasource: "payfrit" });
} else {
// Create new user record
// New phone - create user record
userUUID = replace(createUUID(), "-", "", "all");
queryTimed("
INSERT INTO Users (
@ -142,9 +128,9 @@ try {
try {
smsResult = application.twilioObj.sendSMS(
recipientNumber: "+1" & phone,
messageBody: "Your Payfrit verification code is: " & otp
messageBody: "Your Payfrit code is: " & otp
);
smsMessage = smsResult.success ? "Verification code sent" : "SMS failed - please try again";
smsMessage = smsResult.success ? "Code sent" : "SMS failed - please try again";
} catch (any smsErr) {
smsMessage = "SMS error: " & smsErr.message;
}

View file

@ -5,14 +5,14 @@
<cfscript>
/**
* Verify OTP and activate user account
* Unified OTP Verify: Works for both login and signup
*
* POST: { "uuid": "...", "otp": "123456" }
*
* Returns: { OK: true, UserID: 123, Token: "...", NeedsProfile: true/false }
*
* On success, marks phone as verified and returns auth token.
* NeedsProfile indicates if user still needs to provide name/email.
* - If account is complete (has FirstName) → NeedsProfile: false, user is logged in
* - If account is incomplete → NeedsProfile: true, user continues to registration
*/
function apiAbort(required struct payload) {
@ -40,14 +40,12 @@ try {
apiAbort({ "OK": false, "ERROR": "missing_fields", "MESSAGE": "UUID and OTP are required" });
}
// Find unverified user with matching UUID and OTP
// Magic OTP only bypasses Twilio SMS (in sendOTP.cfm), not OTP verification
// Find user with matching UUID and OTP (any verification status)
qUser = queryTimed("
SELECT ID, FirstName, LastName, EmailAddress, IsEmailVerified
SELECT ID, FirstName, LastName, EmailAddress, IsContactVerified, IsEmailVerified, IsActive
FROM Users
WHERE UUID = :uuid
AND MobileVerifyCode = :otp
AND IsContactVerified = 0
LIMIT 1
", {
uuid: { value: userUUID, cfsqltype: "cf_sql_varchar" },
@ -57,7 +55,7 @@ try {
if (qUser.recordCount == 0) {
// Check if UUID exists but OTP is wrong
qCheck = queryTimed("
SELECT ID FROM Users WHERE UUID = :uuid AND IsContactVerified = 0
SELECT ID FROM Users WHERE UUID = :uuid
", { uuid: { value: userUUID, cfsqltype: "cf_sql_varchar" } }, { datasource: "payfrit" });
if (qCheck.recordCount > 0) {
@ -67,15 +65,29 @@ try {
}
}
// Clear the OTP code (one-time use) but DON'T mark as verified yet
// Account will be marked verified after profile completion
// Check if profile is complete (has first name)
needsProfile = !len(trim(qUser.FirstName));
// Clear the OTP code (one-time use)
// If profile is complete, mark as verified and active (login case)
// If profile incomplete, leave unverified until profile completion
if (needsProfile) {
queryTimed("
UPDATE Users
SET MobileVerifyCode = ''
WHERE ID = :userId
", { userId: { value: qUser.ID, cfsqltype: "cf_sql_integer" } }, { datasource: "payfrit" });
} else {
queryTimed("
UPDATE Users
SET MobileVerifyCode = '',
IsContactVerified = 1,
IsActive = 1
WHERE ID = :userId
", { userId: { value: qUser.ID, cfsqltype: "cf_sql_integer" } }, { datasource: "payfrit" });
}
// Create auth token (needed for completeProfile call)
// Create auth token
token = replace(createUUID(), "-", "", "all");
queryTimed("
INSERT INTO UserTokens (UserID, Token) VALUES (:userId, :token)
@ -84,10 +96,6 @@ try {
token: { value: token, cfsqltype: "cf_sql_varchar" }
}, { datasource: "payfrit" });
// Check if profile is complete (has first name)
// For new signups, this will always be true
needsProfile = !len(trim(qUser.FirstName));
try{logPerf(0);}catch(any e){}
writeOutput(serializeJSON({
"OK": true,