深入解析数美滑块验证码破解:从抓包到实现(Java版)

数美滑块验证码是一种常见的安全验证方式,用于防止自动化脚本和机器人进行恶意操作。本文将详细介绍如何通过抓包获取滑块图片信息、分析加密参数、生成滑动轨迹等步骤,成功破解数美滑块验证码。以下是我们要实现的主要步骤:

抓取滑块图片信息并计算滑动距离
分析加密参数
生成滑动轨迹并进行DES加密
验证破解结果
1. 抓取滑块图片信息并计算滑动距离
首先,进入数美滑块验证码的演示页面,进行抓包操作以获取滑块验证码的前景和背景图片。我们通过HttpClient从网站获取图片:

java

import java.io.InputStream;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.Point;
import org.opencv.core.Size;
import org.opencv.imgcodecs.Imgcodecs;
import org.opencv.imgproc.Imgproc;
import org.opencv.core.Core;

public class CaptchaBreaker {

    static {
        System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
    }

    public static void main(String[] args) throws Exception {
        String registerUrl = "https://www.ishumei.com/trial/captcha.html";
        var images = getImages(registerUrl);
        Mat fg = images[0];
        Mat bg = images[1];
        int distance = getDistance(fg, bg);

        System.out.println("Calculated distance: " + distance);
    }

    public static Mat[] getImages(String registerUrl) throws Exception {
        HttpClient client = HttpClient.newHttpClient();
        HttpRequest request = HttpRequest.newBuilder()
            .uri(URI.create(registerUrl))
            .POST(HttpRequest.BodyPublishers.ofString("{}"))
            .build();

        HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());
        String responseBody = response.body();

        // 假设返回的JSON包含前景和背景图片URL
        String fgUrl = parseImageUrl(responseBody, "fg");
        String bgUrl = parseImageUrl(responseBody, "bg");

        Mat fgImage = downloadImage(fgUrl);
        Mat bgImage = downloadImage(bgUrl);

        return new Mat[] { fgImage, bgImage };
    }

    public static String parseImageUrl(String json, String key) {
        // 实现简单的JSON解析来提取图片URL   更多内容联系1436423940
        int index = json.indexOf(key);
        int start = json.indexOf(":", index) + 2;
        int end = json.indexOf("\"", start);
        return json.substring(start, end);
    }

    public static Mat downloadImage(String imageUrl) throws Exception {
        HttpClient client = HttpClient.newHttpClient();
        HttpRequest request = HttpRequest.newBuilder()
            .uri(URI.create(imageUrl))
            .build();

        HttpResponse<InputStream> response = client.send(request, HttpResponse.BodyHandlers.ofInputStream());
        return Imgcodecs.imdecode(new Mat(response.body().readAllBytes(), CvType.CV_8UC1), Imgcodecs.IMREAD_COLOR);
    }

    public static int getDistance(Mat fg, Mat bg) {
        Mat result = new Mat();
        Imgproc.matchTemplate(fg, bg, result, Imgproc.TM_CCOEFF_NORMED);
        Core.MinMaxLocResult mmr = Core.minMaxLoc(result);
        Point matchLoc = mmr.maxLoc;
        return (int) matchLoc.x;
    }
}
2. 分析加密参数
在滑动滑块的过程中,我们需要分析加密参数。抓包查看请求信息,可以看到有一个fverify的验证信息。如果正确的话,riskLevel返回值为PASS,失败为REJECT。

通过在浏览器中设置断点,我们可以找到参数aw等的生成函数,并确定这些参数的加密方式。使用断点调试后,发现是通过DES加密。

以下是通过javax.crypto库实现DES加密的示例代码:

java

import javax.crypto.Cipher;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESKeySpec;
import java.util.Base64;

public class DESEncryption {

    public static void main(String[] args) throws Exception {
        String message = "Hello, World!";
        String key = "your_key_here";
        boolean flag = true;

        String encryptedMessage = encryptContent(message, key, flag);
        System.out.println("Encrypted message: " + encryptedMessage);
    }

