Просмотр исходного кода

feat:app im发课、标签、欢迎语等

caoliqin 3 недель назад
Родитель
Сommit
e43a3e13c4
95 измененных файлов с 5743 добавлено и 19 удалено
  1. 22 0
      fs-admin/src/main/java/com/fs/app/controller/AppArticleController.java
  2. 23 0
      fs-admin/src/main/java/com/fs/app/controller/AppCivilianGoodsController.java
  3. 87 0
      fs-admin/src/main/java/com/fs/app/controller/AppCompanyUserDeptController.java
  4. 45 0
      fs-admin/src/main/java/com/fs/app/controller/AppCourseController.java
  5. 88 0
      fs-admin/src/main/java/com/fs/app/controller/AppCourseFinishTempController.java
  6. 30 0
      fs-admin/src/main/java/com/fs/app/controller/AppCoursePlaySourceConfigController.java
  7. 109 0
      fs-admin/src/main/java/com/fs/app/controller/AppCustomerRoleController.java
  8. 40 0
      fs-admin/src/main/java/com/fs/app/controller/AppGenerateController.java
  9. 12 0
      fs-admin/src/main/java/com/fs/app/controller/AppGroupController.java
  10. 75 0
      fs-admin/src/main/java/com/fs/app/controller/AppInvitationCodeController.java
  11. 33 0
      fs-admin/src/main/java/com/fs/app/controller/AppLiveController.java
  12. 23 0
      fs-admin/src/main/java/com/fs/app/controller/AppMedicinesController.java
  13. 22 0
      fs-admin/src/main/java/com/fs/app/controller/AppOpenClassVideoController.java
  14. 22 0
      fs-admin/src/main/java/com/fs/app/controller/AppShortVideoController.java
  15. 122 0
      fs-admin/src/main/java/com/fs/app/controller/AppSopController.java
  16. 62 0
      fs-admin/src/main/java/com/fs/app/controller/AppSopLogsController.java
  17. 154 0
      fs-admin/src/main/java/com/fs/app/controller/AppSopTempController.java
  18. 62 0
      fs-admin/src/main/java/com/fs/app/controller/AppSopUserLogController.java
  19. 47 0
      fs-admin/src/main/java/com/fs/app/controller/AppSopUserLogInfoController.java
  20. 74 0
      fs-admin/src/main/java/com/fs/app/controller/AppUrgentClassTaskController.java
  21. 103 0
      fs-admin/src/main/java/com/fs/app/controller/AppUserChatLogsController.java
  22. 80 0
      fs-admin/src/main/java/com/fs/app/controller/AppUserController.java
  23. 144 0
      fs-admin/src/main/java/com/fs/app/controller/AppUserPortraitController.java
  24. 93 0
      fs-admin/src/main/java/com/fs/app/controller/AppWelcomeController.java
  25. 186 0
      fs-admin/src/main/java/com/fs/app/controller/CommonV2Controller.java
  26. 36 0
      fs-admin/src/main/java/com/fs/app/controller/FsUserInfoController.java
  27. 34 0
      fs-admin/src/main/java/com/fs/app/controller/VipTagController.java
  28. 98 0
      fs-admin/src/main/java/com/fs/app/controller/VipTagGroupController.java
  29. 79 0
      fs-admin/src/main/java/com/fs/app/controller/VipUserCustomerController.java
  30. 78 0
      fs-admin/src/main/java/com/fs/app/controller/VipUserTagController.java
  31. 116 0
      fs-admin/src/main/java/com/fs/app/job/UserActivityTask.java
  32. 97 0
      fs-admin/src/main/java/com/fs/app/job/UserConsumptionAmountTask.java
  33. 24 0
      fs-admin/src/main/java/com/fs/app/job/UserTagTask.java
  34. 201 0
      fs-admin/src/main/java/com/fs/app/job/VideoTrackTask.java
  35. 6 0
      fs-admin/src/main/java/com/fs/company/controller/CompanyUserController.java
  36. 514 0
      fs-admin/src/main/java/com/fs/course/controller/FsAppCourseWatchLogController.java
  37. 5 0
      fs-admin/src/main/java/com/fs/his/controller/FsPackageController.java
  38. 308 0
      fs-admin/src/main/java/com/fs/utils/AudioUtils.java
  39. 136 0
      fs-admin/src/main/java/com/fs/web/controller/system/AppDeptController.java
  40. 39 0
      fs-common/src/main/java/com/fs/common/constant/VideoRedisKeyConst.java
  41. 198 0
      fs-common/src/main/java/com/fs/common/core/domain/entity/AppDept.java
  42. 3 1
      fs-common/src/main/java/com/fs/common/enums/DataSourceType.java
  43. 4 2
      fs-qw-api-msg/src/main/java/com/fs/app/msgarchives/job/QwMsgAuditScheduleJob.java
  44. 267 0
      fs-service/src/main/java/com/fs/app/civilgoods/domain/CivilGoods.java
  45. 17 0
      fs-service/src/main/java/com/fs/app/civilgoods/dto/CivilGoodsDTO.java
  46. 11 0
      fs-service/src/main/java/com/fs/app/civilgoods/mapper/CivilGoodsMapper.java
  47. 16 0
      fs-service/src/main/java/com/fs/app/civilgoods/service/ICivilGoodsService.java
  48. 69 0
      fs-service/src/main/java/com/fs/app/civilgoods/service/impl/CivilGoodsServiceImpl.java
  49. 10 0
      fs-service/src/main/java/com/fs/app/civilgoods/vo/CivilGoodsVO.java
  50. 252 0
      fs-service/src/main/java/com/fs/app/medicines/domain/AppFsStoreProduct.java
  51. 17 0
      fs-service/src/main/java/com/fs/app/medicines/dto/AppFsStoreProductDTO.java
  52. 9 0
      fs-service/src/main/java/com/fs/app/medicines/mapper/AppFsStoreProductMapper.java
  53. 16 0
      fs-service/src/main/java/com/fs/app/medicines/service/IAppFsStoreProductService.java
  54. 69 0
      fs-service/src/main/java/com/fs/app/medicines/service/impl/AppFsStoreProductServiceImpl.java
  55. 10 0
      fs-service/src/main/java/com/fs/app/medicines/vo/AppFsStoreProductVO.java
  56. 1 1
      fs-service/src/main/java/com/fs/app/sender/properties/ConfigProperties.java
  57. 4 0
      fs-service/src/main/java/com/fs/app/sop/dto/AppSopUserLogInfoDTO.java
  58. 2 0
      fs-service/src/main/java/com/fs/app/sop/mapper/AppSopUserLogInfoMapper.java
  59. 2 1
      fs-service/src/main/java/com/fs/app/sop/service/IAppSopUserLogInfoService.java
  60. 17 6
      fs-service/src/main/java/com/fs/app/sop/service/impl/AppSopUserLogInfoServiceImpl.java
  61. 3 0
      fs-service/src/main/java/com/fs/company/mapper/CompanyUserMapper.java
  62. 2 0
      fs-service/src/main/java/com/fs/company/service/ICompanyUserService.java
  63. 16 0
      fs-service/src/main/java/com/fs/company/service/impl/CompanyUserServiceImpl.java
  64. 15 0
      fs-service/src/main/java/com/fs/course/dto/VideoUpdateDTO.java
  65. 2 0
      fs-service/src/main/java/com/fs/course/mapper/FsUserCourseVideoMapper.java
  66. 17 0
      fs-service/src/main/java/com/fs/course/mapper/FsUserVideoMapper.java
  67. 2 0
      fs-service/src/main/java/com/fs/course/service/IFsUserCourseVideoService.java
  68. 4 0
      fs-service/src/main/java/com/fs/course/service/IFsUserVideoService.java
  69. 6 0
      fs-service/src/main/java/com/fs/course/service/impl/FsUserCourseVideoServiceImpl.java
  70. 11 0
      fs-service/src/main/java/com/fs/course/service/impl/FsUserVideoServiceImpl.java
  71. 3 0
      fs-service/src/main/java/com/fs/his/mapper/FsArticleMapper.java
  72. 2 0
      fs-service/src/main/java/com/fs/his/mapper/FsPackageMapper.java
  73. 14 0
      fs-service/src/main/java/com/fs/his/mapper/FsStoreOrderMapper.java
  74. 1 1
      fs-service/src/main/java/com/fs/his/mapper/FsUserIntegralLogsMapper.java
  75. 2 0
      fs-service/src/main/java/com/fs/his/service/IFsArticleService.java
  76. 2 0
      fs-service/src/main/java/com/fs/his/service/IFsPackageService.java
  77. 4 0
      fs-service/src/main/java/com/fs/his/service/IFsStoreOrderService.java
  78. 6 0
      fs-service/src/main/java/com/fs/his/service/impl/FsArticleServiceImpl.java
  79. 5 0
      fs-service/src/main/java/com/fs/his/service/impl/FsPackageServiceImpl.java
  80. 6 0
      fs-service/src/main/java/com/fs/his/service/impl/FsStoreOrderServiceImpl.java
  81. 91 0
      fs-service/src/main/java/com/fs/qw/mapper/QwWatchLogMapper.java
  82. 6 0
      fs-service/src/main/java/com/fs/qw/param/QwWatchLogStatisticsListParam.java
  83. 9 0
      fs-service/src/main/java/com/fs/qw/service/IQwWatchLogService.java
  84. 106 0
      fs-service/src/main/java/com/fs/qw/service/impl/QwWatchLogServiceImpl.java
  85. 116 0
      fs-service/src/main/java/com/fs/system/mapper/AppDeptMapper.java
  86. 114 0
      fs-service/src/main/java/com/fs/system/service/IAppDeptService.java
  87. 295 0
      fs-service/src/main/java/com/fs/system/service/impl/AppDeptServiceImpl.java
  88. 2 2
      fs-service/src/main/resources/application-config-druid-hst.yml
  89. 54 0
      fs-service/src/main/resources/mapper/app/AppSopUserLogInfoMapper.xml
  90. 19 0
      fs-service/src/main/resources/mapper/company/CompanyUserMapper.xml
  91. 24 0
      fs-service/src/main/resources/mapper/course/FsUserCourseVideoMapper.xml
  92. 24 0
      fs-service/src/main/resources/mapper/course/FsUserVideoMapper.xml
  93. 29 5
      fs-service/src/main/resources/mapper/his/FsArticleMapper.xml
  94. 28 0
      fs-service/src/main/resources/mapper/his/FsPackageMapper.xml
  95. 180 0
      fs-service/src/main/resources/mapper/system/AppDeptMapper.xml

+ 22 - 0
fs-admin/src/main/java/com/fs/app/controller/AppArticleController.java

@@ -0,0 +1,22 @@
+package com.fs.app.controller;
+
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.his.service.IFsArticleService;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/article")
+@RequiredArgsConstructor
+public class AppArticleController {
+
+    private final IFsArticleService fsArticleService;
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(String keyword, Long metaId, Long limit) {
+        return AjaxResult.success(fsArticleService.findOptions(keyword, metaId, limit));
+    }
+
+}

+ 23 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCivilianGoodsController.java

@@ -0,0 +1,23 @@
+package com.fs.app.controller;
+
+import com.fs.app.civilgoods.dto.CivilGoodsDTO;
+import com.fs.app.civilgoods.service.ICivilGoodsService;
+import com.fs.common.core.domain.AjaxResult;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/civilianGoods")
+@RequiredArgsConstructor
+public class AppCivilianGoodsController {
+
+    private final ICivilGoodsService appFsStoreProductScrmService;
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(CivilGoodsDTO req) {
+        return AjaxResult.success(appFsStoreProductScrmService.findOptions(req));
+    }
+
+}

+ 87 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCompanyUserDeptController.java

@@ -0,0 +1,87 @@
+package com.fs.app.controller;
+
+import com.fs.app.comuserdept.domain.AppCompanyUserDept;
+import com.fs.app.comuserdept.dto.AppCompanyUserDeptDTO;
+import com.fs.app.comuserdept.service.IAppCompanyUserDeptService;
+import com.fs.app.comuserdept.vo.AppCompanyUserDeptVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/comuserdept")
+public class AppCompanyUserDeptController extends BaseController {
+
+    @Autowired
+    private IAppCompanyUserDeptService appCompanyUserDeptService;
+
+    /**
+     * 查询app-销售/客服绑定部门(与销售原部门无关联)列表
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:list')")
+    @GetMapping("/list")
+    public TableDataInfo list(AppCompanyUserDeptDTO appCompanyUserDept) {
+        startPage();
+        List<AppCompanyUserDeptVO> list = appCompanyUserDeptService.selectAppCompanyUserDeptList(appCompanyUserDept);
+        return getDataTable(list);
+    }
+
+    /**
+     * 导出app-销售/客服绑定部门(与销售原部门无关联)列表
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:export')")
+    @Log(title = "app-销售/客服绑定部门(与销售原部门无关联)", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppCompanyUserDeptDTO appCompanyUserDept) {
+        List<AppCompanyUserDeptVO> list = appCompanyUserDeptService.selectAppCompanyUserDeptList(appCompanyUserDept);
+        ExcelUtil<AppCompanyUserDeptVO> util = new ExcelUtil<>(AppCompanyUserDeptVO.class);
+        return util.exportExcel(list, "app-销售/客服绑定部门(与销售原部门无关联)数据");
+    }
+
+    /**
+     * 获取app-销售/客服绑定部门(与销售原部门无关联)详细信息
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:list')")
+    @GetMapping(value = "/getById/{id}")
+    public AjaxResult getById(@PathVariable Long id) {
+        return AjaxResult.success(appCompanyUserDeptService.selectAppCompanyUserDeptById(id));
+    }
+
+    /**
+     * 新增app-销售/客服绑定部门(与销售原部门无关联)
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:add')")
+    @Log(title = "app-销售/客服绑定部门(与销售原部门无关联)", businessType = BusinessType.INSERT)
+    @PostMapping("/add")
+    public AjaxResult add(@RequestBody AppCompanyUserDept appCompanyUserDept) {
+        return toAjax(appCompanyUserDeptService.insertAppCompanyUserDept(appCompanyUserDept));
+    }
+
+    /**
+     * 修改app-销售/客服绑定部门(与销售原部门无关联)
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:edit')")
+    @Log(title = "app-销售/客服绑定部门(与销售原部门无关联)", businessType = BusinessType.UPDATE)
+    @PostMapping("/edit")
+    public AjaxResult edit(@RequestBody AppCompanyUserDept appCompanyUserDept) {
+        return toAjax(appCompanyUserDeptService.updateAppCompanyUserDept(appCompanyUserDept));
+    }
+
+    /**
+     * 删除app-销售/客服绑定部门(与销售原部门无关联)
+     */
+    @PreAuthorize("@ss.hasPermi('app:comuserdept:delete')")
+    @Log(title = "app-销售/客服绑定部门(与销售原部门无关联)", businessType = BusinessType.DELETE)
+    @DeleteMapping("/deleteByIds/{ids}")
+    public AjaxResult remove(@PathVariable Long[] ids) {
+        return toAjax(appCompanyUserDeptService.deleteAppCompanyUserDeptByIds(ids));
+    }
+}

+ 45 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCourseController.java

@@ -0,0 +1,45 @@
+package com.fs.app.controller;
+
+import com.fs.app.course.service.IAppFsUserCourseService;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.R;
+import com.fs.his.vo.OptionsVO;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/course")
+@RequiredArgsConstructor
+public class AppCourseController extends BaseController {
+
+    private final IAppFsUserCourseService appFsUserCourseService;
+
+    /**
+     * 获取课程选项
+     *
+     * @param keyword
+     * @return
+     */
+    @GetMapping("/courseList")
+    public R courseList(String keyword, String selectedId) {
+        startPage();
+        return R.ok().put("list", appFsUserCourseService.findOptions(keyword, selectedId));
+    }
+
+    /**
+     * 获取指定课程的所有视频列表
+     * @param courseId
+     * @param title
+     * @return
+     */
+    @GetMapping(value = "/videoList")
+    public R videoList(Long courseId, String title) {
+        List<OptionsVO> optionsVOS = appFsUserCourseService.findVideoOptionsByCourse(courseId, title);
+        return R.ok().put("list", optionsVOS);
+    }
+
+}

+ 88 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCourseFinishTempController.java

@@ -0,0 +1,88 @@
+package com.fs.app.controller;
+
+import com.fs.app.course.dto.AppCourseFinishTempDTO;
+import com.fs.app.course.service.IAppCourseFinishTempService;
+import com.fs.app.course.vo.AppCourseFinishTempVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/courseFinishTemp")
+public class AppCourseFinishTempController extends BaseController {
+
+    @Autowired
+    private IAppCourseFinishTempService appCourseFinishTempService;
+
+    /**
+     * 查询app-完课模板列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:list')")
+    @GetMapping("/findList")
+    public TableDataInfo list(AppCourseFinishTempDTO appCourseFinishTemp) {
+        startPage();
+        List<AppCourseFinishTempVO> list = appCourseFinishTempService.selectAppCourseFinishTempList(appCourseFinishTemp);
+        return getDataTable(list);
+    }
+
+    /**
+     * 导出app-完课模板列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:export')")
+    @Log(title = "app-完课模板", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppCourseFinishTempDTO appCourseFinishTemp) {
+        List<AppCourseFinishTempVO> list = appCourseFinishTempService.selectAppCourseFinishTempList(appCourseFinishTemp);
+        ExcelUtil<AppCourseFinishTempVO> util = new ExcelUtil<>(AppCourseFinishTempVO.class);
+        return util.exportExcel(list, "app-完课模板数据");
+    }
+
+    /**
+     * 获取app-完课模板详细信息
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:list')")
+    @Log(title = "app-完课模板", businessType = BusinessType.OTHER)
+    @GetMapping(value = "/getById/{id}")
+    public AjaxResult getInfo(@PathVariable Long id) {
+        return AjaxResult.success(appCourseFinishTempService.selectAppCourseFinishTempById(id));
+    }
+
+    /**
+     * 新增app-完课模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:add')")
+    @Log(title = "app-完课模板", businessType = BusinessType.INSERT)
+    @PostMapping("/add")
+    public AjaxResult add(@RequestBody AppCourseFinishTempDTO appCourseFinishTemp) {
+        return toAjax(appCourseFinishTempService.insertAppCourseFinishTemp(appCourseFinishTemp));
+    }
+
+    /**
+     * 修改app-完课模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:edit')")
+    @Log(title = "app-完课模板", businessType = BusinessType.UPDATE)
+    @PostMapping("/edit")
+    public AjaxResult edit(@RequestBody AppCourseFinishTempDTO appCourseFinishTemp) {
+        return toAjax(appCourseFinishTempService.updateAppCourseFinishTemp(appCourseFinishTemp));
+    }
+
+    /**
+     * 删除app-完课模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:finishTemp:delete')")
+    @Log(title = "app-完课模板", businessType = BusinessType.DELETE)
+    @DeleteMapping("/deleteByIds/{ids}")
+    public AjaxResult remove(@PathVariable List<Long> ids) {
+        return toAjax(appCourseFinishTempService.deleteAppCourseFinishTempByIds(ids));
+    }
+
+}

+ 30 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCoursePlaySourceConfigController.java

@@ -0,0 +1,30 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.domain.AppFsCoursePlaySourceConfig;
+import com.fs.app.sop.service.IAppCoursePlaySourceConfigService;
+import com.fs.app.sop.vo.AppCoursePlaySourceConfigVO;
+import com.fs.common.BeanCopyUtils;
+import com.fs.common.core.domain.R;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/cpsc")
+@RequiredArgsConstructor
+public class AppCoursePlaySourceConfigController {
+
+    private final IAppCoursePlaySourceConfigService appCoursePlaySourceConfigService;
+
+    @GetMapping(value = "/getList")
+    public R getCompanyInfo() {
+        //查询小程序名称
+        List<AppFsCoursePlaySourceConfig> cpsc = appCoursePlaySourceConfigService.lambdaQuery().list();
+        List<AppCoursePlaySourceConfigVO> vos = BeanCopyUtils.copyList(cpsc, AppCoursePlaySourceConfigVO.class);
+        return R.ok().put("data", vos);
+    }
+
+}

+ 109 - 0
fs-admin/src/main/java/com/fs/app/controller/AppCustomerRoleController.java

@@ -0,0 +1,109 @@
+package com.fs.app.controller;
+
+import cn.hutool.http.HttpRequest;
+import com.fs.app.cusrole.dto.AppCustomerRoleDTO;
+import com.fs.app.cusrole.service.IAppCustomerRoleService;
+import com.fs.app.cusrole.vo.AppCustomerRoleVO;
+import com.fs.app.user.vo.AppUserVO;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.utils.StringUtils;
+import com.fs.im.service.OpenIMService;
+import io.swagger.annotations.ApiOperation;
+import lombok.RequiredArgsConstructor;
+import org.json.JSONArray;
+import org.json.JSONObject;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+@RestController
+@RequestMapping("/app/cusrole")
+@RequiredArgsConstructor
+public class AppCustomerRoleController extends BaseController {
+
+    private final IAppCustomerRoleService appCustomerRoleService;
+
+    @Autowired
+    private OpenIMService openIMService;
+
+    @GetMapping("/list")
+    public TableDataInfo list(AppCustomerRoleDTO dto) {
+        startPage();
+        List<AppCustomerRoleVO> list = appCustomerRoleService.findList(dto);
+        return getDataTable(list);
+    }
+
+    @ApiOperation("校验客服是否注册新的im")
+    @PostMapping("/accountCheck")
+    public R accountCheck(@RequestBody Map<String, String> userIdMap){
+        //获取管理员token
+        String userId = userIdMap.get("userId");
+        String adminToken = openIMService.getAdminToken();
+        JSONObject requestBody = new JSONObject();
+        // 解析响应
+        if (StringUtils.isNotEmpty(adminToken)) {
+            //查询用户是否注册
+            ArrayList<String> userIds = new ArrayList<>();
+            requestBody = new JSONObject();
+            userIds.add(userId);
+            requestBody.put("checkUserIDs", userIds);
+            String body = HttpRequest.post("https://web.im.cdwjyyh.com/api/user/account_check")
+                    .header("operationID", String.valueOf(System.currentTimeMillis()))
+                    .header("token", adminToken)
+                    .body(requestBody.toString())
+                    .execute()
+                    .body();
+            JSONObject jsonObject = new JSONObject(body);
+            JSONArray results = jsonObject.getJSONObject("data").getJSONArray("results");
+            if (results != null && results.length() > 0) {
+                JSONObject resultObj = results.getJSONObject(0);
+                int accountStatus = resultObj.getInt("accountStatus");
+                //未注册自动注册
+                if (accountStatus==0){
+                    String s = userId.replaceFirst("^C", "");
+                    AppUserVO appUserVO = appCustomerRoleService.selectById(Long.parseLong(s));
+                    if (null==appUserVO){
+                        return R.error("用户不存在");
+                    }
+                    ArrayList<Object> users = new ArrayList<>();
+                    HashMap<String, String> map = new HashMap<>();
+                    map.put("userID",userId);
+                    map.put("nickname",appUserVO.getUsername());
+                    map.put("faceURL",appUserVO.getAvatar());
+                    users.add(map);
+                    requestBody = new JSONObject();
+                    userIds.add(userId);
+                    requestBody.put("users", users);
+                    HttpRequest.post("https://web.im.cdwjyyh.com/api/user/user_register")
+                            .header("operationID", String.valueOf(System.currentTimeMillis()))
+                            .header("token", adminToken).body(requestBody.toString()).execute().body();
+                }
+            } else {
+                return R.error("返回结果为空");
+            }
+           /* HashMap<String, String> tokenMap = new HashMap<>();
+            tokenMap.put("platformID","1");
+            tokenMap.put("userID",userId);*/
+            requestBody = new JSONObject();
+            requestBody.put("platformID",5);
+            requestBody.put("userID",userId);
+            String body1 = HttpRequest.post("https://web.im.cdwjyyh.com/api/auth/get_user_token")
+                    .header("operationID", String.valueOf(System.currentTimeMillis()))
+                    .header("token", adminToken)
+                    .body(requestBody.toString()).execute().body();
+            JSONObject userJson = new JSONObject(body1);
+            JSONObject userData = userJson.getJSONObject("data");
+            String userToken = userData.getString("token");
+            return R.ok().put("token", userToken);
+        } else {
+            return R.error("获取管理员token失败");
+        }
+    }
+
+}

+ 40 - 0
fs-admin/src/main/java/com/fs/app/controller/AppGenerateController.java

@@ -0,0 +1,40 @@
+package com.fs.app.controller;
+
+import com.fs.app.cusrole.mapper.AppCustomerRoleMapper;
+import com.fs.app.sop.service.impl.AppSopUserLogsInfoHandle;
+import com.fs.common.core.domain.R;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.PathVariable;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/generate")
+@RequiredArgsConstructor
+public class AppGenerateController {
+
+    private final AppCustomerRoleMapper appCustomerRoleMapper;
+
+    private final AppSopUserLogsInfoHandle appSopUserLogsInfoHandle;
+
+    /**
+     * 加入或创建营期
+     *
+     * @param userIds
+     * @return
+     */
+    @GetMapping("/joinOrCreateUserLogs/{userIds}")
+    public R joinOrCreateUserLogs(@PathVariable List<Long> userIds) {
+        for (Long userId : userIds) {
+            List<Long> customerIds = appCustomerRoleMapper.getCustomerIdsByUserId(userId);
+            customerIds.forEach(customerId -> {
+                this.appSopUserLogsInfoHandle.joinOrCreateSopUserLogsHandle(customerId, userId);
+            });
+        }
+        return R.ok();
+    }
+
+}

+ 12 - 0
fs-admin/src/main/java/com/fs/app/controller/AppGroupController.java

@@ -0,0 +1,12 @@
+package com.fs.app.controller;
+
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/groupChat")
+@RequiredArgsConstructor
+public class AppGroupController {
+
+}

+ 75 - 0
fs-admin/src/main/java/com/fs/app/controller/AppInvitationCodeController.java

@@ -0,0 +1,75 @@
+package com.fs.app.controller;
+
+import com.fs.app.invitation.dto.AppInvitationCodeDTO;
+import com.fs.app.invitation.service.IAppInvitationCodeService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ * app-邀请码管理
+ */
+@RestController
+@RequestMapping("/app/invCode")
+@RequiredArgsConstructor
+public class AppInvitationCodeController extends BaseController {
+
+    private final IAppInvitationCodeService appInvitationCodeService;
+
+    /**
+     * 列表查询
+     * @param reqData
+     * @return
+     */
+    @GetMapping("/findList")
+    @PreAuthorize("@ss.hasAnyPermi('app:invite:list')")
+    public TableDataInfo findList(AppInvitationCodeDTO reqData) {
+        startPage();
+        return getDataTable(appInvitationCodeService.findList(reqData));
+    }
+
+    /**
+     * 新增编辑保存
+     * @param reqData
+     * @return
+     */
+    @PostMapping("/save")
+    @PreAuthorize("@ss.hasAnyPermi('app:invite:add,app:invite:edit')")
+    @Log(title = "邀请码", businessType = BusinessType.UPDATE)
+    public R saveAddOrEdit(@RequestBody AppInvitationCodeDTO reqData) {
+        return appInvitationCodeService.saveAddOrEdit(reqData);
+    }
+
+    /**
+     * 根据id获取指定记录详情
+     * @param reqData
+     * @return
+     */
+    @GetMapping("/getById")
+    @PreAuthorize("@ss.hasAnyPermi('app:invite:list')")
+    public AjaxResult getById(AppInvitationCodeDTO reqData) {
+        return AjaxResult.success(appInvitationCodeService.getById(reqData));
+    }
+
+    /**
+     * 邀请码删除
+     * @param ids
+     * @return
+     */
+    @DeleteMapping("/deleteByIds/{ids}")
+    @PreAuthorize("@ss.hasAnyPermi('app:invite:delete')")
+    @Log(title = "邀请码", businessType = BusinessType.DELETE)
+    public R deleteByIds(@PathVariable List<Long> ids) {
+        this.appInvitationCodeService.deleteByIds(ids);
+        return R.ok();
+    }
+
+}

+ 33 - 0
fs-admin/src/main/java/com/fs/app/controller/AppLiveController.java

