• 17、生成长图,并上传至服务器


    生成长图

    1、wkhtmltopdf介绍

    • wkhtmltopdf url file
    • wkhtmltoimage url file

    2、java中使用wkhtmltopdf(异步执行)

    Runtime.getRuntime().exec(cmd) 服务器把定义的cmd命令交给操作系统后,便继续向下执行了,即异步。

    package com.nowcoder.community;
    
    import java.io.IOException;
    
    public class WkTests {
    
        public static void main(String[] args) {
            String cmd = "d:/work/wkhtmltopdf/bin/wkhtmltoimage --quality 75  https://www.nowcoder.com d:/work/data/wk-images/3.png";
            try {
                Runtime.getRuntime().exec(cmd);
                System.out.println("ok.");
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    3、服务器端生成长图

    检查并创建目录

    @Configuration
    public class WkConfig {
    
        private static final Logger logger = LoggerFactory.getLogger(WkConfig.class);
    
        @Value("${wk.image.storage}")
        private String wkImageStorage;
    
        @PostConstruct
        public void init() {
            // 创建WK图片目录
            File file = new File(wkImageStorage);
            if (!file.exists()) {
                file.mkdir();
                logger.info("创建WK图片目录: " + wkImageStorage);
            }
        }
    
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    将指定 url 生成长图,用于分享

    @Controller
    public class ShareController implements CommunityConstant {
    
        private static final Logger logger = LoggerFactory.getLogger(ShareController.class);
    
        @Autowired
        private EventProducer eventProducer;
    
        @Value("${community.path.domain}")
        private String domain;
    
        @Value("${server.servlet.context-path}")
        private String contextPath;
    
        @Value("${wk.image.storage}")
        private String wkImageStorage;
    
        @RequestMapping(path = "/share", method = RequestMethod.GET)
        @ResponseBody
        public String share(String htmlUrl) {
            // 文件名
            String fileName = CommunityUtil.generateUUID();
    
            // 异步生成长图
            Event event = new Event()
                    .setTopic(TOPIC_SHARE)
                    .setData("htmlUrl", htmlUrl)
                    .setData("fileName", fileName)
                    .setData("suffix", ".png");
            eventProducer.fireEvent(event);
    
            // 返回访问路径
            Map<String, Object> map = new HashMap<>();
            map.put("shareUrl", domain + contextPath + "/share/image/" + fileName);  // [http://localhost:8080/community]/share/image/XXX.png
    
            return CommunityUtil.getJSONString(0, null, map);
        }
    
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40

    A、share 方法是用来生成长图的,并返回一个json字符串,如果成功,则code=0,分享url即是json中"shareUrl"的值。
    1)生成图片是异步的方式,使用kafka
    在 消费TOPIC_SHARE 方法里,使用 wkhtmltopdf 生成长图,并放在指定的本机文件夹中。

        // 消费分享事件A
        @KafkaListener(topics = TOPIC_SHARE)
        public void handleShareMessage(ConsumerRecord record) {
            if (record == null || record.value() == null) {
                logger.error("消息的内容为空!");
                return;
            }
    
            Event event = JSONObject.parseObject(record.value().toString(), Event.class);
            if (event == null) {
                logger.error("消息格式错误!");
                return;
            }
    
            String htmlUrl = (String) event.getData().get("htmlUrl");
            String fileName = (String) event.getData().get("fileName");
            String suffix = (String) event.getData().get("suffix");
    
            String cmd = wkImageCommand + " --quality 75 "
                    + htmlUrl + " " + wkImageStorage + "/" + fileName + suffix;
            try {
                Runtime.getRuntime().exec(cmd);
                logger.info("生成长图成功: " + cmd);
            } catch (IOException e) {
                logger.error("生成长图失败: " + e.getMessage());
            }
        }
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29

    B、用户访问 “shareUrl” 即可获取长图(即访问 [http://localhost:8080/community]/share/image/XXX.png 时,调用了 getShareImage 方法)

      // 获取长图
        @RequestMapping(path = "/share/image/{fileName}", method = RequestMethod.GET)
        public void getShareImage(@PathVariable("fileName") String fileName, HttpServletResponse response) {
            if (StringUtils.isBlank(fileName)) {
                throw new IllegalArgumentException("文件名不能为空!");
            }
    
            response.setContentType("image/png");  //
            File file = new File(wkImageStorage + "/" + fileName + ".png");  //本地 d:/JavaWork/data/wk-images/XXX.png
            try {
                OutputStream os = response.getOutputStream();  // 输出流,将文件写入 respone
                FileInputStream fis = new FileInputStream(file);  // 输入流,读取文件
                byte[] buffer = new byte[1024];
                int b = 0;
                while ((b = fis.read(buffer)) != -1) {
                    os.write(buffer, 0, b);
                }
            } catch (IOException e) {
                logger.error("获取长图失败: " + e.getMessage());
            }
        }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    上传至服务器

    客户端上传:

    • 客户端将数据提交给云服务器,并等待其响应
    • 本项目中,用户上传头像时,将表单数据提交给云服务器。
      服务器直传:
    • 应用服务器将数据直接提交给云服务器,并等待其响应
    • 本项目中,分享时,服务器将自动生成的图片,直接提交给云服务器。

    服务器直传

    ShareController.java:将访问路径shareUrl 修改为 云服务器上对应的存储路径

        @RequestMapping(path = "/share", method = RequestMethod.GET)
        @ResponseBody
        public String share(String htmlUrl) {
            // 文件名
            String fileName = CommunityUtil.generateUUID();
    
            // 异步生成长图
            Event event = new Event()
                    .setTopic(TOPIC_SHARE)
                    .setData("htmlUrl", htmlUrl)
                    .setData("fileName", fileName)
                    .setData("suffix", ".png");
            eventProducer.fireEvent(event);
    
            // 返回访问路径
            Map<String, Object> map = new HashMap<>();
    //        map.put("shareUrl", domain + contextPath + "/share/image/" + fileName);
            map.put("shareUrl", shareBucketUrl + "/" + fileName);
    
            return CommunityUtil.getJSONString(0, null, map);
        }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    废弃 getShareImage() 方法(该方法用来查看图片)

    由于 WK 生成长图是异步操作,所以使用定时器监视该图片,一旦生成了,则上传至七牛云。如果生成失败,或上传失败到一定次数,则中止该定时器。

    使用的类:Future、ThreadPoolTaskScheduler (?)
    这里没有使用Quartz 是因为, 即使在分布式情况下,每台服务器都部署了consumer,但消费者有个抢占机制,所以只会某个服务器上执行。
    (同一消费组下的各个消费者在消费消息是是互斥的,也即是说,同一条消息,只能被同一个消费组下的某个消费者消费,不能被其它组的消费者消费

        @Value("${wk.image.command}")
        private String wkImageCommand;
    
        @Value("${wk.image.storage}")
        private String wkImageStorage;
    
        @Value("${qiniu.key.access}")
        private String accessKey;
    
        @Value("${qiniu.key.secret}")
        private String secretKey;
    
        @Value("${qiniu.bucket.share.name}")
        private String shareBucketName;
    
        @Autowired
        private ThreadPoolTaskScheduler taskScheduler;
    
        // 消费分享事件
        @KafkaListener(topics = TOPIC_SHARE)
        public void handleShareMessage(ConsumerRecord record) {
            if (record == null || record.value() == null) {
                logger.error("消息的内容为空!");
                return;
            }
    
            Event event = JSONObject.parseObject(record.value().toString(), Event.class);
            if (event == null) {
                logger.error("消息格式错误!");
                return;
            }
    
            String htmlUrl = (String) event.getData().get("htmlUrl");
            String fileName = (String) event.getData().get("fileName");
            String suffix = (String) event.getData().get("suffix");
    
            String cmd = wkImageCommand + " --quality 75 "
                    + htmlUrl + " " + wkImageStorage + "/" + fileName + suffix;
            try {
                Runtime.getRuntime().exec(cmd);
                logger.info("生成长图成功: " + cmd);
            } catch (IOException e) {
                logger.error("生成长图失败: " + e.getMessage());
            }
    
            // 启用定时器,监视该图片,一旦生成了,则上传至七牛云.
            UploadTask task = new UploadTask(fileName, suffix);
            Future future = taskScheduler.scheduleAtFixedRate(task, 500);
            task.setFuture(future);
        }
    
    
        class UploadTask implements Runnable {
    
            // 文件名称
            private String fileName;
            // 文件后缀
            private String suffix;
            // 启动任务的返回值
            private Future future;
            // 开始时间
            private long startTime;
            // 上传次数
            private int uploadTimes;
    
            public UploadTask(String fileName, String suffix) {
                this.fileName = fileName;
                this.suffix = suffix;
                this.startTime = System.currentTimeMillis();
            }
    
            public void setFuture(Future future) {
                this.future = future;
            }
    
            @Override
            public void run() {
                // 生成失败
                if (System.currentTimeMillis() - startTime > 30000) {
                    logger.error("执行时间过长,终止任务:" + fileName);
                    future.cancel(true);
                    return;
                }
                // 上传失败
                if (uploadTimes >= 3) {
                    logger.error("上传次数过多,终止任务:" + fileName);
                    future.cancel(true);
                    return;
                }
    
                String path = wkImageStorage + "/" + fileName + suffix;
                File file = new File(path);
                if (file.exists()) {
                    logger.info(String.format("开始第%d次上传[%s].", ++uploadTimes, fileName));
                    // 设置响应信息
                    StringMap policy = new StringMap();
                    policy.put("returnBody", CommunityUtil.getJSONString(0));
                    // 生成上传凭证
                    Auth auth = Auth.create(accessKey, secretKey);
                    String uploadToken = auth.uploadToken(shareBucketName, fileName, 3600, policy);
                    // 指定上传机房
                    UploadManager manager = new UploadManager(new Configuration(Zone.zone1()));
                    try {
                        // 开始上传图片
                        Response response = manager.put(
                                path, fileName, uploadToken, null, "image/" + suffix, false);
                        // 处理响应结果
                        JSONObject json = JSONObject.parseObject(response.bodyString());
                        if (json == null || json.get("code") == null || !json.get("code").toString().equals("0")) {
                            logger.info(String.format("第%d次上传失败[%s].", uploadTimes, fileName));
                        } else {
                            logger.info(String.format("第%d次上传成功[%s].", uploadTimes, fileName));
                            future.cancel(true);
                        }
                    } catch (QiniuException e) {
                        logger.info(String.format("第%d次上传失败[%s].", uploadTimes, fileName));
                    }
                } else {
                    logger.info("等待图片生成[" + fileName + "].");
                }
            }
        }
    
    }
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
  • 相关阅读:
    重构项目 vue2 => vue3 & nuxt2 => nuxt3 遇到的问题
    毕设-基于Javaweb药品销售管理系统
    【TypeScript】深入学习TypeScript对象类型
    ISCSLP 2022 | AccentSpeech—从众包数据中学习口音来构建目标说话人的口音语音合成系统
    二维码怎么分解成链接?线上快速解码教学
    【前端】WebWorker 在前端SPA框架的应用
    IO流文件相关部分
    8. Go实现Gin服务优雅关机与重启
    认知电子战 | 无线电中的认知理论
    文本直接生成20多种背景音乐,免费版Stable Audio来了!
  • 原文地址:https://blog.csdn.net/nice___amusin/article/details/126092756