    public static String pad(String data) {
        int blockSize = 8;
        int paddingSize = blockSize - (data.length() % blockSize);
        return data + "\0".repeat(paddingSize);
    }

    public static String encryptContent(String message, String key, boolean flag) throws Exception {
        DESKeySpec desKeySpec = new DESKeySpec(key.getBytes());
        SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
        Cipher cipher = Cipher.getInstance("DES/ECB/NoPadding");
        cipher.init(flag ? Cipher.ENCRYPT_MODE : Cipher.DECRYPT_MODE, keyFactory.generateSecret(desKeySpec));

        byte[] paddedMessage = pad(message.replace(" ", "")).getBytes();
        byte[] result = cipher.doFinal(paddedMessage);
        return Base64.getEncoder().encodeToString(result);
    }
}
3. 生成滑动轨迹并进行DES加密
滑动轨迹可以通过模拟人的滑动行为来生成,包括一些随机的抖动和速度变化。以下是生成滑动轨迹的示例代码:

java

import java.util.ArrayList;
import java.util.List;
import java.util.Random;

public class SlideTrackGenerator {

    public static void main(String[] args) {
        int distance = 100; // 假设滑动距离为100
        List<int[]> track = getRandom(distance);

        for (int[] step : track) {
            System.out.println("x: " + step[0] + ", y: " + step[1] + ", t: " + step[2]);
        }
    }

    public static List<int[]> getRandom(int distance) {
        List<int[]> track = new ArrayList<>();
        track.add(new int[] { 0, 0, 0 });
        Random random = new Random();

        for (int i = 0; i < 10; i++) {
            int x = 0;
            int y = random.nextInt(3) - 1;
            int t = 100 * (i + 1) + random.nextInt(3);
            track.add(new int[] { x, y, t });
        }

        for (int i = 1; i < track.size() - 5; i++) {
            track.get(i)[0] = distance / 2;
        }

        for (int i = track.size() - 5; i < track.size() - 1; i++) {
            track.get(i)[0] = distance + random.nextInt(4);
        }

        track.get(track.size() - 1)[0] = distance;
        return track;
    }
}
4. 验证破解结果
最后,结合前面的步骤,实现滑块破解。以下是完整的示例代码:

java

import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.util.Base64;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import com.google.gson.Gson;

public class CaptchaBreaker {
    static {
        System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
    }

    public static void main(String[] args) throws Exception {
        String registerUrl = "https://www.ishumei.com/trial/captcha.html";
        var images = getImages(registerUrl);
        Mat fg = images[0];
        Mat bg = images[1];
        int distance = getDistance(fg, bg);
        List<int[]> track = getRandom(distance);

        String nm = new Gson().toJson(track);
        String dy = String.valueOf(track.get(track.size() - 1)[2]);

        String key = "your_key_here";
        boolean flag = true;

        String encryptedNm = encryptContent(nm, key, flag);
        String encryptedDy = encryptContent(dy, key, flag);

        Map<String, String> verifyData = new HashMap<>();
        verifyData.put("nm", encryptedNm);
        verifyData.put("dy", encryptedDy);
        verifyData.put("rid", "register_id_here");
        verifyData.put("dl", String.valueOf(distance));

        Gson gson = new Gson();
        String json = gson.toJson(verifyData);

        HttpRequest request = HttpRequest.newBuilder()
                .uri(URI.create("https://www.ishumei.com/trial/fverify"))
                .POST(HttpRequest.BodyPublishers.ofString(json))
                .header("Content-Type", "application/json")
                .build();

        HttpClient client = HttpClient.newHttpClient();
        HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());
        Map<String, Object> result = gson.fromJson(response.body(), Map.class);

