添加分页查询单位内外的车辆任务接口

main
吴顺杰 2 weeks ago
parent f66794f3d8
commit adcac54754

@ -18,6 +18,20 @@
<dependencies>
<!-- 空间坐标计算-->
<dependency>
<groupId>org.gavaghan</groupId>
<artifactId>geodesy</artifactId>
<version>1.1.3</version>
</dependency>
<!-- https://mvnrepository.com/artifact/jakarta.validation/jakarta.validation-api -->
<dependency>
<groupId>jakarta.validation</groupId>
<artifactId>jakarta.validation-api</artifactId>
<version>3.1.1</version>
</dependency>
<!-- SpringBoot 拦截器 -->
<dependency>
<groupId>org.springframework.boot</groupId>

@ -0,0 +1,24 @@
package com.ykMap.base.config;
/**
* mybatis-plus
*
* @author wu
* @since 2025/3/19 16:55
*/
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class MyBatisPlusConfig {
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
interceptor.addInnerInterceptor(new PaginationInnerInterceptor());
return interceptor;
}
}

@ -0,0 +1,42 @@
package com.ykMap.controller;
import com.ykMap.base.controller.BaseController;
import com.ykMap.base.domain.AjaxResult;
import com.ykMap.entity.request.CarPageRequest;
import com.ykMap.entity.response.CarPageResponse;
import com.ykMap.service.CarService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import jakarta.validation.Valid;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;
/**
*
*
* @author wu
* @since 2025/3/12 10:22
*/
@RestController
@RequestMapping("/ykmap/car")
@Api(tags = "车辆实时定位接口")
public class CarController extends BaseController {
@Resource
private CarService carService;
/**
*
*
* @param req
* @return
*/
@GetMapping("page")
@ApiOperation(value = "分页查询单位内外的车辆任务", response = CarPageResponse.class)
public AjaxResult page( @Valid CarPageRequest req) {
return success(carService.page(req));
}
}

@ -0,0 +1,19 @@
package com.ykMap.entity;
import lombok.Data;
import java.io.Serializable;
/**
* @author wu
* @since 2025/3/19 17:04
*/
@Data
public class ColDTO implements Serializable {
private static final long serialVersionUID = -8045055883099483791L;
private String value;
private String label;
}

@ -0,0 +1,44 @@
package com.ykMap.entity.request;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import jakarta.validation.constraints.NotNull;
import lombok.Data;
import java.io.Serializable;
/**
*
*
* @author wu
* @since 2025/3/14 14:22
*/
@Data
@ApiModel("分页查询单位内外的车辆任务请求类")
public class CarPageRequest implements Serializable {
private static final long serialVersionUID = 1031466955513130671L;
/**
*
*/
@ApiModelProperty(value = "当前页 默认1")
public Integer pageNo = 1;
/**
*
*/
@ApiModelProperty(value = "每页显示条数 默认10")
public Integer pageSize = 10;
/**
*
*/
@ApiModelProperty(value = "距离单位距离(米) 默认200米")
public double distance = 200;
/**
* 1 2
*/
@ApiModelProperty(value = "单位内外 1单位外 2单位内", required = true)
@NotNull(message = "单位内外不能为空")
private Integer type;
}

@ -0,0 +1,31 @@
package com.ykMap.entity.response;
import lombok.Data;
import java.io.Serializable;
/**
*
*
* @author wu
* @since 2025/3/14 15:02
*/
@Data
public class CarGPSResponse implements Serializable {
private static final long serialVersionUID = 6524106295010311569L;
/**
*
*/
public String carPlate;
/**
* 84
*/
public String lat;
/**
* 84
*/
public String lng;
}

@ -0,0 +1,30 @@
package com.ykMap.entity.response;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import java.io.Serializable;
/**
*
*
* @author wu
* @since 2025/3/14 14:32
*/
@Data
@ApiModel("分页查询单位内外的车辆任务响应类")
public class CarPageResponse implements Serializable {
/**
*
*/
@ApiModelProperty(value = "车牌号")
public String carNumber;
/**
*
*/
@ApiModelProperty(value = "车辆关联任务")
public String taskTitle;
}

@ -0,0 +1,30 @@
package com.ykMap.entity.response;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import java.io.Serializable;
/**
*
*
* @author wu
* @since 2025/3/14 14:32
*/
@Data
@ApiModel("分页查询单位内外的车辆任务响应类")
public class CarResponse implements Serializable {
/**
*
*/
@ApiModelProperty(value = "车牌号")
public String carNumber;
/**
*
*/
@ApiModelProperty(value = "车辆关联任务")
public String taskTitle;
}