@@ -0,0 +1,33 @@
+package com.fs.app.controller;
+
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.live.domain.Live;
+import com.fs.live.service.ILiveService;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/live")
+@RequiredArgsConstructor
+public class AppLiveController extends BaseController {
+
+    private final ILiveService liveService;
+
+    /**
+     * 查询直播列表
+     */
+//    @PreAuthorize("@ss.hasPermi('live:live:list')")
+    @GetMapping("/list")
+    public TableDataInfo list(Live live)
+    {
+        startPage();
+        List<Live> list = liveService.selectLiveList(live);
+        return getDataTable(list);
+    }
+
+}

+ 23 - 0
fs-admin/src/main/java/com/fs/app/controller/AppMedicinesController.java

@@ -0,0 +1,23 @@
+package com.fs.app.controller;
+
+import com.fs.app.medicines.dto.AppFsStoreProductDTO;
+import com.fs.app.medicines.service.IAppFsStoreProductService;
+import com.fs.common.core.domain.AjaxResult;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/medicines")
+@RequiredArgsConstructor
+public class AppMedicinesController {
+
+    private final IAppFsStoreProductService appFsStoreProductService;
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(AppFsStoreProductDTO req) {
+        return AjaxResult.success(appFsStoreProductService.findOptions(req));
+    }
+
+}

+ 22 - 0
fs-admin/src/main/java/com/fs/app/controller/AppOpenClassVideoController.java

@@ -0,0 +1,22 @@
+package com.fs.app.controller;
+
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.course.service.IFsUserCourseVideoService;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/openClassVideo")
+@RequiredArgsConstructor
+public class AppOpenClassVideoController {
+
+    private final IFsUserCourseVideoService fsUserCourseVideoService;
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(String keyword, Long metaId, Long limit) {
+        return AjaxResult.success(fsUserCourseVideoService.findOptions(keyword, metaId, limit));
+    }
+
+}

+ 22 - 0
fs-admin/src/main/java/com/fs/app/controller/AppShortVideoController.java

@@ -0,0 +1,22 @@
+package com.fs.app.controller;
+
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.course.service.IFsUserVideoService;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/app/shortVideo")
+@RequiredArgsConstructor
+public class AppShortVideoController {
+
+    private final IFsUserVideoService fsUserVideoService;
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(String keyword, Long metaId, Long limit) {
+        return AjaxResult.success(fsUserVideoService.findOptions(keyword, metaId, limit));
+    }
+
+}

+ 122 - 0
fs-admin/src/main/java/com/fs/app/controller/AppSopController.java

@@ -0,0 +1,122 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.dto.AppSopDTO;
+import com.fs.app.sop.service.IAppSopService;
+import com.fs.app.sop.vo.AppSopVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/sop")
+@RequiredArgsConstructor
+public class AppSopController extends BaseController {
+
+
+    private final IAppSopService appSopService;
+
+    /**
+     * 查询 sop 列表
+     */
+    @PreAuthorize("@ss.hasPermi('app:sop:list')")
+    @GetMapping("/list")
+    public TableDataInfo list(AppSopDTO sopDTO) {
+        startPage();
+        List<AppSopVO> list = appSopService.findList(sopDTO);
+        return getDataTable(list);
+    }
+
+    /**
+     * 新增 sop
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:add')")
+    @Log(title = "appSop", businessType = BusinessType.INSERT)
+    @PostMapping("/add")
+    public AjaxResult add(@RequestBody AppSopDTO sop) {
+        int count = appSopService.add(sop);
+//        if (count > 0) {
+//            if (qwSop.getQwUserIds() != null) {
+//                updateTempVoiceInfo(qwSop);
+//            }
+//        }
+        return toAjax(count);
+    }
+
+    /**
+     * 修改 sop
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:edit')")
+    @Log(title = "修改appSop", businessType = BusinessType.UPDATE)
+    @PostMapping("/edit")
+    public R edit(@RequestBody AppSopDTO sopDTO) {
+        R sop = appSopService.edit(sopDTO);
+//        String code = sop.get("code").toString();
+//        if(code.equals("200")){
+//            if(qwSop != null && qwSop.getQwUserIds() != null){
+//                updateTempVoiceInfo(qwSop);
+//            }
+//        }
+        return sop;
+    }
+
+    /**
+     * 删除 sop
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:delete')")
+    @Log(title = "删除appSop", businessType = BusinessType.DELETE)
+    @DeleteMapping("/delete/{ids}")
+    public AjaxResult remove(@PathVariable List<String> ids) {
+        return toAjax(appSopService.deleteSopByIds(ids));
+    }
+
+    /**
+     * 根据id获取 sop详情
+     */
+    @PreAuthorize("@ss.hasPermi('app:sop:list')")
+    @GetMapping("/getById")
+    public AjaxResult getById(String id) {
+        return AjaxResult.success(appSopService.getById(id));
+    }
+
+    /**
+     * 修改 sop 自动创建时间
+     */
+    @Log(title = "appSop自动创建时间", businessType = BusinessType.UPDATE)
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:edit')")
+    @PostMapping("/updateAutoSopTime")
+    public AjaxResult updateAutoSopTime(@RequestBody AppSopDTO appSopReq) {
+        return toAjax(appSopService.updateAutoSopTime(appSopReq));
+    }
+
+    /**
+     * 更新执行状态
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:exec')")
+    @Log(title = "修改执行状态", businessType = BusinessType.UPDATE)
+    @GetMapping(value = "/updateStatus/{ids}")
+    public R updateStatus(@PathVariable List<String> ids) {
+        return appSopService.updateStatusByIds(ids);
+    }
+
+    /**
+     * 导出sop列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:sop:export')")
+    @Log(title = "sop", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppSopDTO sopDTO) {
+        List<AppSopVO> list = appSopService.findList(sopDTO);
+        ExcelUtil<AppSopVO> util = new ExcelUtil<>(AppSopVO.class);
+        return util.exportExcel(list, "自动化数据");
+    }
+
+}

+ 62 - 0
fs-admin/src/main/java/com/fs/app/controller/AppSopLogsController.java

@@ -0,0 +1,62 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.dto.AppSopLogDTO;
+import com.fs.app.sop.service.IAppSopLogsService;
+import com.fs.app.sop.vo.AppSopLogsVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/sopLogs")
+@RequiredArgsConstructor
+public class AppSopLogsController extends BaseController {
+
+    private final IAppSopLogsService appSopLogsService;
+
+    @GetMapping("/findList")
+    public TableDataInfo findList(AppSopLogDTO param) {
+        startPage();
+        return getDataTable(this.appSopLogsService.findList(param));
+    }
+
+    /**
+     * 删除SOP  定时任务
+     */
+//    @PreAuthorize("@ss.hasPermi('qw:sopLogs:remove')")
+    @Log(title = "SOP  定时任务", businessType = BusinessType.DELETE)
+    @DeleteMapping("/deleteByIds/{ids}")
+    public AjaxResult remove(@PathVariable Long[] ids) {
+        return toAjax(appSopLogsService.deleteByIds(ids));
+    }
+
+    /**
+     * 修改SOP  定时任务-只修改完课的补发
+     */
+//    @PreAuthorize("@ss.hasPermi('qw:sopLogs:editCourse')")
+    @Log(title = "修改SOP只修改完课", businessType = BusinessType.UPDATE)
+    @PutMapping("/editCourseSopLogs/{ids}")
+    public AjaxResult editCourseQwSopLogs(@PathVariable Long[] ids) {
+        return toAjax(appSopLogsService.editCourseSopLogs(ids));
+    }
+
+    /**
+     * 导出企业微信SOP  定时任务列表
+     */
+//    @PreAuthorize("@ss.hasPermi('qw:sopLogs:export')")
+    @Log(title = "SOP  定时任务", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppSopLogDTO param) {
+        List<AppSopLogsVO> list = appSopLogsService.findList(param);
+        ExcelUtil<AppSopLogsVO> util = new ExcelUtil<>(AppSopLogsVO.class);
+        return util.exportExcel(list, "SOP 定时任务数据");
+    }
+
+}

+ 154 - 0
fs-admin/src/main/java/com/fs/app/controller/AppSopTempController.java

@@ -0,0 +1,154 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.dto.AppSopTempDTO;
+import com.fs.app.sop.dto.AppSopTempDayDTO;
+import com.fs.app.sop.dto.AppSopTempDaySortDTO;
+import com.fs.app.sop.service.IAppSopTempService;
+import com.fs.app.sop.vo.AppSopTempVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ *
+ * @author hulin
+ */
+@RestController
+@RequestMapping("/app/sopTemp")
+@RequiredArgsConstructor
+public class AppSopTempController extends BaseController {
+
+    private final IAppSopTempService appSopTempService;
+
+    /**
+     * 查询 sop 模板列表
+     */
+    @GetMapping("/list")
+    @PreAuthorize("@ss.hasAnyPermi('app:template:list')")
+    public TableDataInfo list(AppSopTempDTO param) {
+        startPage();
+        List<AppSopTempVO> list = appSopTempService.findList(param);
+        return getDataTable(list);
+    }
+
+    /**
+     * 导出sop模板列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:template:export')")
+    @Log(title = "sop模板", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppSopTempDTO param) {
+        List<AppSopTempVO> list = appSopTempService.findList(param);
+        ExcelUtil<AppSopTempVO> util = new ExcelUtil(AppSopTempVO.class);
+        return util.exportExcel(list, "sop模板数据");
+    }
+
+    /**
+     * 获取 sop 模板详细信息
+     */
+    @GetMapping(value = "/{id}")
+    @PreAuthorize("@ss.hasAnyPermi('app:template:list')")
+    public AjaxResult getInfo(@PathVariable String id) {
+        return AjaxResult.success(appSopTempService.getOneById(id));
+    }
+
+    /**
+     * 新增 sop 模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:template:add')")
+    @Log(title = "sop模板", businessType = BusinessType.INSERT)
+    @PostMapping("/add")
+    public AjaxResult add(@RequestBody AppSopTempDTO param) {
+        return toAjax(appSopTempService.addSopTemp(param));
+    }
+
+    /**
+     * 修改 sop 模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:template:edit')")
+    @Log(title = "sop模板", businessType = BusinessType.UPDATE)
+    @PostMapping("/edit")
+    public AjaxResult edit(@RequestBody AppSopTempDTO param) {
+        return toAjax(appSopTempService.editSopTemp(param));
+    }
+
+    /**
+     * 删除 sop 模板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:template:delete,app:template:stop')")
+    @Log(title = "sop模板", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{ids}")
+    public AjaxResult remove(@PathVariable String[] ids) {
+        return toAjax(appSopTempService.deleteSopTemp(ids));
+    }
+
+
+    /**
+     * 模板复制
+     *
+     * @param param
+     * @return
+     */
+    @Log(title = "sop模板复制", businessType = BusinessType.INSERT)
+    @PreAuthorize("@ss.hasAnyPermi('app:template:copy')")
+    @PostMapping("/copy")
+    public AjaxResult copyTemplate(@RequestBody AppSopTempDTO param) {
+        appSopTempService.copyTemplate(param);
+        return toAjax(1);
+    }
+
+    /**
+     * 新增/编辑 保存模板天数
+     *
+     * @param day
+     * @return
+     */
+    @Log(title = "添加修改sop模板天数", businessType = BusinessType.UPDATE)
+    @PreAuthorize("@ss.hasAnyPermi('app:template:edit')")
+    @PostMapping("/addOrUpdateSetting")
+    public AjaxResult addOrUpdateSetting(@RequestBody AppSopTempDayDTO day) {
+        return AjaxResult.success(appSopTempService.addOrUpdateSetting(day));
+    }
+
+    /**
+     * 获取规则信息
+     *
+     * @param id
+     * @return
+     */
+    @GetMapping("/selectRulesInfo")
+    public AjaxResult selectRulesInfo(Long id) {
+        return AjaxResult.success(appSopTempService.selectRulesInfo(id));
+    }
+
+
+    @GetMapping("/dayList")
+    public AjaxResult dayList(String id) {
+        return AjaxResult.success(appSopTempService.dayList(id));
+    }
+
+    @Log(title = "排序天数", businessType = BusinessType.UPDATE)
+    @PostMapping("/sortDay")
+    public AjaxResult sortDay(@RequestBody List<AppSopTempDaySortDTO> list) {
+        appSopTempService.sortDay(list);
+        return toAjax(1);
+    }
+
+    // 更新模板图片
+    @Log(title = "更新模板图片", businessType = BusinessType.UPDATE)
+    @PreAuthorize("@ss.hasAnyPermi('app:template:updateImage')")
+    @GetMapping("/updateImage")
+    public AjaxResult updateImage(String id){
+        appSopTempService.updateImage(id);
+        return AjaxResult.success();
+    }
+
+}

+ 62 - 0
fs-admin/src/main/java/com/fs/app/controller/AppSopUserLogController.java