        if ("PASS".equals(result.get("riskLevel"))) {
            System.out.println("验证成功");
        } else {
            System.out.println("验证失败");
        }
    }
}

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
验证码是一种常见的人机交互验证方式,主要用于防止恶意攻击和机器人攻击。下面是一个简单的Java实现验证码的示例代码: 1. 首先,需要在前端页面上实现一个块组件,并在后台生成一个随机的验证码图片。 2. 然后,用户需要按住块并将其拖动到正确的位置,以验证自己是一个真正的人类用户。 3. 在后台,需要验证用户拖动块的位置是否正确,以确保用户通过了验证。 下面是一个基于Spring Boot框架的简单示例代码: 1. 在前端页面中添加如下代码: ```html <div class="slider-container"> <div class="slider-background"></div> <div class="slider-handle"></div> </div> ``` 2. 在后台代码中,需要生成一个随机的验证码图片,并将验证码信息保存在Session中,以便后续验证。以下是一个简单的验证码生成器示例代码: ```java import java.awt.*; import java.awt.image.BufferedImage; import java.util.Random; public class CaptchaGenerator { private static final int IMAGE_WIDTH = 200; private static final int IMAGE_HEIGHT = 80; private static final int LINE_COUNT = 20; private static final int CHAR_COUNT = 4; private static final String CHAR_SET = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"; private static final int CHAR_SPACE = 20; private static final int CHAR_FONT_SIZE = 50; public static BufferedImage generate(String captcha) { BufferedImage image = new BufferedImage(IMAGE_WIDTH, IMAGE_HEIGHT, BufferedImage.TYPE_INT_RGB); Graphics2D g = image.createGraphics(); g.setColor(Color.WHITE); g.fillRect(0, 0, IMAGE_WIDTH, IMAGE_HEIGHT); g.setColor(Color.BLACK); Random random = new Random(); for (int i = 0; i < LINE_COUNT; i++) { int x1 = random.nextInt(IMAGE_WIDTH); int y1 = random.nextInt(IMAGE_HEIGHT); int x2 = random.nextInt(IMAGE_WIDTH); int y2 = random.nextInt(IMAGE_HEIGHT); g.drawLine(x1, y1, x2, y2); } Font font = new Font("Arial", Font.BOLD, CHAR_FONT_SIZE); g.setFont(font); int x = (IMAGE_WIDTH - CHAR_COUNT * CHAR_FONT_SIZE - (CHAR_COUNT - 1) * CHAR_SPACE) / 2; int y = (IMAGE_HEIGHT - CHAR_FONT_SIZE) / 2 + CHAR_FONT_SIZE; for (int i = 0; i < captcha.length(); i++) { char c = captcha.charAt(i); g.drawString(String.valueOf(c), x, y); x += CHAR_FONT_SIZE + CHAR_SPACE; } g.dispose(); return image; } public static String generateCaptcha() { Random random = new Random(); StringBuilder sb = new StringBuilder(); for (int i = 0; i < CHAR_COUNT; i++) { char c = CHAR_SET.charAt(random.nextInt(CHAR_SET.length())); sb.append(c); } return sb.toString(); } } ``` 3. 在Controller中,需要处理验证请求,并进行验证验证。以下是一个简单的Controller示例代码: ```java import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.*; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import java.awt.image.BufferedImage; import java.io.IOException; @Controller public class CaptchaController { @GetMapping("/captcha") public void captcha(HttpServletRequest request, HttpServletResponse response) throws IOException { String captcha = CaptchaGenerator.generateCaptcha(); BufferedImage image = CaptchaGenerator.generate(captcha); HttpSession session = request.getSession(); session.setAttribute("captcha", captcha); response.setContentType("image/png"); response.getOutputStream().write(ImageUtil.toByteArray(image)); } @PostMapping("/captcha/verify") @ResponseBody public boolean verify(@RequestParam String captcha, HttpSession session) { String expectedCaptcha = (String) session.getAttribute("captcha"); return captcha.equals(expectedCaptcha); } } ``` 以上是一个简单的Java实现验证码的示例代码。为了实现更好的安全性,实际应用中需要进一步加强验证机制,例如添加时间限制、IP限制等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值