Blob存储:SpringBoot实战之文件上传微软云(Azure Storage)

小晨Maste 2019-09-27 16:35:36 ⋅ 1122 阅读

前言

上传文件到Azure Storage 的案例比较少,只能到官网去研究,并且也不一定拿来就可以使用。

Blob 存储简介

为任何种类的非结构化数据使用可进行大规模缩放的对象存储


第一步:配置pom.xml

<!-- https://mvnrepository.com/artifact/com.microsoft.azure/azure-storage -->
<dependency>
<groupId>com.microsoft.azure</groupId>
<artifactId>azure-storage</artifactId>
<version>8.4.0</version>
</dependency>

第二步:增加azure blob配置

可以配置到项目中的.properties和yml 文件中

# properties 配置如下
azureblob.defaultEndpointsProtocol=https
azureblob.blobEndpoint=https://teststorage.blob.core.chinacloudapi.cn/
azureblob.queueEndpoint=https://teststorage.queue.core.chinacloudapi.cn/
azureblob.tableEndpoint=https://teststorage.table.core.chinacloudapi.cn/
azureblob.accountName=teststorage
azureblob.accountKey=accountkey
# yml 配置如下
azureblob:
defaultEndpointsProtocol: https
blobEndpoint: https://teststorage.queue.core.chinacloudapi.cn/
queueEndpoint: https://teststorage.queue.core.chinacloudapi.cn/
tableEndpoint: https://teststorage.queue.core.chinacloudapi.cn/
accountName: teststorage
accountKey: accountkey

第三步:编写配置信息类(StorageConfig)

import lombok.Data;
import lombok.NoArgsConstructor;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
@Configuration
@ConfigurationProperties(prefix = "azureblob")
@NoArgsConstructor
@Data
public class StorageConfig {
@Value("${azureblob.defaultEndpointsProtocol}")
private String defaultEndpointsProtocol;
@Value("${azureblob.blobEndpoint}")
private String blobEndpoint;
@Value("${azureblob.queueEndpoint}")
private String queueEndpoint;
@Value("${azureblob.tableEndpoint}")
private String tableEndpoint;
@Value("${azureblob.accountName}")
private String accountName;
@Value("${azureblob.accountKey}")
private String accountKey;
}

备注:这里用了lombok注解,也可以手写get/set

第四步:编写上传文件类(BlobHelper)

import com.microsoft.azure.storage.CloudStorageAccount;
import com.microsoft.azure.storage.blob.BlobContainerPermissions;
import com.microsoft.azure.storage.blob.BlobContainerPublicAccessType;
import com.microsoft.azure.storage.blob.CloudBlobClient;
import com.microsoft.azure.storage.blob.CloudBlobContainer;

public class BlobHelper {

public static CloudBlobContainer getBlobContainer(String containerName, StorageConfig storageConfig)
{
try
{
String blobStorageConnectionString = String.format("DefaultEndpointsProtocol=%s;"
+ "BlobEndpoint=%s;"
+ "QueueEndpoint=%s;"
+ "TableEndpoint=%s;"
+ "AccountName=%s;"
+ "AccountKey=%s",
storageConfig.getDefaultEndpointsProtocol(), storageConfig.getBlobEndpoint(),
storageConfig.getQueueEndpoint(), storageConfig.getTableEndpoint(),
storageConfig.getAccountName(), storageConfig.getAccountKey());

CloudStorageAccount account = CloudStorageAccount.parse(blobStorageConnectionString);
CloudBlobClient serviceClient = account.createCloudBlobClient();

CloudBlobContainer container = serviceClient.getContainerReference(containerName);

// Create a permissions object.
BlobContainerPermissions containerPermissions = new BlobContainerPermissions();

// Include public access in the permissions object.
containerPermissions.setPublicAccess(BlobContainerPublicAccessType.CONTAINER);
// Set the permissions on the container.
container.uploadPermissions(containerPermissions);
container.createIfNotExists();
return container;
}
catch(Exception e)
{
// 加载上传文件启动异常
return null;
}
}
}

第五步:增加工具类(MyUtil)