@@ -0,0 +1,62 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.dto.AppGroupSendMessageDTO;
+import com.fs.app.sop.dto.AppSopUserLogDTO;
+import com.fs.app.sop.service.IAppSopUserLogService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/userLogs")
+@RequiredArgsConstructor
+public class AppSopUserLogController extends BaseController {
+
+    private final IAppSopUserLogService appSopUserLogService;
+
+    @GetMapping("/list")
+    public TableDataInfo list(AppSopUserLogDTO dto) {
+        startPage();
+        return getDataTable(this.appSopUserLogService.findList(dto));
+    }
+
+    /**
+     * 删除sopUserLogs
+     */
+//    @PreAuthorize("@ss.hasPermi('qwSop:sopUserLogs:remove')")
+    @Log(title = "删除sopUserLogs", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{ids}")
+    public AjaxResult remove(@PathVariable List<String> ids) {
+        return toAjax(appSopUserLogService.deleteByIds(ids));
+    }
+
+
+    /**
+     * 修改营期时间
+     */
+//    @PreAuthorize("@ss.hasPermi('qwSop:sopUserLogs:updateTime')")
+    @Log(title = "批量修改 sopUserLogs", businessType = BusinessType.UPDATE)
+    @PostMapping("/batchUpdateSopUserLogsToTime")
+    public R updateTime(@RequestBody AppSopUserLogDTO dto) {
+        return appSopUserLogService.batchUpdateSopUserLogsToTime(dto);
+    }
+
+    /**
+     * 一键群发
+     * @param appGroupSendMessageDTO
+     * @return
+     */
+    @PostMapping("/groupSendMessage")
+    public R groupSendMessage(@RequestBody AppGroupSendMessageDTO appGroupSendMessageDTO) {
+        this.appSopUserLogService.groupSendMessage(appGroupSendMessageDTO);
+        return R.ok();
+    }
+
+}

+ 47 - 0
fs-admin/src/main/java/com/fs/app/controller/AppSopUserLogInfoController.java

@@ -0,0 +1,47 @@
+package com.fs.app.controller;
+
+import com.fs.app.sop.dto.AppSopUserLogInfoDTO;
+import com.fs.app.sop.service.IAppSopUserLogInfoService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+@RestController
+@RequestMapping("/app/userLogsInfo")
+@RequiredArgsConstructor
+public class AppSopUserLogInfoController {
+
+    private final IAppSopUserLogInfoService appSopUserLogInfoService;
+
+    @GetMapping("/list")
+    public TableDataInfo list(AppSopUserLogInfoDTO dto) {
+        return this.appSopUserLogInfoService.findList(dto);
+    }
+
+    /**
+     * 删除 sopUserLogsInfo
+     */
+//    @PreAuthorize("@ss.hasPermi('qw:sopUserLogsInfo:remove')")
+    @Log(title = "deleteByIds", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{ids}")
+    public AjaxResult remove(@PathVariable List<Long> ids) {
+        return appSopUserLogInfoService.deleteByIds(ids) > 0 ? AjaxResult.success() : AjaxResult.error();
+    }
+
+    /**
+     * 营期详情内,修改具体用户所属的营期
+     */
+//    @PreAuthorize("@ss.hasPermi('qw:sopUserLogsInfo:edit')")
+    @Log(title = "updateSopUserLogsInfo", businessType = BusinessType.UPDATE)
+    @PostMapping("/batchUpdateSopUserLogsInfoToTime")
+    public R edit(@RequestBody AppSopUserLogInfoDTO dto) {
+        return appSopUserLogInfoService.batchUpdateSopUserLogsInfoToTime(dto);
+    }
+
+}

+ 74 - 0
fs-admin/src/main/java/com/fs/app/controller/AppUrgentClassTaskController.java

@@ -0,0 +1,74 @@
+package com.fs.app.controller;
+
+import com.fs.app.task.dto.AppWorkTaskDTO;
+import com.fs.app.task.service.IAppWorkTaskService;
+import com.fs.app.task.vo.AppWorkTaskVO;
+import com.fs.app.watchlog.service.IAppCourseWatchLogService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.StringUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ * app-任务看板Controller
+ *
+ * @author fs
+ * @date 2026-03-03
+ */
+@RestController
+@RequestMapping("/app/urgentClass")
+public class AppUrgentClassTaskController extends BaseController {
+
+    @Autowired
+    private IAppWorkTaskService appWorkTaskService;
+
+    @Autowired
+    private IAppCourseWatchLogService appCourseWatchLogService;
+
+
+    /**
+     * 查询app-任务看板列表
+     */
+    @GetMapping("/list")
+    @PreAuthorize("@ss.hasAnyPermi('app:urgentClass:list')")
+    public TableDataInfo list(AppWorkTaskDTO appWorkTask) {
+        startPage();
+        List<AppWorkTaskVO> list = appWorkTaskService.selectAppWorkTaskList(appWorkTask);
+        for (AppWorkTaskVO appWorkTaskVO : list) {
+            appWorkTaskVO.setLatestCourseName(appCourseWatchLogService.recentWatchVideoName(appWorkTaskVO.getFsUserId(), appWorkTaskVO.getCustomerRoleId()));
+            appWorkTaskVO.setUserWatchLogs(this.appCourseWatchLogService.getAppUserWatchLogByDaysAndUserId(7, appWorkTaskVO.getFsUserId(), appWorkTaskVO.getCustomerRoleId()));
+        }
+        return getDataTable(list);
+    }
+
+    /**
+     * 修改app-任务看板
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:urgentClass:handle')")
+    @Log(title = "催课看板", businessType = BusinessType.UPDATE)
+    @PostMapping("/setStatus")
+    public AjaxResult setStatus(@RequestBody AppWorkTaskDTO appWorkTask) {
+        return toAjax(appWorkTaskService.setStatus(appWorkTask));
+    }
+
+    /**
+     * 查询app-任务看板用户电话
+     */
+    @GetMapping("/getFsUserPhone/{fsUserId}")
+    @PreAuthorize("@ss.hasAnyPermi('app:urgentClass:getFsUserPhone')")
+    public AjaxResult getFsUserPhone(@PathVariable Long fsUserId) {
+        String phone = appWorkTaskService.getFsUserPhone(fsUserId);
+        if(StringUtils.isNotBlank(phone)){
+            return AjaxResult.success("查询成功!",phone);
+        }else{
+            return AjaxResult.error("用户电话不存在");
+        }
+    }
+}

+ 103 - 0
fs-admin/src/main/java/com/fs/app/controller/AppUserChatLogsController.java

@@ -0,0 +1,103 @@
+package com.fs.app.controller;
+
+import com.fs.app.chat.domain.AppUserChatLogs;
+import com.fs.app.chat.dto.AppUserChatLogsDTO;
+import com.fs.app.chat.service.IAppUserChatLogsService;
+import com.fs.app.chat.vo.AppUserChatLogsVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ * app-客户与客服对话(问答)Controller
+ *
+ * @author fs
+ * @date 2026-04-07
+ */
+@RestController
+@RequestMapping("/app/userChatLogs")
+public class AppUserChatLogsController extends BaseController {
+
+    @Autowired
+    private IAppUserChatLogsService appUserChatLogsService;
+
+    /**
+     * 查询app-客户与客服对话(问答)列表
+     */
+//    @PreAuthorize("@ss.hasPermi('app:userChatLogs:list')")
+    @GetMapping("/findList")
+    public TableDataInfo findList(AppUserChatLogsDTO appUserChatLogs) {
+        startPage();
+        List<AppUserChatLogsVO> list = appUserChatLogsService.selectAppUserChatLogsList(appUserChatLogs);
+        return getDataTable(list);
+    }
+
+    /**
+     * 导出app-客户与客服对话(问答)列表
+     */
+    @PreAuthorize("@ss.hasPermi('app:userChatLogs:export')")
+    @Log(title = "app-客户与客服对话(问答)", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppUserChatLogsDTO appUserChatLogs) {
+        List<AppUserChatLogsVO> list = appUserChatLogsService.selectAppUserChatLogsList(appUserChatLogs);
+        ExcelUtil<AppUserChatLogsVO> util = new ExcelUtil<>(AppUserChatLogsVO.class);
+        return util.exportExcel(list, "app-客户与客服对话(问答)数据");
+    }
+
+    /**
+     * 获取app-客户与客服对话(问答)详细信息
+     */
+    @PreAuthorize("@ss.hasPermi('app:userChatLogs:query')")
+    @GetMapping(value = "/{id}")
+    public AjaxResult getInfo(@PathVariable Long id) {
+        return AjaxResult.success(appUserChatLogsService.selectAppUserChatLogsById(id));
+    }
+
+    /**
+     * 新增app-客户与客服对话(问答)
+     */
+    @PreAuthorize("@ss.hasPermi('app:userChatLogs:add')")
+    @Log(title = "app-客户与客服对话(问答)", businessType = BusinessType.INSERT)
+    @PostMapping
+    public AjaxResult add(@RequestBody AppUserChatLogs appUserChatLogs) {
+        return toAjax(appUserChatLogsService.insertAppUserChatLogs(appUserChatLogs));
+    }
+
+    /**
+     * 修改app-客户与客服对话(问答)
+     */
+//    @PreAuthorize("@ss.hasPermi('app:userChatLogs:edit')")
+    @Log(title = "app-客户与客服对话(问答)", businessType = BusinessType.UPDATE)
+    @PostMapping("/edit")
+    public AjaxResult edit(@RequestBody AppUserChatLogsDTO req) {
+        return AjaxResult.success("ok", appUserChatLogsService.updateAppUserChatLogs(req));
+    }
+
+    /**
+     * 删除app-客户与客服对话(问答)
+     */
+    @PreAuthorize("@ss.hasPermi('app:userChatLogs:remove')")
+    @Log(title = "app-客户与客服对话(问答)", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{ids}")
+    public AjaxResult remove(@PathVariable Long[] ids) {
+        return toAjax(appUserChatLogsService.deleteAppUserChatLogsByIds(ids));
+    }
+
+    /**
+     * 查询当前用户和当前客服的最新对话信息
+     *
+     * @return
+     */
+    @GetMapping("/findChatLogsByUserIdAndCustomerId")
+    public AjaxResult findChatLogsByUserIdAndCustomerId(@RequestParam Long userId, @RequestParam Long customerId) {
+        return AjaxResult.success(appUserChatLogsService.findChatLogsByUserIdAndCustomerId(userId, customerId));
+    }
+}

+ 80 - 0
fs-admin/src/main/java/com/fs/app/controller/AppUserController.java

@@ -0,0 +1,80 @@
+package com.fs.app.controller;
+
+import cn.hutool.core.util.ObjectUtil;
+import com.fs.app.user.dto.AppUserDTO;
+import com.fs.app.user.service.IAppUserService;
+import com.fs.app.user.vo.AppUserVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.util.List;
+
+import static com.fs.his.utils.PhoneUtil.encryptPhone;
+import static com.fs.his.utils.PhoneUtil.encryptPhoneOldKey;
+
+@RestController
+@RequestMapping("/app/user")
+@RequiredArgsConstructor
+public class AppUserController {
+
+    private final IAppUserService userService;
+
+
+
+    /**
+     * 筛选列表
+     *
+     * @param userDTO
+     * @return
+     */
+    @GetMapping("/list")
+    public TableDataInfo list(AppUserDTO userDTO) {
+        return userService.list(userDTO);
+    }
+
+    /**
+     * 数据列表
+     *
+     * @param userDTO
+     * @return
+     */
+    @GetMapping("/findList")
+    @PreAuthorize("@ss.hasPermi('app:user:list')")
+    public TableDataInfo findList(AppUserDTO userDTO) {
+        //原本有两种手机号加密方式,为了兼容,故两种都需要计算
+        if (ObjectUtil.isNotEmpty(userDTO.getPhone())) {
+            userDTO.setEncryptPhone(encryptPhone(userDTO.getPhone()));
+            userDTO.setEncryptPhone2(encryptPhoneOldKey(userDTO.getPhone()));
+        }
+        return userService.findList(userDTO);
+    }
+
+    /**
+     * 导出用户列表
+     */
+    @PreAuthorize("@ss.hasPermi('app:user:export')")
+    @Log(title = "用户", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppUserDTO userDTO) {
+        //原本有两种手机号加密方式,为了兼容,故两种都需要计算
+        if (ObjectUtil.isNotEmpty(userDTO.getPhone())) {
+            userDTO.setEncryptPhone(encryptPhone(userDTO.getPhone()));
+            userDTO.setEncryptPhone2(encryptPhoneOldKey(userDTO.getPhone()));
+        }
+        List<AppUserVO> list = userService.tableList(userDTO);
+        if (list.size() > 10000) {
+            return AjaxResult.error("导出数据不可超过1w条");
+        }
+        ExcelUtil<AppUserVO> util = new ExcelUtil<>(AppUserVO.class);
+        return util.exportExcel(list, "用户数据");
+    }
+
+}

+ 144 - 0
fs-admin/src/main/java/com/fs/app/controller/AppUserPortraitController.java

@@ -0,0 +1,144 @@
+package com.fs.app.controller;
+
+import cn.hutool.core.collection.CollUtil;
+import cn.hutool.core.util.ObjectUtil;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.fs.app.tag.domain.FsTag;
+import com.fs.app.tag.dto.FsContentTagDTO;
+import com.fs.app.tag.service.IFsTagService;
+import com.fs.app.user.domain.FsUserPortrait;
+import com.fs.app.user.domain.FsUserPortraitTag;
+import com.fs.app.user.service.IFsUserPortraitContentTagService;
+import com.fs.app.user.service.IFsUserPortraitService;
+import com.fs.app.user.service.IFsUserPortraitTagService;
+import com.fs.common.constant.HttpStatus;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.his.utils.PhoneUtil;
+import lombok.RequiredArgsConstructor;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.stream.Collectors;
+
+@RestController
+@RequestMapping("/app/user/portrait")
+@RequiredArgsConstructor
+public class AppUserPortraitController extends BaseController {
+
+    @Autowired
+    private IFsUserPortraitService userService;
+
+
+    @Autowired
+    private IFsUserPortraitContentTagService contentTagService;
+    @Autowired
+    private IFsUserPortraitTagService userPortraitTagService;
+    @Autowired
+    private IFsTagService fsTagService;
+
+    /**
+     * 数据列表
+     *
+     * @param userDTO
+     * @return
+     */
+    @GetMapping("/findList")
+    //@PreAuthorize("@ss.hasPermi('app:user:list')")
+    public TableDataInfo findList(FsUserPortrait userDTO) {
+        LambdaQueryWrapper<FsUserPortrait> wrapper = new LambdaQueryWrapper<>();
+        //原本有两种手机号加密方式,为了兼容,故两种都需要计算
+        if (ObjectUtil.isNotEmpty(userDTO.getPhone())) {
+            String s = PhoneUtil.encryptPhone(userDTO.getPhone());
+            String s1 = PhoneUtil.encryptPhoneOldKey(userDTO.getPhone());
+            wrapper.in(FsUserPortrait::getPhone, s, s1);
+        }
+        wrapper.eq(ObjectUtil.isNotEmpty(userDTO.getNickName()),FsUserPortrait::getNickName, userDTO.getNickName());
+        wrapper.eq(ObjectUtil.isNotEmpty(userDTO.getUserId()),FsUserPortrait::getUserId, userDTO.getUserId());
+        wrapper.eq(ObjectUtil.isNotEmpty(userDTO.getActivityLevel()),FsUserPortrait::getActivityLevel, userDTO.getActivityLevel());
+        wrapper.eq(ObjectUtil.isNotEmpty(userDTO.getConsumptionLevel()),FsUserPortrait::getConsumptionLevel, userDTO.getConsumptionLevel());
+        int total = userService.count(wrapper);
+        startPage();
+        List<FsUserPortrait> list = userService.list(wrapper);
+        if (CollUtil.isNotEmpty(list)){
+            List<Long> userIds = list.stream()
+                    .map(FsUserPortrait::getUserId)
+                    .collect(Collectors.toList());
+
+            List<FsUserPortraitTag> relationList = userPortraitTagService.list(
+                    new LambdaQueryWrapper<FsUserPortraitTag>()
+                            .in(FsUserPortraitTag::getUserId, userIds)
+            );
+
+            Set<Long> tagIds = relationList.stream()
+                    .map(FsUserPortraitTag::getTagId)
+                    .collect(Collectors.toSet());
+
+            if (CollUtil.isNotEmpty(tagIds)) {
+                Map<Long, String> tagMap = fsTagService.list(
+                        new LambdaQueryWrapper<FsTag>()
+                                .in(FsTag::getId, tagIds)
+                ).stream().collect(Collectors.toMap(FsTag::getId, FsTag::getName));
+
+                Map<Long, List<FsUserPortraitTag>> userTagMap = relationList.stream()
+                        .collect(Collectors.groupingBy(FsUserPortraitTag::getUserId));
+
+                for (FsUserPortrait user : list) {
+                    List<FsUserPortraitTag> tags = userTagMap.get(user.getUserId());
+                    if (tags == null || tags.isEmpty()) {
+                        user.setInterestTags("");
+                        continue;
+                    }
+                    String result = tags.stream()
+                            .map(t -> {
+                                String tagName = tagMap.get(t.getTagId());
+                                Integer weight = t.getWeight(); // 假设字段叫 weight
+                                return tagName + "|" + weight;
+                            })
+                            .collect(Collectors.joining(","));
+                    user.setInterestTags(result);
+                }
+            }
+        }
+        TableDataInfo tableDataInfo = new TableDataInfo();
+        tableDataInfo.setCode(HttpStatus.SUCCESS);
+        tableDataInfo.setMsg("查询成功");
+        tableDataInfo.setTotal(total);
+        tableDataInfo.setRows(list);
+        return tableDataInfo;
+    }
+//
+//    /**
+//     * 导出用户列表
+//     */
+//    @PreAuthorize("@ss.hasPermi('app:user:export')")
+//    @Log(title = "用户", businessType = BusinessType.EXPORT)
+//    @GetMapping("/export")
+//    public AjaxResult export(AppUserDTO userDTO) {
+//        //原本有两种手机号加密方式,为了兼容,故两种都需要计算
+//        if (ObjectUtil.isNotEmpty(userDTO.getPhone())) {
+//            userDTO.setEncryptPhone(encryptPhone(userDTO.getPhone()));
+//            userDTO.setEncryptPhone2(encryptPhoneOldKey(userDTO.getPhone()));
+//        }
+//        List<AppUserVO> list = userService.tableList(userDTO);
+//        if (list.size() > 10000) {
+//            return AjaxResult.error("导出数据不可超过1w条");
+//        }
+//        ExcelUtil<AppUserVO> util = new ExcelUtil<>(AppUserVO.class);
+//        return util.exportExcel(list, "用户数据");
+//    }
+
+    /**
+     * 批量增加内容的标签
+     */
+    @PostMapping("/bind/tag")
+    public AjaxResult bindTag(@RequestBody FsContentTagDTO fsUserTagDTO) {
+        contentTagService.bindTag(fsUserTagDTO);
+        return AjaxResult.success();
+    }
+
+}

+ 93 - 0
fs-admin/src/main/java/com/fs/app/controller/AppWelcomeController.java

@@ -0,0 +1,93 @@
+package com.fs.app.controller;
+
+import com.fs.app.welcome.dto.AppWelcomeDTO;
+import com.fs.app.welcome.service.IAppWelcomeService;
+import com.fs.app.welcome.vo.AppWelcomeVO;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ * app-欢迎语维护Controller
+ *
+ * @author fs
+ * @date 2026-02-27
+ */
+@RestController
+@RequestMapping("/app/welcome")
+public class AppWelcomeController extends BaseController {
+
+    @Autowired
+    private IAppWelcomeService appWelcomeService;
+
+    /**
+     * 查询app-欢迎语维护列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:list')")
+    @GetMapping("/list")
+    public TableDataInfo list(AppWelcomeDTO appWelcome) {
+        startPage();
+        List<AppWelcomeVO> list = appWelcomeService.selectAppWelcomeList(appWelcome);
+        return getDataTable(list);
+    }
+
+    /**
+     * 导出app-欢迎语维护列表
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:export')")
+    @Log(title = "app欢迎语", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppWelcomeDTO appWelcome) {
+        List<AppWelcomeVO> list = appWelcomeService.selectAppWelcomeList(appWelcome);
+        ExcelUtil<AppWelcomeVO> util = new ExcelUtil<>(AppWelcomeVO.class);
+        return util.exportExcel(list, "app-欢迎语维护数据");
+    }
+
+    /**
+     * 获取app-欢迎语维护详细信息
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:list')")
+    @GetMapping(value = "/getById/{id}")
+    public AjaxResult getInfo(@PathVariable("id") Long id) {
+        return AjaxResult.success(appWelcomeService.selectAppWelcomeById(id));
+    }
+
+    /**
+     * 保存 新增/编辑 欢迎语
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:add,app:welcome:edit')")
+    @Log(title = "app欢迎语", businessType = BusinessType.UPDATE)
+    @PostMapping(value = "/save")
+    public AjaxResult save(@RequestBody AppWelcomeDTO appWelcome) {
+        return AjaxResult.success(appWelcomeService.saveWelcome(appWelcome));
+    }
+
+
+    /**
+     * 删除app-欢迎语维护
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:delete')")
+    @Log(title = "app欢迎语", businessType = BusinessType.DELETE)
+    @DeleteMapping("/deleteByIds/{ids}")
+    public AjaxResult remove(@PathVariable List<Long> ids) {
+        return toAjax(appWelcomeService.deleteAppWelcomeByIds(ids));
+    }
+
+    /**
+     * 删除app-设置欢迎语状态
+     */
+    @PreAuthorize("@ss.hasAnyPermi('app:welcome:edit')")
+    @Log(title = "app欢迎语", businessType = BusinessType.UPDATE)
+    @PostMapping("/setSendStatus")
+    public AjaxResult setSendStatus(@RequestBody AppWelcomeDTO appWelcomeDTO) {
+        return toAjax(appWelcomeService.setSendStatus(appWelcomeDTO));
+    }
+}

+ 186 - 0
fs-admin/src/main/java/com/fs/app/controller/CommonV2Controller.java

@@ -0,0 +1,186 @@
+package com.fs.app.controller;
+
+import com.fs.common.config.FSConfig;
+import com.fs.common.constant.Constants;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.exception.file.OssException;
+import com.fs.common.utils.StringUtils;
+import com.fs.common.utils.file.FileUploadUtils;
+import com.fs.common.utils.file.FileUtils;
+import com.fs.framework.config.ServerConfig;
+import com.fs.system.oss.CloudStorageService;
+import com.fs.system.oss.OSSFactory;
+import com.fs.utils.AudioUtils;
+import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.http.MediaType;
+import org.springframework.web.bind.annotation.*;
+import org.springframework.web.multipart.MultipartFile;
+
+import javax.imageio.ImageIO;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.awt.image.BufferedImage;
+import java.io.ByteArrayOutputStream;
+import java.io.InputStream;
+
+/**
+ * 通用请求处理
+ *
+ */
+@RestController
+@RequiredArgsConstructor
+@Slf4j
+@RequestMapping("/app/common")
+public class CommonV2Controller {
+
+    private final ServerConfig serverConfig;
+
+
+    /**
+     * 通用下载请求
+     *
+     * @param fileName 文件名称
+     * @param delete   是否删除
+     */
+    @GetMapping("/download")
+    public void fileDownload(String fileName, Boolean delete, HttpServletResponse response, HttpServletRequest request) {
+        try {
+            if (!FileUtils.checkAllowDownload(fileName)) {
+                throw new Exception(StringUtils.format("文件名称({})非法,不允许下载。 ", fileName));
+            }
+            String realFileName = System.currentTimeMillis() + fileName.substring(fileName.indexOf("_") + 1);
+            String filePath = FSConfig.getDownloadPath() + fileName;
+
+            response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
+            FileUtils.setAttachmentResponseHeader(response, realFileName);
+            FileUtils.writeBytes(filePath, response.getOutputStream());
+            if (delete) {
+                FileUtils.deleteFile(filePath);
+            }
+        } catch (Exception e) {
+            log.error("下载文件失败", e);
+        }
+    }
+
+    /**
+     * 通用上传请求
+     */
+    @PostMapping("/upload")
+    public AjaxResult uploadFile(MultipartFile file) throws Exception {
+        try {
+            // 上传文件路径
+            String filePath = FSConfig.getUploadPath();
+            // 上传并返回新文件名称
+            String fileName = FileUploadUtils.upload(filePath, file);
+            String url = serverConfig.getUrl() + fileName;
+            AjaxResult ajax = AjaxResult.success();
+            ajax.put("fileName", fileName);
+            ajax.put("url", url);
+            return ajax;
+        } catch (Exception e) {
+            return AjaxResult.error(e.getMessage());
+        }
+    }
+
+    /**
+     * 本地资源通用下载
+     */
+    @GetMapping("/download/resource")
+    public void resourceDownload(String resource, HttpServletRequest request, HttpServletResponse response)
+            throws Exception {
+        try {
+            if (!FileUtils.checkAllowDownload(resource)) {
+                throw new Exception(StringUtils.format("资源文件({})非法,不允许下载。 ", resource));
+            }
+            // 本地资源路径
+            String localPath = FSConfig.getProfile();
+            // 数据库资源地址
+            String downloadPath = localPath + StringUtils.substringAfter(resource, Constants.RESOURCE_PREFIX);
+            // 下载名称
+            String downloadName = StringUtils.substringAfterLast(downloadPath, "/");
+            response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
+            FileUtils.setAttachmentResponseHeader(response, downloadName);
+            FileUtils.writeBytes(downloadPath, response.getOutputStream());
+        } catch (Exception e) {
+            log.error("下载文件失败", e);
+        }
+    }
+
+    @PostMapping("/uploadOSS")
+    public R uploadOSS(@RequestParam("file") MultipartFile file) throws Exception {
+        if (file.isEmpty()) {
+            throw new OssException("上传文件不能为空");
+        }
+        // 上传文件
+        String fileName = file.getOriginalFilename();
+        String suffix = fileName.substring(fileName.lastIndexOf("."));
+        CloudStorageService storage = OSSFactory.build();
+        String url = storage.uploadSuffix(file.getBytes(), suffix);
+        return R.ok().put("url", url);
+    }
+
+    @PostMapping("/uploadOSS2")
+    public R uploadOSS2(@RequestParam("file") MultipartFile file) throws Exception {
+        if (file.isEmpty()) {
+            throw new OssException("上传文件不能为空");
+        }
+        // 上传文件
+        String fileName = file.getOriginalFilename();
+        String suffix = fileName.substring(fileName.lastIndexOf("."));
+        String prefix = fileName.substring(0, fileName.lastIndexOf("."));
+        CloudStorageService storage = OSSFactory.build();
+        String url = storage.upload(file.getBytes(), prefix + System.currentTimeMillis() + suffix);
+        return R.ok().put("url", url);
+    }
+
+    @PostMapping("/uploadOSSByHOOKImage")
+    public R uploadOSSByHOOK(@RequestParam("file") MultipartFile file) throws Exception {
+        if (file.isEmpty()) {
+            throw new OssException("上传文件不能为空");
+        }
+        InputStream inputStream = file.getInputStream();
+        BufferedImage bufferedImage = ImageIO.read(inputStream);
+        if (bufferedImage == null) {
+            throw new OssException("无法解析图片,请检查文件内容是否正确");
+        }
+
+        // 将 BufferedImage 转换为 PNG 格式字节数组
+        ByteArrayOutputStream pngOutputStream = new ByteArrayOutputStream();
+        ImageIO.write(bufferedImage, "png", pngOutputStream);
+
+        // 获取 PNG 格式的字节数组
+        byte[] pngBytes = pngOutputStream.toByteArray();
+
+        // 上传文件
+        CloudStorageService storage = OSSFactory.build();
+        String url = storage.uploadSuffix(pngBytes, ".png");
+
+        return R.ok().put("url", url);
+    }
+
+
+    //MP3上传且转化为Sile,并且返回MP3的地址和sile的地址
+    @PostMapping("/uploadOSSByHOOKVoice")
+    public R uploadOSSByQw(@RequestParam("file") MultipartFile file) throws Exception {
+        if (file.isEmpty()) {
+            throw new OssException("上传文件不能为空");
+        }
+
+        // 上传文件
+        String fileName = file.getOriginalFilename();
+        String suffix = fileName.substring(fileName.lastIndexOf("."));
+        CloudStorageService storage = OSSFactory.build();
+        String mp3Url = storage.uploadSuffix(file.getBytes(), suffix);
+        String silkUrl = "";
+        try {
+            silkUrl = AudioUtils.transferAudioSilkFromUrl(mp3Url, true);
+        } catch (Exception e) {
+            throw new Exception("音频转换失败:" + e);
+        }
+
+        return R.ok().put("mp3Url", mp3Url).put("silkUrl", silkUrl);
+    }
+
+}

+ 36 - 0
fs-admin/src/main/java/com/fs/app/controller/FsUserInfoController.java

@@ -0,0 +1,36 @@
+package com.fs.app.controller;
+
+import com.fs.app.user.dto.FsUserInfoDTO;
+import com.fs.app.user.service.IFsUserInfoService;
+import com.fs.common.core.domain.AjaxResult;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.*;
+
+@RestController
+@RequestMapping("/app/fsUserInfo")
+public class FsUserInfoController {
+
+    @Autowired
+    private IFsUserInfoService fsUserInfoService;
+
+    @GetMapping("/getByUserId/{userId}")
+    public AjaxResult getByUserId(@PathVariable Long userId) {
+        return AjaxResult.success(fsUserInfoService.selectFsUserInfoByUserId(userId));
+    }
+
+    @PostMapping("/saveUserInfo")
+    public AjaxResult saveUserInfo(@RequestBody FsUserInfoDTO reqParam) {
+        return AjaxResult.success(fsUserInfoService.saveUserInfo(reqParam));
+    }
+
+    /**
+     * 获取用户兴趣标签
+     * @param userId
+     * @return
+     */
+    @GetMapping("/getUserTags/{userId}")
+    public AjaxResult getUserTags(@PathVariable Long userId) {
+        return AjaxResult.success(fsUserInfoService.getUserTags(userId));
+    }
+
+}

+ 34 - 0
fs-admin/src/main/java/com/fs/app/controller/VipTagController.java

@@ -0,0 +1,34 @@
+package com.fs.app.controller;
+
+import com.fs.app.tag.param.FsTagParam;
+import com.fs.app.tag.service.IFsTagService;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.page.TableDataInfo;
+import lombok.RequiredArgsConstructor;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+/**
+ * 会员标签组
+ */
+@RestController
+@RequestMapping("/vip/tag")
+@RequiredArgsConstructor
+public class VipTagController extends BaseController {
+
+    private final IFsTagService fsTagService;
+
+    /**
+     * 列表
+     * @param param
+     * @return
+     */
+    @GetMapping("/list")
+//    @PreAuthorize("@ss.hasPermi('vip:tagGroup:list')")
+    public TableDataInfo list(FsTagParam param) {
+        startPage();
+        return getDataTable(fsTagService.findList(param));
+    }
+
+}

+ 98 - 0
fs-admin/src/main/java/com/fs/app/controller/VipTagGroupController.java

@@ -0,0 +1,98 @@
+package com.fs.app.controller;
+
+import com.fs.app.tag.dto.FsTagGroupDTO;
+import com.fs.app.tag.param.FsTagGroupParam;
+import com.fs.app.tag.service.IFsTagGroupService;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.page.TableDataInfo;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.List;
+
+/**
+ * 会员标签组
+ */
+@RestController
+@RequestMapping("/vip/tagGroup")
+@RequiredArgsConstructor
+public class VipTagGroupController extends BaseController {
+
+    private final IFsTagGroupService fsTagGroupService;
+
+    /**
+     * 列表
+     *
+     * @param param
+     * @return
+     */
+    @GetMapping("/list")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:list')")
+    public TableDataInfo list(FsTagGroupParam param) {
+        startPage();
+        return getDataTable(fsTagGroupService.findList(param));
+    }
+
+    /**
+     * 用户绑定标签,标签(组)列表
+     *
+     * @param param
+     * @return
+     */
+    @GetMapping("/listForUserBindTag")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:list')")
+    public TableDataInfo listForUserBindTag(FsTagGroupParam param) {
+        return fsTagGroupService.listForUserBindTag(param);
+    }
+
+
+    /**
+     * 新增
+     *
+     * @param dto
+     * @return
+     */
+    @PostMapping("/add")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:add')")
+    public AjaxResult add(@RequestBody FsTagGroupDTO dto) {
+        return toAjax(fsTagGroupService.addGroup(dto));
+    }
+
+    /**
+     * 编辑回显
+     *
+     * @param id
+     * @return
+     */
+    @GetMapping("/getById/{id}")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:edit')")
+    public AjaxResult getById(@PathVariable Long id) {
+        return AjaxResult.success(fsTagGroupService.getById(id));
+    }
+
+    /**
+     * 根据 id 列表获取所有信息
+     *
+     * @param ids
+     * @return
+     */
+    @GetMapping("/getTagByIds/{ids}")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:edit')")
+    public AjaxResult getTagByIds(@PathVariable List<Long> ids) {
+        return AjaxResult.success(fsTagGroupService.getTagByIds(ids));
+    }
+
+    /**
+     * 编辑保存
+     *
+     * @param dto
+     * @return
+     */
+    @PostMapping("/edit")
+    @PreAuthorize("@ss.hasPermi('vip:tagGroup:edit')")
+    public AjaxResult edit(@RequestBody FsTagGroupDTO dto) {
+        return toAjax(fsTagGroupService.editGroup(dto));
+    }
+}

+ 79 - 0
fs-admin/src/main/java/com/fs/app/controller/VipUserCustomerController.java

@@ -0,0 +1,79 @@
+package com.fs.app.controller;
+
+import com.fs.app.tag.dto.FsUserCustomerDTO;
+import com.fs.app.tag.service.IFsUserCustomerService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.enums.BusinessType;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestBody;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/vip/userCusRole")
+@RequiredArgsConstructor
+public class VipUserCustomerController {
+
+    private final IFsUserCustomerService fsUserCustomerService;
+
+
+    /**
+     * 用户绑定客服-老版本,后续会弃用
+     *
+     * @param fsUserCustomerDTO
+     * @return
+     */
+    @PostMapping("/bind")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:bindCustomer,app:user:batchBindCustomer')")
+    @Log(title = "APP会员绑定客服", businessType = BusinessType.OTHER)
+    public AjaxResult bind(@RequestBody FsUserCustomerDTO fsUserCustomerDTO) {
+        fsUserCustomerService.bind(fsUserCustomerDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 用户解绑客服-老版本,后续会弃用
+     *
+     * @param fsUserCustomerDTO
+     * @return
+     */
+    @PostMapping("/unbind")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:unbindCustomer,app:user:batchUnbindCustomer')")
+    @Log(title = "APP会员解绑客服", businessType = BusinessType.OTHER)
+    public AjaxResult unbind(@RequestBody FsUserCustomerDTO fsUserCustomerDTO) {
+        fsUserCustomerService.unbind(fsUserCustomerDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 用户绑定客服-新版本,后续需调用这个版本
+     *
+     * @param fsUserCustomerDTO
+     * @return
+     */
+    @PostMapping("/bind/v2")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:bindCustomer,app:user:batchBindCustomer')")
+    @Log(title = "APP会员绑定客服", businessType = BusinessType.OTHER)
+    public AjaxResult bindV2(@RequestBody FsUserCustomerDTO fsUserCustomerDTO) {
+        fsUserCustomerService.bindV2(fsUserCustomerDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 用户解绑客服-新版本,后续需调用这个版本
+     *
+     * @param fsUserCustomerDTO
+     * @return
+     */
+    @PostMapping("/unbind/v2")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:unbindCustomer,app:user:batchUnbindCustomer')")
+    @Log(title = "APP会员解绑客服", businessType = BusinessType.OTHER)
+    public AjaxResult unbindV2(@RequestBody FsUserCustomerDTO fsUserCustomerDTO) {
+        fsUserCustomerService.unbindV2(fsUserCustomerDTO);
+        return AjaxResult.success();
+    }
+
+}

+ 78 - 0
fs-admin/src/main/java/com/fs/app/controller/VipUserTagController.java

@@ -0,0 +1,78 @@
+package com.fs.app.controller;
+
+import com.fs.app.tag.dto.FsUserTagDTO;
+import com.fs.app.tag.service.IFsUserTagService;
+import com.fs.app.user.service.IFsUserPortraitTagService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.enums.BusinessType;
+import lombok.RequiredArgsConstructor;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestBody;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+@RequestMapping("/vip/userTag")
+@RequiredArgsConstructor
+public class VipUserTagController extends BaseController {
+
+    private final IFsUserTagService fsUserTagService;
+    private final IFsUserPortraitTagService fsUserPortraitTagService;
+    /**
+     * 用户绑定标签
+     *
+     * @param fsUserTagDTO 用户,标签绑定载体
+     */
+    @PostMapping("/bind")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:bindTag,app:user:batchBindTag')")
+    @Log(title = "APP会员绑定标签", businessType = BusinessType.OTHER)
+    public AjaxResult bind(@RequestBody FsUserTagDTO fsUserTagDTO) {
+        fsUserTagService.bindUserTag(fsUserTagDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 解绑用户标签
+     *
+     * @param fsUserTagDTO 用户,标签绑定载体
+     */
+    @PostMapping("/unbind")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:unbindTag,app:user:batchUnbindTag')")
+    @Log(title = "APP会员解绑标签", businessType = BusinessType.OTHER)
+    public AjaxResult unbind(@RequestBody FsUserTagDTO fsUserTagDTO) {
+        fsUserTagService.unbindUserTag(fsUserTagDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 用户绑定标签-新版本,需后续切换到此版本
+     *
+     * @param fsUserTagDTO 用户,标签绑定载体
+     */
+    @PostMapping("/bind/v2")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:bindTag,app:user:batchBindTag')")
+    @Log(title = "APP会员绑定标签", businessType = BusinessType.OTHER)
+    public AjaxResult bindV2(@RequestBody FsUserTagDTO fsUserTagDTO) {
+        fsUserPortraitTagService.bindTag(fsUserTagDTO.getUserId(),fsUserTagDTO.getTagId());
+        fsUserTagService.bindUserTagV2(fsUserTagDTO);
+        return AjaxResult.success();
+    }
+
+    /**
+     * 解绑用户标签-新版本,需后续切换到此版本
+     *
+     * @param fsUserTagDTO 用户,标签绑定载体
+     */
+    @PostMapping("/unbind/v2")
+    @PreAuthorize("@ss.hasAnyPermi('app:user:unbindTag,app:user:batchUnbindTag')")
+    @Log(title = "APP会员解绑标签", businessType = BusinessType.OTHER)
+    public AjaxResult unbindV2(@RequestBody FsUserTagDTO fsUserTagDTO) {
+        fsUserPortraitTagService.unBindTag(fsUserTagDTO.getUserId(),fsUserTagDTO.getTagId());
+        fsUserTagService.unbindUserTagV2(fsUserTagDTO);
+        return AjaxResult.success();
+    }
+
+}

+ 116 - 0
fs-admin/src/main/java/com/fs/app/job/UserActivityTask.java

@@ -0,0 +1,116 @@
+package com.fs.app.job;
+
+import com.fs.app.user.domain.FsUserPortrait;
+import com.fs.app.user.service.IFsUserPortraitService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.redis.core.Cursor;
+import org.springframework.data.redis.core.ScanOptions;
+import org.springframework.data.redis.core.StringRedisTemplate;
+import org.springframework.scheduling.annotation.Scheduled;
+import org.springframework.stereotype.Component;
+
+import java.time.LocalDate;
+import java.time.format.DateTimeFormatter;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Map;
+
+@Component
+@Slf4j
+public class UserActivityTask {
+    @Autowired
+    private StringRedisTemplate stringRedisTemplate;
+
+    @Autowired
+    private IFsUserPortraitService portraitService;
+
+    private static final String LASTS_EEN_KEY = "userPortrait:last_seen";
+    private static final String TOTAL_7D_KEY = "userPortrait:total:7d";
+    /**
+     * 入口 1:每天凌晨 3 点定时执行
+     */
+    @Scheduled(cron = "0 0 3 * * ?")
+    public void cronTask() {
+        // 昨天
+        String d1 = LocalDate.now().minusDays(1).format(DateTimeFormatter.ofPattern("yyyyMMdd"));
+        // 8天前
+        String d8 = LocalDate.now().minusDays(8).format(DateTimeFormatter.ofPattern("yyyyMMdd"));
+
+        log.info("开始滑动窗口计算: 增加 {}, 减去 {}", d1, d8);
+
+        String dailyD1Key = "userPortrait:daily:duration:" + d1;
+        String dailyD8Key = "userPortrait:daily:duration:" + d8;
+
+
+
+        Cursor<Map.Entry<Object, Object>> cursor = stringRedisTemplate.opsForHash().scan(LASTS_EEN_KEY,
+                ScanOptions.scanOptions().count(1000).build());
+
+        List<FsUserPortrait> batchList = new ArrayList<>();
+
+        while (cursor.hasNext()) {
+            Map.Entry<Object, Object> entry = cursor.next();
+            String userId = entry.getKey().toString();
+            long lastSeenTs = Long.parseLong(entry.getValue().toString());
+
+            // 1. 获取昨天的增量
+            Object d1Incr = stringRedisTemplate.opsForHash().get(dailyD1Key, userId);
+            long plusSeconds = d1Incr != null ? Long.parseLong(d1Incr.toString()) : 0L;
+
+            // 2. 获取8天前的过期量
+            Object d8Decr = stringRedisTemplate.opsForHash().get(dailyD8Key, userId);
+            long minusSeconds = d8Decr != null ? Long.parseLong(d8Decr.toString()) : 0L;
+
+            // 3. 更新 Redis 中的 7 日滑动窗口总值
+            long current7dTotal = stringRedisTemplate.opsForHash().increment(TOTAL_7D_KEY, userId, plusSeconds - minusSeconds);
+
+            // 防止出现负数
+            if (current7dTotal < 0) {
+                stringRedisTemplate.opsForHash().put(TOTAL_7D_KEY, userId, "0");
+                current7dTotal = 0;
+            }
+
+            // 4. 计算活跃度状态
+            FsUserPortrait activity = calculateStatus(Long.valueOf(userId), lastSeenTs, current7dTotal);
+            batchList.add(activity);
+
+            if (batchList.size() >= 500) {
+                portraitService.updateActivityLevelBatch(batchList);
+                batchList.clear();
+            }
+        }
+        if (!batchList.isEmpty()) {
+            portraitService.updateActivityLevelBatch(batchList);
+            batchList.clear();
+        }
+
+        // 5. 清理 8 天前的数据,释放内存
+        stringRedisTemplate.delete(dailyD8Key);
+    }
+
+    private FsUserPortrait calculateStatus(Long userId, long lastSeenTs, long total7dSeconds) {
+        long now = System.currentTimeMillis() / 1000;
+        long daysInactive = (now - lastSeenTs) / (24 * 3600);
+        double hours7d = total7dSeconds / 3600.0;
+
+        String activityLevel;
+        if (daysInactive >= 30) {
+            activityLevel = "流失";
+            // 删除流失用户数据
+            stringRedisTemplate.opsForHash().delete(LASTS_EEN_KEY, userId);
+            stringRedisTemplate.opsForHash().delete(TOTAL_7D_KEY, userId);
+        } else if (daysInactive >= 7) {
+            activityLevel = "沉默";
+        } else {
+            if (hours7d < 3) activityLevel = "低活跃";
+            else if (hours7d < 7) activityLevel = "活跃";
+            else activityLevel = "高活跃";
+        }
+        FsUserPortrait fsUserPortrait = new FsUserPortrait();
+        fsUserPortrait.setUserId(userId);
+        fsUserPortrait.setActivityLevel(activityLevel);
+        return fsUserPortrait;
+    }
+
+}

+ 97 - 0
fs-admin/src/main/java/com/fs/app/job/UserConsumptionAmountTask.java

@@ -0,0 +1,97 @@
+package com.fs.app.job;
+
+import cn.hutool.core.collection.CollUtil;
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.fs.app.user.domain.FsUserPortrait;
+import com.fs.app.user.service.IFsUserPortraitService;
+import com.fs.his.service.IFsStoreOrderService;
+import com.fs.his.vo.UserAmountVO;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.scheduling.annotation.Scheduled;
+import org.springframework.stereotype.Component;
+
+import java.math.BigDecimal;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Map;
+import java.util.stream.Collectors;
+
+@Component
+@Slf4j
+public class UserConsumptionAmountTask {
+    @Autowired
+    private IFsUserPortraitService portraitService;
+    @Autowired
+    private IFsStoreOrderService storeOrderService;
+
+    @Scheduled(cron = "0 0 2 * * ?") // 确保只执行一次
+    public void cronTask() {
+        LocalDateTime start = LocalDateTime.of(LocalDate.now().minusDays(1), LocalTime.MIN);
+        LocalDateTime end = LocalDateTime.of(LocalDate.now(), LocalTime.MIN);
+        long startPlay = System.currentTimeMillis();
+        log.error("开始增量统计消费数据: {} - {}", start, end);
+
+        // 1. 订单量大了之后这里要做分页
+        List<UserAmountVO> dailySums = storeOrderService.getSumAmount(start, end);
+        if (CollUtil.isEmpty(dailySums)) {
+            log.info("昨日无新增消费记录");
+            return;
+        }
+
+        // 2. 分批处理(例如每 1000 个用户一批)
+        List<List<UserAmountVO>> batches = CollUtil.split(dailySums, 1000);
+        for (List<UserAmountVO> batch : batches) {
+            processBatch(batch);
+        }
+        long endPlay = System.currentTimeMillis();
+        log.error("增量统计消费数据完成:{}", endPlay - startPlay);
+    }
+
+    public void processBatch(List<UserAmountVO> batch) {
+        List<Long> userIds = batch.stream().map(UserAmountVO::getUserId).collect(Collectors.toList());
+        Map<Long, FsUserPortrait> portraitMap = portraitService.list(
+                new LambdaQueryWrapper<FsUserPortrait>().in(FsUserPortrait::getUserId, userIds)
+        ).stream().collect(Collectors.toMap(FsUserPortrait::getUserId, p -> p));
+
+        List<FsUserPortrait> updateList = new ArrayList<>();
+
+        for (UserAmountVO dailySum : batch) {
+            FsUserPortrait portrait = portraitMap.get(dailySum.getUserId());
+            if (portrait == null) continue;
+            updatePortraitData(portrait, dailySum);
+            updateList.add(portrait);
+        }
+
+        // 3. 批量更新:一次 IO 更新 1000 行
+        portraitService.updateBatchById(updateList);
+    }
+
+    private void updatePortraitData(FsUserPortrait portrait, UserAmountVO dailySum) {
+        // 累加数据
+        portrait.setConsumptionAmount(portrait.getConsumptionAmount().add(dailySum.getDailySum()));
+        portrait.setSmallOrderCount(portrait.getSmallOrderCount() + dailySum.getSmallCount());
+
+        // 重新计算等级
+        portrait.setConsumptionLevel(calculateLevel(portrait.getConsumptionAmount(), portrait.getSmallOrderCount()));
+    }
+
+    private String calculateLevel(BigDecimal amount, int smallCount) {
+        if (amount.compareTo(new BigDecimal("3000")) > 0) return "S+";
+        if (amount.compareTo(new BigDecimal("500")) > 0) return "S";
+        if (amount.compareTo(new BigDecimal("150")) > 0) {
+            return smallCount > 20 ? "E" : "A";
+        }
+        if (smallCount > 20) return "E";
+        if (smallCount > 8) return "D";
+        if (smallCount > 3) return "C";
+        return "B"; // 补全默认等级
+    }
+
+}
+
+
+

+ 24 - 0
fs-admin/src/main/java/com/fs/app/job/UserTagTask.java

@@ -0,0 +1,24 @@
+package com.fs.app.job;
+
+import com.fs.app.user.service.IFsUserPortraitTagService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.scheduling.annotation.Scheduled;
+import org.springframework.stereotype.Component;
+
+@Component
+@Slf4j
+public class UserTagTask {
+
+    @Autowired
+    private IFsUserPortraitTagService portraitTagService;
+    // 每两天5点执行一次
+    @Scheduled(cron = "0 0 5 1/2 * ?")
+    public void syncDataToDb() {
+        // 权重衰减
+        portraitTagService.weightAttenuation();
+    }
+
+
+
+}

+ 201 - 0
fs-admin/src/main/java/com/fs/app/job/VideoTrackTask.java

@@ -0,0 +1,201 @@
+package com.fs.app.job;
+
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
+import com.fs.app.user.domain.FsUserPortraitContentTag;
+import com.fs.app.user.domain.FsUserPortraitTag;
+import com.fs.app.user.service.IFsUserPortraitContentTagService;
+import com.fs.app.user.service.IFsUserPortraitTagService;
+import com.fs.common.constant.VideoRedisKeyConst;
+import com.fs.course.dto.VideoUpdateDTO;
+import com.fs.course.service.IFsUserVideoService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.context.annotation.Lazy;
+import org.springframework.data.redis.core.StringRedisTemplate;
+import org.springframework.scheduling.annotation.Scheduled;
+import org.springframework.stereotype.Component;
+
+import java.util.*;
+import java.util.stream.Collectors;
+
+@Component
+@Slf4j
+public class VideoTrackTask {
+    @Autowired
+    private StringRedisTemplate stringRedisTemplate;
+    @Autowired
+    @Lazy
+    private IFsUserVideoService userVideoService;
+    @Autowired
+    private IFsUserPortraitContentTagService contentTagService;
+    @Autowired
+    private IFsUserPortraitTagService portraitTagService;
+
+
+    @Scheduled(cron = "0 0/2 * * * ?")
+    public void syncDataToDb() {
+        syncVideoMetrics();
+        syncUserTags();
+    }
+
+    private void syncVideoMetrics() {
+        // 每次最多从待处理池中弹出 1000 个视频ID进行同步 (控制单次处理量)
+
+        List<String> pendingVideoIds = stringRedisTemplate.opsForSet().pop(VideoRedisKeyConst.PENDING_SYNC_VIDEOS, 1000);
+
+        if (pendingVideoIds == null || pendingVideoIds.isEmpty()) {
+            return;
+        }
+
+        // 用于收集批量更新参数的 List
+        List<VideoUpdateDTO> updateBatchList = new ArrayList<>();
+
+        for (String
+                videoIdStr : pendingVideoIds) {
+            String key = VideoRedisKeyConst.VIDEO_METRICS_PREFIX + videoIdStr + VideoRedisKeyConst.KEY_END;
+            String syncKey = key + VideoRedisKeyConst.SYNC_SUFFIX;
+
+            try {
+                if (stringRedisTemplate.hasKey(key) && stringRedisTemplate.renameIfAbsent(key, syncKey)) {
+                    Long videoId = Long.parseLong(videoIdStr);
+                    Map<Object, Object> metrics = stringRedisTemplate.opsForHash().entries(syncKey);
+
+                    long play = parseLong(metrics.get("PLAY"));
+                    long play3s = parseLong(metrics.get("PLAY_3S"));
+                    long complete = parseLong(metrics.get("COMPLETE"));
+
+                    updateBatchList.add(new VideoUpdateDTO(videoId, play, play3s, complete));
+                    stringRedisTemplate.delete(syncKey);
+                } else {
+                    // 重命名失败,说明该Key正在被处理,为了防丢,把 videoId 塞回池子里下次处理
+                    stringRedisTemplate.opsForSet().add(VideoRedisKeyConst.PENDING_SYNC_VIDEOS, videoIdStr );
+                }
+            } catch (Exception e) {
+                stringRedisTemplate.opsForSet().add(VideoRedisKeyConst.PENDING_SYNC_VIDEOS, videoIdStr);
+            }
+        }
+
+
+        if (!updateBatchList.isEmpty()) {
+            userVideoService.batchIncrementMetrics(updateBatchList);
+        }
+    }
+
+    private void syncUserTags() {
+        // 1. 从待处理池中一次性弹出最多 1000 个需要更新的用户 ID
+        List<String> pendingUserIds = stringRedisTemplate.opsForSet().pop(VideoRedisKeyConst.PENDING_SYNC_USERS, 1000);
+
+        if (pendingUserIds == null || pendingUserIds.isEmpty()) {
+            return;
+        }
+
+        // userId -> { videoId -> weight }
+        Map<Long, Map<Long, Integer>> pendingSyncData = new HashMap<>();
+        // videoIds
+        Set<Long> allVideoIdsToQuery = new HashSet<>();
+        List<String> syncKeysToDelete = new ArrayList<>();
+
+        // 2. 遍历处理这批 userId
+        for (String userIdStr : pendingUserIds) {
+            String key = VideoRedisKeyConst.USER_VIDEO_WEIGHT_PREFIX + userIdStr + VideoRedisKeyConst.KEY_END;
+            String syncKey = key + VideoRedisKeyConst.SYNC_SUFFIX;
+
+            try {
+                // 原子重命名
+                if (stringRedisTemplate.hasKey(key) && stringRedisTemplate.renameIfAbsent(key, syncKey)) {
+                    Long userId = Long.parseLong(userIdStr);
+                    Map<Object, Object> videoWeights = stringRedisTemplate.opsForHash().entries(syncKey);
+
+                    Map<Long, Integer> userVideoMap = new HashMap<>();
+                    for (Map.Entry<Object, Object> entry : videoWeights.entrySet()) {
+                        Long videoId = Long.parseLong(String.valueOf(entry.getKey()));
+                        Integer weight = Integer.parseInt(String.valueOf(entry.getValue()));
+
+                        userVideoMap.put(videoId, weight);
+                        allVideoIdsToQuery.add(videoId);
+                    }
+
+                    pendingSyncData.put(userId, userVideoMap);
+                    syncKeysToDelete.add(syncKey);
+                } else {
+                    // 重命名失败说明该 key 正处于其它同步流程中,防丢:塞回待处理池下次再试
+                    stringRedisTemplate.opsForSet().add(VideoRedisKeyConst.PENDING_SYNC_USERS, userIdStr);
+                }
+            } catch (Exception e) {
+                log.error("转移用户视频权重 Redis Key 异常, userId: {}", userIdStr, e);
+                // 发生异常防丢:塞回待处理池下次再试
+                stringRedisTemplate.opsForSet().add(VideoRedisKeyConst.PENDING_SYNC_USERS, userIdStr);
+            }
+        }
+
+        // 如果没有任何有效的视频ID需要查询,直接清理 keys 并返回
+        if (allVideoIdsToQuery.isEmpty()) {
+            if (!syncKeysToDelete.isEmpty()) {
+                stringRedisTemplate.delete(syncKeysToDelete);
+            }
+            return;
+        }
+
+        // 3. 批量查询 DB 获取这些视频的标签 (1次网络IO)
+        List<FsUserPortraitContentTag> videoTags = contentTagService.list(new LambdaQueryWrapper<FsUserPortraitContentTag>()
+                .in(FsUserPortraitContentTag::getContentId, allVideoIdsToQuery));
+
+        // 按照 contentId 分组,收集每个 contentId 对应的 tagId 集合
+        Map<Long, List<Long>> videoTagMap = videoTags.stream()
+                .collect(Collectors.groupingBy(
+                        FsUserPortraitContentTag::getContentId,
+                        Collectors.mapping(FsUserPortraitContentTag::getTagId, Collectors.toList())
+                ));
+
+
+        // 用于收集将要批量写入数据库的对象集合
+        List<FsUserPortraitTag> tagBatchList = new ArrayList<>();
+
+        // 4. 在内存中将 videoId 的权重转化为 tag 的权重
+        for (Map.Entry<Long, Map<Long, Integer>> userEntry : pendingSyncData.entrySet()) {
+            Long userId = userEntry.getKey();
+            Map<Long, Integer> videoWeightsMap = userEntry.getValue();
+
+            // 聚合同一个用户在这批数据里的各个标签总权重: tag -> totalWeight
+            Map<Long, Integer> userTagAggregator = new HashMap<>();
+
+            for (Map.Entry<Long, Integer> vwEntry : videoWeightsMap.entrySet()) {
+                Long videoId = vwEntry.getKey();
+                Integer weight = vwEntry.getValue();
+
+                List<Long> tags = videoTagMap.get(videoId);
+                if (tags != null) {
+                    for (Long tag : tags) {
+                        userTagAggregator.put(tag, userTagAggregator.getOrDefault(tag, 0) + weight);
+                    }
+                }
+            }
+
+            // 组装成实体类放入 List
+            for (Map.Entry<Long, Integer> tagEntry : userTagAggregator.entrySet()) {
+                if (tagEntry.getValue() != 0) { // 只处理权重有净增减的
+                    FsUserPortraitTag utw = new FsUserPortraitTag();
+                    utw.setUserId(userId);
+                    utw.setTagId(tagEntry.getKey());
+                    utw.setWeight(tagEntry.getValue());
+                    tagBatchList.add(utw);
+                }
+            }
+        }
+
+        // 5. 排序防死锁并批量落库
+        if (!tagBatchList.isEmpty()) {
+            tagBatchList.sort(Comparator.comparing(FsUserPortraitTag::getUserId).thenComparing(FsUserPortraitTag::getTagId));
+            portraitTagService.batchAddOrUpdateWeight(tagBatchList);
+        }
+
+        // 落库成功,批量清理 Redis 中的临时 syncKeys
+        if (!syncKeysToDelete.isEmpty()) {
+            stringRedisTemplate.delete(syncKeysToDelete);
+        }
+    }
+
+    private long parseLong(Object obj) {
+        return obj == null ? 0L : Long.parseLong(String.valueOf(obj));
+    }
+}

+ 6 - 0
fs-admin/src/main/java/com/fs/company/controller/CompanyUserController.java

@@ -214,4 +214,10 @@ public class CompanyUserController extends BaseController
         List<Company> list = companyUserService.getCompanyList(corpId);
         return  R.ok().put("data",list);
     }
+
+    @GetMapping("/findOptions")
+    public AjaxResult findOptions(String keyword, Long selectedId, Integer pageNumber,  Integer pageSize) {
+        return AjaxResult.success(companyUserService.findCompanyUserOptions(keyword, selectedId, pageNumber, pageSize));
+    }
+
 }

+ 514 - 0
fs-admin/src/main/java/com/fs/course/controller/FsAppCourseWatchLogController.java

@@ -0,0 +1,514 @@
+package com.fs.course.controller;
+
+import com.fs.app.cusrole.service.IAppCustomerRoleService;
+import com.fs.app.watchlog.dto.AppCourseWatchLogDto;
+import com.fs.app.watchlog.param.AppCourseWatchLogListParam;
+import com.fs.app.watchlog.service.IAppCourseWatchLogService;
+import com.fs.common.annotation.Log;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.R;
+import com.fs.common.core.domain.model.LoginUser;
+import com.fs.common.core.page.TableDataInfo;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.poi.ExcelUtil;
+import com.fs.company.vo.OptionVO;
+import com.fs.course.service.IFsCourseWatchLogService;
+import com.fs.his.domain.FsExportTask;
+import com.fs.his.service.IFsExportTaskService;
+import com.fs.qw.param.QwWatchLogStatisticsListParam;
+import com.fs.qw.service.IQwWatchLogService;
+import com.github.pagehelper.PageHelper;
+import com.github.pagehelper.PageInfo;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RequestParam;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.time.LocalDate;
+import java.time.format.DateTimeFormatter;
+import java.time.temporal.ChronoUnit;
+import java.util.Date;
+import java.util.List;
+
+/**
+ * 短链课程看课记录Controller
+ *
+ * @author fs
+ * @date 2024-10-24
+ */
+@RestController
+@RequestMapping("/course/appCourseWatchLog")
+public class FsAppCourseWatchLogController extends BaseController {
+    @Autowired
+    private IFsCourseWatchLogService fsCourseWatchLogService;
+
+    @Autowired
+    private IQwWatchLogService qwWatchLogService;
+    @Autowired
+    private IFsExportTaskService exportTaskService;
+    @Autowired
+    private IAppCourseWatchLogService appCourseWatchLogService;
+    @Autowired
+    private IAppCustomerRoleService appCustomerRoleService;
+
+
+    /**
+     * 查询app课程看课记录列表
+     */
+    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:list')")
+    @GetMapping("/list")
+    public TableDataInfo list(AppCourseWatchLogListParam param) {
+        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+
+        List<AppCourseWatchLogDto> list = appCourseWatchLogService.selectAppCourseWatchLogListVO(param);
+        return getDataTable(list);
+    }
+
+//    /**
+//     * 查询短链课程 我的部门 看课记录列表
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:deptList')")
+//    @GetMapping("/deptList")
+//    public TableDataInfo deptList(FsCourseWatchLogListParam param) {
+//
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//
+//
+//        List<Long> combinedList = new ArrayList<>();
+//        //本部门
+//        Long deptId = loginUser.getUser().getDeptId();
+//        if (deptId!=null){
+//            combinedList.add(deptId);
+//        }
+//        //本部门的下级部门
+//        List<Long> deptList = companyDeptService.selectCompanyDeptByParentId(deptId);
+//        if (!deptList.isEmpty()){
+//            combinedList.addAll(deptList);
+//        }
+//
+//        param.setCuDeptIdList(combinedList);
+//        param.setUserType(loginUser.getUser().getUserType());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//
+//        startPage();
+//        List<FsCourseWatchLogListVO> list = fsCourseWatchLogService.selectFsCourseWatchLogListVO(param);
+//
+//        TableDataInfo rspData = new TableDataInfo();
+//        rspData.setCode(HttpStatus.SUCCESS);
+//        rspData.setMsg("查询成功");
+//        rspData.setRows(list);
+//        rspData.setTotal(fsCourseWatchLogService.selectFsCourseWatchLogListVOCount(param));
+//        return rspData;
+//    }
+//    /**
+//     * 查询短链课程看课记录列表
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:myList')")
+//    @GetMapping("/myList")
+//    public TableDataInfo myList(FsCourseWatchLogListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyUserId(loginUser.getUser().getUserId());
+//        List<FsCourseWatchLogListVO> list = fsCourseWatchLogService.selectFsCourseWatchLogListVO(param);
+//
+//        TableDataInfo rspData = new TableDataInfo();
+//        rspData.setCode(HttpStatus.SUCCESS);
+//        rspData.setMsg("查询成功");
+//        rspData.setRows(list);
+//        rspData.setTotal(fsCourseWatchLogService.selectFsCourseWatchLogListVOCount(param));
+//        return rspData;
+//    }
+
+
+    /**
+     * 判断传入的日期是否是7天之前
+     * @param dateStr
+     * @return
+     */
+    public boolean isDateBefore7Days(String dateStr) {
+        // 定义日期格式
+        DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd");
+
+        // 解析输入日期
+        LocalDate inputDate = LocalDate.parse(dateStr, formatter);
+
+        // 获取当前日期(2025-05-23)
+        LocalDate currentDate = LocalDate.now();
+
+        // 计算日期差
+        long daysBetween = ChronoUnit.DAYS.between(inputDate, currentDate);
+
+        // 判断是否超过7天
+        return daysBetween >= 7;
+    }
+
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:statisticsList')")
+//    @GetMapping("/statisticsList")
+//    public TableDataInfo statisticsList(FsCourseWatchLogStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<FsCourseWatchLogStatisticsListVO> list = fsCourseWatchLogService.selectFsCourseWatchLogStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+//    @GetMapping("/qwWatchLogStatisticsList")
+//    public TableDataInfo qwWatchLogStatisticsList(QwWatchLogStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<QwWatchLogStatisticsListVO> list = qwWatchLogService.selectQwWatchLogStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+//    @GetMapping("/exportWatchLogStatistics")
+//    public AjaxResult exportWatchLogStatistics(QwWatchLogStatisticsListParam param) {
+//
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setExcelName("进线客户统计记录导出");
+//
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return AjaxResult.error("请筛选时间后导出");
+//        }
+//
+//        boolean res = qwWatchLogService.dateDifferenceMoreThan7Days(param.getSTime(), param.getETime());
+//
+//        if (res) {
+//            return AjaxResult.error("导出数据时间差不能超过七天");
+//        }
+//        AjaxResult returnRes = new AjaxResult();
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        try {
+//            Class<? extends IQwWatchLogService> clazz = qwWatchLogService.getClass();
+//            returnRes = exportCommon(
+//                    param,
+//                    loginUser.getUser().getUserId(),
+//                    "进线客户统计记录",
+//                    10000L,
+//                    "getQwWatchLogStatisticsCount",
+//                    "exportQwWatchLogStatistics",
+//                    clazz);
+//        } catch (Exception ex) {
+//            ex.printStackTrace();
+//            return AjaxResult.error("导出报错了");
+//        }
+//        return returnRes;
+
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setExcelName("进线客户统计记录导出");
+//        return exportQwWatchLogStatistics(param, loginUser.getUser().getUserId());
+//    }
+
+//    @GetMapping("/myQwWatchLogStatisticsList")
+//    public TableDataInfo myQwWatchLogStatisticsList(QwWatchLogStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setCompanyUserId(loginUser.getUser().getUserId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<QwWatchLogStatisticsListVO> list = qwWatchLogService.selectQwWatchLogStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+    public AjaxResult exportQwWatchLogStatistics(QwWatchLogStatisticsListParam param,Long userId){
+
+        if (param.getSTime() == null || param.getETime() == null) {
+            return AjaxResult.error("请筛选时间后导出");
+        }
+
+        boolean res = qwWatchLogService.dateDifferenceMoreThan7Days(param.getSTime(), param.getETime());
+
+        if (res) {
+            return AjaxResult.error("导出数据时间差不能超过七天");
+        }
+
+        Long count = qwWatchLogService.getQwWatchLogStatisticsCount(param);
+        if (count > 10000) {
+            return AjaxResult.error("导出数据不可超过1w条");
+        }
+//        Long userId = loginUser.getUser().getUserId();
+        FsExportTask task = new FsExportTask();
+        task.setTaskType(2);
+        task.setStatus(0);
+        task.setStartTime(new Date());
+        task.setRemark("客户统计记录");
+        task.setSysType(2);
+        task.setCompanyUserId(userId);
+        exportTaskService.insertFsExportTask(task);
+        param.setTaskId(task.getTaskId());
+        qwWatchLogService.exportQwWatchLogStatistics(param, count);
+
+        return new AjaxResult(200, "后台正在导出,请等待...任务ID:" + task.getTaskId(), task.getTaskId());
+    }
+
+//    @GetMapping("/exportMyQwWatchLogStatistics")
+//    public AjaxResult exportMyQwWatchLogStatistics(QwWatchLogStatisticsListParam param) {
+//
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setCompanyUserId(loginUser.getUser().getUserId());
+//        param.setExcelName("我的客户统计记录导出");
+//        return exportQwWatchLogStatistics(param, loginUser.getUser().getUserId());
+//    }
+
+
+
+//    @GetMapping("/qwWatchLogAllStatisticsList")
+//    public TableDataInfo qwWatchLogAllStatisticsList(QwWatchLogStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<QwWatchLogAllStatisticsListVO> list = qwWatchLogService.selectQwWatchLogAllStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+//    @GetMapping("/myQwWatchLogAllStatisticsList")
+//    public TableDataInfo myQwWatchLogAllStatisticsList(QwWatchLogStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setCompanyUserId(loginUser.getUser().getUserId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<QwWatchLogAllStatisticsListVO> list = qwWatchLogService.selectQwWatchLogAllStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+
+    public AjaxResult exportQwWatchLogAllStatistics(QwWatchLogStatisticsListParam param, Long userId) {
+
+        if (param.getSTime() == null || param.getETime() == null) {
+            return AjaxResult.error("请筛选时间后导出");
+        }
+
+        boolean res = qwWatchLogService.dateDifferenceMoreThan7Days(param.getSTime(), param.getETime());
+
+        if (res) {
+            return AjaxResult.error("导出数据时间差不能超过七天");
+        }
+
+        Long count = qwWatchLogService.getQwWatchLogStatisticsCount(param);
+        if (count > 10000) {
+            return AjaxResult.error("导出数据不可超过1w条");
+        }
+//        Long userId = loginUser.getUser().getUserId();
+        FsExportTask task = new FsExportTask();
+        task.setTaskType(2);
+        task.setStatus(0);
+        task.setStartTime(new Date());
+        task.setRemark("我的客户统计记录");
+        task.setSysType(2);
+        task.setCompanyUserId(userId);
+        exportTaskService.insertFsExportTask(task);
+        param.setTaskId(task.getTaskId());
+        qwWatchLogService.exportQwWatchLogAllStatistics(param, count);
+
+        return new AjaxResult(200, "后台正在导出,请等待...任务ID:" + task.getTaskId(), task.getTaskId());
+    }
+
+//    @GetMapping("/exportMyQwWatchLogAllStatistics")
+//    public AjaxResult exportMyQwWatchLogAllStatistics(QwWatchLogStatisticsListParam param) {
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setCompanyUserId(loginUser.getUser().getUserId());
+//        param.setExcelName("我的数据汇总记录导出");
+//        return exportQwWatchLogAllStatistics(param, loginUser.getUser().getUserId());
+//    }
+
+//    @GetMapping("/exportQwWatchLogStatisticsAll")
+//    public AjaxResult exportQwWatchLogStatisticsAll(QwWatchLogStatisticsListParam param) {
+//
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        param.setExcelName("数据汇总记录导出");
+//        return exportQwWatchLogAllStatistics(param, loginUser.getUser().getUserId());
+//
+//    }
+
+//    @GetMapping("/watchLogStatistics")
+//    public TableDataInfo watchLogStatistics(FsCourseOverParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<FsCourseOverVO> list = fsCourseWatchLogService.selectFsCourseWatchLogOverStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+//
+//    @GetMapping("/watchLogStatisticsExport")
+//    public AjaxResult watchLogStatisticsExport(FsCourseOverParam param) {
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return AjaxResult.error("请选择时间");
+//        }
+//        List<FsCourseOverVO> list = fsCourseWatchLogService.selectFsCourseWatchLogOverStatisticsListVO(param);
+//        ExcelUtil<FsCourseOverVO> util = new ExcelUtil<FsCourseOverVO>(FsCourseOverVO.class);
+//        return util.exportExcel(list, "完课数据");
+//    }
+
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:userStatisticsList')")
+//    @GetMapping("/userStatisticsList")
+//    public TableDataInfo userStatisticsList(FsCourseUserStatisticsListParam param) {
+//        startPage();
+//        LoginUser loginUser = tokenService.getLoginUser(ServletUtils.getRequest());
+//        param.setCompanyId(loginUser.getCompany().getCompanyId());
+//        if (param.getSTime() == null || param.getETime() == null) {
+//            return getDataTable(new ArrayList<>());
+//        }
+//        List<FsCourseUserStatisticsListVO> list = fsCourseWatchLogService.selectFsCourseUserStatisticsListVO(param);
+//        return getDataTable(list);
+//    }
+
+    /**
+     * 导出短链课程看课记录列表
+     */
+    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:export')")
+    @Log(title = "短链课程看课记录", businessType = BusinessType.EXPORT)
+    @GetMapping("/export")
+    public AjaxResult export(AppCourseWatchLogListParam param)
+    {
+        // 导出
+        List<AppCourseWatchLogDto> list = appCourseWatchLogService.selectAppCourseWatchLogListVO(param);
+        ExcelUtil<AppCourseWatchLogDto> util = new ExcelUtil<AppCourseWatchLogDto>(AppCourseWatchLogDto.class);
+        return util.exportExcel(list, "短链课程看课记录数据");
+//        return exportCourseWatchLog(param);// 放导出任务
+    }
+
+    /**
+     * 导出短链课程看客记录
+     *
+     * @param param 参数
+     * @return AjaxResult
+     */
+    private AjaxResult exportCourseWatchLog(AppCourseWatchLogListParam param) {
+        LoginUser loginUser = getLoginUser();
+        logger.info("导出看课记录");
+        if (appCourseWatchLogService.isEntityNull(param)) {
+            return AjaxResult.error("请筛选数据导出");
+        }
+
+        Long userId = loginUser.getUser().getUserId();
+        Integer exportType = exportTaskService.isExportType1(userId);
+        if (exportType > 0) {
+            return AjaxResult.error("你已经有正在导出的任务");
+        }
+
+        if(param.getSTime() == null || param.getETime() == null || (param.getSTime().getTime() > param.getETime().getTime())) {
+            return AjaxResult.error("导出任务的创建时间范围只能是一天!");
+        }
+
+        Long count = appCourseWatchLogService.selectAppCourseWatchLogListVOCount(param);
+        if (count > 10000) {
+            return AjaxResult.error("导出数据不可超过1w条");
+        }
+
+        FsExportTask task = new FsExportTask();
+        task.setTaskType(2);
+        task.setStatus(0);
+        task.setStartTime(new Date());
+        task.setRemark("app短链课程看课记录");
+        task.setSysType(1);
+        task.setCompanyUserId(userId);
+        exportTaskService.insertFsExportTask(task);
+        param.setTaskId(task.getTaskId());
+
+        appCourseWatchLogService.exportData(param, count);
+        return new AjaxResult(200, "后台正在导出,请等待...任务ID:" + task.getTaskId(), task.getTaskId());
+    }
+
+//    private AjaxResult exportCommon(Object param, Long userId, String remark, Long limitCount,  String getCountMethodName, String exportMethodName, Class<?> clazz) {
+//        AjaxResult res = new AjaxResult();
+//        try {
+//            res = fsCourseWatchLogService.exportCommon(param, userId, remark, limitCount, getCountMethodName, exportMethodName, clazz);
+//        } catch (Exception ex) {
+//            ex.printStackTrace();
+//        }
+//        return res;
+//    }
+
+//    /**
+//     * 导出短链课程看课记录列表
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:myExport')")
+//    @Log(title = "短链课程看课记录", businessType = BusinessType.EXPORT)
+//    @GetMapping("/myExport")
+//    public AjaxResult myExport(FsCourseWatchLogListParam param)
+//    {
+//        // 导出
+//        return exportCourseWatchLog(param);
+//    }
+//    /**
+//     * 获取短链课程看课记录详细信息
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:query')")
+//    @GetMapping(value = "/{logId}")
+//    public AjaxResult getInfo(@PathVariable("logId") Long logId) {
+//        return AjaxResult.success(fsCourseWatchLogService.selectFsCourseWatchLogByLogId(logId));
+//    }
+//
+//    /**
+//     * 新增短链课程看课记录
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:add')")
+//    @Log(title = "短链课程看课记录", businessType = BusinessType.INSERT)
+//    @PostMapping
+//    public AjaxResult add(@RequestBody FsCourseWatchLog fsCourseWatchLog) {
+//        return toAjax(fsCourseWatchLogService.insertFsCourseWatchLog(fsCourseWatchLog));
+//    }
+//
+//    /**
+//     * 修改短链课程看课记录
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:edit')")
+//    @Log(title = "短链课程看课记录", businessType = BusinessType.UPDATE)
+//    @PutMapping
+//    public AjaxResult edit(@RequestBody FsCourseWatchLog fsCourseWatchLog) {
+//        return toAjax(fsCourseWatchLogService.updateFsCourseWatchLog(fsCourseWatchLog));
+//    }
+//
+//    /**
+//     * 删除短链课程看课记录
+//     */
+//    @PreAuthorize("@ss.hasPermi('course:courseWatchLog:remove')")
+//    @Log(title = "短链课程看课记录", businessType = BusinessType.DELETE)
+//    @DeleteMapping("/{logIds}")
+//    public AjaxResult remove(@PathVariable Long[] logIds) {
+//        return toAjax(fsCourseWatchLogService.deleteFsCourseWatchLogByLogIds(logIds));
+//    }
+
+    /**
+     * 昵称模糊查询客服分页
+     */
+    @GetMapping("/getCustomerListLikeName")
+    public R getCustomerListLikeName(@RequestParam(required = false) String name,
+                                     @RequestParam(required = false, defaultValue = "1") Integer pageNum,
+                                     @RequestParam(required = false, defaultValue = "10") Integer pageSize) {
+        PageHelper.startPage(pageNum,pageSize);
+        List<OptionVO> appCustomerRoles = appCustomerRoleService.getCustomerListLikeName(name);
+        return R.ok().put("data",new PageInfo<>(appCustomerRoles));
+    }
+
+
+}

+ 5 - 0
fs-admin/src/main/java/com/fs/his/controller/FsPackageController.java

@@ -207,4 +207,9 @@ public class FsPackageController extends BaseController
         return R.ok().put("data", new PageInfo<>(list));
     }
 
+    @GetMapping("/getOptions")
+    public AjaxResult getOptions(String keyword, Long metaId, Long limit) {
+        return AjaxResult.success(fsPackageService.getOptions(keyword, metaId, limit));
+    }
+
 }

+ 308 - 0
fs-admin/src/main/java/com/fs/utils/AudioUtils.java

@@ -0,0 +1,308 @@
+package com.fs.utils;
+
+import com.fs.common.exception.ServiceException;
+import com.fs.system.oss.CloudStorageService;
+import com.fs.system.oss.OSSFactory;
+
+import java.io.File;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.net.HttpURLConnection;
+import java.net.URL;
+import java.nio.file.Files;
+import java.text.SimpleDateFormat;
+import java.util.ArrayList;
+import java.util.Date;
+import java.util.List;
+
+public class AudioUtils {
+
+    /**
+     * 工具地址
+     **/
+    static String path = "c:\\";
+    static String destinationDir = "c:\\hook\\";
+
+    /**
+     * 从网络 URL 转换 MP3/WAV到SILk格式
+     *
+     * @param audioUrl  音频文件的 URL
+     * @param isSource  是否清空原文件
+     * @return SILK 文件路径
+     */
+    public static String transferAudioSilkFromUrl(String audioUrl, boolean isSource) {
+        try {
+            // 下载文件到本地临时路径
+            File tempFile = downloadFileFromUrl(audioUrl);
+            if (tempFile == null) {
+                throw new ServiceException("下载文件失败");
+            }
+
+            String silkUrl = transferAudioSilk(tempFile.getParent()+"\\", tempFile.getName(), isSource);
+            // 删除临时文件
+            tempFile.delete();
+
+            // 调用原来的转换方法
+            return silkUrl;
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+        return null;
+    }
+    /**
+     * MP3/WAV转SILk格式
+     *
+     * @param path 文件路径 例:D:\\file\\
+     * @param name 文件名称 例:audio.mp3/audio.wav
+     * @param isSource 是否清空原文件
+     * @return silk文件路径
+     * @throws Exception
+     */
+    public static String transferAudioSilk(String path, String name, boolean isSource) {
+        try {
+            // 判断后缀格式
+            String suffix = name.split("\\.")[1];
+            if (!suffix.toLowerCase().equals("mp3") && !suffix.toLowerCase().equals("wav")) {
+                throw new ServiceException("文件格式必须是mp3/wav");
+            }
+            String filePath = path + name;
+            File file = new File(filePath);
+            if (!file.exists()) {
+                throw new Exception("文件不存在!");
+            }
+            // 文件名时拼接
+            SimpleDateFormat ttime = new SimpleDateFormat("yyyyMMddhhMMSS");
+            String time = ttime.format(new Date());
+            // 导出的pcm格式路径
+            String pcmPath = path + "PCM_" + time + ".pcm";
+            // 先将mp3/wav转换成pcm格式
+            transferAudioPcm(filePath, pcmPath);
+            // 导出的silk格式路径
+            String silkPath = path + "SILK_" + time + ".silk";
+            // 转换成silk格式
+            transferPcmSilk(pcmPath, silkPath);
+            // 删除pcm文件
+            File pcmFile = new File(pcmPath);
+            if (pcmFile.exists()) {
+                pcmFile.delete();
+            }
+            if (isSource) {
+                File audioFile = new File(filePath);
+
+                if (audioFile.exists()) {
+                    audioFile.delete();
+                }
+            }
+
+            File silkFile = new File(silkPath);
+            //上传silk文件
+            String silkPathSuffix = silkPath.split("\\.")[1];
+            CloudStorageService storage = OSSFactory.build();
+            byte[] fileBytes = Files.readAllBytes(silkFile.toPath());
+            String silkUrl = storage.uploadSuffix(fileBytes, "."+silkPathSuffix);
+
+            if (silkFile.exists()) {
+                silkFile.delete();
+            }
+            return silkUrl;
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+        return null;
+    }
+
+    /**
+     * mp3/wav 通用
+     * @param fpath
+     * @param target
+     */
+    private static void transferAudioPcm(String fpath, String target) {
+        List<String> commend = new ArrayList<String>();
+        commend.add(path + "ffmpeg.exe");
+        commend.add("-y");
+        commend.add("-i");
+        commend.add(fpath);
+        commend.add("-f");
+        commend.add("s16le");
+        commend.add("-ar");
+        commend.add("24000");
+        commend.add("-ac");
+        commend.add("1");
+        commend.add(target);
+        try {
+            ProcessBuilder builder = new ProcessBuilder();
+            builder.command(commend);
+            Process p = builder.start();
+            p.waitFor();
+            p.destroy();
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+    }
+
+    /**
+     * silk_v3_encoder.exe,转成Silk格式
+     * @param pcmPath pcm 文件地址
+     * @param target  转换后的silk地址
+     * silk_v3_encoder.exe 路径
+     * pcm文件地址
+     * silk输出地址
+     * -Fs_API <Hz>            : API sampling rate in Hz, default: 24000
+     * -Fs_maxInternal <Hz>    : Maximum internal sampling rate in Hz, default: 24000
+     * -packetlength <ms>      : Packet interval in ms, default: 20
+     * -rate <bps>            : Target bitrate;   default: 25000
+     * -loss <perc>          : Uplink loss estimate, in percent (0-100);  default: 0
+     * -complexity <comp>   : Set complexity, 0: low, 1: medium, 2: high; default: 2
+     * -DTX <flag>          : Enable DTX (0/1); default: 0
+     * -quiet               : Print only some basic values
+     * -tencent             : Compatible with QQ/Wechat
+     */
+    public static void transferPcmSilk(String pcmPath, String target) {
+        Process process = null;
+        try {
+            /**
+             // 1、这一节的,语音长度太长会使音频长度丢失
+             List<String> commend = new ArrayList<>();
+             // 指令,可参照方法注释, 请不要在commend.add()里同时写【-参数 值】
+             commend.add(path + "silk_v3_encoder.exe");
+             commend.add(pcmPath);
+             commend.add(target);
+             commend.add("-tencent");
+             ProcessBuilder builder = new ProcessBuilder();
+             builder.command(commend);
+             process = builder.start();
+             // 如果删除下班这行写process.waitFor() ,太长的语音会阻塞,BufferedReader 打印出来太长的语音也会阻塞
+             process = Runtime.getRuntime().exec("taskkill -f -t -im silk_v3_encoder.exe");
+             */
+            // 方法2,除了会弹出弹窗,没什么问题 cmd /c 极为重要,执行完毕后会自动关闭
+            process = Runtime.getRuntime().exec("cmd /c start " + path + "silk_v3_encoder.exe " + pcmPath + " " + target + " -tencent");
+            process .waitFor();
+            Thread.sleep(1000);
+            // 有更好的方法会后续慢慢更新..
+        } catch (Exception e) {
+            e.printStackTrace();
+        } finally {
+            if (process != null) {
+                process.destroy();
+            }
+        }
+    }
+    /**
+     * 下载文件从网络 URL 到本地临时文件夹
+     *
+     * @param fileUrl 文件的网络 URL
+     * @return 下载到本地的临时文件
+     */
+    private static File downloadFileFromUrl(String fileUrl) {
+        InputStream inputStream = null;
+        FileOutputStream outputStream = null;
+        try {
+            // 创建 HTTP 连接
+            URL url = new URL(fileUrl);
+            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
+            // 设置Referer请求头
+            connection.setRequestProperty("Referer", "cos.his.cdwjyyh.com");
+            connection.setRequestMethod("GET");
+            connection.connect();
+
+            // 检查是否成功连接
+            if (connection.getResponseCode() != 200) {
+                throw new ServiceException("无法下载音频文件,HTTP 响应码:" + connection.getResponseCode());
+            }
+
+            // 获取输入流
+            inputStream = connection.getInputStream();
+
+            // 创建临时文件,并指定存放地址
+            String tempFileName = "temp_" + System.currentTimeMillis() + "_" + getFileExtension(fileUrl);
+            File destinationDirectory = new File(destinationDir);
+
+            // 确保目标目录存在
+            if (!destinationDirectory.exists()) {
+                destinationDirectory.mkdirs();
+            }
+
+            // 将文件保存到指定路径
+            File tempFile = new File(destinationDirectory, tempFileName);
+
+            // 写入文件
+            outputStream = new FileOutputStream(tempFile);
+            byte[] buffer = new byte[8192];
+            int bytesRead;
+            while ((bytesRead = inputStream.read(buffer)) != -1) {
+                outputStream.write(buffer, 0, bytesRead);
+            }
+
+            return tempFile;
+        } catch (Exception e) {
+            e.printStackTrace();
+        } finally {
+            try {
+                if (inputStream != null) inputStream.close();
+                if (outputStream != null) outputStream.close();
+            } catch (IOException e) {
+                e.printStackTrace();
+            }
+        }
+        return null;
+    }
+
+//    private static File downloadFileFromUrl(String fileUrl)
+//    {
+//        InputStream inputStream = null;
+//        FileOutputStream outputStream = null;
+//        try {
+//            // 创建 HTTP 连接
+//            URL url = new URL(fileUrl);
+//            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
+//            connection.setRequestMethod("GET");
+//            connection.connect();
+//
+//            // 检查是否成功连接
+//            if (connection.getResponseCode() != 200) {
+//                throw new ServiceException("无法下载音频文件,HTTP 响应码:" + connection.getResponseCode());
+//            }
+//
+//            // 获取输入流
+//            inputStream = connection.getInputStream();
+//
+//            // 创建临时文件
+//            String tempFileName = "temp_" + System.currentTimeMillis() + "_" + getFileExtension(fileUrl);
+//            File tempFile = new File(tempFileName);
+//
+//            // 写入文件
+//            outputStream = new FileOutputStream(tempFile);
+//            byte[] buffer = new byte[8192];
+//            int bytesRead;
+//            while ((bytesRead = inputStream.read(buffer)) != -1) {
+//                outputStream.write(buffer, 0, bytesRead);
+//            }
+//
+//            return tempFile;
+//        } catch (Exception e) {
+//            e.printStackTrace();
+//        } finally {
+//            try {
+//                if (inputStream != null) inputStream.close();
+//                if (outputStream != null) outputStream.close();
+//            } catch (IOException e) {
+//                e.printStackTrace();
+//            }
+//        }
+//        return null;
+//    }
+
+    /**
+     * 获取文件扩展名
+     *
+     * @param fileUrl 文件路径
+     * @return 文件扩展名
+     */
+    private static String getFileExtension(String fileUrl) {
+        return fileUrl.substring(fileUrl.lastIndexOf("."));
+    }
+
+    // 省略其他方法的实现
+
+}

+ 136 - 0
fs-admin/src/main/java/com/fs/web/controller/system/AppDeptController.java

@@ -0,0 +1,136 @@
+package com.fs.web.controller.system;
+
+import com.fs.common.annotation.Log;
+import com.fs.common.constant.UserConstants;
+import com.fs.common.core.controller.BaseController;
+import com.fs.common.core.domain.AjaxResult;
+import com.fs.common.core.domain.entity.AppDept;
+import com.fs.common.enums.BusinessType;
+import com.fs.common.utils.StringUtils;
+import com.fs.system.service.IAppDeptService;
+import org.apache.commons.lang3.ArrayUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.security.access.prepost.PreAuthorize;
+import org.springframework.validation.annotation.Validated;
+import org.springframework.web.bind.annotation.*;
+
+import java.util.Iterator;
+import java.util.List;
+
+/**
+ * 部门信息
+ */
+@RestController
+@RequestMapping("/app/dept")
+public class AppDeptController extends BaseController {
+    @Autowired
+    private IAppDeptService deptService;
+
+    /**
+     * 获取部门列表
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:list')")
+    @GetMapping("/list")
+    public AjaxResult list(AppDept dept) {
+        List<AppDept> depts = deptService.selectDeptList(dept);
+        return AjaxResult.success(depts);
+    }
+
+    /**
+     * 查询部门列表(排除节点)
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:list')")
+    @GetMapping("/list/exclude/{deptId}")
+    public AjaxResult excludeChild(@PathVariable(value = "deptId", required = false) Long deptId) {
+        List<AppDept> depts = deptService.selectDeptList(new AppDept());
+        Iterator<AppDept> it = depts.iterator();
+        while (it.hasNext()) {
+            AppDept d = (AppDept) it.next();
+            if (d.getDeptId().intValue() == deptId
+                    || ArrayUtils.contains(StringUtils.split(d.getAncestors(), ","), deptId + "")) {
+                it.remove();
+            }
+        }
+        return AjaxResult.success(depts);
+    }
+
+    /**
+     * 根据部门编号获取详细信息
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:query')")
+    @GetMapping(value = "/{deptId}")
+    public AjaxResult getInfo(@PathVariable Long deptId) {
+        deptService.checkDeptDataScope(deptId);
+        return AjaxResult.success(deptService.selectDeptById(deptId));
+    }
+
+    /**
+     * 获取部门下拉树列表
+     */
+    @GetMapping("/treeselect")
+    public AjaxResult treeselect(AppDept dept) {
+        List<AppDept> depts = deptService.selectDeptList(dept);
+        return AjaxResult.success(deptService.buildDeptTreeSelect(depts));
+    }
+
+    /**
+     * 加载对应角色部门列表树
+     */
+    @GetMapping(value = "/roleDeptTreeselect/{roleId}")
+    public AjaxResult roleDeptTreeselect(@PathVariable("roleId") Long roleId) {
+        List<AppDept> depts = deptService.selectDeptList(new AppDept());
+        AjaxResult ajax = AjaxResult.success();
+        ajax.put("checkedKeys", deptService.selectDeptListByRoleId(roleId));
+        ajax.put("depts", deptService.buildDeptTreeSelect(depts));
+        return ajax;
+    }
+
+    /**
+     * 新增部门
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:add')")
+    @Log(title = "部门管理", businessType = BusinessType.INSERT)
+    @PostMapping
+    public AjaxResult add(@Validated @RequestBody AppDept dept) {
+        if (UserConstants.NOT_UNIQUE.equals(deptService.checkDeptNameUnique(dept))) {
+            return AjaxResult.error("新增部门'" + dept.getDeptName() + "'失败,部门名称已存在");
+        }
+        dept.setCreateBy(getUsername());
+        return toAjax(deptService.insertDept(dept));
+    }
+
+    /**
+     * 修改部门
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:edit')")
+    @Log(title = "部门管理", businessType = BusinessType.UPDATE)
+    @PutMapping
+    public AjaxResult edit(@Validated @RequestBody AppDept dept) {
+        if (UserConstants.NOT_UNIQUE.equals(deptService.checkDeptNameUnique(dept))) {
+            return AjaxResult.error("修改部门'" + dept.getDeptName() + "'失败,部门名称已存在");
+        } else if (dept.getParentId().equals(dept.getDeptId())) {
+            return AjaxResult.error("修改部门'" + dept.getDeptName() + "'失败,上级部门不能是自己");
+        } else if (StringUtils.equals(UserConstants.DEPT_DISABLE, dept.getStatus())
+                && deptService.selectNormalChildrenDeptById(dept.getDeptId()) > 0) {
+            return AjaxResult.error("该部门包含未停用的子部门!");
+        }
+        dept.setUpdateBy(getUsername());
+        return toAjax(deptService.updateDept(dept));
+    }
+
+    /**
+     * 删除部门
+     */
+    @PreAuthorize("@ss.hasPermi('system:dept:remove')")
+    @Log(title = "部门管理", businessType = BusinessType.DELETE)
+    @DeleteMapping("/{deptId}")
+    public AjaxResult remove(@PathVariable Long deptId) {
+        if (deptService.hasChildByDeptId(deptId)) {
+            return AjaxResult.error("存在下级部门,不允许删除");
+        }
+        if (deptService.checkDeptExistUser(deptId)) {
+            return AjaxResult.error("部门存在邀请码,不允许删除");
+        }
+        return toAjax(deptService.deleteDeptById(deptId));
+    }
+}

+ 39 - 0
fs-common/src/main/java/com/fs/common/constant/VideoRedisKeyConst.java

@@ -0,0 +1,39 @@
+package com.fs.common.constant;
+
+public interface VideoRedisKeyConst {
+
+    String ROOT_PREFIX = "video_app:";
+
+    /**
+     * 防刷与幂等校验 Key 前缀
+     * 结构: video_app:event:check:{userId}:{videoId}:{event}
+     */
+    String EVENT_CHECK_PREFIX = ROOT_PREFIX + "event:check:";
+
+    /**
+     * 视频统计指标 Hash Key 前缀
+     * 结构: video_app:metrics:{videoId}
+     */
+    String VIDEO_METRICS_PREFIX = ROOT_PREFIX + "metrics:{";
+
+    /**
+     * 用户对视频的临时权重增量 Hash Key 前缀
+     * 结构: video_app:user_weight:{userId}
+     */
+    String USER_VIDEO_WEIGHT_PREFIX = ROOT_PREFIX + "user_weight:{";
+
+    /**
+     * 定时任务同步专用的后缀
+     */
+    String SYNC_SUFFIX = ":sync";
+    String KEY_END = "}";
+    /**
+     * 待同步到 MySQL 的视频 ID 集合
+     */
+    String PENDING_SYNC_VIDEOS = ROOT_PREFIX + "pending:videos";
+
+    /**
+     * 待同步到 MySQL 的用户 ID 集合
+     */
+    String PENDING_SYNC_USERS = ROOT_PREFIX + "pending:users";
+}

+ 198 - 0
fs-common/src/main/java/com/fs/common/core/domain/entity/AppDept.java

@@ -0,0 +1,198 @@
+package com.fs.common.core.domain.entity;
+
+import com.fs.common.core.domain.BaseEntity;
+import org.apache.commons.lang3.builder.ToStringBuilder;
+import org.apache.commons.lang3.builder.ToStringStyle;
+
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.Size;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * app部门表 app_dept
+ */
+public class AppDept extends BaseEntity {
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 部门ID
+     */
+    private Long deptId;
+
+    /**
+     * 父部门ID
+     */
+    private Long parentId;
+
+    /**
+     * 祖级列表
+     */
+    private String ancestors;
+
+    /**
+     * 部门名称
+     */
+    private String deptName;
+
+    /**
+     * 显示顺序
+     */
+    private String orderNum;
+
+    /**
+     * 负责人
+     */
+    private String leader;
+
+    /**
+     * 联系电话
+     */
+    private String phone;
+
+    /**
+     * 邮箱
+     */
+    private String email;
+
+    /**
+     * 部门状态:0正常,1停用
+     */
+    private String status;
+
+    /**
+     * 删除标志(0代表存在 2代表删除)
+     */
+    private String delFlag;
+
+    /**
+     * 父部门名称
+     */
+    private String parentName;
+
+    /**
+     * 子部门
+     */
+    private List<AppDept> children = new ArrayList<AppDept>();
+
+    public Long getDeptId() {
+        return deptId;
+    }
+
+    public void setDeptId(Long deptId) {
+        this.deptId = deptId;
+    }
+
+    public Long getParentId() {
+        return parentId;
+    }
+
+    public void setParentId(Long parentId) {
+        this.parentId = parentId;
+    }
+
+    public String getAncestors() {
+        return ancestors;
+    }
+
+    public void setAncestors(String ancestors) {
+        this.ancestors = ancestors;
+    }
+
+    @NotBlank(message = "部门名称不能为空")
+    @Size(min = 0, max = 30, message = "部门名称长度不能超过30个字符")
+    public String getDeptName() {
+        return deptName;
+    }
+
+    public void setDeptName(String deptName) {
+        this.deptName = deptName;
+    }
+
+
+    public String getOrderNum() {
+        return orderNum;
+    }
+
+    public void setOrderNum(String orderNum) {
+        this.orderNum = orderNum;
+    }
+
+    public String getLeader() {
+        return leader;
+    }
+
+    public void setLeader(String leader) {
+        this.leader = leader;
+    }
+
+    @Size(min = 0, max = 11, message = "联系电话长度不能超过11个字符")
+    public String getPhone() {
+        return phone;
+    }
+
+    public void setPhone(String phone) {
+        this.phone = phone;
+    }
+
+
+    public String getEmail() {
+        return email;
+    }
+
+    public void setEmail(String email) {
+        this.email = email;
+    }
+
+    public String getStatus() {
+        return status;
+    }
+
+    public void setStatus(String status) {
+        this.status = status;
+    }
+
+    public String getDelFlag() {
+        return delFlag;
+    }
+
+    public void setDelFlag(String delFlag) {
+        this.delFlag = delFlag;
+    }
+
+    public String getParentName() {
+        return parentName;
+    }
+
+    public void setParentName(String parentName) {
+        this.parentName = parentName;
+    }
+
+    public List<AppDept> getChildren() {
+        return children;
+    }
+
+    public void setChildren(List<AppDept> children) {
+        this.children = children;
+    }
+
+    @Override
+    public String toString() {
+        return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE)
+                .append("deptId", getDeptId())
+                .append("parentId", getParentId())
+                .append("ancestors", getAncestors())
+                .append("deptName", getDeptName())
+                .append("orderNum", getOrderNum())
+                .append("leader", getLeader())
+                .append("phone", getPhone())
+                .append("email", getEmail())
+                .append("status", getStatus())
+                .append("delFlag", getDelFlag())
+                .append("createBy", getCreateBy())
+                .append("createTime", getCreateTime())
+                .append("updateBy", getUpdateBy())
+                .append("updateTime", getUpdateTime())
+                .toString();
+    }
+}

+ 3 - 1
fs-common/src/main/java/com/fs/common/enums/DataSourceType.java

@@ -22,5 +22,7 @@ public enum DataSourceType
      * 从库
      */
     SLAVE,
-    SopREAD
+    SopREAD,
+    ZEROMALL,
+    CIVILGOODS,
 }

+ 4 - 2
fs-qw-api-msg/src/main/java/com/fs/app/msgarchives/job/QwMsgAuditScheduleJob.java

@@ -13,7 +13,9 @@ import lombok.extern.slf4j.Slf4j;
 import org.springframework.scheduling.annotation.Scheduled;
 import org.springframework.stereotype.Component;
 import org.springframework.util.StringUtils;
+import org.springframework.web.bind.annotation.GetMapping;
 
+import java.util.Collections;
 import java.util.List;
 
 /**
@@ -37,7 +39,7 @@ public class QwMsgAuditScheduleJob {
     /**
      * 每1分钟执行一次,拉取会话存档消息并存库
      */
-    @Scheduled(cron = "0 */1 * * * ?")
+//    @Scheduled(cron = "0 */1 * * * ?")
     public void pullQwMsgAndStore() {
         List<String> corpIds = qwCompanyMapper.selectQwCompanyCorpIdListByAll();
         if (corpIds == null || corpIds.isEmpty()) {
@@ -55,7 +57,7 @@ public class QwMsgAuditScheduleJob {
     /**
      * 每2分钟执行一次,扫描 media_oss_url为空的媒体消息(语音/视频/图片),下载并上传至 OSS
      */
-    @Scheduled(cron = "0 */2 * * * ?")
+//    @Scheduled(cron = "0 */2 * * * ?")
     public void uploadMediaToOss() {
         List<String> corpIds = qwCompanyMapper.selectQwCompanyCorpIdListByAll();
         if (corpIds == null || corpIds.isEmpty()) {

+ 267 - 0
fs-service/src/main/java/com/fs/app/civilgoods/domain/CivilGoods.java

@@ -0,0 +1,267 @@
+package com.fs.app.civilgoods.domain;
+
+import com.baomidou.mybatisplus.annotation.IdType;
+import com.baomidou.mybatisplus.annotation.TableId;
+import com.baomidou.mybatisplus.annotation.TableName;
+import lombok.Data;
+
+import java.math.BigDecimal;
+import java.time.LocalDateTime;
+
+@Data
+@TableName("fs_store_product")
+public class CivilGoods {
+
+    /**
+     * 商品id
+     */
+    @TableId(type = IdType.AUTO)
+    private Long productId;
+
+    /**
+     * 商品图片
+     */
+    private String image;
+
+    /**
+     * 轮播图
+     */
+    private String sliderImage;
+
+    /**
+     * 商品名称
+     */
+    private String productName;
+
+    /**
+     * 商品简介
+     */
+    private String productInfo;
+
+    /**
+     * 关键字
+     */
+    private String keyword;
+
+    /**
+     * 产品条码(一维码)
+     */
+    private String barCode;
+
+    /**
+     * 分类id
+     */
+    private Long cateId;
+
+    /**
+     * 商品价格
+     */
+    private BigDecimal price;
+
+    /**
+     * 会员价格
+     */
+    private BigDecimal vipPrice;
+
+    /**
+     * 市场价
+     */
+    private BigDecimal otPrice;
+
+    /**
+     * 代理价格
+     */
+    private BigDecimal agentPrice;
+
+    /**
+     * 邮费
+     */
+    private BigDecimal postage;
+
+    /**
+     * 单位名
+     */
+    private String unitName;
+
+    /**
+     * 排序
+     */
+    private Short sort;
+
+    /**
+     * 销量
+     */
+    private Integer sales;
+
+    /**
+     * 库存
+     */
+    private Integer stock;
+
+    /**
+     * 状态(0:未上架,1:上架)
+     */
+    private Integer isShow;
+
+    /**
+     * 是否热卖
+     */
+    private Integer isHot;
+
+    /**
+     * 是否优惠
+     */
+    private Integer isBenefit;
+
+    /**
+     * 是否精品
+     */
+    private Integer isBest;
+
+    /**
+     * 是否新品
+     */
+    private Integer isNew;
+
+    /**
+     * 产品描述
+     */
+    private String description;
+
+    /**
+     * 添加时间
+     */
+    private LocalDateTime createTime;
+
+    /**
+     * 更新时间
+     */
+    private LocalDateTime updateTime;
+
+    /**
+     * 是否包邮
+     */
+    private Integer isPostage;
+
+    /**
+     * 是否删除
+     */
+    private Integer isDel;
+
+    /**
+     * 获得积分
+     */
+    private BigDecimal giveIntegral;
+
+    /**
+     * 成本价
+     */
+    private BigDecimal cost;
+
+    /**
+     * 是否优品推荐
+     */
+    private Integer isGood;
+
+    /**
+     * 浏览量
+     */
+    private Integer browse;
+
+    /**
+     * 产品二维码地址(用户小程序海报)
+     */
+    private String codePath;
+
+    /**
+     * 运费模板ID
+     */
+    private Integer tempId;
+
+    /**
+     * 规格 0单 1多
+     */
+    private Integer specType;
+
+    /**
+     * 是否开启积分兑换
+     */
+    private Integer isIntegral;
+
+    /**
+     * 需要多少积分兑换 只在开启积分兑换时生效
+     */
+    private Integer integral;
+
+    /**
+     * 商品类型:1非处方 2处方
+     */
+    private Integer productType;
+
+    /**
+     * 国药准字
+     */
+    private String prescribeCode;
+
+    /**
+     * 规格
+     */
+    private String prescribeSpec;
+
+    /**
+     * 生产厂家
+     */
+    private String prescribeFactory;
+
+    /**
+     * 处方名
+     */
+    private String prescribeName;
+
+    /**
+     * 是否在商品展示
+     */
+    private Integer isDisplay;
+
+    /**
+     * 商品推广分类
+     */
+    private Integer tuiCateId;
+
+    /**
+     * 仓库id
+     */
+    private Integer warehouseId;
+
+    /**
+     * 仓库code
+     */
+    private String warehouseCode;
+
+    /**
+     * 服用方法
+     */
+    private String usageMethod;
+
+    /**
+     * 一日几次
+     */
+    private String frequency;
+
+    /**
+     * 用药数量
+     */
+    private String dosage;
+
+    /**
+     * 税收分类码
+     */
+    private String taxClassificationCode;
+
+    /**
+     * 商品发票名称
+     */
+    private String invoiceName;
+
+
+}

+ 17 - 0
fs-service/src/main/java/com/fs/app/civilgoods/dto/CivilGoodsDTO.java

@@ -0,0 +1,17 @@
+package com.fs.app.civilgoods.dto;
+
+import com.fs.app.civilgoods.domain.CivilGoods;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+@Data
+public class CivilGoodsDTO extends CivilGoods {
+
+    private String keyword;
+
+    private Long metaId;
+
+    private Long limit = 50L;
+
+}

+ 11 - 0
fs-service/src/main/java/com/fs/app/civilgoods/mapper/CivilGoodsMapper.java

@@ -0,0 +1,11 @@
+package com.fs.app.civilgoods.mapper;
+
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import com.fs.app.civilgoods.domain.CivilGoods;
+import org.apache.ibatis.annotations.Mapper;
+
+@Mapper
+public interface CivilGoodsMapper extends BaseMapper<CivilGoods> {
+
+
+}

+ 16 - 0
fs-service/src/main/java/com/fs/app/civilgoods/service/ICivilGoodsService.java

@@ -0,0 +1,16 @@
+package com.fs.app.civilgoods.service;
+
+import com.baomidou.mybatisplus.extension.service.IService;
+import com.fs.app.civilgoods.domain.CivilGoods;
+import com.fs.app.civilgoods.dto.CivilGoodsDTO;
+import com.fs.app.civilgoods.vo.CivilGoodsVO;
+
+import java.util.List;
+
+public interface ICivilGoodsService extends IService<CivilGoods> {
+
+    CivilGoods getByProductId(Long productId);
+
+    List<CivilGoodsVO> findOptions(CivilGoodsDTO req);
+
+}

+ 69 - 0
fs-service/src/main/java/com/fs/app/civilgoods/service/impl/CivilGoodsServiceImpl.java

@@ -0,0 +1,69 @@
+package com.fs.app.civilgoods.service.impl;
+
+import cn.hutool.core.util.ObjectUtil;
+import com.baomidou.mybatisplus.extension.service.additional.query.impl.LambdaQueryChainWrapper;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import com.fs.app.civilgoods.domain.CivilGoods;
+import com.fs.app.civilgoods.dto.CivilGoodsDTO;
+import com.fs.app.civilgoods.mapper.CivilGoodsMapper;
+import com.fs.app.civilgoods.service.ICivilGoodsService;
+import com.fs.app.civilgoods.vo.CivilGoodsVO;
+import com.fs.common.BeanCopyUtils;
+import com.fs.common.annotation.DataSource;
+import com.fs.common.enums.DataSourceType;
+import org.springframework.stereotype.Service;
+
+import java.util.ArrayList;
+import java.util.List;
+
+@Service
+@DataSource(DataSourceType.CIVILGOODS)
+public class CivilGoodsServiceImpl extends ServiceImpl<CivilGoodsMapper, CivilGoods> implements ICivilGoodsService {
+
+    @Override
+    public CivilGoods getByProductId(Long productId) {
+        return this.lambdaQuery()
+                .eq(CivilGoods::getProductId, productId)
+                .one();
+    }
+
+    @Override
+    public List<CivilGoodsVO> findOptions(CivilGoodsDTO req) {
+        if (ObjectUtil.isNotEmpty(req.getMetaId())) {
+            CivilGoods topProduct = this.lambdaQuery()
+                    .eq(CivilGoods::getProductId, req.getMetaId())
+                    .eq(CivilGoods::getIsShow, 1)
+                    .eq(CivilGoods::getIsDel, 0)
+                    .one();
+
+            LambdaQueryChainWrapper<CivilGoods> otherQuery = this.lambdaQuery();
+            otherQuery.eq(CivilGoods::getIsShow, 1);
+            otherQuery.eq(CivilGoods::getIsDel, 0);
+
+            if (ObjectUtil.isNotEmpty(req.getKeyword())) {
+                otherQuery.and(q -> q.like(CivilGoods::getProductName, req.getKeyword())
+                        .or()
+                        .eq(CivilGoods::getProductId, req.getKeyword()));
+            }
+            otherQuery.ne(CivilGoods::getProductId, req.getMetaId())
+                    .last(" LIMIT " + (req.getLimit() - 1L));
+            List<CivilGoods> otherList = otherQuery.list();
+            List<CivilGoods> finalList = new ArrayList<>();
+            if (topProduct != null) {
+                finalList.add(topProduct);
+            }
+            finalList.addAll(otherList);
+            return BeanCopyUtils.copyList(finalList, CivilGoodsVO.class);
+        }
+        LambdaQueryChainWrapper<CivilGoods> lambdaQuery = this.lambdaQuery();
+        lambdaQuery.eq(CivilGoods::getIsShow, 1);
+        lambdaQuery.eq(CivilGoods::getIsDel, 0);
+        if (ObjectUtil.isNotEmpty(req.getKeyword())) {
+            lambdaQuery.and(q -> q.like(CivilGoods::getProductName, req.getKeyword())
+                    .or()
+                    .eq(CivilGoods::getProductId, req.getKeyword()));
+        }
+        lambdaQuery.last(" LIMIT " + req.getLimit());
+        return BeanCopyUtils.copyList(lambdaQuery.list(), CivilGoodsVO.class);
+    }
+}

+ 10 - 0
fs-service/src/main/java/com/fs/app/civilgoods/vo/CivilGoodsVO.java

@@ -0,0 +1,10 @@
+package com.fs.app.civilgoods.vo;
+
+import com.fs.app.civilgoods.domain.CivilGoods;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+@Data
+public class CivilGoodsVO extends CivilGoods {
+}

+ 252 - 0
fs-service/src/main/java/com/fs/app/medicines/domain/AppFsStoreProduct.java

@@ -0,0 +1,252 @@
+package com.fs.app.medicines.domain;
+
+import com.baomidou.mybatisplus.annotation.IdType;
+import com.baomidou.mybatisplus.annotation.TableId;
+import com.baomidou.mybatisplus.annotation.TableName;
+import lombok.Data;
+
+import java.math.BigDecimal;
+import java.time.LocalDateTime;
+
+@Data
+@TableName("fs_store_product")
+public class AppFsStoreProduct {
+
+    /**
+     * 主键ID(根据你的表自行调整)
+     */
+    @TableId(type = IdType.AUTO)
+    private Long productId;
+
+    /**
+     * 商品图片
+     */
+    private String image;
+
+    /**
+     * 轮播图
+     */
+    private String sliderImage;
+
+    /**
+     * 商品名称
+     */
+    private String productName;
+
+    /**
+     * 商品简介
+     */
+    private String productInfo;
+
+    /**
+     * 关键字
+     */
+    private String keyword;
+
+    /**
+     * 产品条码(一维码)
+     */
+    private String barCode;
+
+    /**
+     * 分类id
+     */
+    private Long cateId;
+
+    /**
+     * 商品价格
+     */
+    private BigDecimal price;
+
+    /**
+     * 会员价格
+     */
+    private BigDecimal vipPrice;
+
+    /**
+     * 市场价
+     */
+    private BigDecimal otPrice;
+
+    /**
+     * 代理价格
+     */
+    private BigDecimal agentPrice;
+
+    /**
+     * 邮费
+     */
+    private BigDecimal postage;
+
+    /**
+     * 单位名
+     */
+    private String unitName;
+
+    /**
+     * 排序
+     */
+    private Short sort;
+
+    /**
+     * 销量
+     */
+    private Integer sales;
+
+    /**
+     * 库存
+     */
+    private Integer stock;
+
+    /**
+     * 状态(0:未上架,1:上架)
+     */
+    private Integer isShow;
+
+    /**
+     * 是否热卖
+     */
+    private Integer isHot;
+
+    /**
+     * 是否优惠
+     */
+    private Integer isBenefit;
+
+    /**
+     * 是否精品
+     */
+    private Integer isBest;
+
+    /**
+     * 是否新品
+     */
+    private Integer isNew;
+
+    /**
+     * 产品描述
+     */
+    private String description;
+
+    /**
+     * 添加时间
+     */
+    private LocalDateTime createTime;
+
+    /**
+     * 更新时间
+     */
+    private LocalDateTime updateTime;
+
+    /**
+     * 是否包邮
+     */
+    private Integer isPostage;
+
+    /**
+     * 是否删除
+     */
+    private Integer isDel;
+
+    /**
+     * 获得积分
+     */
+    private BigDecimal giveIntegral;
+
+    /**
+     * 成本价
+     */
+    private BigDecimal cost;
+
+    /**
+     * 是否优品推荐
+     */
+    private Integer isGood;
+
+    /**
+     * 浏览量
+     */
+    private Integer browse;
+
+    /**
+     * 产品二维码地址(用户小程序海报)
+     */
+    private String codePath;
+
+    /**
+     * 运费模板ID
+     */
+    private Integer tempId;
+
+    /**
+     * 规格 0单 1多
+     */
+    private Integer specType;
+
+    /**
+     * 是开启积分兑换
+     */
+    private Integer isIntegral;
+
+    /**
+     * 需要多少积分兑换 只在开启积分兑换时生效
+     */
+    private Integer integral;
+
+    /**
+     * 商品类型:1非处方 2处方
+     */
+    private Integer productType;
+
+    /**
+     * 国药准字
+     */
+    private String prescribeCode;
+
+    /**
+     * 规格
+     */
+    private String prescribeSpec;
+
+    /**
+     * 生产厂家
+     */
+    private String prescribeFactory;
+
+    /**
+     * 处方名
+     */
+    private String prescribeName;
+
+    /**
+     * 仓库id
+     */
+    private Long warehouseId;
+
+    /**
+     * 仓库code
+     */
+    private String warehouseCode;
+
+    /**
+     * 是否在商品展示
+     */
+    private Integer isDisplay;
+
+    /**
+     * 商品推广分类
+     */
+    private Integer tuiCateId;
+
+    /**
+     * 是否免除运费
+     */
+    private Integer isFreePostage;
+
+    /**
+     * 是否免除服务费
+     */
+    private Integer isFreeService;
+
+
+}

+ 17 - 0
fs-service/src/main/java/com/fs/app/medicines/dto/AppFsStoreProductDTO.java

@@ -0,0 +1,17 @@
+package com.fs.app.medicines.dto;
+
+import com.fs.app.medicines.domain.AppFsStoreProduct;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+@Data
+public class AppFsStoreProductDTO extends AppFsStoreProduct {
+
+    private Long metaId;
+
+    private String keyword;
+
+    private Long limit;
+
+}

+ 9 - 0
fs-service/src/main/java/com/fs/app/medicines/mapper/AppFsStoreProductMapper.java

@@ -0,0 +1,9 @@
+package com.fs.app.medicines.mapper;
+
+import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import com.fs.app.medicines.domain.AppFsStoreProduct;
+import org.apache.ibatis.annotations.Mapper;
+
+@Mapper
+public interface AppFsStoreProductMapper extends BaseMapper<AppFsStoreProduct> {
+}

+ 16 - 0
fs-service/src/main/java/com/fs/app/medicines/service/IAppFsStoreProductService.java

@@ -0,0 +1,16 @@
+package com.fs.app.medicines.service;
+
+import com.baomidou.mybatisplus.extension.service.IService;
+import com.fs.app.medicines.domain.AppFsStoreProduct;
+import com.fs.app.medicines.dto.AppFsStoreProductDTO;
+import com.fs.app.medicines.vo.AppFsStoreProductVO;
+
+import java.util.List;
+
+public interface IAppFsStoreProductService extends IService<AppFsStoreProduct> {
+
+    AppFsStoreProduct getByProductId(Long productId);
+
+    List<AppFsStoreProductVO> findOptions(AppFsStoreProductDTO req);
+
+}

+ 69 - 0
fs-service/src/main/java/com/fs/app/medicines/service/impl/AppFsStoreProductServiceImpl.java

@@ -0,0 +1,69 @@
+package com.fs.app.medicines.service.impl;
+
+import cn.hutool.core.util.ObjectUtil;
+import com.baomidou.mybatisplus.extension.service.additional.query.impl.LambdaQueryChainWrapper;
+import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
+import com.fs.app.medicines.domain.AppFsStoreProduct;
+import com.fs.app.medicines.dto.AppFsStoreProductDTO;
+import com.fs.app.medicines.mapper.AppFsStoreProductMapper;
+import com.fs.app.medicines.service.IAppFsStoreProductService;
+import com.fs.app.medicines.vo.AppFsStoreProductVO;
+import com.fs.common.BeanCopyUtils;
+import com.fs.common.annotation.DataSource;
+import com.fs.common.enums.DataSourceType;
+import org.springframework.stereotype.Service;
+
+import java.util.ArrayList;
+import java.util.List;
+
+@Service
+@DataSource(DataSourceType.ZEROMALL)//零利润商城/药品商城
+public class AppFsStoreProductServiceImpl extends ServiceImpl<AppFsStoreProductMapper, AppFsStoreProduct> implements IAppFsStoreProductService {
+
+
+    @Override
+    public AppFsStoreProduct getByProductId(Long productId) {
+        return this.getById(productId);
+    }
+
+    @Override
+    public List<AppFsStoreProductVO> findOptions(AppFsStoreProductDTO req) {
+        if (ObjectUtil.isNotEmpty(req.getMetaId())) {
+            AppFsStoreProduct topProduct = this.lambdaQuery()
+                    .eq(AppFsStoreProduct::getProductId, req.getMetaId())
+                    .eq(AppFsStoreProduct::getIsShow, 1)
+                    .eq(AppFsStoreProduct::getIsDel, 0)
+                    .one();
+
+            LambdaQueryChainWrapper<AppFsStoreProduct> otherQuery = this.lambdaQuery();
+            otherQuery.eq(AppFsStoreProduct::getIsShow, 1);
+            otherQuery.eq(AppFsStoreProduct::getIsDel, 0);
+            if (ObjectUtil.isNotEmpty(req.getKeyword())) {
+                otherQuery.and(q -> q.like(AppFsStoreProduct::getProductName, req.getKeyword())
+                        .or()
+                        .eq(AppFsStoreProduct::getProductId, req.getKeyword()));
+            }
+            otherQuery.ne(AppFsStoreProduct::getProductId, req.getMetaId())
+                    .last(" LIMIT " + (req.getLimit() - 1L));
+
+            List<AppFsStoreProduct> otherList = otherQuery.list();
+            List<AppFsStoreProduct> finalList = new ArrayList<>();
+            if (topProduct != null) {
+                finalList.add(topProduct);
+            }
+            finalList.addAll(otherList);
+            return BeanCopyUtils.copyList(finalList, AppFsStoreProductVO.class);
+        }
+
+        LambdaQueryChainWrapper<AppFsStoreProduct> lambdaQuery = this.lambdaQuery();
+        lambdaQuery.eq(AppFsStoreProduct::getIsShow, 1);
+        lambdaQuery.eq(AppFsStoreProduct::getIsDel, 0);
+        if (ObjectUtil.isNotEmpty(req.getKeyword())) {
+            lambdaQuery.and(q -> q.like(AppFsStoreProduct::getProductName, req.getKeyword())
+                    .or()
+                    .eq(AppFsStoreProduct::getProductId, req.getKeyword()));
+        }
+        lambdaQuery.last(" LIMIT " + req.getLimit());
+        return BeanCopyUtils.copyList(lambdaQuery.list(), AppFsStoreProductVO.class);
+    }
+}

+ 10 - 0
fs-service/src/main/java/com/fs/app/medicines/vo/AppFsStoreProductVO.java

@@ -0,0 +1,10 @@
+package com.fs.app.medicines.vo;
+
+import com.fs.app.medicines.domain.AppFsStoreProduct;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+@Data
+public class AppFsStoreProductVO extends AppFsStoreProduct {
+}

+ 1 - 1
fs-service/src/main/java/com/fs/app/sender/properties/ConfigProperties.java

@@ -18,7 +18,7 @@ public class ConfigProperties {
      * IM默认接收人,当且仅当isDebug为true时生效
      * 测试环境尽量开启,避免配置错误,导致发送给了真实客户
      */
-    private String receiver = "4052861305";
+    private String receiver = "";
 
     /**
      * IM发送人账号前缀

+ 4 - 0
fs-service/src/main/java/com/fs/app/sop/dto/AppSopUserLogInfoDTO.java

@@ -32,4 +32,8 @@ public class AppSopUserLogInfoDTO {
 
     private Long[] tagIds;
 
+    private Long startIndex;
+
+    private Long pageLimit;
+
 }

+ 2 - 0
fs-service/src/main/java/com/fs/app/sop/mapper/AppSopUserLogInfoMapper.java

@@ -26,4 +26,6 @@ public interface AppSopUserLogInfoMapper extends BaseMapper<AppSopUserLogsInfos>
 
     int handleRepeatData();
 
+    Long findListTotal(AppSopUserLogInfoDTO dto);
+
 }

+ 2 - 1
fs-service/src/main/java/com/fs/app/sop/service/IAppSopUserLogInfoService.java

@@ -5,12 +5,13 @@ import com.fs.app.sop.domain.AppSopUserLogsInfos;
 import com.fs.app.sop.dto.AppSopUserLogInfoDTO;
 import com.fs.app.sop.vo.AppSopUserLogInfoVO;
 import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
 
 import java.util.List;
 
 public interface IAppSopUserLogInfoService extends IService<AppSopUserLogsInfos> {
 
-    List<AppSopUserLogInfoVO> findList(AppSopUserLogInfoDTO dto);
+    TableDataInfo findList(AppSopUserLogInfoDTO dto);
 
     int deleteByIds(List<Long> ids);
 

+ 17 - 6
fs-service/src/main/java/com/fs/app/sop/service/impl/AppSopUserLogInfoServiceImpl.java

@@ -11,7 +11,9 @@ import com.fs.app.sop.service.IAppSopService;
 import com.fs.app.sop.service.IAppSopUserLogInfoService;
 import com.fs.app.sop.service.IAppSopUserLogService;
 import com.fs.app.sop.vo.AppSopUserLogInfoVO;
+import com.fs.common.constant.HttpStatus;
 import com.fs.common.core.domain.R;
+import com.fs.common.core.page.TableDataInfo;
 import lombok.extern.slf4j.Slf4j;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.beans.factory.annotation.Value;
@@ -22,10 +24,7 @@ import org.springframework.transaction.annotation.Transactional;
 
 import java.time.LocalDateTime;
 import java.time.ZoneId;
-import java.util.Arrays;
-import java.util.Date;
-import java.util.List;
-import java.util.UUID;
+import java.util.*;
 
 @Slf4j
 @Service
@@ -43,8 +42,20 @@ public class AppSopUserLogInfoServiceImpl extends ServiceImpl<AppSopUserLogInfoM
 
 
     @Override
-    public List<AppSopUserLogInfoVO> findList(AppSopUserLogInfoDTO dto) {
-        return baseMapper.findList(dto);
+    public TableDataInfo findList(AppSopUserLogInfoDTO dto) {
+        Long listTotal = this.baseMapper.findListTotal(dto);
+        TableDataInfo tableDataInfo = new TableDataInfo();
+        tableDataInfo.setCode(HttpStatus.SUCCESS);
+        tableDataInfo.setMsg("查询成功");
+        tableDataInfo.setTotal(listTotal);
+        tableDataInfo.setRows(new ArrayList<>());
+        if (listTotal > 0L) {
+            Long startIndex = (dto.getStartIndex() - 1L) * dto.getPageLimit();
+            dto.setStartIndex(startIndex);
+            List<AppSopUserLogInfoVO> userLogInfoVOS = baseMapper.findList(dto);
+            tableDataInfo.setRows(userLogInfoVOS);
+        }
+        return tableDataInfo;
     }
 
     @Override

+ 3 - 0
fs-service/src/main/java/com/fs/company/mapper/CompanyUserMapper.java

@@ -368,4 +368,7 @@ public interface CompanyUserMapper
             "update company_user set ai_sip_call_user_id=#{aiSipCallId} where user_id=#{companyUserId} " +
             "</script>")
     public int updateCompanyUserByAiSipCall(@Param("companyUserId") Long companyUserId, @Param("aiSipCallId") Long aiSipCallId);
+
+    List<CompanyUser> findCompanyUserOptions(@Param("keyword") String keyword, @Param("selectedId") Long selectedId,
+                                             @Param("pageNumber") Integer pageNumber,  @Param("pageSize") Integer pageSize);
 }

+ 2 - 0
fs-service/src/main/java/com/fs/company/service/ICompanyUserService.java

@@ -311,4 +311,6 @@ public interface ICompanyUserService {
      * 获取销售绑定的fs_user
      */
     int countCompanyUserByUserId(Long userId);
+
+    List<OptionsVO> findCompanyUserOptions(String keyword, Long selectedId, Integer pageNumber,  Integer pageSize);
 }

+ 16 - 0
fs-service/src/main/java/com/fs/company/service/impl/CompanyUserServiceImpl.java

@@ -1227,4 +1227,20 @@ public class CompanyUserServiceImpl implements ICompanyUserService
         return 0;
     }
 
+    @Override
+    public List<OptionsVO> findCompanyUserOptions(String keyword, Long selectedId, Integer pageNumber, Integer pageSize) {
+        //计算起始位置
+        pageNumber = (pageNumber - 1) * pageSize;
+        return Optional.ofNullable(companyUserMapper.findCompanyUserOptions(keyword, selectedId, pageNumber, pageSize))
+                .orElse(new ArrayList<>())
+                .stream()
+                .map(cu -> {
+                    OptionsVO vo = new OptionsVO();
+                    vo.setDictLabel(cu.getNickName());
+                    vo.setDictValue(cu.getUserId());
+                    return vo;
+                })
+                .collect(Collectors.toList());
+    }
+
 }

+ 15 - 0
fs-service/src/main/java/com/fs/course/dto/VideoUpdateDTO.java

@@ -0,0 +1,15 @@
+package com.fs.course.dto;
+
+import lombok.AllArgsConstructor;
+import lombok.Data;
+
+import java.io.Serializable;
+
+@Data
+@AllArgsConstructor
+public class VideoUpdateDTO implements Serializable {
+    private Long id;
+    private long playCount;
+    private long play3sCount;
+    private long completeCount;
+}

+ 2 - 0
fs-service/src/main/java/com/fs/course/mapper/FsUserCourseVideoMapper.java

@@ -350,4 +350,6 @@ public interface FsUserCourseVideoMapper extends BaseMapper<FsUserCourseVideo> {
             "</script>")
     int countOnShelfCourseByVideoIds(@Param("videoIds") String[] videoIds);
 
+    List<FsUserCourseVideo> findOptions(@Param("keyword") String keyword, @Param("metaId") Long metaId, @Param("limit") Long limit);
+
 }

+ 17 - 0
fs-service/src/main/java/com/fs/course/mapper/FsUserVideoMapper.java

@@ -2,6 +2,7 @@ package com.fs.course.mapper;
 
 import java.util.List;
 import com.fs.course.domain.FsUserVideo;
+import com.fs.course.dto.VideoUpdateDTO;
 import com.fs.course.param.FsUserVideoAuditParam;
 import com.fs.course.param.FsUserVideoListUParam;
 import com.fs.course.vo.FsUserVideoListPVO;
@@ -9,6 +10,7 @@ import com.fs.course.vo.FsUserVideoPVO;
 import com.fs.course.param.FsUserVideoParam;
 import com.fs.course.vo.FsUserVideoListUVO;
 import com.fs.course.vo.VideoListVO;
+import org.apache.ibatis.annotations.Insert;
 import org.apache.ibatis.annotations.Param;
 import org.apache.ibatis.annotations.Select;
 import org.apache.ibatis.annotations.Update;
@@ -292,5 +294,20 @@ public interface FsUserVideoMapper
             "</script>"})
     List<VideoListVO> getVideoByIds(@Param("ids") List<String> ids);
 
+    @Insert("<script>" +
+            "INSERT INTO fs_user_video (video_id, views, play_3s_count, complete_count) VALUES " +
+            "<foreach collection='list' item='item' separator=','>" +
+            "(#{item.id}, #{item.playCount}, #{item.play3sCount}, #{item.completeCount})" +
+            "</foreach> " +
+            "AS new_data " +
+            "ON DUPLICATE KEY UPDATE " +
+            "views = fs_user_video.views + new_data.views, " +
+            "play_3s_count = fs_user_video.play_3s_count + new_data.play_3s_count, " +
+            "complete_count = fs_user_video.complete_count + new_data.complete_count" +
+            "</script>")
+    void batchIncrementMetrics(@Param("list") List<VideoUpdateDTO> list);
+
+    List<FsUserVideo> findOptions(@Param("keyword") String keyword, @Param("metaId") Long metaId, @Param("limit") Long limit);
+
 }
 

+ 2 - 0
fs-service/src/main/java/com/fs/course/service/IFsUserCourseVideoService.java

@@ -296,4 +296,6 @@ public interface IFsUserCourseVideoService extends IService<FsUserCourseVideo> {
     * app发红包 1 自动发课 2 手动发课
     */
     R withdrawal(FsCourseSendRewardUParam param,Integer type);
+
+    List<FsUserCourseVideo> findOptions(String keyword, Long metaId, Long limit);
 }

+ 4 - 0
fs-service/src/main/java/com/fs/course/service/IFsUserVideoService.java

@@ -4,6 +4,7 @@ import java.util.List;
 
 import com.fs.common.core.domain.R;
 import com.fs.course.domain.FsUserVideo;
+import com.fs.course.dto.VideoUpdateDTO;
 import com.fs.course.param.*;
 import com.fs.course.vo.FsUserVideoListPVO;
 import com.fs.course.vo.FsUserVideoPVO;
@@ -116,4 +117,7 @@ public interface IFsUserVideoService {
 
     List<VideoListVO> getVideoByIds(List<String> ids);
 
+    void batchIncrementMetrics(List<VideoUpdateDTO> updateBatchList);
+
+    List<FsUserVideo> findOptions(String keyword, Long metaId, Long limit);
 }

+ 6 - 0
fs-service/src/main/java/com/fs/course/service/impl/FsUserCourseVideoServiceImpl.java

@@ -5636,5 +5636,11 @@ public class FsUserCourseVideoServiceImpl extends ServiceImpl<FsUserCourseVideoM
 
         return user.getMpOpenId();
     }
+
+    @Override
+    public List<FsUserCourseVideo> findOptions(String keyword, Long metaId, Long limit) {
+        return fsUserCourseVideoMapper.findOptions(keyword, metaId, limit);
+    }
+
 }
 

+ 11 - 0
fs-service/src/main/java/com/fs/course/service/impl/FsUserVideoServiceImpl.java

@@ -16,6 +16,7 @@ import com.fs.common.utils.bean.BeanUtils;
 import com.fs.course.domain.FsUserCourseVideo;
 import com.fs.course.dto.VideoFavoriteStatusDTO;
 import com.fs.course.dto.VideoLikeStatusDTO;
+import com.fs.course.dto.VideoUpdateDTO;
 import com.fs.course.mapper.*;
 import com.fs.course.param.*;
 import com.fs.course.service.IRecommendationService;
@@ -578,4 +579,14 @@ public class FsUserVideoServiceImpl implements IFsUserVideoService {
         return fsUserVideoMapper.getVideoByIds(ids);
     }
 
+    @Override
+    public void batchIncrementMetrics(List<VideoUpdateDTO> updateBatchList) {
+        fsUserVideoMapper.batchIncrementMetrics(updateBatchList);
+    }
+
+    @Override
+    public List<FsUserVideo> findOptions(String keyword, Long metaId, Long limit) {
+        return fsUserVideoMapper.findOptions(keyword, metaId, limit);
+    }
+
 }

+ 3 - 0
fs-service/src/main/java/com/fs/his/mapper/FsArticleMapper.java

@@ -112,4 +112,7 @@ public interface FsArticleMapper
             "</if>" +
             "</script>"})
     List<FsArticleListQueryVO> selectFsArticleListQuery(@Param("maps") FsArticleQueryParam param);
+
+    List<FsArticle> findOptions(@Param("keyword") String keyword, @Param("metaId") Long metaId, @Param("limit") Long limit);
+
 }

+ 2 - 0
fs-service/src/main/java/com/fs/his/mapper/FsPackageMapper.java

@@ -166,4 +166,6 @@ public interface FsPackageMapper
      * 根据套餐包id集合查询列表
      */
     List<FsGoodsVO> getFsGoodsVOListByIds(@Param("packageIds") List<Long> packageIds);
+
+    List<FsPackageListVO> getOptions(@Param("keyword") String keyword, @Param("metaId") Long metaId, @Param("limit") Long limit);
 }

+ 14 - 0
fs-service/src/main/java/com/fs/his/mapper/FsStoreOrderMapper.java

@@ -1,6 +1,7 @@
 package com.fs.his.mapper;
 
 import java.math.BigDecimal;
+import java.time.LocalDateTime;
 import java.util.List;
 import java.util.Map;
 
@@ -1209,4 +1210,17 @@ public interface FsStoreOrderMapper
     FsStoreOrderAmountStatsVo selectFsStoreOrderAmountStats(FsStoreOrderAmountStatsQueryDto queryDto);
 
     List<FsStoreOrder> selectOutTimeOrderList(@Param("unPayTime")Integer unPayTime);
+
+    @Select("<script>" +
+            "SELECT o.user_id, SUM(o.pay_price) as daily_sum " +
+            "SUM(o.pay_price &lt; 8) as small_count " +
+            "FROM fs_store_order o " +
+            "INNER JOIN fs_user u ON o.user_id = u.user_id " +
+            "WHERE o.status = 4 " +
+            "AND u.history_app IS NOT NULL " +
+            "AND o.finish_time &gt;= #{startTime} " +
+            "AND o.finish_time &lt; #{endTime} " +
+            "GROUP BY o.user_id " +
+            "</script>")
+    List<UserAmountVO> getSumAmount(@Param("startTime") LocalDateTime startTime, @Param("endTime") LocalDateTime endTime);
 }

+ 1 - 1
fs-service/src/main/java/com/fs/his/mapper/FsUserIntegralLogsMapper.java

@@ -77,7 +77,7 @@ public interface FsUserIntegralLogsMapper
             "LEFT JOIN fs_user u ON u.user_id=l.user_id "+
             " <where>  \n" +
             "            <if test=\"userId != null \"> and l.user_id = #{userId}</if>\n" +
-            "            <if test=\"nick_name != null \"> and u.nick_name = #{nickName}</if>\n" +
+            "            <if test=\"nickName != null \"> and u.nick_name = #{nickName}</if>\n" +
             "            <if test=\"logType != null  and logType != ''\"> and log_type = #{logType}</if>\n" +
             "            <if test=\"phone != null \"> and u.phone = #{phone}</if>\n" +
             "            <if test=\"businessId != null  and businessId != ''\"> and business_id = #{businessId}</if>\n" +

+ 2 - 0
fs-service/src/main/java/com/fs/his/service/IFsArticleService.java

@@ -73,4 +73,6 @@ public interface IFsArticleService
 
 
     List<FsArticleListQueryVO> selectFsArticleListQuery(FsArticleQueryParam param);
+
+    List<FsArticle> findOptions(String keyword, Long metaId, Long limit);
 }

