阿里云 Oss 防刷实现

公司私有Oss服务满且需设权限,避免文件泄露。作者翻阅阿里云Oss权限文档,整合常用方法,包含文件增删、修改权限、获取签名等。经测试,上传文件设公共读可正常访问,修改权限后需签名才能访问,删除文件传全路径即可。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

最近公司的私有 Oss 服务满了,且 Oss 地址需要设置权限,只有当前系统的登录用户才能访问 Oss 下载地址。一开始想着用 Nginx 做个转发来着,Nginx 每当检测当前请求包含特定的 Oss 地址就转发到我们的统一鉴权接口上去,但是紧接着又细想了一下,转发后的地址被恶意分享出去了,不也还是存在文件泄露的风险吗?于是又去翻阅了一下阿里云的 Oss 权限相关的文档。借此整合一些常用的方法,机械代码自留也是分享给大家

完整代码

里面整合了文件的增、删、修改权限、获取签名等方法,各位替换成各自的 ak、sk 即可

package com.queyi.qykgjx.util;

import com.aliyun.oss.*;
import com.aliyun.oss.internal.OSSHeaders;
import com.aliyun.oss.model.*;
import org.apache.http.ParseException;
import org.springframework.web.multipart.MultipartFile;

import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;


public class OssUtilCsdn {
    private final static String ENDPOINT = "端点";
    private final static String ACCESS_KEY_ID = "ak";
    private final static String ACCESS_KEY_SECRET = "sk";
    private final static String BUCKET_NAME = "容器名";
    private static String HTTPS_URL_PREFIX = ENDPOINT.replace("https://", ("https://" + BUCKET_NAME + "."));
    private final static String FILE_CATALOG = "文件上传目录";
    
    public static void cpoy(String sourceKey, String dstKey, Boolean deleteSourceFile) {
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        CopyObjectRequest copyObjectRequest = new CopyObjectRequest(BUCKET_NAME, sourceKey, BUCKET_NAME, dstKey);
        ObjectMetadata objectMetadata = new ObjectMetadata();
        objectMetadata.setObjectAcl(CannedAccessControlList.Private);
        copyObjectRequest.setNewObjectMetadata(objectMetadata);
        ossClient.copyObject(copyObjectRequest);
        //删除被拷贝的文件
        if (deleteSourceFile) ossClient.deleteObject(BUCKET_NAME, sourceKey);
        ossClient.shutdown();
    }

    public static void setAcl(String bucketName, String url, Boolean privateAcl) {
        ClientBuilderConfiguration config = new ClientBuilderConfiguration();
        config.setSupportCname(false);
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        try {
            ossClient.setObjectAcl(bucketName == null ? BUCKET_NAME : bucketName,
                    getObjectNameByFullUrl(url),
                    privateAcl ? CannedAccessControlList.Private : CannedAccessControlList.PublicRead);
        } catch (OSSException oe) {
            oe.printStackTrace();
        } catch (ClientException ce) {
            ce.printStackTrace();
        } finally {
            if (ossClient != null) {
                ossClient.shutdown();
            }
        }
    }

    /**
     * @param bucketName 容器名称
     * @param file       待上传的文件
     * @param acl        文件权限
     */
    public static String upload(String bucketName, MultipartFile file, String acl) throws IOException {
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        if (null == bucketName || bucketName.length() > 0) bucketName = BUCKET_NAME;
        InputStream inputStream = file.getInputStream();
        String filename = file.getOriginalFilename();
        String[] split = filename.split("\\.");
        filename = new Date().getTime() + "." + split[split.length - 1];
        String s = FILE_CATALOG + "/" + filename.replaceAll("/", "");
        PutObjectRequest o = new PutObjectRequest(bucketName, s, inputStream);
        ObjectMetadata metadata = new ObjectMetadata();
        metadata.setHeader(OSSHeaders.OSS_STORAGE_CLASS, StorageClass.Standard.toString());
        if ("private".equals(acl)) metadata.setObjectAcl(CannedAccessControlList.Private);
        else if ("public".equals(acl)) metadata.setObjectAcl(CannedAccessControlList.PublicRead);
        else metadata.setObjectAcl(CannedAccessControlList.Private);
        o.setMetadata(metadata);
        ossClient.putObject(o);
        ossClient.shutdown();
        inputStream.close();
        return HTTPS_URL_PREFIX + "/" + s;
    }