@ -0,0 +1,24 @@
package com.ykMap.entity.response;
import com.ykMap.entity.ColDTO;
import lombok.Data;
import java.io.Serializable;
import java.util.List;
/**
* @author wu
* @since 2025/3/19 17:02
*/
@Data
public class ContentResponse implements Serializable {
private static final long serialVersionUID = -8078281805074879229L;
private List<ColDTO> col;
private List<CarPageResponse> data;
private Long totalCount = 0L;
private Long pageNum = 1L;
private Long pageSize = 10L;
}

@ -0,0 +1,110 @@
package com.ykMap.exception;
import com.ykMap.base.domain.AjaxResult;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.validation.BindException;
import org.springframework.web.HttpRequestMethodNotSupportedException;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.MissingPathVariableException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import org.springframework.web.method.annotation.MethodArgumentTypeMismatchException;
import javax.servlet.http.HttpServletRequest;
import java.util.Objects;
/**
*
*
* @author wu
* @since 2025/3/13 17:28
*/
@RestControllerAdvice
public class GlobalExceptionHandler {
private static final Logger log = LoggerFactory.getLogger(GlobalExceptionHandler.class);
/**
*
*/
@ExceptionHandler(HttpRequestMethodNotSupportedException.class)
public AjaxResult handleHttpRequestMethodNotSupported(HttpRequestMethodNotSupportedException e,
HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("请求地址'{}',不支持'{}'请求", requestURI, e.getMethod());
return AjaxResult.error(e.getMessage());
}
/**
*
*/
@ExceptionHandler(ServiceException.class)
public AjaxResult handleServiceException(ServiceException e) {
log.error(e.getMessage(), e);
Integer code = e.getCode();
return code != null ? AjaxResult.error(code, e.getMessage()) : AjaxResult.error(e.getMessage());
}
/**
*
*/
@ExceptionHandler(MissingPathVariableException.class)
public AjaxResult handleMissingPathVariableException(MissingPathVariableException e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("请求路径中缺少必需的路径变量'{}',发生系统异常.", requestURI, e);
return AjaxResult.error(String.format("请求路径中缺少必需的路径变量[%s]", e.getVariableName()));
}
/**
*
*/
@ExceptionHandler(MethodArgumentTypeMismatchException.class)
public AjaxResult handleMethodArgumentTypeMismatchException(MethodArgumentTypeMismatchException e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("请求参数类型不匹配'{}',发生系统异常.", requestURI, e);
return AjaxResult.error(String.format("请求参数类型不匹配,参数[%s]要求类型为:'%s',但输入值为:'%s'",
e.getName(), Objects.requireNonNull(e.getRequiredType()).getName(), e.getValue()));
}
/**
*
*/
@ExceptionHandler(RuntimeException.class)
public AjaxResult handleRuntimeException(RuntimeException e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("请求地址'{}',发生未知异常.", requestURI, e);
return AjaxResult.error(e.getMessage());
}
/**
*
*/
@ExceptionHandler(Exception.class)
public AjaxResult handleException(Exception e, HttpServletRequest request) {
String requestURI = request.getRequestURI();
log.error("请求地址'{}',发生系统异常.", requestURI, e);
return AjaxResult.error(e.getMessage());
}
/**
*
*/
@ExceptionHandler(BindException.class)
public AjaxResult handleBindException(BindException e) {
log.error(e.getMessage(), e);
String message = e.getAllErrors().get(0).getDefaultMessage();
return AjaxResult.error(message);
}
/**
*
*/
@ExceptionHandler(MethodArgumentNotValidException.class)
public Object handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {
log.error(e.getMessage(), e);
String message = Objects.requireNonNull(e.getBindingResult().getFieldError()).getDefaultMessage();
return AjaxResult.error(message);
}
}

@ -0,0 +1,64 @@
package com.ykMap.exception;
/**
*
*
* @author wu
* @since 2025/3/13 17:28
*/
public final class ServiceException extends RuntimeException {
private static final long serialVersionUID = 1L;
/**
*
*/
private Integer code;
/**
*
*/
private String message;
/**
*
*/
private String detailMessage;
/**
*
*/
public ServiceException() {
}
public ServiceException(String message) {
this.message = message;
}
public ServiceException(String message, Integer code) {
this.message = message;
this.code = code;
}
public String getDetailMessage() {
return detailMessage;
}
public ServiceException setDetailMessage(String detailMessage) {
this.detailMessage = detailMessage;
return this;
}
@Override
public String getMessage() {
return message;
}
public ServiceException setMessage(String message) {
this.message = message;
return this;
}
public Integer getCode() {
return code;
}
}

