Solução de Problemas

Lidando com reCAPTCHA v2 e v3 na mesma página

Alguns sites implementam reCAPTCHA v2 e v3 na mesma página. O padrão típico é: a v3 é executada de forma invisível em segundo plano e, se a pontuação for muito baixa, a v2 aparece como um desafio alternativo visível. Essa implementação dupla cria confusão para a automação porque você precisa lidar com dois tipos diferentes de CAPTCHA com métodos de resolução diferentes. Este guia cobre detecção, estratégias de resolução e casos extremos comuns.


Por que os sites usam v2 e v3 juntas

User visits page
    ↓
reCAPTCHA v3 runs invisibly in background
    ↓
Score returned to server (e.g., 0.4)
    ↓
Score below threshold (e.g., < 0.7)?
    ├─ YES → Show reCAPTCHA v2 checkbox/image challenge
    └─ NO  → Allow action without visible CAPTCHA

Este padrão oferece o mais adequado dos dois mundos:

  • A maioria dos usuários (pontuação v3 alta) não vê CAPTCHA → baixo atrito
  • Usuários suspeitos (pontuação v3 baixa) veja o desafio v2 → substituto de segurança
  • Operador de site controla o limite entre invisível e visível

Padrões de implementação dupla

Padrão 1: pré-avaliação v3 + substituto v2

O padrão mais comum. v3 é executado primeiro e v2 aparece somente se necessário.

<!-- Both scripts loaded -->
<script src="https://www.google.com/recaptcha/api.js?render=V3_SITE_KEY"></script>
<script src="https://www.google.com/recaptcha/api.js?render=explicit" async defer></script>

<form id="loginForm">
    <!-- v2 widget (hidden initially) -->
    <div id="recaptcha-v2-container" style="display:none;">
        <div class="g-recaptcha" data-sitekey="V2_SITE_KEY"></div>
    </div>
    <button type="submit">Login</button>
</form>

<script>
// First attempt: v3 invisible
grecaptcha.ready(function() {
    grecaptcha.execute('V3_SITE_KEY', {action: 'login'}).then(function(v3Token) {
        fetch('/api/verify-v3', {
            method: 'POST',
            body: JSON.stringify({token: v3Token})
        })
        .then(r => r.json())
        .then(data => {
            if (data.score < 0.7) {
                // Score too low → show v2 fallback
                document.getElementById('recaptcha-v2-container').style.display = 'block';
                grecaptcha.render('recaptcha-v2-container', {sitekey: 'V2_SITE_KEY'});
            } else {
                // Score OK → submit form directly
                document.getElementById('loginForm').submit();
            }
        });
    });
});
</script>

Padrão 2: chaves de site diferentes para ações diferentes

Alguns sites usam v3 para monitoramento passivo e v2 para ações específicas de alto risco:

Homepage → v3 only (passive score)
Login page → v3 assessment, v2 fallback
Checkout → v2 always (high security)
Contact form → v3 only

Padrão 3: script único, modo duplo

O Google suporta o carregamento de um único script reCAPTCHA que lida com v2 e v3:

<script src="https://www.google.com/recaptcha/api.js?render=V3_SITE_KEY"></script>
<script>
    // v3 execute
    grecaptcha.execute('V3_SITE_KEY', {action: 'login'});

    // v2 render (uses a different site key)
    grecaptcha.render('v2-container', {sitekey: 'V2_SITE_KEY'});
</script>

Detectando implementação dupla

Detecção de Python

import requests
import re

def detect_dual_recaptcha(url):
    """Detect if a page uses both reCAPTCHA v2 and v3."""
    headers = {
        "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) "
                      "AppleWebKit/537.36 (KHTML, like Gecko) "
                      "Chrome/120.0.0.0 Safari/537.36",
    }
    html = requests.get(url, headers=headers, timeout=15).text

    result = {
        "has_v3": False,
        "has_v2": False,
        "v3_site_key": None,
        "v2_site_key": None,
        "dual": False,
        "pattern": None,
    }

    # Detect v3 (render parameter or enterprise.execute)
    v3_match = re.search(r"api\.js\?render=([A-Za-z0-9_-]+)", html)
    if v3_match and v3_match.group(1) != "explicit":
        result["has_v3"] = True
        result["v3_site_key"] = v3_match.group(1)

    # Detect v3 in execute calls
    v3_execute = re.search(
        r"grecaptcha\.(?:enterprise\.)?execute\s*\(\s*['\"]([^'\"]+)['\"]",
        html,
    )
    if v3_execute:
        result["has_v3"] = True
        if not result["v3_site_key"]:
            result["v3_site_key"] = v3_execute.group(1)

    # Detect v2 (g-recaptcha class or explicit render)
    v2_match = re.search(r'data-sitekey="([^"]+)"', html)
    if v2_match:
        key = v2_match.group(1)
        if key != result.get("v3_site_key"):
            result["has_v2"] = True
            result["v2_site_key"] = key

    # Check for explicit v2 render
    v2_render = re.search(
        r"grecaptcha\.render\s*\([^,]+,\s*\{[^}]*sitekey:\s*['\"]([^'\"]+)",
        html,
    )
    if v2_render:
        result["has_v2"] = True
        if not result["v2_site_key"]:
            result["v2_site_key"] = v2_render.group(1)

    result["dual"] = result["has_v3"] and result["has_v2"]

    if result["dual"]:
        # Determine pattern
        if "display:none" in html or "display: none" in html:
            result["pattern"] = "v3_pre_assessment_v2_fallback"
        else:
            result["pattern"] = "v2_v3_simultaneous"

    return result

