对于小视频的功能的开发,核心点就是:存储 + 推荐 + 加载速度 。
搭建服务
我们使用docker进行搭建。
#拉取镜像
docker pull delron/fastdfs
#创建tracker容器
docker run -d --network=host --restart always --name tracker -v /var/fdfs/tracker:/var/fdfs delron/fastdfs tracker
#创建storage容器
docker run -d --network=host --restart always --name storage -e TRACKER_SERVER=192.168.56.11:22122 -v /var/fdfs/storage:/var/fdfs -e GROUP_NAME=group1 delron/fastdfs storage
#进入storage容器,到storage的配置文件中配置http访问的端口,配置文件在/etc/fdfs目录下的storage.conf。
docker exec -it storage /bin/bash
#默认的http端口为8888,可以修改也可以配置
#the port of the web server on this storage server
http.server_port=8888
#配置nginx,在/usr/local/nginx目录下,修改nginx.conf文件
#默认配置如下:
server {
listen 8888;
server_name localhost;
location ~/group[0-9]/ {
ngx_fastdfs_module;
}
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root html;
}
}
#默认的存储路径为/var/fdfs/data
server项目导入依赖:
<dependency>
<groupId>com.github.tobato</groupId>
<artifactId>fastdfs-client</artifactId>
<version>1.26.7</version>
<exclusions>
<exclusion>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
</exclusion>
</exclusions>
</dependency>
application.properties
# ===================================================================
#
# ===================================================================
fdfs.so-timeout = 1501
fdfs.connect-timeout = 601
fdfs.thumb-image.width= 150
fdfs.thumb-image.height= 150
fdfs.tracker-list=192.168.56.11:22122
spring.servlet.multipart.max-file-size=30MB
spring.servlet.multipart.max-request-size=30MB
fdfs.web-server-url=http://192.168.56.11:8888
package com.tanhua.server;
import com.github.tobato.fastdfs.domain.fdfs.StorePath;
import com.github.tobato.fastdfs.service.FastFileStorageClient;
import org.apache.commons.io.FileUtils;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import java.io.File;
import java.io.IOException;
@RunWith(SpringRunner.class)
@SpringBootTest
public class TestFastDFS {
@Autowired
protected FastFileStorageClient storageClient;
@Test
public void testUpload(){
String path = "C:\\Users\\zhijun\\Desktop\\pics\\1.jpg";
File file = new File(path);
try {
StorePath storePath = this.storageClient.uploadFile(FileUtils.openInputStream(file), file.length(), "jpg", null);
System.out.println(storePath); //StorePath [group=group1, path=M00/00/00/wKgfUV2GJSuAOUd_AAHnjh7KpOc1.1.jpg]
System.out.println(storePath.getFullPath());//group1/M00/00/00/wKgfUV2GJSuAOUd_AAHnjh7KpOc1.1.jpg
} catch (IOException e) {
e.printStackTrace();
}
}
}
在接口工程中编写pojo
package com.***.dubbo.server.pojo;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.bson.types.ObjectId;
import org.springframework.data.mongodb.core.mapping.Document;
import java.util.List;
@Data
@NoArgsConstructor
@AllArgsConstructor
@Document(collection = "video")
public class Video implements java.io.Serializable {
private static final long serialVersionUID = -3136732836884933873L;
private ObjectId id; //主键id
private Long userId;
private String text; //文字
private String picUrl; //视频封面文件
private String videoUrl; //视频文件
private Long created; //创建时间
private Integer seeType; // 谁可以看,1-公开,2-私密,3-部分可见,4-不给谁看
private List<Long> seeList; //部分可见的列表
private List<Long> notSeeList; //不给谁看的列表
private String longitude; //经度
private String latitude; //纬度
private String locationName; //位置名称
}
package com.***.dubbo.server.api;
import com.***.dubbo.server.pojo.Video;
public interface VideoApi {
/**
* 保存小视频
*
* @param video
* @return
*/
Boolean saveVideo(Video video);
}
package com.tanhua.dubbo.server.api;
import com.alibaba.dubbo.config.annotation.Service;
import com.tanhua.dubbo.server.pojo.Video;
import org.bson.types.ObjectId;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.mongodb.core.MongoTemplate;
@Service(version = "1.0.0")
public class VideoApiImpl implements VideoApi {
@Autowired
private MongoTemplate mongoTemplate;
@Override
public Boolean saveVideo(Video video) {
if(video.getUserId() == null){
return false;
}
video.setId(ObjectId.get());
video.setCreated(System.currentTimeMillis());
this.mongoTemplate.save(video);
return true;
}
}
package com.***.server.controller;
import com.***.server.service.VideoService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;
@RestController
@RequestMapping("smallVideos")
public class VideoController {
@Autowired
private VideoService videoService;
@PostMapping
public ResponseEntity<Void> saveVideo(@RequestParam(value = "videoThumbnail", required = false) MultipartFile picFile,
@RequestParam(value = "videoFile", required = false) MultipartFile videoFile) {
try {
Boolean bool = this.videoService.saveVideo(picFile, videoFile);
if(bool){
return ResponseEntity.ok(null);
}
} catch (Exception e) {
e.printStackTrace();
}
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build();
}
}
VideoService
package com.***.server.service;
import com.alibaba.dubbo.config.annotation.Reference;
import com.github.tobato.fastdfs.domain.conn.FdfsWebServer;
import com.github.tobato.fastdfs.domain.fdfs.StorePath;
import com.github.tobato.fastdfs.service.FastFileStorageClient;
import com.***.dubbo.server.api.VideoApi;
import com.***.dubbo.server.pojo.Video;
import com.***.server.pojo.User;
import com.***.server.utils.UserThreadLocal;
import com.***.server.vo.PicUploadResult;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
@Service
public class VideoService {
@Autowired
private PicUploadService picUploadService;
@Autowired
protected FastFileStorageClient storageClient;
@Autowired
private FdfsWebServer fdfsWebServer;
@Reference(version = "1.0.0")
private VideoApi videoApi;
public Boolean saveVideo(MultipartFile picFile, MultipartFile videoFile) {
User user = UserThreadLocal.get();
Video video = new Video();
video.setUserId(user.getId());
video.setSeeType(1);
try {
//上传封面图片
PicUploadResult picUploadResult = this.picUploadService.upload(picFile);
video.setPicUrl(picUploadResult.getName()); //图片路径
//上传视频
StorePath storePath = storageClient.uploadFile(videoFile.getInputStream(),
videoFile.getSize(),
StringUtils.substringAfter(videoFile.getOriginalFilename(), "."),
null);
video.setVideoUrl(fdfsWebServer.getWebServerUrl() + "/" + storePath.getFullPath());
this.videoApi.saveVideo(video);
return true;
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
}
如果上传视频,会导致异常,是因为请求太大的缘故:
解决:application.properties
spring.servlet.multipart.max-file-size=30MB
spring.servlet.multipart.max-request-size=30MB