+ 2 - 0
fs-service/src/main/java/com/fs/his/service/IFsPackageService.java

@@ -97,4 +97,6 @@ public interface IFsPackageService
      * 获取套餐包选择列表
      */
     List<FsPackageChooseVO> getChoosePackageListByMap(Map<String, Object> params);
+
+    List<FsPackageListVO> getOptions(String keyword, Long metaId, Long limit);
 }

+ 4 - 0
fs-service/src/main/java/com/fs/his/service/IFsStoreOrderService.java

@@ -3,6 +3,7 @@ package com.fs.his.service;
 import java.io.IOException;
 import java.math.BigDecimal;
 import java.text.ParseException;
+import java.time.LocalDateTime;
 import java.util.List;
 import java.util.Map;
 
@@ -294,4 +295,7 @@ public interface IFsStoreOrderService
     void weizouPushIntergral(Long l);
 
     void weizouPushScrm(Long l);
+
+    // 统计用户累计消费金额
+    List<UserAmountVO> getSumAmount(LocalDateTime startTime, LocalDateTime endTime);
 }

+ 6 - 0
fs-service/src/main/java/com/fs/his/service/impl/FsArticleServiceImpl.java

@@ -129,4 +129,10 @@ public class FsArticleServiceImpl implements IFsArticleService
         }
         return list;
     }