    /**
     * 获取签名
     */
    public static String getSign(String key, int timeOut) throws ParseException {
        ClientBuilderConfiguration config = new ClientBuilderConfiguration();
        config.setSupportCname(false);
        Date expiration = new Date(new Date().getTime() + timeOut * 1000L);
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET, config);
        URL url = ossClient.generatePresignedUrl(BUCKET_NAME, key, expiration);
        return url.toString().split("\\?")[1];
    }

    public static String getAclPath(String url, int timeOut) throws ParseException {
        ClientBuilderConfiguration config = new ClientBuilderConfiguration();
        config.setSupportCname(false);
        Date expiration = new Date(new Date().getTime() + timeOut * 1000L);
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET, config);
        return ossClient.generatePresignedUrl(BUCKET_NAME, getObjectNameByFullUrl(url), expiration).toString();
    }

    public static String getObjectNameByFullUrl(String url) {
        if (!url.contains(FILE_CATALOG)) return null;
        /**
         * a/a.pdf
         */
        return FILE_CATALOG + url.split(FILE_CATALOG)[1];
    }

    /**
     * 删
     */
    public static List<String> deleteObject(String bucketName, List<String> keys) {
        ArrayList<String> newKeys = new ArrayList<>();
        keys.stream().forEach(key -> {
            newKeys.add(key.replace(HTTPS_URL_PREFIX + "/", ""));
        });
        OSS ossClient = new OSSClientBuilder().build(ENDPOINT, ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        DeleteObjectsResult deleteObjectsResult = ossClient
                .deleteObjects(new DeleteObjectsRequest(bucketName == null ? BUCKET_NAME : bucketName)
                        .withKeys(newKeys));
        List<String> deletedObjects = deleteObjectsResult.getDeletedObjects();
        ossClient.shutdown();
        return deletedObjects;
    }

}

整合接口

 @PostMapping("csdnUpFile")
    public Result csdnUpFile(@RequestParam("file") MultipartFile multipartFile
            , @RequestParam("acl") String perm) throws IOException {
        HashMap<String, String> map = new HashMap<>();
        map.put("fileName", multipartFile.getOriginalFilename());
        map.put("fileUrl", OssUtil.zzhUpload(null, multipartFile, perm));
        return Result.success(map);
    }
    @PostMapping("csdnSetAcl")
    public Result csdnSetAcl(@RequestParam("filePath") String filePath) throws IOException {
        OssUtil.setAcl(null, filePath, true);
        return Result.success("ok");
    }
    @PostMapping("csdnGetAclPath")
    public Result csdnGetAclPath(@RequestParam("filePath") String url) throws IOException {
        return Result.success(OssUtil.getAclPath(url, 60));
    }

    @PostMapping("csdnDeleteFile")
    public Result csdnDeleteDile(@RequestBody List<String> urls) throws IOException {
        return Result.success(OssUtil.deleteObject(null, urls));
    }

测试

上传文件设置成公共读
在这里插入图片描述
文件正常访问

在这里插入图片描述
将全路径地址作为参数掉修改权限接口

在这里插入图片描述
再次访问此地址提示无权限

在这里插入图片描述

加上签名访问文件可以正常访问

其他方法不做一一测试了,删除文件也是传上传接口返回的全路径即可,可直接删除文件。其他操作看下阿里的Oss 文档即可,机械代码不做过多描述

在这里插入图片描述

