package cc.mrbird.febs.ai.controller.productPoint;
|
|
import cc.mrbird.febs.ai.req.memberTalkStream.ApiMemberTalkReloadStreamDto;
|
import cc.mrbird.febs.ai.req.product.ApiProductInfoDto;
|
import cc.mrbird.febs.ai.req.product.ApiProductPageDto;
|
import cc.mrbird.febs.ai.req.productPoint.ApiProductPointInfoDto;
|
import cc.mrbird.febs.ai.req.productPoint.ApiProductPointPageDto;
|
import cc.mrbird.febs.ai.req.productPoint.ApiProductPointRecommendDto;
|
import cc.mrbird.febs.ai.res.product.ApiProductInfoVo;
|
import cc.mrbird.febs.ai.res.product.ApiProductVo;
|
import cc.mrbird.febs.ai.res.productPoint.ApiProductPointInfoVo;
|
import cc.mrbird.febs.ai.res.productPoint.ApiProductPointListVo;
|
import cc.mrbird.febs.ai.res.productPoint.ApiProductPointRecommendVo;
|
import cc.mrbird.febs.ai.service.AiProductPointService;
|
import cc.mrbird.febs.common.entity.FebsResponse;
|
import cc.mrbird.febs.common.utils.LoginUserUtil;
|
import io.swagger.annotations.Api;
|
import io.swagger.annotations.ApiOperation;
|
import io.swagger.annotations.ApiResponse;
|
import io.swagger.annotations.ApiResponses;
|
import lombok.RequiredArgsConstructor;
|
import lombok.extern.slf4j.Slf4j;
|
import org.springframework.http.HttpStatus;
|
import org.springframework.validation.annotation.Validated;
|
import org.springframework.web.bind.annotation.*;
|
|
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletResponse;
|
import java.io.InputStream;
|
import java.io.OutputStream;
|
import java.io.RandomAccessFile;
|
import java.nio.file.Files;
|
import java.nio.file.Path;
|
import java.nio.file.Paths;
|
|
/**
|
* @author Administrator
|
*/
|
@Slf4j
|
@Validated
|
@RestController
|
@RequiredArgsConstructor
|
@RequestMapping(value = "/api/ai/productPoint")
|
@Api(value = "ApiProductPointController", tags = "AI-知识点(学习)")
|
public class ApiProductPointController {
|
|
private final AiProductPointService aiProductPointService;
|
|
@ApiOperation(value = "知识点列表", notes = "知识点列表")
|
@ApiResponses({
|
@ApiResponse(code = 200, message = "success", response = ApiProductPointListVo.class)
|
})
|
@PostMapping(value = "/list")
|
public FebsResponse list(@RequestBody @Validated ApiProductPointPageDto dto) {
|
|
return aiProductPointService.productPointList(dto);
|
}
|
|
@ApiOperation(value = "知识点详情", notes = "知识点详情")
|
@ApiResponses({
|
@ApiResponse(code = 200, message = "success", response = ApiProductPointInfoVo.class)
|
})
|
@PostMapping(value = "/info")
|
public FebsResponse info(@RequestBody @Validated ApiProductPointInfoDto dto) {
|
|
return aiProductPointService.productPointInfo(dto);
|
}
|
|
|
@ApiOperation(value = "推荐", notes = "推荐")
|
@ApiResponses({
|
@ApiResponse(code = 200, message = "success", response = ApiProductPointRecommendVo.class)
|
})
|
@PostMapping(value = "/recommend")
|
public FebsResponse recommend(@RequestBody @Validated ApiProductPointRecommendDto dto) {
|
|
return aiProductPointService.recommend(dto);
|
}
|
|
|
public static final String baseUploadDir = "/home/javaweb/webresource/ai/file";
|
/**
|
* 播放视频文件
|
*/
|
@GetMapping("/play/{fileName}")
|
public void playVideo(@PathVariable("fileName") String fileName,
|
HttpServletRequest request,
|
HttpServletResponse response) {
|
try {
|
String companyId = LoginUserUtil.getLoginUser().getCompanyId();
|
String uploadDir = baseUploadDir + "/" + companyId;
|
Path filePath = Paths.get(uploadDir, fileName);
|
if (!Files.exists(filePath)) {
|
response.setStatus(HttpStatus.NOT_FOUND.value());
|
return;
|
}
|
|
// 获取文件大小
|
long fileSize = Files.size(filePath);
|
|
// 检查是否有Range请求头
|
String rangeHeader = request.getHeader("Range");
|
if (rangeHeader != null) {
|
// 解析Range头,格式为"bytes=start-end"
|
String[] ranges = rangeHeader.replace("bytes=", "").split("-");
|
long start = 0, end = fileSize - 1;
|
|
if (!ranges[0].isEmpty()) {
|
start = Long.parseLong(ranges[0]);
|
}
|
if (ranges.length > 1 && !ranges[1].isEmpty()) {
|
end = Long.parseLong(ranges[1]);
|
} else {
|
end = fileSize - 1;
|
}
|
|
// 确保范围有效
|
if (start > end || start >= fileSize) {
|
response.setStatus(HttpStatus.REQUESTED_RANGE_NOT_SATISFIABLE.value());
|
return;
|
}
|
|
// 限制结束位置不超过文件大小
|
if (end >= fileSize) {
|
end = fileSize - 1;
|
}
|
|
// 计算内容长度
|
long contentLength = end - start + 1;
|
|
// 设置响应状态和头部信息
|
response.setStatus(HttpStatus.PARTIAL_CONTENT.value());
|
response.setContentType(getContentTypeByExtension(fileName));
|
response.setContentLengthLong(contentLength);
|
response.setHeader("Content-Range", "bytes " + start + "-" + end + "/" + fileSize);
|
response.setHeader("Accept-Ranges", "bytes");
|
response.setHeader("Content-Disposition", "inline; filename=\"" + fileName + "\"");
|
|
// 从指定位置开始读取文件
|
try (RandomAccessFile randomAccessFile = new RandomAccessFile(filePath.toFile(), "r");
|
OutputStream outputStream = response.getOutputStream()) {
|
|
randomAccessFile.seek(start);
|
byte[] buffer = new byte[8192];
|
long remaining = contentLength;
|
int bytesRead;
|
|
while (remaining > 0 && (bytesRead = randomAccessFile.read(buffer, 0, (int) Math.min(buffer.length, remaining))) != -1) {
|
outputStream.write(buffer, 0, bytesRead);
|
remaining -= bytesRead;
|
}
|
}
|
} else {
|
|
// 没有Range头,返回前30秒内容(如果可能的话)
|
// 首先尝试估计前30秒对应的字节数(假设平均比特率为5 Mbps)
|
long estimatedFirst30Seconds = (long) (5 * 1024 * 1024 / 8 * 30); // 5 Mbps => bytes for 30 seconds
|
long actualEnd = Math.min(estimatedFirst30Seconds, fileSize - 1);
|
|
// 设置响应状态和头部信息
|
response.setStatus(HttpStatus.PARTIAL_CONTENT.value()); // 使用206状态码,因为只返回部分内容
|
response.setContentType(getContentTypeByExtension(fileName));
|
response.setContentLengthLong(actualEnd + 1);
|
response.setHeader("Content-Range", "bytes 0-" + actualEnd + "/" + fileSize);
|
response.setHeader("Accept-Ranges", "bytes");
|
response.setHeader("Content-Disposition", "inline; filename=\"" + fileName + "\"");
|
|
// 流式传输文件的前30秒
|
try (InputStream inputStream = Files.newInputStream(filePath);
|
OutputStream outputStream = response.getOutputStream()) {
|
byte[] buffer = new byte[8192];
|
long written = 0;
|
int bytesRead;
|
|
while (written <= actualEnd && (bytesRead = inputStream.read(buffer)) != -1) {
|
long toWrite = Math.min(bytesRead, (int)(actualEnd + 1 - written));
|
outputStream.write(buffer, 0, (int)toWrite);
|
written += toWrite;
|
|
// 如果已经写完前30秒的内容,则停止
|
if (written > actualEnd) {
|
break;
|
}
|
}
|
}
|
}
|
} catch (Exception e) {
|
e.printStackTrace();
|
response.setStatus(HttpStatus.INTERNAL_SERVER_ERROR.value());
|
}
|
}
|
|
/**
|
* 根据文件扩展名获取MIME类型
|
*/
|
private String getContentTypeByExtension(String fileName) {
|
String extension = "";
|
int lastDotIndex = fileName.lastIndexOf('.');
|
if (lastDotIndex > 0) {
|
extension = fileName.substring(lastDotIndex).toLowerCase();
|
}
|
|
switch (extension) {
|
case ".mp4":
|
return "video/mp4";
|
case ".avi":
|
return "video/x-msvideo";
|
case ".mov":
|
return "video/quicktime";
|
case ".wmv":
|
return "video/x-ms-wmv";
|
case ".flv":
|
return "video/x-flv";
|
case ".webm":
|
return "video/webm";
|
case ".mkv":
|
return "video/x-matroska";
|
case ".m3u8":
|
return "application/vnd.apple.mpegurl";
|
case ".ts":
|
return "video/MP2T";
|
default:
|
return "application/octet-stream";
|
}
|
}
|
|
}
|