+
+    @Override
+    public List<FsArticle> findOptions(String keyword, Long metaId, Long limit) {
+        return fsArticleMapper.findOptions(keyword, metaId, limit);
+    }
+
 }

+ 5 - 0
fs-service/src/main/java/com/fs/his/service/impl/FsPackageServiceImpl.java

@@ -354,5 +354,10 @@ public class FsPackageServiceImpl implements IFsPackageService {
     public List<FsPackageChooseVO> getChoosePackageListByMap(Map<String, Object> params) {
         return fsPackageMapper.getChoosePackageListByMap(params);
     }
+
+    @Override
+    public List<FsPackageListVO> getOptions(String keyword, Long metaId, Long limit) {
+        return fsPackageMapper.getOptions(keyword, metaId, limit);
+    }
 }
 

+ 6 - 0
fs-service/src/main/java/com/fs/his/service/impl/FsStoreOrderServiceImpl.java

@@ -5351,4 +5351,10 @@ public class FsStoreOrderServiceImpl implements IFsStoreOrderService {
         }
         log.info("微走推送结果:{}",s);
     }
+
+    @Override
+    public List<UserAmountVO> getSumAmount(LocalDateTime startTime, LocalDateTime endTime) {
+        return fsStoreOrderMapper.getSumAmount(startTime, endTime);
+    }
+
 }