public class MyUtils {
private static char hexdigits[] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e',
'f' };
public static String getMD5(String inStr) {
MessageDigest md5 = null;
try {
md5 = MessageDigest.getInstance("MD5");
} catch (Exception e) {
e.printStackTrace();
return "";
}
char[] charArray = inStr.toCharArray();
byte[] byteArray = new byte[charArray.length];
for (int i = 0; i < charArray.length; i++)
byteArray[i] = (byte) charArray[i];
byte[] md5Bytes = md5.digest(byteArray);
StringBuffer hexValue = new StringBuffer();
for (int i = 0; i < md5Bytes.length; i++) {
int val = ((int) md5Bytes[i]) & 0xff;
if (val < 16)
hexValue.append("0");
hexValue.append(Integer.toHexString(val));
}
return hexValue.toString();
}
public static String getMD5(InputStream fileStream) {
try {
MessageDigest md = MessageDigest.getInstance("MD5");
byte[] buffer = new byte[2048];
int length = -1;
while ((length = fileStream.read(buffer)) != -1) {
md.update(buffer, 0, length);
}
byte[] b = md.digest();
return byteToHexString(b);
} catch (Exception ex) {
ex.printStackTrace();
return null;
}finally{
try {
fileStream.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
private static String byteToHexString(byte[] tmp) {
String s;
// 用字节表示就是 16 个字节
char str[] = new char[16 * 2]; // 每个字节用 16 进制表示的话,使用两个字符,
// 所以表示成 16 进制需要 32 个字符
int k = 0; // 表示转换结果中对应的字符位置
for (int i = 0; i < 16; i++) { // 从第一个字节开始,对 MD5 的每一个字节
// 转换成 16 进制字符的转换
byte byte0 = tmp[i]; // 取第 i 个字节
str[k++] = hexdigits[byte0 >>> 4 & 0xf]; // 取字节中高 4 位的数字转换,
// >>> 为逻辑右移,将符号位一起右移
str[k++] = hexdigits[byte0 & 0xf]; // 取字节中低 4 位的数字转换
}
s = new String(str); // 换后的结果转换为字符串
return s;
}
}

第六步:上传文件接口( MultipartFile)

@Api(value = "微软云存储", description = "微软云存储")
@RestController
@RequestMapping("azure")
@Slf4j
public class FileUploadController {
@Autowired
private StorageConfig storageConfig;
@ApiOperation(value = "图片上传Azure", notes = "图片上传Azure")
@RequestMapping(value = "/uploadImg", method = RequestMethod.POST, consumes = "multipart/*", headers = "content-type=multipart/form-data")
public Object uploadImg(@RequestBody MultipartFile file) {
try {
if (file != null) {
//获取或创建container
CloudBlobContainer blobContainer = BlobHelper.getBlobContainer(blobContainerName, storageConfig);
if (!file.isEmpty()) {
try {

//拼装blob的名称(前缀名称+文件的md5值+文件扩展名称)
String checkSum = MyUtils.getMD5(file.getInputStream());
String fileExtension = getFileExtension(file.getOriginalFilename()).toLowerCase();
String preName = getBlobPreName(0, false).toLowerCase();
String blobName = preName + checkSum + fileExtension;
log.info(blobName);
//设置文件类型,并且上传到azure blob
CloudBlockBlob blob = blobContainer.getBlockBlobReference(blobName);
blob.getProperties().setContentType(file.getContentType());
blob.upload(file.getInputStream(), file.getSize());
//将上传后的图片URL返回
return blob.getUri().toString();
} catch (Exception e) {
log.error("upload azure oss error:{}", e);
}
}
}
// }
} catch (Exception e) {
log.error("upload azure oss error:{}", e);
}
}
return null;
}

结束

好了,一个简单的SpringBoot 上传文件至微软云的小案例就完成啦,当然如果是图片、视频等可能还需要进行文件格式的拦截,代码如下:

if (!(file.getContentType().toLowerCase().equals("image/jpg")
|| file.getContentType().toLowerCase().equals("image/jpeg")
|| file.getContentType().toLowerCase().equals("image/png"))) {
infoUniformResultTemplate.setCode(Code.FAIL.getCode());
log.info("图片格式不正确");
}

扩展


Azure 信息保护客户端支持的文件类型如下:

  • Adobe 可移植文档格式:pdf

  • Microsoft Project:.mpp、.mpt

  • Microsoft Publisher:.pub

  • Microsoft XPS:.xps .oxps

  • 图像:.jpg、.jpe、.jpeg、.jif、.jfif、.jfi、 .png、.tif、.tiff

  • Autodesk Design Review 2013:.dwfx

  • Adobe Photoshop:.psd

  • 数码底片:.dng

  • Microsoft Office:*

官网地址:https://docs.microsoft.com/zh-cn/azure/information-protection/rms-client/client-admin-guide-file-types




全部评论: 0

    我有话说:

    改进 Chromium 标签页恢复功能

      正在为 Chromium 改进标签页恢复功能,以提升 Edge 和 Chrome 的可靠性。 当 Chrome/Edge 遭遇意外关闭或崩溃时,再次启动后会提供恢复标签页及其会

    信小程序电商实战-首页(

    一篇:信小程序电商实战-入门篇 嗨,大家好!经过近两周的精心准备终于开始信小程序电商实战路喽。那么最终会做成什么样呢?好了,不啰嗦了 我们先看首页长什么样吧!   首页效果图

    Fluid 0.3 正式发布:实现原生场景通用化数据加速

    简介 为了解决大数据、AI 等数据密集型应用在原生计算存储分离场景下,存在的数据访问延时高、联合分析难、多维管理杂等痛点问题,南京大学 PASALab、阿里巴巴、Alluxio 在 2020 年

    信小程序电商实战-购物车(

    好久没更新小程序电商实战文章了,是因为最近一直做整体架构调整,一些准备工作也是比较耗费时间的。在这几天将会有新版的 小程序电商教程推出.......

    「转载」蘑菇街消息系统实践

    小编又来啦~本周要推荐给大家的是一篇跟中间件相关的技术文章,这里面详细的记录了,蘑菇街自研消息系统的全过程,也是市面开放出来为数不多的企业自研组件实践。有相关需求的同学可以好好学习下

    信小程序抖音实战-首页(

    你也可以用信小程序编写一个抖音

    信小程序电商实战-商品详情(

    先看一下今天要实现的小程序商品详情页吧!

    信小程序实战篇:小程序页面数据传递

    我们在写小程序的时候经常会遇到子页面向主页面回数据或者普通页面跳转到tabBar 页面携带数......

    原生信小程序开发实战

    腾讯高级工程师 和 腾讯技术产品经理 联合打造原生信小程序开发实战课程

    信小程序电商实战-首页(下)

    一篇:信小程序电商实战-首页()好了,一期我们把首页搜索、导航栏和广告轮播给做完了,那么接下来会继续

    信小程序商城(四):动态API实现商品详情页(

    1、实现商品详情页面布局(这篇实现3个模块,头部商品图片轮播、商品价格和商品描述、商品详情展示) 2、根据用户点击不同的商品请求API动态加载数据

    Node文件文件夹(八)

    学习时候的错觉就是你写bug时候的感觉。没错、老铁。

    SpringBoot+zk+dubbo架构实践(一):本地部署zookeeper

    SpringBoot+zk+dubbo架构实践系列实现目标:自己动手搭建服务架构

    Syncthing 1.11.0 和 1.11.1 发布,连续文件同步工具

    Syncthing 是一个免费开源的工具,它能在你的各个网络计算机间同步文件/文件夹,它的同步数据是直接从一个系统中直接传输到另一个系统的,并且它是安全且私密的。 Syncthing 1

    信小程序商城(九):信授权并实现个人中心页面页面

    实现商城的信授权并获取用户信息和个人中心页面布局

    信小程序营销大转盘(二)

    第一个版本的大转盘都是用图片做的,奖品等信息都是不无法修改的,说白了就是没啥实际用途,作者我就直接用canvas撸了一个全手工绘制的大转盘分享给大家

    Edge 88 beta 适配 M1,弃用 Flash

    如果你想在 M1 芯片的 Mac 电脑使用基于 Chromnium 开发的浏览器,从现在起不必再选择 Google Chrome。 根据 Windows Central 的报道,最新推出的

    精品推荐:服务架构下静态数据通用缓存机制

    在分布式系统中,特别是最近很火的服务架构下,有没有或者能不能总结出一个业务静态数据的通用缓存处理机制或方案,这篇文章将结合一些实际的研发经验,尝试理清其中存在的关键问题以及探寻通用的解决道。

    信小程序电商实战-商品列表流式布局

    今天给大家分享一下信小程序中商品列表的流式布局方式,根据文章内容操作就可以看到效果哦~~~