@ -65,4 +65,10 @@ public interface LineMapper{
* @return
*/
LocalDateTime findLineTemplateTime();
/**
*
* @return
*/
LocalDateTime findCityTime();
}

@ -1,7 +1,9 @@
package com.ykMap.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.ykMap.entity.TaskMission;
import com.ykMap.entity.response.CarPageResponse;
import org.apache.ibatis.annotations.Param;
import java.util.List;
@ -29,5 +31,14 @@ public interface TaskMissionMapper extends BaseMapper<TaskMission> {
* @return
*/
List<TaskMission> findByPlateNums(@Param("array") String[] arr);
/**
*
*
* @param page
* @param list
* @return
*/
Page<CarPageResponse> pageByPlateNums(Page<CarPageResponse> page, @Param("list") List<String> list);
}

@ -0,0 +1,21 @@
package com.ykMap.service;
import com.ykMap.entity.request.CarPageRequest;
import com.ykMap.entity.response.ContentResponse;
/**
*
*
* @author wu
* @since 2025/3/13 17:28
*/
public interface CarService {
/**
*
*
* @param req
* @return
*/
ContentResponse page(CarPageRequest req);
}

@ -1,7 +1,9 @@
package com.ykMap.service;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService;
import com.ykMap.entity.TaskMission;
import com.ykMap.entity.response.CarPageResponse;
import java.util.List;
@ -29,5 +31,14 @@ public interface TaskMissionService extends IService<TaskMission> {
* @return
*/
List<TaskMission> findByPlateNums(String plateNum);
/**
*
*
* @param page
* @param plateNums
* @return
*/
Page<CarPageResponse> pageByPlateNums(Page<CarPageResponse> page, List<String> plateNums);
}