+ 91 - 0
fs-service/src/main/java/com/fs/qw/mapper/QwWatchLogMapper.java

@@ -1,6 +1,8 @@
 package com.fs.qw.mapper;
 
 import com.baomidou.mybatisplus.core.mapper.BaseMapper;
+import com.fs.common.annotation.DataSource;
+import com.fs.common.enums.DataSourceType;
 import com.fs.course.param.FsCourseWatchLogListParam;
 import com.fs.qw.domain.QwExternalContact;
 import com.fs.qw.domain.QwWatchLog;
@@ -430,5 +432,94 @@ public interface QwWatchLogMapper extends BaseMapper<QwWatchLog>{
     @Select("SELECT count(1) from qw_watch_log where fs_user_id=#{userId} and `day`=0")
     int selectQwWatchLogIsFirstByUserId(Long userId);
 
+    @Select({"<script> " +
+            "select count(*) from ( \n" +
+            "\t\t\t\t\t\tSELECT \n" +
+            "\t\t\t\t\t\tqec.qw_user_id\n" +
+            "            FROM\n" +
+            "                qw_external_contact qec\n" +
+            "              LEFT JOIN company_user cu on qec.company_user_id=cu.user_id \n" +
+            "WHERE\n" +
+            "    DATE(qec.create_time) &gt;= DATE(#{sTime}) and  DATE(qec.create_time) &lt;= DATE(#{eTime}) and qec.company_id =#{companyId} " +
+            "<if test ='nickName !=null and nickName!=\"\"'>\n" +
+            "   and qu.qw_user_name like concat( #{nickName}, '%')\n" +
+            "</if>" +
+            "<if test ='ids !=null and ids!=\"\"'>\n" +
+            "   and qec.qw_user_id in (${ids})\n" +
+            "</if>" +
+            "<if test = 'deptId != null    '>   AND (cu.dept_id = #{deptId} OR cu.dept_id IN ( SELECT t.dept_id FROM company_dept t WHERE find_in_set(#{deptId}, ancestors) )) </if>" +
+            "            GROUP BY\n" +
+            "                qec.qw_user_id, DATE(qec.create_time) \n" +
+            " ) t1" +
+            "</script>"})
+    @DataSource(DataSourceType.SLAVE)
+    Long getQwWatchLogStatisticsCount(QwWatchLogStatisticsListParam param);
+
+
+    @Select({"<script> " +
+            "SELECT\n" +
+            "    qec.qw_user_id id,\n" +
+            "    qu.qw_user_name AS qw_user_name, \n" +
+            "    DATE(qec.create_time) AS create_time, \n" +
+            "    COUNT(1) AS line,\n" +
+            "    COUNT(CASE WHEN qec.is_interact = 1 THEN 1 END) AS interact,\n" +
+            "    COUNT(CASE WHEN qec.`level` = 1 THEN 1 END) AS A,\n" +
+            "    COUNT(CASE WHEN qec.`level` = 2 THEN 1 END) AS B,\n" +
+            "    COUNT(CASE WHEN qec.`level` = 3 THEN 1 END) AS C,\n" +
+            "    COUNT(CASE WHEN qec.`level` = 4 THEN 1 END) AS D,\n" +
+            "    COUNT(CASE WHEN qec.fs_user_id IS NOT NULL THEN 1 END) AS sign,\n" +
+            "    COUNT(CASE WHEN qec.`status` =3 THEN 1 END) AS los,\n" +
+            "    COUNT(CASE WHEN qec.`status` IN (4, 5,6) THEN 1 END) AS del\n" +
+            "FROM\n" +
+            "    qw_external_contact qec\n" +
+            "JOIN\n" +
+            "    qw_user qu ON qec.qw_user_id = qu.id \n" +
+            "  LEFT JOIN company_user cu on qec.company_user_id=cu.user_id \n" +
+            "WHERE\n" +
+            "    DATE(qec.create_time) &gt;= DATE(#{sTime}) and  DATE(qec.create_time) &lt;= DATE(#{eTime}) and qec.company_id =#{companyId} " +
+            "<if test ='nickName !=null and nickName!=\"\"'>\n" +
+            "   and qu.qw_user_name like concat( #{nickName}, '%')\n" +
+            "</if>" +
+            "<if test ='ids !=null and ids!=\"\"'>\n" +
+            "   and qec.qw_user_id in (${ids})\n" +
+            "</if>" +
+            "<if test = 'deptId != null    '>   AND (cu.dept_id = #{deptId} OR cu.dept_id IN ( SELECT t.dept_id FROM company_dept t WHERE find_in_set(#{deptId}, ancestors) )) </if>" +
+            "GROUP BY\n" +
+            "    qec.qw_user_id, DATE(qec.create_time) \n" +
+            "ORDER BY\n" +
+            "    DATE(qec.create_time) "+
+            " <if test='limitSize != null'> LIMIT #{offset}, #{limitSize}</if>" +
+            "</script>"})
+    @DataSource(DataSourceType.SLAVE)
+    List<QwWatchLogStatisticsListVO> selectExportQwWatchLogStatistics(QwWatchLogStatisticsListParam param);
+
+    @Select({"<script> " +
+            "SELECT\n" +
+            "    qec.qw_user_id id,\n" +
+            "    qu.qw_user_name AS qw_user_name, \n" +
+            "    DATE(qec.create_time) AS create_time, \n" +
+            "    COUNT(1) AS line\n" +
+            "FROM\n" +
+            "    qw_external_contact qec\n" +
+            "JOIN\n" +
+            "    qw_user qu ON qec.qw_user_id = qu.id \n" +
+            "  LEFT JOIN company_user cu on qec.company_user_id=cu.user_id \n" +
+            "WHERE\n" +
+            "    DATE(qec.create_time) &gt;= DATE(#{sTime}) and  DATE(qec.create_time) &lt;= DATE(#{eTime}) and qec.company_id =#{companyId} " +
+            "<if test ='ids !=null and ids!=\"\"'>\n" +
+            "   and qec.qw_user_id in (${ids})\n" +
+            "</if>" +
+            "<if test ='nickName !=null and nickName!=\"\"'>\n" +
+            "   and qu.qw_user_name like concat( #{nickName}, '%')\n" +
+            "</if>" +
+            "<if test = 'deptId != null    '>   AND (cu.dept_id = #{deptId} OR cu.dept_id IN ( SELECT t.dept_id FROM company_dept t WHERE find_in_set(#{deptId}, ancestors) )) </if>" +
+            "GROUP BY\n" +
+            "    qec.qw_user_id, DATE(qec.create_time) \n" +
+            "ORDER BY\n" +
+            "    DATE(qec.create_time) "+
+            " <if test='limitSize != null'> LIMIT #{offset}, #{limitSize}</if>" +
+            "</script>"})
+    @DataSource(DataSourceType.SLAVE)
+    List<QwWatchLogAllStatisticsListVO> selectExportQwWatchLogAllStatistics(QwWatchLogStatisticsListParam param);
 
 }