<think>好的,我现在需要帮用户介绍一下基于微信小程序的二手交易平台的技术准备和总体设计。首先,我需要理清楚用户的需求是什么。他们可能想要了解从技术角度出发,如何搭建这样一个平台,包括前端、后端、数据库、第三方服务集成等方面。用户可能是一个开发者或者创业者,想了解技术上的可行性和需要的资源。 首先,我应该从技术架构的分层开始,通常分为前端、后端、数据库和第三方服务。微信小程序作为前端,所以需要讲小程序的开发技术,比如WXML、WXSS、JavaScript,还有小程序框架的部分,比如页面路由、组件库等。然后,后端部分可能需要用Node.js或者Java、Python之类的语言,选择云开发还是自建服务器。数据库方面,微信有云数据库,或者自己用MySQL、MongoDB之类的。第三方服务可能包括支付、地图、即时通讯,比如微信支付、腾讯地图、WebSocket或者第三方IM服务。 接下来要考虑模块设计。用户模块需要注册登录,用户信息管理。商品模块包括发布、展示、搜索,可能需要图片上传和分类。交易模块涉及订单、支付、状态管理。消息模块需要即时通讯和通知。评价与售后模块有评价系统和纠纷处理。这些都是常见的模块,需要详细说明每个模块的功能和技术实现。 然后,技术选型是关键。前端用小程序原生开发,或者用框架如Taro、Uni-app实现跨平台。后端可以用云开发,比如微信云开发,或者传统的Spring Boot、Express、Django。数据库方面,云数据库适合快速开发,自建的话MySQL或MongoDB。第三方服务集成微信支付、地图、OSS存储,消息推送用模板消息,即时通讯可能需要融云或环信。 安全性和性能优化也不能忽视。接口要HTTPS,SQL注入和XSS攻击,用户数据加密,权限控制。性能方面,图片懒加载、缓存策略、CDN加速,数据库索引优化,负载均衡和自动扩展。 部署与运维部分,云服务的选择,自动化部署工具,监控和日志分析,还有版本更新和回滚策略。 最后,开发准备需要团队分工,开发工具如微信开发者工具、IDE选择,测试流程包括单元测试、压力测试、兼容性测试,以及上线前的审核和灰度发布。 需要注意的是,用户可能对某些技术细节不太了解,所以需要解释清楚每个部分的作用和选择的原因。同时,要确保结构清晰,逐步引导他们理解整个设计流程。可能还需要提醒他们根据项目规模和团队能力调整技术选型,比如小团队用云开发更高效,大项目可能需要自建后端。同时,安全性和性能是必须考虑的重点,不能遗漏。还要检查是否符合之前的系统级指令,比如数学公式的正确使用,但在这个问题中可能涉及较少,不过如果有技术参数或算法,需要正确应用LaTeX格式。</think>以下为基于微信小程序的二手交易平台技术准备及总体设计的结构化分析,分步骤说明核心要点: --- ### 一、技术架构分层设计 #### 1. **前端层(微信小程序)** - **技术栈** - 使用WXML+WXSS+JavaScript开发基础页面 - 采用小程序原生框架实现页面路由、数据绑定 - 集成`Vant Weapp`等UI组件库提升交互体验 - **关键能力** - 调用微信原生API(如相机、定位、扫码) - 实现图片上传压缩(通过`wx.compressImage`) - 本地缓存管理(`wx.setStorageSync`优化加载速度) #### 2. **后端服务层** - **方案选择** - **轻量级方案**:微信云开发(云函数+云数据库+云存储) - **高定制化方案**:自建后端(Node.js/Spring Boot/Django) - **核心模块** - RESTful API设计(商品管理、订单处理、用户鉴权) - 接口安全(JWT令牌验证+接口限流) - 异步任务处理(如消息队列处理订单超时) #### 3. **数据存储层** - **数据库选型** - 关系型数据库(MySQL):存储用户、订单等强一致性数据 - 文档数据库(MongoDB):存储商品信息、评论等半结构化数据 - 缓存层(Redis):热点数据缓存(如首页商品列表) - **设计要点** - 分库分表策略(按用户ID哈希分片) - 索引优化(为高频查询字段建立组合索引) #### 4. **第三方服务集成** - **支付系统**:微信支付(需申请商户资质) - **地图服务**:腾讯位置服务(实现同城交易定位) - **即时通讯**:融云SDK或自研WebSocket消息系统 - **文件存储**:阿里云OSS(存储用户上传的图片/视频) --- ### 二、核心功能模块设计 #### 1. **用户系统** - **OpenID绑定**:通过`wx.login`获取微信身份标识 - **信息管理**: ```javascript // 小程序端获取用户信息 wx.getUserProfile({ desc: '用于完善会员资料', success: (res) => { this.setData({ userInfo: res.userInfo }) } }) ``` #### 2. **商品系统** - **发布流程** - 多图上传(限制9张+EXIF信息过滤) - 智能分类(NLP分词+分类模型预测) - **搜索优化** - 倒排索引实现关键词搜索 - 协同过滤算法推荐相似商品 #### 3. **交易系统** - **订单状态机**: ```mermaid graph LR A[待付款] -->|支付超时| B[已关闭] A -->|成功支付| C[待发货] C -->|卖家发货| D[待收货] D -->|确认收货| E[交易完成] ``` - **支付回调**:处理微信支付异步通知,保证最终一致性 #### 4. **消息系统** - **实时通知**:WebSocket推送订单状态变更 - **离线消息**:通过订阅消息模板补推(如`wx.requestSubscribeMessage`) --- ### 三、关键技术实现方案 #### 1. **图片优化策略** - **前端压缩**:使用`canvas`调整图片分辨率 - **CDN加速**:对静态资源进行分区域缓存 - **防盗链**:OSS设置Referer白名单 #### 2. **安全护措施** - **敏感操作验证**:重要操作(如删除商品)需短信确认 - **机制**: - 接口调用频率限制(Redis记录IP请求次数) - 图形验证码触发条件(1分钟内错误3次) #### 3. **性能优化点** - **首屏加载**:骨架屏+分页懒加载 - **数据库查询**: ```sql /* 添加覆盖索引示例 */ CREATE INDEX idx_goods_search ON goods(title, price, location) USING BTREE; ``` --- ### 四、部署与运维方案 1. **多环境配置** - 开发环境:本地Mock数据 - 测试环境:Docker容器化部署 - 生产环境:Kubernetes集群+自动扩缩容 2. **监控体系** - 小程序错误监控(接入Fundebug) - 业务指标埋点(UV/PV、交易转化率) 3. **灰度发布** - 按用户ID分段逐步开放新功能 - 通过AB测试验证功能效果 --- ### 五、开发准备清单 | 类别 | 具体内容 | |------------|-----------------------------------| | 资质申请 | 微信小程序企业认证、支付商户号 | | 开发工具 | 微信开发者工具+Postman+Jmeter | | 文档资源 | 小程序开发文档、OpenAPI接口文档 | | 测试设备 | iOS/Android多机型覆盖 | --- 以上设计需根据实际业务规模动态调整,初期建议采用微信云开发快速验证商业模式,后期逐步迁移至微服务架构。关键成功因素在于交易流程的流畅性与信任体系构建(如实名认证+信用评分)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小咸鱼的技术窝

你的鼓励将是我最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值