@ -0,0 +1,155 @@
package com.ykMap.service.impl;
import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.NumberUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.http.HttpRequest;
import cn.hutool.json.JSONUtil;
import com.alibaba.fastjson2.JSONObject;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.ykMap.entity.ColDTO;
import com.ykMap.entity.request.CarPageRequest;
import com.ykMap.entity.response.CarGPSResponse;
import com.ykMap.entity.response.CarPageResponse;
import com.ykMap.entity.response.ContentResponse;
import com.ykMap.exception.ServiceException;
import com.ykMap.service.CarService;
import com.ykMap.service.TaskMissionService;
import com.ykMap.utils.SpaceUtils;
import lombok.extern.slf4j.Slf4j;
import org.gavaghan.geodesy.GlobalCoordinates;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;
/**
*
*
* @author wu
* @since 2025/3/13 17:28
*/
@Service("CarService")
@Slf4j
public class CarServiceImpl implements CarService {
@Resource
private TaskMissionService taskMissionService;
/**
*
*
* @param req
* @return
*/
@Override
public ContentResponse page(CarPageRequest req) {
// 查询sessionId
String sessionId = findSessionId();
// 获取车辆实时定位
List<CarGPSResponse> carGPSList = findCarGPS(sessionId);
GlobalCoordinates companyLocation = new GlobalCoordinates(31.226963,120.634355);
// 过滤后的车辆
List<CarGPSResponse> list = new ArrayList<>();
if (CollectionUtil.isEmpty(carGPSList)) {
return null;
}
for (CarGPSResponse res : carGPSList) {
GlobalCoordinates carLocation = new GlobalCoordinates(Double.parseDouble(res.getLat()), Double.parseDouble(res.getLng()));
double distanceMeter = SpaceUtils.getDistanceMeter(companyLocation, carLocation);
if (req.getType() == 1) {
if (NumberUtil.compare(req.getDistance(), distanceMeter) != 1) {
// 单位外
list.add(res);
}
} else {
if (NumberUtil.compare(req.getDistance(), distanceMeter) == 1) {
// 单位内
list.add(res);
}
}
}
if (CollectionUtil.isEmpty(list)) {
return null;
}
List<String> plateNums = list.stream().map(CarGPSResponse::getCarPlate).collect(Collectors.toList());
Page<CarPageResponse> page = new Page<>();
page.setCurrent(req.getPageNo());
page.setSize(req.getPageSize());
Page<CarPageResponse> carPageResponsePage = taskMissionService.pageByPlateNums(page, plateNums);
ContentResponse contentResponse = new ContentResponse();
contentResponse.setTotalCount(carPageResponsePage.getTotal());
contentResponse.setPageNum(carPageResponsePage.getCurrent());
contentResponse.setPageSize(carPageResponsePage.getSize());
ArrayList<ColDTO> colDTOS = new ArrayList<>();
ColDTO colDTO = new ColDTO();
colDTO.setValue("carNumber");
colDTO.setLabel("车辆号码");
ColDTO colDTO2 = new ColDTO();
colDTO2.setValue("Title");
colDTO2.setLabel("任务名称");
colDTOS.add(colDTO);
colDTOS.add(colDTO2);
contentResponse.setCol(colDTOS);
contentResponse.setData(carPageResponsePage.getRecords());
return contentResponse;
}
/**
* sessionId
*
* @return sessionId
*/
private String findSessionId() {
String body = HttpRequest.get("http://192.168.0.91/gps-web/api/login.jsp")
.form("userId", 99999)
.form("password", "6d0fd76ab81325588d8065ec1036fdd3")
.form("loginType", "user")
.execute().body();
JSONObject json;
log.info("================获取sessionId数据=================");
log.info(body);
try {
json = JSONObject.parse(body);
} catch (Exception e) {
log.error(e.getMessage());
throw new ServiceException("未获取到sessionId");
}
return json.get("sessionId").toString();
}
/**
*
*
* @return
*/
private List<CarGPSResponse> findCarGPS(String sessionId) {
String body = HttpRequest.get("http://192.168.0.91/gps-web/api/get_gps_r.jsp")
.form("teamId", 2)
.form("sessionId", sessionId)
.execute().body();
JSONObject json;
log.info("================获取车辆实时定位数据=================");
log.info(body);
try {
json = JSONObject.parse(body);
} catch (Exception e) {
log.error(e.getMessage());
throw new ServiceException("获取车辆实时定位失败");
}
List<CarGPSResponse> list = Collections.emptyList();
Object listStr = json.get("list");
String recordsStr = JSONUtil.toJsonStr(listStr);
if (StrUtil.isNotEmpty(recordsStr)) {
// 获取所有车辆经纬度
list = JSONUtil.toList(recordsStr, CarGPSResponse.class);
}
return list;
}
}