+ 6 - 0
fs-service/src/main/java/com/fs/qw/param/QwWatchLogStatisticsListParam.java

@@ -46,6 +46,12 @@ public class QwWatchLogStatisticsListParam {
     private Long pageSize;
     private List<Long> filterDeptIds;
 
+    // 导出任务ID
+    private Long taskId;
+    private Integer limitSize;
+    private Long offset;
+    private String excelName;
+
     /**
      * 添加方式
      */

+ 9 - 0
fs-service/src/main/java/com/fs/qw/service/IQwWatchLogService.java

@@ -8,6 +8,7 @@ import com.fs.qw.param.QwWatchLogStatisticsListParam;
 import com.fs.qw.vo.QwWatchLogAllStatisticsListVO;
 import com.fs.qw.vo.QwWatchLogStatisticsListVO;
 
+import java.util.Date;
 import java.util.List;
 
 /**
@@ -83,4 +84,12 @@ public interface IQwWatchLogService extends IService<QwWatchLog>{
     List<QwWatchLogStatisticsListVO> selectQwWatchLogStatisticsListVOExport(FsCourseWatchLogListParam param);
 
     List<QwWatchLogStatisticsListVO> selectQwWatchLogStatisticsListVOExportExcludeTransfer(FsCourseWatchLogListParam param);
+
+    boolean dateDifferenceMoreThan7Days(Date date1, Date date2);
+
+    Long getQwWatchLogStatisticsCount(QwWatchLogStatisticsListParam param);
+
+    void exportQwWatchLogStatistics(QwWatchLogStatisticsListParam param,Long count);
+
+    void exportQwWatchLogAllStatistics(QwWatchLogStatisticsListParam param,Long count);
 }

+ 106 - 0
fs-service/src/main/java/com/fs/qw/service/impl/QwWatchLogServiceImpl.java

@@ -3,10 +3,12 @@ package com.fs.qw.service.impl;
 import com.baomidou.mybatisplus.core.toolkit.ObjectUtils;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 import com.fs.common.constant.HttpStatus;
+import com.fs.common.core.domain.AjaxResult;
 import com.fs.common.core.page.TableDataInfo;
 import com.fs.common.exception.CustomException;
 import com.fs.common.utils.DateUtils;
 import com.fs.common.utils.DictUtils;
+import com.fs.common.utils.poi.ExcelUtil;
 import com.fs.company.cache.ICompanyCacheService;
 import com.fs.company.cache.ICompanyUserCacheService;
 import com.fs.company.domain.Company;
@@ -18,7 +20,9 @@ import com.fs.course.domain.FsUserCourse;
 import com.fs.course.domain.FsUserCourseVideo;
 import com.fs.course.param.FsCourseWatchLogListParam;
 import com.fs.course.service.cache.IFsUserCourseVideoCacheService;
+import com.fs.his.domain.FsExportTask;
 import com.fs.his.domain.FsUser;
+import com.fs.his.service.IFsExportTaskService;
 import com.fs.qw.domain.QwWatchLog;
 import com.fs.qw.mapper.QwExternalContactMapper;
 import com.fs.qw.mapper.QwUserMapper;
@@ -33,9 +37,11 @@ import com.hc.openapi.tool.util.StringUtils;
 import org.apache.commons.collections4.CollectionUtils;
 import org.jetbrains.annotations.Nullable;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.scheduling.annotation.Async;
 import org.springframework.stereotype.Service;
 
 import java.util.*;
+import java.util.concurrent.TimeUnit;
 import java.util.stream.Collectors;
 
 /**
@@ -72,6 +78,9 @@ public class QwWatchLogServiceImpl extends ServiceImpl<QwWatchLogMapper, QwWatch
     @Autowired
     private IFsUserCacheService fsUserCacheService;
 
+    @Autowired
+    private IFsExportTaskService exportTaskService;
+
 
     /**
      * 查询企微看课
@@ -556,6 +565,103 @@ public class QwWatchLogServiceImpl extends ServiceImpl<QwWatchLogMapper, QwWatch
         return rspData;
     }
 
+    @Override
+    public boolean dateDifferenceMoreThan7Days(Date date1, Date date2) {
+        long diffInMillies = Math.abs(date2.getTime() - date1.getTime());
+        long diffInDays = TimeUnit.MILLISECONDS.toDays(diffInMillies);
+        return diffInDays > 7;
+    }
+
+    @Override
+    public Long getQwWatchLogStatisticsCount(QwWatchLogStatisticsListParam param) {
+
+        if (param.getCompanyUserId() != null) {
+            param.setIds(companyUserMapper.selectQwUserIdsByCompany(param.getCompanyUserId()));
+        }
+
+        return qwWatchLogMapper.getQwWatchLogStatisticsCount(param);
+    }
+
+    @Override
+    @Async
+    public void exportQwWatchLogStatistics(QwWatchLogStatisticsListParam param, Long count) {
+        if (param.getCompanyUserId() != null) {
+            param.setIds(companyUserMapper.selectQwUserIdsByCompany(param.getCompanyUserId()));
+        }
+        List<QwWatchLogStatisticsListVO> exportList = new ArrayList<>();
+        Long fileSize = 1000L;
+        Long totalPage = count % fileSize == 0 ? count / fileSize : count / fileSize + 1;
+        Long offset = 0L;
+        for (int i = 0; i < totalPage; i++) {
+            param.setLimitSize(fileSize.intValue());
+            param.setOffset(offset);
+            List<QwWatchLogStatisticsListVO> pageList = qwWatchLogMapper.selectExportQwWatchLogStatistics(param);
+            exportList.addAll(pageList);
+            offset = offset + fileSize;
+        }
+
+        if (null != exportList && !exportList.isEmpty()) {
+            for (QwWatchLogStatisticsListVO vo : exportList) {
+                Long id = vo.getId();
+                Date createTime = vo.getCreateTime();
+                QwWatchLogStatisticsListVO stat = qwWatchLogMapper.selectQwWatchLogByQwUserId(id, createTime);
+                vo.setD1Online(stat.getD1Online());
+                vo.setD1Over(stat.getD1Over());
+                vo.setFirstOnline(stat.getFirstOnline());
+                vo.setFirstOver(stat.getFirstOver());
+            }
+        }
+
+        ExcelUtil<QwWatchLogStatisticsListVO> util = new ExcelUtil<>(QwWatchLogStatisticsListVO.class);
+        AjaxResult result = util.exportExcel(exportList, param.getExcelName());
+
+        FsExportTask task = exportTaskService.selectFsExportTaskByTaskId(param.getTaskId());
+        task.setFinishTime(new Date());
+        task.setStatus(1);
+        task.setFileUrl(result.get("msg").toString());
+        exportTaskService.updateFsExportTask(task);
+    }
+
+    @Override
+    @Async
+    public void exportQwWatchLogAllStatistics(QwWatchLogStatisticsListParam param, Long count) {
+        if (param.getCompanyUserId() != null) {
+            param.setIds(companyUserMapper.selectQwUserIdsByCompany(param.getCompanyUserId()));
+        }
+        List<QwWatchLogAllStatisticsListVO> exportList = new ArrayList<>();
+        Long fileSize = 1000L;
+        Long totalPage = count % fileSize == 0 ? count / fileSize : count / fileSize + 1;
+        Long offset = 0L;
+        for (int i = 0; i < totalPage; i++) {
+            param.setLimitSize(fileSize.intValue());
+            param.setOffset(offset);
+            List<QwWatchLogAllStatisticsListVO> pageList = qwWatchLogMapper.selectExportQwWatchLogAllStatistics(param);
+            exportList.addAll(pageList);
+            offset = offset + fileSize;
+        }
+
+        ArrayList<QwWatchLogAllStatisticsListVO> list = new ArrayList<>();
+
+        for (QwWatchLogAllStatisticsListVO vo : exportList) {
+            Long id = vo.getId();
+            Date createTime = vo.getCreateTime();
+            QwWatchLogAllStatisticsListVO stat = qwWatchLogMapper.selectQwWatchLogAllStatisticsListVO(id, createTime);
+            stat.setCreateTime(vo.getCreateTime());
+            stat.setQwUserName(vo.getQwUserName());
+            stat.setLine(vo.getLine());
+            list.add(stat);
+        }
+
+        ExcelUtil<QwWatchLogAllStatisticsListVO> util = new ExcelUtil<>(QwWatchLogAllStatisticsListVO.class);
+        AjaxResult result = util.exportExcel(list, param.getExcelName());
+
+        FsExportTask task = exportTaskService.selectFsExportTaskByTaskId(param.getTaskId());
+        task.setFinishTime(new Date());
+        task.setStatus(1);
+        task.setFileUrl(result.get("msg").toString());
+        exportTaskService.updateFsExportTask(task);
+    }
+
     public List<Date> getDatesBetween(Date sTime, Date eTime) {
         List<Date> dates = new ArrayList<>();
         Calendar calendar = Calendar.getInstance();

+ 116 - 0
fs-service/src/main/java/com/fs/system/mapper/AppDeptMapper.java

@@ -0,0 +1,116 @@
+package com.fs.system.mapper;
+
+import com.fs.common.core.domain.entity.AppDept;
+import org.apache.ibatis.annotations.Param;
+
+import java.util.List;
+
+/**
+ * 部门管理 数据层
+ */
+public interface AppDeptMapper {
+    /**
+     * 查询部门管理数据
+     *
+     * @param dept 部门信息
+     * @return 部门信息集合
+     */
+    public List<AppDept> selectDeptList(AppDept dept);
+
+    /**
+     * 根据角色ID查询部门树信息
+     *
+     * @param roleId            角色ID
+     * @param deptCheckStrictly 部门树选择项是否关联显示
+     * @return 选中部门列表
+     */
+    public List<Integer> selectDeptListByRoleId(@Param("roleId") Long roleId, @Param("deptCheckStrictly") boolean deptCheckStrictly);
+
+    /**
+     * 根据部门ID查询信息
+     *
+     * @param deptId 部门ID
+     * @return 部门信息
+     */
+    public AppDept selectDeptById(Long deptId);
+
+    /**
+     * 根据ID查询所有子部门
+     *
+     * @param deptId 部门ID
+     * @return 部门列表
+     */
+    public List<AppDept> selectChildrenDeptById(Long deptId);
+
+    /**
+     * 根据ID查询所有子部门(正常状态)
+     *
+     * @param deptId 部门ID
+     * @return 子部门数
+     */
+    public int selectNormalChildrenDeptById(Long deptId);
+
+    /**
+     * 是否存在子节点
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    public int hasChildByDeptId(Long deptId);
+
+    /**
+     * 查询部门是否存在用户
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    public int checkDeptExistUser(Long deptId);
+
+    /**
+     * 校验部门名称是否唯一
+     *
+     * @param deptName 部门名称
+     * @param parentId 父部门ID
+     * @return 结果
+     */
+    public AppDept checkDeptNameUnique(@Param("deptName") String deptName, @Param("parentId") Long parentId);
+
+    /**
+     * 新增部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    public int insertDept(AppDept dept);
+
+    /**
+     * 修改部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    public int updateDept(AppDept dept);
+
+    /**
+     * 修改所在部门正常状态
+     *
+     * @param deptIds 部门ID组
+     */
+    public void updateDeptStatusNormal(Long[] deptIds);
+
+    /**
+     * 修改子元素关系
+     *
+     * @param depts 子元素
+     * @return 结果
+     */
+    public int updateDeptChildren(@Param("depts") List<AppDept> depts);
+
+    /**
+     * 删除部门管理信息
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    public int deleteDeptById(Long deptId);
+}

+ 114 - 0
fs-service/src/main/java/com/fs/system/service/IAppDeptService.java

@@ -0,0 +1,114 @@
+package com.fs.system.service;
+
+import com.fs.common.core.domain.TreeSelect;
+import com.fs.common.core.domain.entity.AppDept;
+
+import java.util.List;
+
+/**
+ * 部门管理 服务层
+ */
+public interface IAppDeptService {
+    /**
+     * 查询部门管理数据
+     *
+     * @param dept 部门信息
+     * @return 部门信息集合
+     */
+    public List<AppDept> selectDeptList(AppDept dept);
+
+    /**
+     * 构建前端所需要树结构
+     *
+     * @param depts 部门列表
+     * @return 树结构列表
+     */
+    public List<AppDept> buildDeptTree(List<AppDept> depts);
+
+    /**
+     * 构建前端所需要下拉树结构
+     *
+     * @param depts 部门列表
+     * @return 下拉树结构列表
+     */
+    public List<TreeSelect> buildDeptTreeSelect(List<AppDept> depts);
+
+    /**
+     * 根据角色ID查询部门树信息
+     *
+     * @param roleId 角色ID
+     * @return 选中部门列表
+     */
+    public List<Integer> selectDeptListByRoleId(Long roleId);
+
+    /**
+     * 根据部门ID查询信息
+     *
+     * @param deptId 部门ID
+     * @return 部门信息
+     */
+    public AppDept selectDeptById(Long deptId);
+
+    /**
+     * 根据ID查询所有子部门(正常状态)
+     *
+     * @param deptId 部门ID
+     * @return 子部门数
+     */
+    public int selectNormalChildrenDeptById(Long deptId);
+
+    /**
+     * 是否存在部门子节点
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    public boolean hasChildByDeptId(Long deptId);
+
+    /**
+     * 查询部门是否存在用户
+     *
+     * @param deptId 部门ID
+     * @return 结果 true 存在 false 不存在
+     */
+    public boolean checkDeptExistUser(Long deptId);
+
+    /**
+     * 校验部门名称是否唯一
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    public String checkDeptNameUnique(AppDept dept);
+
+    /**
+     * 校验部门是否有数据权限
+     *
+     * @param deptId 部门id
+     */
+    public void checkDeptDataScope(Long deptId);
+
+    /**
+     * 新增保存部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    public int insertDept(AppDept dept);
+
+    /**
+     * 修改保存部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    public int updateDept(AppDept dept);
+
+    /**
+     * 删除部门管理信息
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    public int deleteDeptById(Long deptId);
+}

+ 295 - 0
fs-service/src/main/java/com/fs/system/service/impl/AppDeptServiceImpl.java

@@ -0,0 +1,295 @@
+package com.fs.system.service.impl;
+
+import com.fs.common.annotation.DataScope;
+import com.fs.common.constant.UserConstants;
+import com.fs.common.core.domain.TreeSelect;
+import com.fs.common.core.domain.entity.AppDept;
+import com.fs.common.core.domain.entity.SysRole;
+import com.fs.common.core.domain.entity.SysUser;
+import com.fs.common.core.text.Convert;
+import com.fs.common.exception.ServiceException;
+import com.fs.common.utils.SecurityUtils;
+import com.fs.common.utils.StringUtils;
+import com.fs.common.utils.spring.SpringUtils;
+import com.fs.system.mapper.AppDeptMapper;
+import com.fs.system.mapper.SysRoleMapper;
+import com.fs.system.service.IAppDeptService;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+import java.util.ArrayList;
+import java.util.Iterator;
+import java.util.List;
+
+
+/**
+ * 部门管理 服务实现
+ */
+@Service
+public class AppDeptServiceImpl implements IAppDeptService {
+    @Autowired
+    private AppDeptMapper deptMapper;
+
+    @Autowired
+    private SysRoleMapper roleMapper;
+
+    /**
+     * 查询部门管理数据
+     *
+     * @param dept 部门信息
+     * @return 部门信息集合
+     */
+    @Override
+    @DataScope(deptAlias = "d")
+    public List<AppDept> selectDeptList(AppDept dept) {
+        return deptMapper.selectDeptList(dept);
+    }
+
+    /**
+     * 构建前端所需要树结构
+     *
+     * @param depts 部门列表
+     * @return 树结构列表
+     */
+    @Override
+    public List<AppDept> buildDeptTree(List<AppDept> depts) {
+        List<AppDept> returnList = new ArrayList<AppDept>();
+        List<Long> tempList = new ArrayList<Long>();
+        for (AppDept dept : depts) {
+            tempList.add(dept.getDeptId());
+        }
+        for (Iterator<AppDept> iterator = depts.iterator(); iterator.hasNext(); ) {
+            AppDept dept = (AppDept) iterator.next();
+            // 如果是顶级节点, 遍历该父节点的所有子节点
+            if (!tempList.contains(dept.getParentId())) {
+                recursionFn(depts, dept);
+                returnList.add(dept);
+            }
+        }
+        if (returnList.isEmpty()) {
+            returnList = depts;
+        }
+        return returnList;
+    }
+
+    /**
+     * 构建前端所需要下拉树结构
+     *
+     * @param depts 部门列表
+     * @return 下拉树结构列表
+     */
+    @Override
+    public List<TreeSelect> buildDeptTreeSelect(List<AppDept> depts) {
+//        List<AppDept> deptTrees = buildDeptTree(depts);
+//        return deptTrees.stream().map(TreeSelect::new).collect(Collectors.toList());
+        return null;
+    }
+
+    /**
+     * 根据角色ID查询部门树信息
+     *
+     * @param roleId 角色ID
+     * @return 选中部门列表
+     */
+    @Override
+    public List<Integer> selectDeptListByRoleId(Long roleId) {
+        SysRole role = roleMapper.selectRoleById(roleId);
+        return deptMapper.selectDeptListByRoleId(roleId, role.isDeptCheckStrictly());
+    }
+
+    /**
+     * 根据部门ID查询信息
+     *
+     * @param deptId 部门ID
+     * @return 部门信息
+     */
+    @Override
+    public AppDept selectDeptById(Long deptId) {
+        return deptMapper.selectDeptById(deptId);
+    }
+
+    /**
+     * 根据ID查询所有子部门(正常状态)
+     *
+     * @param deptId 部门ID
+     * @return 子部门数
+     */
+    @Override
+    public int selectNormalChildrenDeptById(Long deptId) {
+        return deptMapper.selectNormalChildrenDeptById(deptId);
+    }
+
+    /**
+     * 是否存在子节点
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    @Override
+    public boolean hasChildByDeptId(Long deptId) {
+        int result = deptMapper.hasChildByDeptId(deptId);
+        return result > 0 ? true : false;
+    }
+
+    /**
+     * 查询部门是否存在用户
+     *
+     * @param deptId 部门ID
+     * @return 结果 true 存在 false 不存在
+     */
+    @Override
+    public boolean checkDeptExistUser(Long deptId) {
+        int result = deptMapper.checkDeptExistUser(deptId);
+        return result > 0 ? true : false;
+    }
+
+    /**
+     * 校验部门名称是否唯一
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    @Override
+    public String checkDeptNameUnique(AppDept dept) {
+        Long deptId = StringUtils.isNull(dept.getDeptId()) ? -1L : dept.getDeptId();
+        AppDept info = deptMapper.checkDeptNameUnique(dept.getDeptName(), dept.getParentId());
+        if (StringUtils.isNotNull(info) && info.getDeptId().longValue() != deptId.longValue()) {
+            return UserConstants.NOT_UNIQUE;
+        }
+        return UserConstants.UNIQUE;
+    }
+
+    /**
+     * 校验部门是否有数据权限
+     *
+     * @param deptId 部门id
+     */
+    @Override
+    public void checkDeptDataScope(Long deptId) {
+        if (!SysUser.isAdmin(SecurityUtils.getUserId())) {
+            AppDept dept = new AppDept();
+            dept.setDeptId(deptId);
+            List<AppDept> depts = SpringUtils.getAopProxy(this).selectDeptList(dept);
+            if (StringUtils.isEmpty(depts)) {
+                throw new ServiceException("没有权限访问部门数据!");
+            }
+        }
+    }
+
+    /**
+     * 新增保存部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    @Override
+    public int insertDept(AppDept dept) {
+        AppDept info = deptMapper.selectDeptById(dept.getParentId());
+        // 如果父节点不为正常状态,则不允许新增子节点
+        if (!UserConstants.DEPT_NORMAL.equals(info.getStatus())) {
+            throw new ServiceException("部门停用,不允许新增");
+        }
+        dept.setAncestors(info.getAncestors() + "," + dept.getParentId());
+        return deptMapper.insertDept(dept);
+    }
+
+    /**
+     * 修改保存部门信息
+     *
+     * @param dept 部门信息
+     * @return 结果
+     */
+    @Override
+    public int updateDept(AppDept dept) {
+        AppDept newParentDept = deptMapper.selectDeptById(dept.getParentId());
+        AppDept oldDept = deptMapper.selectDeptById(dept.getDeptId());
+        if (StringUtils.isNotNull(newParentDept) && StringUtils.isNotNull(oldDept)) {
+            String newAncestors = newParentDept.getAncestors() + "," + newParentDept.getDeptId();
+            String oldAncestors = oldDept.getAncestors();
+            dept.setAncestors(newAncestors);
+            updateDeptChildren(dept.getDeptId(), newAncestors, oldAncestors);
+        }
+        int result = deptMapper.updateDept(dept);
+        if (UserConstants.DEPT_NORMAL.equals(dept.getStatus()) && StringUtils.isNotEmpty(dept.getAncestors())
+                && !StringUtils.equals("0", dept.getAncestors())) {
+            // 如果该部门是启用状态,则启用该部门的所有上级部门
+            updateParentDeptStatusNormal(dept);
+        }
+        return result;
+    }
+
+    /**
+     * 修改该部门的父级部门状态
+     *
+     * @param dept 当前部门
+     */
+    private void updateParentDeptStatusNormal(AppDept dept) {
+        String ancestors = dept.getAncestors();
+        Long[] deptIds = Convert.toLongArray(ancestors);
+        deptMapper.updateDeptStatusNormal(deptIds);
+    }
+
+    /**
+     * 修改子元素关系
+     *
+     * @param deptId       被修改的部门ID
+     * @param newAncestors 新的父ID集合
+     * @param oldAncestors 旧的父ID集合
+     */
+    public void updateDeptChildren(Long deptId, String newAncestors, String oldAncestors) {
+        List<AppDept> children = deptMapper.selectChildrenDeptById(deptId);
+        for (AppDept child : children) {
+            child.setAncestors(child.getAncestors().replaceFirst(oldAncestors, newAncestors));
+        }
+        if (children.size() > 0) {
+            deptMapper.updateDeptChildren(children);
+        }
+    }
+
+    /**
+     * 删除部门管理信息
+     *
+     * @param deptId 部门ID
+     * @return 结果
+     */
+    @Override
+    public int deleteDeptById(Long deptId) {
+        return deptMapper.deleteDeptById(deptId);
+    }
+
+    /**
+     * 递归列表
+     */
+    private void recursionFn(List<AppDept> list, AppDept t) {
+        // 得到子节点列表
+        List<AppDept> childList = getChildList(list, t);
+        t.setChildren(childList);
+        for (AppDept tChild : childList) {
+            if (hasChild(list, tChild)) {
+                recursionFn(list, tChild);
+            }
+        }
+    }
+
+    /**
+     * 得到子节点列表
+     */
+    private List<AppDept> getChildList(List<AppDept> list, AppDept t) {
+        List<AppDept> tlist = new ArrayList<AppDept>();
+        Iterator<AppDept> it = list.iterator();
+        while (it.hasNext()) {
+            AppDept n = (AppDept) it.next();
+            if (StringUtils.isNotNull(n.getParentId()) && n.getParentId().longValue() == t.getDeptId().longValue()) {
+                tlist.add(n);
+            }
+        }
+        return tlist;
+    }
+
+    /**
+     * 判断是否有子节点
+     */
+    private boolean hasChild(List<AppDept> list, AppDept t) {
+        return getChildList(list, t).size() > 0 ? true : false;
+    }
+}

+ 2 - 2
fs-service/src/main/resources/application-config-druid-hst.yml

@@ -106,12 +106,12 @@ enableRedPackAccount: 1
 
 config:
   debug: true # 部分地方需要通过配置区分是否调试模式,原逻辑无需调整
-  receiver: 4052861305 # 调试课程接收人
+  receiver: 10180 # 调试课程接收人
   sender-prefix: C # IM发送消息,发送人前缀
   receiver-prefix: U # IM发送消息,接收人前缀
   batch-size: 500 # 批量处理数据大小
   load: # 自定义条件加载配置
-    include: "sendSopLogs,generateSopLogs" # 包含需要加载的配置条件,*则是加载全部,如果需要指定加载组件,只需要和组件配置的名称一致即可,多个用英文逗号隔开,不需要用yml数组格式
+    include: "sendSopLogs,generateSopLogs,consumer" # 包含需要加载的配置条件,*则是加载全部,如果需要指定加载组件,只需要和组件配置的名称一致即可,多个用英文逗号隔开,不需要用yml数组格式
 #    exclude: "*" # 排除需要加载的配置条件,*则是排除全部,如果需要指定排除组件,只需要和组件配置的名称一致即可,多个用英文逗号隔开,不需要用yml数组格式,优先级高于include
 
 

+ 54 - 0
fs-service/src/main/resources/mapper/app/AppSopUserLogInfoMapper.xml

@@ -203,4 +203,58 @@
         ON DUPLICATE KEY UPDATE sop_id = sop_id
     </insert>
 
+
+    <select id="findListTotal" resultType="java.lang.Long">
+        SELECT
+        COUNT(1) AS total
+        FROM
+        app_sop_user_logs_infos li
+        <where>
+            li.status = 0
+            <if test="sopId != null and sopId != ''">
+                AND li.sop_id = #{sopId}
+            </if>
+            <if test="userLogsId != null and userLogsId != ''">
+                AND li.user_logs_id = #{userLogsId}
+            </if>
+            <if test="inComTime != null and inComTime != ''">
+                AND DATE(li.create_time) = DATE(#{inComTime})
+            </if>
+            <if test="fsUserId != null or (fsUserName != null and fsUserName != '') or tagIds != null">
+                AND EXISTS (
+                SELECT
+                1
+                FROM
+                fs_user fu
+                <where>
+                    fu.is_del = 0
+                    AND
+                    fu.user_id = li.fs_user_id
+                    AND
+                    fu.history_app IS NOT NULL
+                    <if test="fsUserId != null">
+                        AND fu.user_id = #{fsUserId}
+                    </if>
+                    <if test="fsUserName != null and fsUserName != ''">
+                        AND fu.nick_name LIKE CONCAT('%', #{fsUserName}, '%')
+                    </if>
+                    <if test="tagIds != null">
+                        AND EXISTS (
+                        SELECT
+                        1
+                        FROM
+                        fs_tag ft
+                        INNER JOIN fs_user_tag fut ON fut.tag_id = ft.id AND fu.user_id = fut.user_id
+                        WHERE ft.id IN
+                        <foreach collection='tagIds' item='item' open="(" separator=',' close=")">
+                            #{item}
+                        </foreach>
+                        )
+                    </if>
+                </where>
+                )
+            </if>
+        </where>
+    </select>
+
 </mapper>

+ 19 - 0
fs-service/src/main/resources/mapper/company/CompanyUserMapper.xml

@@ -837,4 +837,23 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
         set cid_server_id = null
         where user_id = #{companyUserId}
     </update>
+
+    <select id="findCompanyUserOptions" resultType="com.fs.company.domain.CompanyUser">
+        SELECT
+        user_id AS userId,
+        nick_name AS nickName
+        FROM
+        company_user
+        WHERE
+        del_flag = '0'
+        <if test="keyword != null and keyword != ''">
+            AND ( user_id = #{keyword} OR nick_name LIKE CONCAT('%', #{keyword}, '%') )
+        </if>
+        <if test="selectedId != null">
+            OR user_id = #{selectedId}
+        </if>
+        ORDER BY
+        create_time DESC
+        LIMIT #{pageNumber}, #{pageSize}
+    </select>
 </mapper>

+ 24 - 0
fs-service/src/main/resources/mapper/course/FsUserCourseVideoMapper.xml

@@ -574,4 +574,28 @@
             #{videoId}
         </foreach>
     </update>
+
+    <select id="findOptions" resultType="com.fs.course.domain.FsUserCourseVideo">
+        SELECT
+        video_id AS videoId,
+        title AS title,
+        thumbnail AS thumbnail
+        FROM
+        fs_user_course_video
+        WHERE
+        open_class = 1
+        <if test="keyword != null and keyword != ''">
+            AND
+            (
+            title LIKE CONCAT('%', #{keyword}, '%')
+            OR
+            video_id = #{keyword}
+            )
+        </if>
+        <if test="metaId != null">
+            OR video_id = #{metaId}
+        </if>
+        LIMIT ${limit}
+    </select>
+
 </mapper>

+ 24 - 0
fs-service/src/main/resources/mapper/course/FsUserVideoMapper.xml

@@ -205,4 +205,28 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
             #{videoId}
         </foreach>
     </update>
+
+    <select id="findOptions" resultType="com.fs.course.domain.FsUserVideo">
+        SELECT
+        video_id AS videoId,
+        title AS title,
+        thumbnail AS thumbnail
+        FROM
+        fs_user_video
+        WHERE
+        is_del = 0
+        <if test="keyword != null and keyword != ''">
+            AND
+            (
+            title LIKE CONCAT('%', #{keyword}, '%')
+            OR
+            video_id = #{keyword}
+            )
+        </if>
+        <if test="metaId != null">
+            OR video_id = #{metaId}
+        </if>
+        LIMIT ${limit}
+    </select>
+
 </mapper>

+ 29 - 5
fs-service/src/main/resources/mapper/his/FsArticleMapper.xml

@@ -3,7 +3,7 @@
 PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
 "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
 <mapper namespace="com.fs.his.mapper.FsArticleMapper">
-    
+
     <resultMap type="FsArticle" id="FsArticleResult">
         <result property="articleId"    column="article_id"    />
         <result property="cateId"    column="cate_id"    />
@@ -25,7 +25,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
 
     <select id="selectFsArticleList" parameterType="FsArticle" resultMap="FsArticleResult">
         <include refid="selectFsArticleVo"/>
-        <where>  
+        <where>
             <if test="cateId != null "> and cate_id = #{cateId}</if>
             <if test="title != null  and title != ''"> and title like concat('%', #{title}, '%')</if>
             <if test="imageUrl != null  and imageUrl != ''"> and image_url = #{imageUrl}</if>
@@ -38,7 +38,7 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
         </where>
         order by article_id desc
     </select>
-    
+
     <select id="selectFsArticleByArticleId" parameterType="Long" resultMap="FsArticleResult">
         <include refid="selectFsArticleVo"/>
         where article_id = #{articleId}
@@ -115,9 +115,33 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
     </delete>
 
     <delete id="deleteFsArticleByArticleIds" parameterType="String">
-        delete from fs_article where article_id in 
+        delete from fs_article where article_id in
         <foreach item="articleId" collection="array" open="(" separator="," close=")">
             #{articleId}
         </foreach>
     </delete>
-</mapper>
+
+    <select id="findOptions" resultType="com.fs.his.domain.FsArticle">
+        SELECT
+        article_id AS articleId,
+        title AS title,
+        image_url AS imageUrl
+        FROM
+        fs_article
+        <where>
+            <if test="keyword != null and keyword != ''">
+                AND
+                (
+                title LIKE CONCAT('%', #{keyword}, '%')
+                OR
+                article_id = #{keyword}
+                )
+            </if>
+            <if test="metaId != null">
+                OR article_id = #{metaId}
+            </if>
+        </where>
+        LIMIT ${limit}
+    </select>
+
+</mapper>

+ 28 - 0
fs-service/src/main/resources/mapper/his/FsPackageMapper.xml

@@ -291,4 +291,32 @@ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
             #{packageId}
         </foreach>
     </delete>
+
+    <select id="getOptions" resultType="com.fs.his.vo.FsPackageListVO">
+        SELECT
+        package_id AS packageId,
+        package_name AS packageName,
+        img_url AS imgUrl,
+        create_time
+        FROM
+        fs_package
+        WHERE
+        status = 1
+        AND
+        is_del = 0
+        <if test="keyword != null and keyword != ''">
+            AND (
+            package_id = CAST(#{keyword} AS UNSIGNED )
+            OR
+            package_name LIKE CONCAT( '%', #{keyword}, '%')
+            )
+        </if>
+        <if test="metaId != null">
+            OR package_id = #{metaId}
+        </if>
+        ORDER BY
+        create_time DESC
+        LIMIT #{limit}
+    </select>
+
 </mapper>

+ 180 - 0
fs-service/src/main/resources/mapper/system/AppDeptMapper.xml

@@ -0,0 +1,180 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<!DOCTYPE mapper
+        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
+        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
+<mapper namespace="com.fs.system.mapper.AppDeptMapper">
+
+    <resultMap type="AppDept" id="AppDeptResult">
+        <id property="deptId" column="dept_id"/>
+        <result property="parentId" column="parent_id"/>
+        <result property="ancestors" column="ancestors"/>
+        <result property="deptName" column="dept_name"/>
+        <result property="orderNum" column="order_num"/>
+        <result property="leader" column="leader"/>
+        <result property="phone" column="phone"/>
+        <result property="email" column="email"/>
+        <result property="status" column="status"/>
+        <result property="delFlag" column="del_flag"/>
+        <result property="parentName" column="parent_name"/>
+        <result property="createBy" column="create_by"/>
+        <result property="createTime" column="create_time"/>
+        <result property="updateBy" column="update_by"/>
+        <result property="updateTime" column="update_time"/>
+    </resultMap>
+
+    <sql id="selectDeptVo">
+        select d.dept_id,
+               d.parent_id,
+               d.ancestors,
+               d.dept_name,
+               d.order_num,
+               d.leader,
+               d.phone,
+               d.email,
+               d.status,
+               d.del_flag,
+               d.create_by,
+               d.create_time
+        from App_dept d
+    </sql>
+
+    <select id="selectDeptList" parameterType="AppDept" resultMap="AppDeptResult">
+        <include refid="selectDeptVo"/>
+        where d.del_flag = '0'
+        <if test="deptId != null and deptId != 0">
+            AND dept_id = #{deptId}
+        </if>
+        <if test="parentId != null and parentId != 0">
+            AND parent_id = #{parentId}
+        </if>
+        <if test="deptName != null and deptName != ''">
+            AND dept_name like concat('%', #{deptName}, '%')
+        </if>
+        <if test="status != null and status != ''">
+            AND status = #{status}
+        </if>
+        <!-- 数据范围过滤 -->
+        ${params.dataScope}
+        order by d.parent_id, d.order_num
+    </select>
+
+    <select id="selectDeptListByRoleId" resultType="Integer">
+        <!--		select d.dept_id-->
+        <!--		from app_dept d-->
+        <!--            left join sys_role_dept rd on d.dept_id = rd.dept_id-->
+        <!--        where rd.role_id = #{roleId}-->
+        <!--            <if test="deptCheckStrictly">-->
+        <!--              and d.dept_id not in (select d.parent_id from sys_dept d inner join sys_role_dept rd on d.dept_id = rd.dept_id and rd.role_id = #{roleId})-->
+        <!--            </if>-->
+        <!--		order by d.parent_id, d.order_num-->
+    </select>
+
+    <select id="selectDeptById" parameterType="Long" resultMap="AppDeptResult">
+        <include refid="selectDeptVo"/>
+        where dept_id = #{deptId}
+    </select>
+
+    <select id="checkDeptExistUser" parameterType="Long" resultType="int">
+        select count(1)
+        from app_invitation_code
+        where dept_id = #{deptId}
+    </select>
+
+    <select id="hasChildByDeptId" parameterType="Long" resultType="int">
+        select count(1)
+        from app_dept
+        where del_flag = '0'
+          and parent_id = #{deptId} limit 1
+    </select>
+
+    <select id="selectChildrenDeptById" parameterType="Long" resultMap="AppDeptResult">
+        select *
+        from app_dept
+        where find_in_set(#{deptId}, ancestors)
+    </select>
+
+    <select id="selectNormalChildrenDeptById" parameterType="Long" resultType="int">
+        select count(*)
+        from app_dept
+        where status = 0
+          and del_flag = '0'
+          and find_in_set(#{deptId}, ancestors)
+    </select>
+
+    <select id="checkDeptNameUnique" resultMap="AppDeptResult">
+        <include refid="selectDeptVo"/>
+        where dept_name=#{deptName} and parent_id = #{parentId} and del_flag = '0' limit 1
+    </select>
+
+    <insert id="insertDept" parameterType="AppDept">
+        insert into app_dept(
+        <if test="deptId != null and deptId != 0">dept_id,</if>
+        <if test="parentId != null and parentId != 0">parent_id,</if>
+        <if test="deptName != null and deptName != ''">dept_name,</if>
+        <if test="ancestors != null and ancestors != ''">ancestors,</if>
+        <if test="orderNum != null and orderNum != ''">order_num,</if>
+        <if test="leader != null and leader != ''">leader,</if>
+        <if test="phone != null and phone != ''">phone,</if>
+        <if test="email != null and email != ''">email,</if>
+        <if test="status != null">status,</if>
+        <if test="createBy != null and createBy != ''">create_by,</if>
+        create_time
+        )values(
+        <if test="deptId != null and deptId != 0">#{deptId},</if>
+        <if test="parentId != null and parentId != 0">#{parentId},</if>
+        <if test="deptName != null and deptName != ''">#{deptName},</if>
+        <if test="ancestors != null and ancestors != ''">#{ancestors},</if>
+        <if test="orderNum != null and orderNum != ''">#{orderNum},</if>
+        <if test="leader != null and leader != ''">#{leader},</if>
+        <if test="phone != null and phone != ''">#{phone},</if>
+        <if test="email != null and email != ''">#{email},</if>
+        <if test="status != null">#{status},</if>
+        <if test="createBy != null and createBy != ''">#{createBy},</if>
+        now()
+        )
+    </insert>
+
+    <update id="updateDept" parameterType="AppDept">
+        update app_dept
+        <set>
+            <if test="parentId != null and parentId != 0">parent_id = #{parentId},</if>
+            <if test="deptName != null and deptName != ''">dept_name = #{deptName},</if>
+            <if test="ancestors != null and ancestors != ''">ancestors = #{ancestors},</if>
+            <if test="orderNum != null and orderNum != ''">order_num = #{orderNum},</if>
+            <if test="leader != null">leader = #{leader},</if>
+            <if test="phone != null">phone = #{phone},</if>
+            <if test="email != null">email = #{email},</if>
+            <if test="status != null and status != ''">status = #{status},</if>
+            <if test="updateBy != null and updateBy != ''">update_by = #{updateBy},</if>
+            update_time = now()
+        </set>
+        where dept_id = #{deptId}
+    </update>
+
+    <update id="updateDeptChildren" parameterType="java.util.List">
+        update app_dept set ancestors =
+        <foreach collection="depts" item="item" index="index"
+                 separator=" " open="case dept_id" close="end">
+            when #{item.deptId} then #{item.ancestors}
+        </foreach>
+        where dept_id in
+        <foreach collection="depts" item="item" index="index"
+                 separator="," open="(" close=")">
+            #{item.deptId}
+        </foreach>
+    </update>
+
+    <update id="updateDeptStatusNormal" parameterType="Long">
+        update app_dept set status = '0' where dept_id in
+        <foreach collection="array" item="deptId" open="(" separator="," close=")">
+            #{deptId}
+        </foreach>
+    </update>
+
+    <delete id="deleteDeptById" parameterType="Long">
+        update app_dept
+        set del_flag = '2'
+        where dept_id = #{deptId}
+    </delete>
+
+</mapper>