detection = detect_dual_recaptcha("https://staging.example.com/qa-login")
print(detection)

Detecção de Node.js

const axios = require("axios");

async function detectDualRecaptcha(url) {
    const { data: html } = await axios.get(url, { timeout: 15000 });

    const result = {
        hasV3: false,
        hasV2: false,
        v3SiteKey: null,
        v2SiteKey: null,
        dual: false,
    };

    // v3 detection
    const v3Match = html.match(/api\.js\?render=([A-Za-z0-9_-]+)/);
    if (v3Match && v3Match[1] !== "explicit") {
        result.hasV3 = true;
        result.v3SiteKey = v3Match[1];
    }

    // v2 detection
    const v2Match = html.match(/data-sitekey="([^"]+)"/);
    if (v2Match && v2Match[1] !== result.v3SiteKey) {
        result.hasV2 = true;
        result.v2SiteKey = v2Match[1];
    }

    result.dual = result.hasV3 && result.hasV2;
    return result;
}

detectDualRecaptcha("https://staging.example.com/qa-login").then(console.log);

Resolvendo estratégias para reCAPTCHA duplo

Estratégia 1: Resolva primeiro a v3 e depois a v2 se necessário

A estratégia ideal reflete o fluxo do próprio site:

import requests
import time

API_KEY = "YOUR_API_KEY"

def solve_v3(site_key, page_url, action="login"):
    """Solve reCAPTCHA v3 and return token."""
    submit = requests.post("https://ocr.captchaai.com/in.php", data={
        "key": API_KEY,
        "method": "userrecaptcha",
        "googlekey": site_key,
        "pageurl": page_url,
        "version": "v3",
        "action": action,
        "score_qa": "0.9",
        "json": 1,
    }).json()

    task_id = submit["request"]

    for _ in range(60):
        time.sleep(5)
        result = requests.get("https://ocr.captchaai.com/res.php", params={
            "key": API_KEY, "action": "get", "id": task_id, "json": 1,
        }).json()
        if result.get("status") == 1:
            return result["request"]

    raise TimeoutError("v3 solve timeout")


def solve_v2(site_key, page_url):
    """Solve reCAPTCHA v2 and return token."""
    submit = requests.post("https://ocr.captchaai.com/in.php", data={
        "key": API_KEY,
        "method": "userrecaptcha",
        "googlekey": site_key,
        "pageurl": page_url,
        "json": 1,
    }).json()

    task_id = submit["request"]

    for _ in range(60):
        time.sleep(5)
        result = requests.get("https://ocr.captchaai.com/res.php", params={
            "key": API_KEY, "action": "get", "id": task_id, "json": 1,
        }).json()
        if result.get("status") == 1:
            return result["request"]

    raise TimeoutError("v2 solve timeout")


def solve_dual_recaptcha(v3_key, v2_key, page_url, action="login"):
    """Handle dual reCAPTCHA: try v3, fall back to v2."""
    # Step 1: Try v3
    v3_token = solve_v3(v3_key, page_url, action)

    # Step 2: Submit v3 token to target
    response = requests.post(f"{page_url}/verify", data={
        "g-recaptcha-response": v3_token,
    })

    # Step 3: Check if v2 fallback is needed
    if "recaptcha" in response.text.lower() and v2_key:
        print("v3 score too low — v2 fallback triggered")
        v2_token = solve_v2(v2_key, page_url)
        return {"version": "v2", "token": v2_token}

    return {"version": "v3", "token": v3_token}


result = solve_dual_recaptcha(
    v3_key="6LcExample_v3_key",
    v2_key="6LcExample_v2_key",
    page_url="https://staging.example.com/qa-login",
)
print(f"Solved with {result['version']}")

Estratégia 2: pule a v3, resolva a v2 diretamente

Se você sabe que o site sempre mostra v2 para tráfego automatizado (a pontuação da v3 será baixa), pule a v3 e resolva a v2 imediatamente:

# If you consistently fail v3 assessment, just solve v2 directly
token = solve_v2(v2_site_key, page_url)
submit_form(token)