@ -99,16 +99,26 @@ public class LineServiceImpl implements LineService {
*/
@Override
public List<SearchCityResponse> searchCity() {
LocalDate now = LocalDate.now();
if (now.getDayOfMonth() < 26) {
LocalDate a1 = now.minusMonths(2).withDayOfMonth(26);
LocalDate a2 = now.minusMonths(1).withDayOfMonth(25);
return lineMapper.searchCity(a1, a2);
} else {
LocalDate a1 = now.minusMonths(1).withDayOfMonth(26);
LocalDate a2 = now.withDayOfMonth(25);
return lineMapper.searchCity(a1, a2);
// LocalDate now = LocalDate.now();
// if (now.getDayOfMonth() < 26) {
// LocalDate a1 = now.minusMonths(2).withDayOfMonth(26);
// LocalDate a2 = now.minusMonths(1).withDayOfMonth(25);
// return lineMapper.searchCity(a1, a2);
// } else {
// LocalDate a1 = now.minusMonths(1).withDayOfMonth(26);
// LocalDate a2 = now.withDayOfMonth(25);
// return lineMapper.searchCity(a1, a2);
// }
// 查询数据库最新时间
LocalDateTime dataTime = lineMapper.findCityTime();
if (dataTime == null) {
return null;
}
// 获取数据库最新时间月份的第一天
LocalDate firstDayOfMonth = dataTime.with(TemporalAdjusters.firstDayOfMonth()).toLocalDate();
// 获取数据库最新时间月份的最后一天
LocalDate lastDayOfMonth = dataTime.with(TemporalAdjusters.lastDayOfMonth()).toLocalDate();
return lineMapper.searchCity(firstDayOfMonth, lastDayOfMonth);
}
/**
@ -116,16 +126,26 @@ public class LineServiceImpl implements LineService {
*/
@Override
public SearchCityCountResponse searchCityCount() {
LocalDate now = LocalDate.now();
if (now.getDayOfMonth() < 26) {
LocalDate a1 = now.minusMonths(2).withDayOfMonth(26);
LocalDate a2 = now.minusMonths(1).withDayOfMonth(25);
return lineMapper.searchCityCount(a1, a2);
} else {
LocalDate a1 = now.minusMonths(1).withDayOfMonth(26);
LocalDate a2 = now.withDayOfMonth(25);
return lineMapper.searchCityCount(a1, a2);
// LocalDate now = LocalDate.now();
// if (now.getDayOfMonth() < 26) {
// LocalDate a1 = now.minusMonths(2).withDayOfMonth(26);
// LocalDate a2 = now.minusMonths(1).withDayOfMonth(25);
// return lineMapper.searchCityCount(a1, a2);
// } else {
// LocalDate a1 = now.minusMonths(1).withDayOfMonth(26);
// LocalDate a2 = now.withDayOfMonth(25);
// return lineMapper.searchCityCount(a1, a2);
// }
// 查询数据库最新时间
LocalDateTime dataTime = lineMapper.findCityTime();
if (dataTime == null) {
return null;
}
// 获取数据库最新时间月份的第一天
LocalDate firstDayOfMonth = dataTime.with(TemporalAdjusters.firstDayOfMonth()).toLocalDate();
// 获取数据库最新时间月份的最后一天
LocalDate lastDayOfMonth = dataTime.with(TemporalAdjusters.lastDayOfMonth()).toLocalDate();
return lineMapper.searchCityCount(firstDayOfMonth, lastDayOfMonth);
}
/**

@ -1,7 +1,9 @@
package com.ykMap.service.impl;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.ykMap.entity.TaskMission;
import com.ykMap.entity.response.CarPageResponse;
import com.ykMap.mapper.TaskMissionMapper;
import com.ykMap.service.TaskMissionService;
import org.springframework.stereotype.Service;
@ -44,5 +46,17 @@ public class TaskMissionServiceImpl extends ServiceImpl<TaskMissionMapper, TaskM
}
}
/**
*
*
* @param page
* @param plateNums
* @return
*/
@Override
public Page<CarPageResponse> pageByPlateNums(Page<CarPageResponse> page, List<String> plateNums) {
return baseMapper.pageByPlateNums(page,plateNums);
}
}

@ -0,0 +1,30 @@
package com.ykMap.utils;
import org.gavaghan.geodesy.Ellipsoid;
import org.gavaghan.geodesy.GeodeticCalculator;
import org.gavaghan.geodesy.GeodeticCurve;
import org.gavaghan.geodesy.GlobalCoordinates;
/**
*
*
* @author wu
* @since 2025/3/14 15:12
*/
public class SpaceUtils {
/**
* ()
*
* @param point1 1
* @param point2 2
* @return ()
*/
public static double getDistanceMeter(GlobalCoordinates point1, GlobalCoordinates point2) {
GeodeticCalculator calc = new GeodeticCalculator();
Ellipsoid ellipsoid = Ellipsoid.WGS84; // 使用 WGS84 椭球模型
GeodeticCurve geoCurve = calc.calculateGeodeticCurve(ellipsoid, point1, point2);
return geoCurve.getEllipsoidalDistance();
}
}

@ -75,4 +75,13 @@
limit 1
</select>
<select id="findCityTime" resultType="java.time.LocalDateTime">
SELECT
data_time
FROM
ysk_ywbmwjdr.operation_expansion_data
order by data_time desc
limit 1
</select>
</mapper>

@ -32,4 +32,24 @@
</foreach>
ORDER BY tm.update_time DESC;
</select>
<select id="pageByPlateNums" resultType="com.ykMap.entity.response.CarPageResponse">
SELECT COALESCE(tm.name, '-') as taskTitle,vi.plate_num as carNumber
FROM vehicle_info vi
INNER JOIN (
SELECT original_vehicle_id, MAX(update_time) AS latest_update_time
FROM task_mission
WHERE update_time IS NOT NULL
GROUP BY original_vehicle_id
) AS latest_tasks ON vi.id = latest_tasks.original_vehicle_id
INNER JOIN task_mission tm ON latest_tasks.original_vehicle_id = tm.original_vehicle_id
AND latest_tasks.latest_update_time = tm.update_time
WHERE vi.plate_num IN
<foreach item="plateNum" collection="list" open="(" separator="," close=")">
#{plateNum}
</foreach>
ORDER BY tm.update_time DESC
</select>
</mapper>

Loading…
Cancel
Save