Isso economiza tempo e custo de uma solução v3 que pode não ultrapassar o limite.

Estratégia 3: Tratamento baseado em navegador

Para implementações complexas, use um navegador para lidar com o fluxo de fallback:

from selenium import webdriver
from selenium.webdriver.common.by import By
import time

driver = webdriver.Chrome()
driver.get("https://staging.example.com/qa-login")
time.sleep(3)

# Check if v2 widget is visible
v2_visible = driver.execute_script("""
    const container = document.querySelector('.g-recaptcha');
    if (!container) return false;
    const style = window.getComputedStyle(container.parentElement);
    return style.display !== 'none' && style.visibility !== 'hidden';
""")

if v2_visible:
    # v2 is showing — solve v2
    sitekey = driver.find_element(
        By.CSS_SELECTOR, "[data-sitekey]"
    ).get_attribute("data-sitekey")
    token = solve_v2(sitekey, driver.current_url)
    driver.execute_script(
        f'document.getElementById("g-recaptcha-response").value = "{token}";'
    )
else:
    # v3 only — solve v3
    # Extract v3 key from page source
    v3_key = driver.execute_script(
        "return document.querySelector('script[src*=\"render=\"]')"
        ".src.match(/render=([^&]+)/)[1];"
    )
    token = solve_v3(v3_key, driver.current_url)
    # Inject v3 token into the form
    driver.execute_script(f"""
        const input = document.createElement('input');
        input.type = 'hidden';
        input.name = 'g-recaptcha-response';
        input.value = '{token}';
        document.querySelector('form').appendChild(input);
    """)

driver.find_element(By.CSS_SELECTOR, "form").submit()

Casos extremos

Duas chaves de site diferentes na mesma página

Sites que usam reCAPTCHA duplo terão DUAS chaves de site diferentes – uma para v3 e outra para v2. A chave v3 aparece na URL do script ?render=KEY e em grecaptcha.execute('KEY', ...). A chave v2 aparece em data-sitekey="KEY" na div do widget. Usar a chave errada para a versão errada produzirá tokens inválidos.

reCAPTCHA Enterprise com substituto v2

Algumas implementações Enterprise usam v3 Enterprise para pontuação e v2 para desafios:

# Detect and handle Enterprise + v2 combo
if "recaptcha/enterprise.js" in html:
    # Use enterprise parameter for v3
    v3_params = {"enterprise": 1, "version": "v3"}
else:
    v3_params = {"version": "v3"}

Vários formulários em uma página

Se uma página tiver vários formulários (login + registro), cada um poderá ter sua própria instância do reCAPTCHA. Extraia a chave do site do formulário específico que você está direcionando:

# Target the login form specifically
login_form = soup.find("form", id="login-form")
widget = login_form.find(attrs={"data-sitekey": True})
sitekey = widget["data-sitekey"]

Perguntas frequentes

Preciso resolver v2 e v3 na mesma página?

Não. Normalmente, você resolve a v3 primeiro (ela é executada automaticamente). Se a pontuação da v3 ultrapassar o limite do site, nenhum desafio da v2 será exibido e pronto. Você só precisa resolver a v2 se a pontuação da v3 acionar o substituto.

Posso usar uma única chamada de API CaptchaAI para reCAPTCHA duplo?

Não. v2 e v3 são tipos separados de CAPTCHA com chaves de site e métodos de resolução diferentes. Cada um requer sua própria chamada de API para CaptchaAI. No entanto, você só precisará fazer uma chamada se a v3 passar sem acionar a v2.

Como posso saber se o fallback v2 foi acionado?

Verifique a resposta do servidor após enviar o token v3. Se a resposta contiver HTML de widget v2 ou acionar um desafio v2 (redirecionamento ou resposta AJAX com HTML CAPTCHA), o substituto foi acionado. Em um navegador, verifique se o contêiner v2 fica visível após o envio da v3.

Qual chave de site devo usar para cada versão?

A chave do site v3 está na URL do script: api.js?render=V3_KEY. A chave do site v2 está no widget HTML: data-sitekey="V2_KEY". São sempre chaves diferentes.


Resumo

As implementações duplas de reCAPTCHA usam v3 para pré-avaliação invisível e v2 como um substituto visível quando a pontuação da v3 é muito baixa. Detecte ambas as versões verificando o parâmetro de renderização (v3) e o widget data-sitekey (v2). A estratégia de automação ideal é: resolver primeiro a v3 comCaptchaAI, envie o token e resolva v2 somente se o substituto for acionado. Cada versão requer uma chamada de API separada com sua própria chave de site.

Artigos relacionados

  • Aplicativo Recaptcha de página única dinâmico
  • Como resolver o retorno de chamada do Recaptcha V2 usando API
  • Torniquete Recaptcha V2 no mesmo local
Os comentários estão desativados para este artigo.